123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320 |
- #include "WSWindow.h"
- #include "WSEvent.h"
- #include "WSEventLoop.h"
- #include "WSScreen.h"
- #include "WSWindowManager.h"
- #include <WindowServer/WSClientConnection.h>
- #include <WindowServer/WindowClientEndpoint.h>
- static String default_window_icon_path()
- {
- return "/res/icons/16x16/window.png";
- }
- static GraphicsBitmap& default_window_icon()
- {
- static GraphicsBitmap* s_icon;
- if (!s_icon)
- s_icon = GraphicsBitmap::load_from_file(default_window_icon_path()).leak_ref();
- return *s_icon;
- }
- WSWindow::WSWindow(CObject& parent, WSWindowType type)
- : CObject(&parent)
- , m_type(type)
- , m_icon(default_window_icon())
- , m_frame(*this)
- {
- WSWindowManager::the().add_window(*this);
- }
- WSWindow::WSWindow(WSClientConnection& client, WSWindowType window_type, int window_id, bool modal, bool resizable, bool fullscreen)
- : CObject(&client)
- , m_client(&client)
- , m_type(window_type)
- , m_modal(modal)
- , m_resizable(resizable)
- , m_fullscreen(fullscreen)
- , m_window_id(window_id)
- , m_icon(default_window_icon())
- , m_frame(*this)
- {
- // FIXME: This should not be hard-coded here.
- if (m_type == WSWindowType::Taskbar) {
- m_wm_event_mask = WSWMEventMask::WindowStateChanges | WSWMEventMask::WindowRemovals | WSWMEventMask::WindowIconChanges;
- m_listens_to_wm_events = true;
- }
- WSWindowManager::the().add_window(*this);
- }
- WSWindow::~WSWindow()
- {
- WSWindowManager::the().remove_window(*this);
- }
- void WSWindow::set_title(const String& title)
- {
- if (m_title == title)
- return;
- m_title = title;
- WSWindowManager::the().notify_title_changed(*this);
- }
- void WSWindow::set_rect(const Rect& rect)
- {
- Rect old_rect;
- if (m_rect == rect)
- return;
- old_rect = m_rect;
- m_rect = rect;
- if (!m_client && (!m_backing_store || old_rect.size() != rect.size())) {
- m_backing_store = GraphicsBitmap::create(GraphicsBitmap::Format::RGB32, m_rect.size());
- }
- m_frame.notify_window_rect_changed(old_rect, rect);
- }
- void WSWindow::handle_mouse_event(const WSMouseEvent& event)
- {
- set_automatic_cursor_tracking_enabled(event.buttons() != 0);
- switch (event.type()) {
- case WSEvent::MouseMove:
- m_client->post_message(WindowClient::MouseMove(m_window_id, event.position(), (u32)event.button(), event.buttons(), event.modifiers(), event.wheel_delta()));
- break;
- case WSEvent::MouseDown:
- m_client->post_message(WindowClient::MouseDown(m_window_id, event.position(), (u32)event.button(), event.buttons(), event.modifiers(), event.wheel_delta()));
- break;
- case WSEvent::MouseDoubleClick:
- m_client->post_message(WindowClient::MouseDoubleClick(m_window_id, event.position(), (u32)event.button(), event.buttons(), event.modifiers(), event.wheel_delta()));
- break;
- case WSEvent::MouseUp:
- m_client->post_message(WindowClient::MouseUp(m_window_id, event.position(), (u32)event.button(), event.buttons(), event.modifiers(), event.wheel_delta()));
- break;
- case WSEvent::MouseWheel:
- m_client->post_message(WindowClient::MouseWheel(m_window_id, event.position(), (u32)event.button(), event.buttons(), event.modifiers(), event.wheel_delta()));
- break;
- default:
- ASSERT_NOT_REACHED();
- }
- }
- void WSWindow::set_minimized(bool minimized)
- {
- if (m_minimized == minimized)
- return;
- m_minimized = minimized;
- if (!minimized)
- request_update({ {}, size() });
- invalidate();
- WSWindowManager::the().notify_minimization_state_changed(*this);
- }
- void WSWindow::set_maximized(bool maximized)
- {
- if (m_maximized == maximized)
- return;
- m_maximized = maximized;
- auto old_rect = m_rect;
- if (maximized) {
- m_unmaximized_rect = m_rect;
- set_rect(WSWindowManager::the().maximized_window_rect(*this));
- } else {
- set_rect(m_unmaximized_rect);
- }
- m_frame.did_set_maximized({}, maximized);
- CEventLoop::current().post_event(*this, make<WSResizeEvent>(old_rect, m_rect));
- }
- void WSWindow::event(CEvent& event)
- {
- if (!m_client) {
- ASSERT(parent());
- event.ignore();
- return;
- }
- if (is_blocked_by_modal_window())
- return;
- if (static_cast<WSEvent&>(event).is_mouse_event())
- return handle_mouse_event(static_cast<const WSMouseEvent&>(event));
- switch (event.type()) {
- case WSEvent::WindowEntered:
- m_client->post_message(WindowClient::WindowEntered(m_window_id));
- break;
- case WSEvent::WindowLeft:
- m_client->post_message(WindowClient::WindowLeft(m_window_id));
- break;
- case WSEvent::KeyDown:
- m_client->post_message(
- WindowClient::KeyDown(m_window_id,
- (u8) static_cast<const WSKeyEvent&>(event).character(),
- (u32) static_cast<const WSKeyEvent&>(event).key(),
- static_cast<const WSKeyEvent&>(event).modifiers()));
- break;
- case WSEvent::KeyUp:
- m_client->post_message(
- WindowClient::KeyUp(m_window_id,
- (u8) static_cast<const WSKeyEvent&>(event).character(),
- (u32) static_cast<const WSKeyEvent&>(event).key(),
- static_cast<const WSKeyEvent&>(event).modifiers()));
- break;
- case WSEvent::WindowActivated:
- m_client->post_message(WindowClient::WindowActivated(m_window_id));
- break;
- case WSEvent::WindowDeactivated:
- m_client->post_message(WindowClient::WindowDeactivated(m_window_id));
- break;
- case WSEvent::WindowCloseRequest:
- m_client->post_message(WindowClient::WindowCloseRequest(m_window_id));
- break;
- case WSEvent::WindowResized:
- m_client->post_message(
- WindowClient::WindowResized(
- m_window_id,
- static_cast<const WSResizeEvent&>(event).old_rect(),
- static_cast<const WSResizeEvent&>(event).rect()));
- break;
- case WSEvent::WM_WindowRemoved: {
- auto& removed_event = static_cast<const WSWMWindowRemovedEvent&>(event);
- m_client->post_message(WindowClient::WM_WindowRemoved(
- removed_event.client_id(),
- removed_event.window_id()));
- break;
- }
- case WSEvent::WM_WindowStateChanged: {
- auto& changed_event = static_cast<const WSWMWindowStateChangedEvent&>(event);
- m_client->post_message(WindowClient::WM_WindowStateChanged(
- changed_event.client_id(),
- changed_event.window_id(),
- changed_event.is_active(),
- changed_event.is_minimized(),
- (i32)(changed_event.window_type()),
- changed_event.title(),
- changed_event.rect()));
- break;
- }
- case WSEvent::WM_WindowIconBitmapChanged: {
- auto& changed_event = static_cast<const WSWMWindowIconBitmapChangedEvent&>(event);
- // FIXME: Perhaps we should update the bitmap sharing list somewhere else instead?
- dbg() << "WindowServer: Sharing icon buffer " << changed_event.icon_buffer_id() << " with PID " << client()->client_pid();
- if (share_buffer_with(changed_event.icon_buffer_id(), m_client->client_pid()) < 0) {
- ASSERT_NOT_REACHED();
- }
- m_client->post_message(
- WindowClient::WM_WindowIconBitmapChanged(
- changed_event.client_id(),
- changed_event.window_id(),
- changed_event.icon_buffer_id(),
- changed_event.icon_size()));
- break;
- }
- case WSEvent::WM_WindowRectChanged: {
- auto& changed_event = static_cast<const WSWMWindowRectChangedEvent&>(event);
- m_client->post_message(
- WindowClient::WM_WindowRectChanged(
- changed_event.client_id(),
- changed_event.window_id(),
- changed_event.rect()));
- break;
- }
- default:
- break;
- }
- }
- void WSWindow::set_global_cursor_tracking_enabled(bool enabled)
- {
- m_global_cursor_tracking_enabled = enabled;
- }
- void WSWindow::set_visible(bool b)
- {
- if (m_visible == b)
- return;
- m_visible = b;
- invalidate();
- }
- void WSWindow::invalidate()
- {
- WSWindowManager::the().invalidate(*this);
- }
- bool WSWindow::is_active() const
- {
- return WSWindowManager::the().active_window() == this;
- }
- bool WSWindow::is_blocked_by_modal_window() const
- {
- return !is_modal() && client() && client()->is_showing_modal_window();
- }
- void WSWindow::set_default_icon()
- {
- m_icon = default_window_icon();
- }
- void WSWindow::request_update(const Rect& rect)
- {
- if (m_pending_paint_rects.is_empty()) {
- deferred_invoke([this](auto&) {
- client()->post_paint_message(*this);
- });
- }
- m_pending_paint_rects.add(rect);
- }
- void WSWindow::popup_window_menu(const Point& position)
- {
- if (!m_window_menu) {
- m_window_menu = WSMenu::construct(nullptr, -1, "(Window Menu)");
- m_window_menu->add_item(make<WSMenuItem>(*m_window_menu, 1, "Minimize"));
- m_window_menu->add_item(make<WSMenuItem>(*m_window_menu, 2, "Unminimize"));
- m_window_menu->add_item(make<WSMenuItem>(*m_window_menu, WSMenuItem::Type::Separator));
- m_window_menu->add_item(make<WSMenuItem>(*m_window_menu, 3, "Close"));
- m_window_menu->on_item_activation = [&](auto& item) {
- switch (item.identifier()) {
- case 1:
- set_minimized(true);
- break;
- case 2:
- set_minimized(false);
- break;
- case 3:
- request_close();
- break;
- }
- };
- }
- m_window_menu->popup(position);
- }
- void WSWindow::request_close()
- {
- WSEvent close_request(WSEvent::WindowCloseRequest);
- event(close_request);
- }
- void WSWindow::set_fullscreen(bool fullscreen)
- {
- if (m_fullscreen == fullscreen)
- return;
- m_fullscreen = fullscreen;
- Rect new_window_rect = m_rect;
- if (m_fullscreen) {
- m_saved_nonfullscreen_rect = m_rect;
- new_window_rect = WSScreen::the().rect();
- } else if (!m_saved_nonfullscreen_rect.is_empty()) {
- new_window_rect = m_saved_nonfullscreen_rect;
- }
- CEventLoop::current().post_event(*this, make<WSResizeEvent>(m_rect, new_window_rect));
- set_rect(new_window_rect);
- }
|