ladybird/Userland/Applets/WorkspacePicker/DesktopStatusWindow.cpp
thankyouverycool 1f1ac99ef9 WindowServer+WorkspacePicker: Adjust active and inactive colors
Now uses the Selection ColorRole for the active desktop and a
slightly darkened Window for inactive ones. Several themes use
the same color for thread highlighting and inactive windows which
was causing frames to draw without the correct perception of depth.
2022-08-16 16:41:47 +02:00

162 lines
5.3 KiB
C++

/*
* Copyright (c) 2021, Peter Elliott <pelliott@serenityos.org>
* Copyright (c) 2022, the SerenityOS developers.
* Copyright (c) 2022, Jakob-Niklas See <git@nwex.de>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include "DesktopStatusWindow.h"
#include <LibCore/Process.h>
#include <LibGUI/ConnectionToWindowManagerServer.h>
#include <LibGUI/Desktop.h>
#include <LibGUI/Menu.h>
#include <LibGUI/Painter.h>
#include <LibGUI/Widget.h>
#include <LibGfx/Palette.h>
class DesktopStatusWidget : public GUI::Widget {
C_OBJECT(DesktopStatusWidget);
public:
virtual ~DesktopStatusWidget() override = default;
Gfx::IntRect rect_for_desktop(unsigned row, unsigned column) const
{
auto& desktop = GUI::Desktop::the();
auto workspace_columns = desktop.workspace_columns();
auto workspace_rows = desktop.workspace_rows();
auto desktop_width = (width() - gap() * (workspace_columns - 1)) / workspace_columns;
auto desktop_height = (height() - gap() * (workspace_rows - 1)) / workspace_rows;
return {
column * (desktop_width + gap()), row * (desktop_height + gap()),
desktop_width, desktop_height
};
}
virtual void paint_event(GUI::PaintEvent& event) override
{
GUI::Widget::paint_event(event);
GUI::Painter painter(*this);
painter.add_clip_rect(event.rect());
painter.fill_rect({ 0, 0, width(), height() }, palette().button());
auto& desktop = GUI::Desktop::the();
auto active_color = palette().selection();
auto inactive_color = palette().window().darkened(0.9f);
for (unsigned row = 0; row < desktop.workspace_rows(); ++row) {
for (unsigned column = 0; column < desktop.workspace_columns(); ++column) {
auto rect = rect_for_desktop(row, column);
painter.fill_rect(rect,
(row == current_row() && column == current_column()) ? active_color : inactive_color);
Gfx::StylePainter::current().paint_frame(painter, rect, palette(), Gfx::FrameShape::Container, Gfx::FrameShadow::Sunken, 1);
}
}
}
virtual void mousedown_event(GUI::MouseEvent& event) override
{
if (event.button() != GUI::MouseButton::Primary)
return;
auto base_rect = rect_for_desktop(0, 0);
auto row = event.y() / (base_rect.height() + gap());
auto column = event.x() / (base_rect.width() + gap());
// Handle case where divider is clicked.
if (rect_for_desktop(row, column).contains(event.position())) {
GUI::ConnectionToWindowManagerServer::the().async_set_workspace(row, column);
set_current_row(row);
set_current_column(column);
update();
}
}
virtual void mousewheel_event(GUI::MouseEvent& event) override
{
auto& desktop = GUI::Desktop::the();
auto column = current_column();
auto row = current_row();
auto workspace_columns = desktop.workspace_columns();
auto workspace_rows = desktop.workspace_rows();
auto direction = event.wheel_delta_y() < 0 ? 1 : -1;
if (event.modifiers() & Mod_Shift)
column = abs((int)column + direction) % workspace_columns;
else
row = abs((int)row + direction) % workspace_rows;
set_current_row(row);
set_current_column(column);
update();
GUI::ConnectionToWindowManagerServer::the().async_set_workspace(row, column);
}
virtual void context_menu_event(GUI::ContextMenuEvent& event) override
{
event.accept();
if (!m_context_menu) {
m_context_menu = GUI::Menu::construct();
auto settings_icon = MUST(Gfx::Bitmap::try_load_from_file("/res/icons/16x16/settings.png"sv));
auto open_workspace_settings_action = GUI::Action::create("Workspace &Settings", *settings_icon, [](auto&) {
auto result = Core::Process::spawn("/bin/DisplaySettings"sv, Array { "--open-tab", "workspaces" }.span());
if (result.is_error()) {
dbgln("Failed to launch DisplaySettings");
}
});
m_context_menu->add_action(open_workspace_settings_action);
}
m_context_menu->popup(event.screen_position());
}
unsigned
current_row() const
{
return m_current_row;
}
void set_current_row(unsigned row) { m_current_row = row; }
unsigned current_column() const { return m_current_column; }
void set_current_column(unsigned column) { m_current_column = column; }
unsigned gap() const { return m_gap; }
private:
DesktopStatusWidget() = default;
unsigned m_gap { 1 };
unsigned m_current_row { 0 };
unsigned m_current_column { 0 };
RefPtr<GUI::Menu> m_context_menu;
};
DesktopStatusWindow::DesktopStatusWindow()
{
set_window_type(GUI::WindowType::Applet);
set_has_alpha_channel(true);
m_widget = &set_main_widget<DesktopStatusWidget>();
}
void DesktopStatusWindow::wm_event(GUI::WMEvent& event)
{
if (event.type() == GUI::Event::WM_WorkspaceChanged) {
auto& changed_event = static_cast<GUI::WMWorkspaceChangedEvent&>(event);
m_widget->set_current_row(changed_event.current_row());
m_widget->set_current_column(changed_event.current_column());
update();
}
}