serenity/Userland/Applets/WorkspacePicker/DesktopStatusWindow.cpp

163 lines
5.3 KiB
C++

/*
* Copyright (c) 2021, Peter Elliott <pelliott@serenityos.org>
* Copyright (c) 2022, the SerenityOS developers.
* Copyright (c) 2022, networkException <networkexception@serenityos.org>
*
* 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::FrameStyle::SunkenPanel);
}
}
}
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::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();
}
}