2018-10-11 17:52:40 +03:00
|
|
|
#pragma once
|
|
|
|
|
2019-01-20 01:22:46 +03:00
|
|
|
#include <SharedGraphics/Rect.h>
|
|
|
|
#include <SharedGraphics/Color.h>
|
|
|
|
#include <SharedGraphics/Painter.h>
|
2019-02-19 16:49:23 +03:00
|
|
|
#include <SharedGraphics/DisjointRectSet.h>
|
2018-10-11 17:52:40 +03:00
|
|
|
#include <AK/HashTable.h>
|
2019-01-09 05:16:58 +03:00
|
|
|
#include <AK/InlineLinkedList.h>
|
2018-10-13 18:00:45 +03:00
|
|
|
#include <AK/WeakPtr.h>
|
2019-02-11 11:47:10 +03:00
|
|
|
#include <AK/HashMap.h>
|
2019-01-26 07:28:02 +03:00
|
|
|
#include "WSMessageReceiver.h"
|
2019-02-11 11:47:10 +03:00
|
|
|
#include "WSMenuBar.h"
|
2019-03-03 17:17:05 +03:00
|
|
|
#include <WindowServer/WSWindowSwitcher.h>
|
2019-02-12 13:53:45 +03:00
|
|
|
#include <WindowServer/WSWindowType.h>
|
2019-03-03 17:17:05 +03:00
|
|
|
#include <WindowServer/WSWindow.h>
|
2019-03-31 23:09:10 +03:00
|
|
|
#include <WindowServer/WSCursor.h>
|
2019-02-26 04:17:25 +03:00
|
|
|
#include <AK/CircularQueue.h>
|
2018-10-11 17:52:40 +03:00
|
|
|
|
2019-02-13 20:48:22 +03:00
|
|
|
class WSAPIClientRequest;
|
2019-01-18 07:41:15 +03:00
|
|
|
class WSScreen;
|
2019-02-11 11:47:10 +03:00
|
|
|
class WSMenuBar;
|
2019-01-21 09:28:04 +03:00
|
|
|
class WSMouseEvent;
|
2019-01-26 07:45:47 +03:00
|
|
|
class WSClientWantsToPaintMessage;
|
2019-01-16 18:03:50 +03:00
|
|
|
class WSWindow;
|
2019-02-14 10:22:47 +03:00
|
|
|
class WSClientConnection;
|
2019-03-03 17:17:05 +03:00
|
|
|
class WSWindowSwitcher;
|
2019-01-12 05:42:50 +03:00
|
|
|
class GraphicsBitmap;
|
2018-10-12 02:10:16 +03:00
|
|
|
|
2019-02-20 17:34:55 +03:00
|
|
|
enum class ResizeDirection { None, Left, UpLeft, Up, UpRight, Right, DownRight, Down, DownLeft };
|
2019-02-13 03:04:11 +03:00
|
|
|
|
2019-01-26 07:28:02 +03:00
|
|
|
class WSWindowManager : public WSMessageReceiver {
|
2019-03-03 17:17:05 +03:00
|
|
|
friend class WSWindowSwitcher;
|
2018-10-11 17:52:40 +03:00
|
|
|
public:
|
2019-01-16 18:03:50 +03:00
|
|
|
static WSWindowManager& the();
|
2019-02-17 02:13:47 +03:00
|
|
|
|
|
|
|
WSWindowManager();
|
|
|
|
virtual ~WSWindowManager() override;
|
|
|
|
|
2019-01-18 07:33:35 +03:00
|
|
|
void add_window(WSWindow&);
|
|
|
|
void remove_window(WSWindow&);
|
2018-10-11 17:52:40 +03:00
|
|
|
|
2019-01-18 07:33:35 +03:00
|
|
|
void notify_title_changed(WSWindow&);
|
|
|
|
void notify_rect_changed(WSWindow&, const Rect& oldRect, const Rect& newRect);
|
2019-02-14 10:22:47 +03:00
|
|
|
void notify_client_changed_app_menubar(WSClientConnection&);
|
2018-10-11 17:52:40 +03:00
|
|
|
|
2019-01-18 17:55:28 +03:00
|
|
|
WSWindow* active_window() { return m_active_window.ptr(); }
|
2019-02-17 10:54:57 +03:00
|
|
|
const WSClientConnection* active_client() const;
|
2018-10-13 18:42:24 +03:00
|
|
|
|
2019-03-03 17:17:05 +03:00
|
|
|
WSWindow* highlight_window() { return m_highlight_window.ptr(); }
|
|
|
|
void set_highlight_window(WSWindow*);
|
|
|
|
|
2019-03-24 15:09:46 +03:00
|
|
|
void move_to_front_and_make_active(WSWindow&);
|
2018-10-14 02:23:01 +03:00
|
|
|
|
2019-02-07 10:53:57 +03:00
|
|
|
void invalidate_cursor();
|
2019-01-13 09:41:31 +03:00
|
|
|
void draw_cursor();
|
2019-02-11 11:47:10 +03:00
|
|
|
void draw_menubar();
|
2019-03-03 17:17:05 +03:00
|
|
|
void draw_window_switcher();
|
2019-02-11 11:47:10 +03:00
|
|
|
|
|
|
|
Rect menubar_rect() const;
|
2019-02-14 10:22:47 +03:00
|
|
|
WSMenuBar* current_menubar() { return m_current_menubar.ptr(); }
|
2019-02-11 11:47:10 +03:00
|
|
|
void set_current_menubar(WSMenuBar*);
|
2019-02-14 10:22:47 +03:00
|
|
|
WSMenu* current_menu() { return m_current_menu.ptr(); }
|
2019-02-11 11:47:10 +03:00
|
|
|
void set_current_menu(WSMenu*);
|
2019-01-11 05:52:09 +03:00
|
|
|
|
2019-01-16 18:03:50 +03:00
|
|
|
void invalidate(const WSWindow&);
|
2019-01-18 06:37:49 +03:00
|
|
|
void invalidate(const WSWindow&, const Rect&);
|
2019-02-17 15:12:59 +03:00
|
|
|
void invalidate(const Rect&, bool should_schedule_compose_event = true);
|
2019-01-12 04:57:14 +03:00
|
|
|
void invalidate();
|
2019-02-17 15:12:59 +03:00
|
|
|
void recompose_immediately();
|
2019-01-12 08:39:34 +03:00
|
|
|
void flush(const Rect&);
|
2019-01-12 04:57:14 +03:00
|
|
|
|
2019-03-09 23:24:12 +03:00
|
|
|
const Font& font() const;
|
|
|
|
const Font& window_title_font() const;
|
|
|
|
const Font& menu_font() const;
|
|
|
|
const Font& app_menu_font() const;
|
2019-02-04 13:37:15 +03:00
|
|
|
|
2019-02-11 15:59:26 +03:00
|
|
|
void close_menu(WSMenu&);
|
2019-02-14 10:22:47 +03:00
|
|
|
void close_menubar(WSMenuBar&);
|
2019-02-12 10:39:19 +03:00
|
|
|
Color menu_selection_color() const { return m_menu_selection_color; }
|
2019-02-13 11:10:32 +03:00
|
|
|
int menubar_menu_margin() const;
|
2019-02-11 15:59:26 +03:00
|
|
|
|
2019-02-17 15:12:59 +03:00
|
|
|
void set_resolution(int width, int height);
|
|
|
|
|
2019-03-21 17:54:19 +03:00
|
|
|
bool set_wallpaper(const String& path);
|
|
|
|
String wallpaper_path() const { return m_wallpaper_path; }
|
|
|
|
|
2019-03-31 23:27:37 +03:00
|
|
|
const WSCursor& active_cursor() const;
|
2019-03-31 23:09:10 +03:00
|
|
|
Rect current_cursor_rect() const;
|
|
|
|
|
2019-04-02 03:34:09 +03:00
|
|
|
const WSCursor& arrow_cursor() const { return *m_arrow_cursor; }
|
|
|
|
const WSCursor& resize_horizontally_cursor() const { return *m_resize_horizontally_cursor; }
|
|
|
|
const WSCursor& resize_vertically_cursor() const { return *m_resize_vertically_cursor; }
|
|
|
|
const WSCursor& resize_diagonally_tlbr_cursor() const { return *m_resize_diagonally_tlbr_cursor; }
|
|
|
|
const WSCursor& resize_diagonally_bltr_cursor() const { return *m_resize_diagonally_bltr_cursor; }
|
|
|
|
const WSCursor& i_beam_cursor() const { return *m_i_beam_cursor; }
|
|
|
|
const WSCursor& disallowed_cursor() const { return *m_disallowed_cursor; }
|
|
|
|
const WSCursor& move_cursor() const { return *m_move_cursor; }
|
|
|
|
|
2019-04-04 15:38:53 +03:00
|
|
|
void set_active_window(WSWindow*);
|
|
|
|
|
2018-10-11 17:52:40 +03:00
|
|
|
private:
|
2019-04-01 20:14:57 +03:00
|
|
|
void process_mouse_event(const WSMouseEvent&, WSWindow*& event_window);
|
|
|
|
bool process_ongoing_window_resize(const WSMouseEvent&, WSWindow*& event_window);
|
|
|
|
bool process_ongoing_window_drag(const WSMouseEvent&, WSWindow*& event_window);
|
|
|
|
void handle_menu_mouse_event(WSMenu&, const WSMouseEvent&);
|
|
|
|
void handle_menubar_mouse_event(const WSMouseEvent&);
|
|
|
|
void handle_close_button_mouse_event(WSWindow&, const WSMouseEvent&);
|
|
|
|
void start_window_resize(WSWindow&, const WSMouseEvent&);
|
|
|
|
void start_window_drag(WSWindow&, const WSMouseEvent&);
|
|
|
|
void handle_client_request(const WSAPIClientRequest&);
|
2019-02-20 12:12:19 +03:00
|
|
|
void set_hovered_window(WSWindow*);
|
2019-02-13 03:04:11 +03:00
|
|
|
template<typename Callback> IterationDecision for_each_visible_window_of_type_from_back_to_front(WSWindowType, Callback);
|
|
|
|
template<typename Callback> IterationDecision for_each_visible_window_of_type_from_front_to_back(WSWindowType, Callback);
|
|
|
|
template<typename Callback> IterationDecision for_each_visible_window_from_front_to_back(Callback);
|
|
|
|
template<typename Callback> IterationDecision for_each_visible_window_from_back_to_front(Callback);
|
2019-04-04 02:44:35 +03:00
|
|
|
template<typename Callback> void for_each_window_listening_to_wm_events(Callback);
|
2019-04-04 14:33:09 +03:00
|
|
|
template<typename Callback> void for_each_window(Callback);
|
2019-02-12 10:47:38 +03:00
|
|
|
template<typename Callback> void for_each_active_menubar_menu(Callback);
|
2019-02-11 11:47:10 +03:00
|
|
|
void close_current_menu();
|
2019-04-01 20:14:57 +03:00
|
|
|
virtual void on_message(const WSMessage&) override;
|
2019-01-12 18:51:14 +03:00
|
|
|
void compose();
|
2019-04-01 20:14:57 +03:00
|
|
|
void paint_window_frame(const WSWindow&);
|
2019-02-07 10:53:57 +03:00
|
|
|
void flip_buffers();
|
2019-02-26 04:17:25 +03:00
|
|
|
void tick_clock();
|
2019-04-04 14:19:26 +03:00
|
|
|
void tell_wm_listeners_window_state_changed(WSWindow&);
|
2019-04-05 16:01:28 +03:00
|
|
|
void tell_wm_listener_about_window(WSWindow& listener, WSWindow&);
|
2019-01-12 18:51:14 +03:00
|
|
|
|
2019-01-18 07:41:15 +03:00
|
|
|
WSScreen& m_screen;
|
2019-01-12 18:51:14 +03:00
|
|
|
Rect m_screen_rect;
|
|
|
|
|
2019-01-25 05:48:50 +03:00
|
|
|
Color m_background_color;
|
2019-01-18 07:33:35 +03:00
|
|
|
Color m_active_window_border_color;
|
2019-01-25 07:01:27 +03:00
|
|
|
Color m_active_window_border_color2;
|
2019-01-18 07:33:35 +03:00
|
|
|
Color m_active_window_title_color;
|
|
|
|
Color m_inactive_window_border_color;
|
2019-01-25 07:01:27 +03:00
|
|
|
Color m_inactive_window_border_color2;
|
2019-01-18 07:33:35 +03:00
|
|
|
Color m_inactive_window_title_color;
|
2019-01-18 17:55:28 +03:00
|
|
|
Color m_dragging_window_border_color;
|
2019-01-25 07:01:27 +03:00
|
|
|
Color m_dragging_window_border_color2;
|
2019-01-18 17:55:28 +03:00
|
|
|
Color m_dragging_window_title_color;
|
2019-03-03 17:17:05 +03:00
|
|
|
Color m_highlight_window_border_color;
|
|
|
|
Color m_highlight_window_border_color2;
|
|
|
|
Color m_highlight_window_title_color;
|
2019-01-18 17:55:28 +03:00
|
|
|
|
2019-02-14 03:21:32 +03:00
|
|
|
HashMap<int, OwnPtr<WSWindow>> m_windows_by_id;
|
2019-01-16 18:03:50 +03:00
|
|
|
HashTable<WSWindow*> m_windows;
|
|
|
|
InlineLinkedList<WSWindow> m_windows_in_order;
|
2018-10-12 03:24:05 +03:00
|
|
|
|
2019-01-17 19:38:04 +03:00
|
|
|
WeakPtr<WSWindow> m_active_window;
|
2019-02-20 12:12:19 +03:00
|
|
|
WeakPtr<WSWindow> m_hovered_window;
|
2019-03-03 17:17:05 +03:00
|
|
|
WeakPtr<WSWindow> m_highlight_window;
|
2018-10-13 18:00:45 +03:00
|
|
|
|
2019-02-20 17:34:55 +03:00
|
|
|
WeakPtr<WSWindow> m_drag_window;
|
2019-01-18 07:33:35 +03:00
|
|
|
Point m_drag_origin;
|
|
|
|
Point m_drag_window_origin;
|
2019-01-09 06:18:28 +03:00
|
|
|
|
2019-02-20 17:34:55 +03:00
|
|
|
WeakPtr<WSWindow> m_resize_window;
|
|
|
|
Rect m_resize_window_original_rect;
|
|
|
|
Point m_resize_origin;
|
|
|
|
ResizeDirection m_resize_direction { ResizeDirection::None };
|
|
|
|
|
2019-01-12 08:39:34 +03:00
|
|
|
Rect m_last_cursor_rect;
|
2019-01-11 05:52:09 +03:00
|
|
|
|
2019-01-19 16:33:59 +03:00
|
|
|
unsigned m_compose_count { 0 };
|
2019-01-16 19:47:18 +03:00
|
|
|
unsigned m_flush_count { 0 };
|
2019-01-12 04:57:14 +03:00
|
|
|
|
2019-01-12 08:39:34 +03:00
|
|
|
RetainPtr<GraphicsBitmap> m_front_bitmap;
|
|
|
|
RetainPtr<GraphicsBitmap> m_back_bitmap;
|
2019-01-12 05:42:50 +03:00
|
|
|
|
2019-02-19 16:49:23 +03:00
|
|
|
DisjointRectSet m_dirty_rects;
|
2019-01-12 08:47:55 +03:00
|
|
|
|
|
|
|
bool m_pending_compose_event { false };
|
2019-01-13 09:17:12 +03:00
|
|
|
|
2019-03-31 23:09:10 +03:00
|
|
|
RetainPtr<WSCursor> m_arrow_cursor;
|
|
|
|
RetainPtr<WSCursor> m_resize_horizontally_cursor;
|
|
|
|
RetainPtr<WSCursor> m_resize_vertically_cursor;
|
|
|
|
RetainPtr<WSCursor> m_resize_diagonally_tlbr_cursor;
|
|
|
|
RetainPtr<WSCursor> m_resize_diagonally_bltr_cursor;
|
|
|
|
RetainPtr<WSCursor> m_i_beam_cursor;
|
|
|
|
RetainPtr<WSCursor> m_disallowed_cursor;
|
2019-03-31 23:42:13 +03:00
|
|
|
RetainPtr<WSCursor> m_move_cursor;
|
2019-01-13 09:41:31 +03:00
|
|
|
|
|
|
|
OwnPtr<Painter> m_back_painter;
|
|
|
|
OwnPtr<Painter> m_front_painter;
|
2019-01-16 18:03:50 +03:00
|
|
|
|
2019-02-17 02:13:47 +03:00
|
|
|
String m_wallpaper_path;
|
2019-02-08 10:45:59 +03:00
|
|
|
RetainPtr<GraphicsBitmap> m_wallpaper;
|
|
|
|
|
2019-02-17 02:13:47 +03:00
|
|
|
bool m_flash_flush { false };
|
2019-02-07 10:53:57 +03:00
|
|
|
bool m_buffers_are_flipped { false };
|
2019-02-11 11:47:10 +03:00
|
|
|
|
2019-03-03 14:56:48 +03:00
|
|
|
byte m_keyboard_modifiers { 0 };
|
|
|
|
|
2019-02-12 10:39:19 +03:00
|
|
|
OwnPtr<WSMenu> m_system_menu;
|
|
|
|
Color m_menu_selection_color;
|
2019-02-14 10:22:47 +03:00
|
|
|
WeakPtr<WSMenuBar> m_current_menubar;
|
|
|
|
WeakPtr<WSMenu> m_current_menu;
|
2019-02-26 04:17:25 +03:00
|
|
|
|
2019-03-03 17:17:05 +03:00
|
|
|
WSWindowSwitcher m_switcher;
|
|
|
|
|
2019-02-26 04:17:25 +03:00
|
|
|
CircularQueue<float, 30> m_cpu_history;
|
2019-03-10 05:16:27 +03:00
|
|
|
|
|
|
|
String m_username;
|
2018-10-11 17:52:40 +03:00
|
|
|
};
|
2019-03-03 17:17:05 +03:00
|
|
|
|
|
|
|
template<typename Callback>
|
|
|
|
IterationDecision WSWindowManager::for_each_visible_window_of_type_from_back_to_front(WSWindowType type, Callback callback)
|
|
|
|
{
|
2019-03-10 04:33:43 +03:00
|
|
|
bool do_highlight_window_at_end = false;
|
2019-03-03 17:17:05 +03:00
|
|
|
for (auto* window = m_windows_in_order.head(); window; window = window->next()) {
|
|
|
|
if (!window->is_visible())
|
|
|
|
continue;
|
|
|
|
if (window->type() != type)
|
|
|
|
continue;
|
2019-03-10 04:33:43 +03:00
|
|
|
if (m_highlight_window.ptr() == window) {
|
|
|
|
do_highlight_window_at_end = true;
|
|
|
|
continue;
|
|
|
|
}
|
2019-03-03 17:17:05 +03:00
|
|
|
if (callback(*window) == IterationDecision::Abort)
|
|
|
|
return IterationDecision::Abort;
|
|
|
|
}
|
2019-03-10 04:33:43 +03:00
|
|
|
if (do_highlight_window_at_end) {
|
|
|
|
if (callback(*m_highlight_window) == IterationDecision::Abort)
|
|
|
|
return IterationDecision::Abort;
|
|
|
|
}
|
2019-03-03 17:17:05 +03:00
|
|
|
return IterationDecision::Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename Callback>
|
|
|
|
IterationDecision WSWindowManager::for_each_visible_window_from_back_to_front(Callback callback)
|
|
|
|
{
|
|
|
|
if (for_each_visible_window_of_type_from_back_to_front(WSWindowType::Normal, callback) == IterationDecision::Abort)
|
|
|
|
return IterationDecision::Abort;
|
2019-03-10 15:30:43 +03:00
|
|
|
if (for_each_visible_window_of_type_from_back_to_front(WSWindowType::Menu, callback) == IterationDecision::Abort)
|
|
|
|
return IterationDecision::Abort;
|
2019-04-03 20:38:44 +03:00
|
|
|
if (for_each_visible_window_of_type_from_back_to_front(WSWindowType::Taskbar, callback) == IterationDecision::Abort)
|
|
|
|
return IterationDecision::Abort;
|
2019-03-10 15:30:43 +03:00
|
|
|
return for_each_visible_window_of_type_from_back_to_front(WSWindowType::WindowSwitcher, callback);
|
2019-03-03 17:17:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename Callback>
|
|
|
|
IterationDecision WSWindowManager::for_each_visible_window_of_type_from_front_to_back(WSWindowType type, Callback callback)
|
|
|
|
{
|
2019-03-10 04:33:43 +03:00
|
|
|
if (m_highlight_window && m_highlight_window->type() == type && m_highlight_window->is_visible()) {
|
|
|
|
if (callback(*m_highlight_window) == IterationDecision::Abort)
|
|
|
|
return IterationDecision::Abort;
|
|
|
|
}
|
|
|
|
|
2019-03-03 17:17:05 +03:00
|
|
|
for (auto* window = m_windows_in_order.tail(); window; window = window->prev()) {
|
|
|
|
if (!window->is_visible())
|
|
|
|
continue;
|
|
|
|
if (window->type() != type)
|
|
|
|
continue;
|
2019-03-10 04:33:43 +03:00
|
|
|
if (window == m_highlight_window.ptr())
|
|
|
|
continue;
|
2019-03-03 17:17:05 +03:00
|
|
|
if (callback(*window) == IterationDecision::Abort)
|
|
|
|
return IterationDecision::Abort;
|
|
|
|
}
|
|
|
|
return IterationDecision::Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename Callback>
|
|
|
|
IterationDecision WSWindowManager::for_each_visible_window_from_front_to_back(Callback callback)
|
|
|
|
{
|
2019-04-03 20:38:44 +03:00
|
|
|
if (for_each_visible_window_of_type_from_front_to_back(WSWindowType::Taskbar, callback) == IterationDecision::Abort)
|
|
|
|
return IterationDecision::Abort;
|
2019-03-03 17:17:05 +03:00
|
|
|
if (for_each_visible_window_of_type_from_front_to_back(WSWindowType::Menu, callback) == IterationDecision::Abort)
|
|
|
|
return IterationDecision::Abort;
|
2019-03-10 15:30:43 +03:00
|
|
|
if (for_each_visible_window_of_type_from_front_to_back(WSWindowType::Normal, callback) == IterationDecision::Abort)
|
|
|
|
return IterationDecision::Abort;
|
|
|
|
return for_each_visible_window_of_type_from_front_to_back(WSWindowType::WindowSwitcher, callback);
|
2019-03-03 17:17:05 +03:00
|
|
|
}
|
2019-04-04 02:44:35 +03:00
|
|
|
|
|
|
|
template<typename Callback>
|
|
|
|
void WSWindowManager::for_each_window_listening_to_wm_events(Callback callback)
|
|
|
|
{
|
|
|
|
for (auto* window = m_windows_in_order.tail(); window; window = window->prev()) {
|
|
|
|
if (!window->listens_to_wm_events())
|
|
|
|
continue;
|
|
|
|
if (callback(*window) == IterationDecision::Abort)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2019-04-04 14:33:09 +03:00
|
|
|
|
|
|
|
template<typename Callback>
|
|
|
|
void WSWindowManager::for_each_window(Callback callback)
|
|
|
|
{
|
|
|
|
for (auto* window = m_windows_in_order.tail(); window; window = window->prev()) {
|
|
|
|
if (callback(*window) == IterationDecision::Abort)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|