mirror of
https://github.com/RGBCube/serenity
synced 2025-10-24 18:42:35 +00:00
162 lines
5.4 KiB
C++
162 lines
5.4 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::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::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>().release_value_but_fixme_should_propagate_errors();
|
|
}
|
|
|
|
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();
|
|
}
|
|
}
|