| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | #include "GWindow.h"
 | 
					
						
							|  |  |  | #include "GEvent.h"
 | 
					
						
							|  |  |  | #include "GEventLoop.h"
 | 
					
						
							| 
									
										
										
										
											2019-01-20 07:03:38 +01:00
										 |  |  | #include "GWidget.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-07 11:46:02 +02:00
										 |  |  | #include <AK/HashMap.h>
 | 
					
						
							| 
									
										
										
										
											2019-06-16 16:55:39 +01:00
										 |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2019-01-20 05:48:43 +01:00
										 |  |  | #include <LibC/stdio.h>
 | 
					
						
							|  |  |  | #include <LibC/stdlib.h>
 | 
					
						
							| 
									
										
										
										
											2019-01-22 16:34:24 +01:00
										 |  |  | #include <LibC/unistd.h>
 | 
					
						
							| 
									
										
										
										
											2019-06-07 11:46:02 +02:00
										 |  |  | #include <LibGUI/GPainter.h>
 | 
					
						
							|  |  |  | #include <SharedGraphics/GraphicsBitmap.h>
 | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-11 08:27:13 +01:00
										 |  |  | //#define UPDATE_COALESCING_DEBUG
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-20 05:48:43 +01:00
										 |  |  | static HashMap<int, GWindow*>* s_windows; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static HashMap<int, GWindow*>& windows() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!s_windows) | 
					
						
							|  |  |  |         s_windows = new HashMap<int, GWindow*>; | 
					
						
							|  |  |  |     return *s_windows; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GWindow* GWindow::from_window_id(int window_id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto it = windows().find(window_id); | 
					
						
							|  |  |  |     if (it != windows().end()) | 
					
						
							|  |  |  |         return (*it).value; | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 17:01:54 +02:00
										 |  |  | GWindow::GWindow(CObject* parent) | 
					
						
							|  |  |  |     : CObject(parent) | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  |     m_rect_when_windowless = { 100, 400, 140, 140 }; | 
					
						
							|  |  |  |     m_title_when_windowless = "GWindow"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GWindow::~GWindow() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-02-05 10:31:37 +01:00
										 |  |  |     if (m_main_widget) | 
					
						
							|  |  |  |         delete m_main_widget; | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  |     hide(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GWindow::close() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-03-19 00:01:02 +01:00
										 |  |  |     if (should_exit_event_loop_on_close()) | 
					
						
							|  |  |  |         GEventLoop::current().quit(0); | 
					
						
							| 
									
										
										
										
											2019-06-22 10:37:03 +02:00
										 |  |  |     if (should_destroy_on_close()) | 
					
						
							|  |  |  |         delete_later(); | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-01 20:10:37 +02:00
										 |  |  | void GWindow::move_to_front() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_window_id) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     WSAPI_ClientMessage request; | 
					
						
							|  |  |  |     request.type = WSAPI_ClientMessage::Type::MoveWindowToFront; | 
					
						
							|  |  |  |     request.window_id = m_window_id; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     GEventLoop::current().connection().post_message_to_server(request); | 
					
						
							| 
									
										
										
										
											2019-06-01 20:10:37 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  | void GWindow::show() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_window_id) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-15 09:17:18 +01:00
										 |  |  |     WSAPI_ClientMessage request; | 
					
						
							|  |  |  |     request.type = WSAPI_ClientMessage::Type::CreateWindow; | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     request.window_id = m_window_id; | 
					
						
							|  |  |  |     request.window.rect = m_rect_when_windowless; | 
					
						
							| 
									
										
										
										
											2019-02-19 01:42:53 +01:00
										 |  |  |     request.window.has_alpha_channel = m_has_alpha_channel; | 
					
						
							| 
									
										
										
										
											2019-03-19 00:52:39 +01:00
										 |  |  |     request.window.modal = m_modal; | 
					
						
							|  |  |  |     request.window.resizable = m_resizable; | 
					
						
							| 
									
										
										
										
											2019-05-17 21:33:44 +02:00
										 |  |  |     request.window.fullscreen = m_fullscreen; | 
					
						
							| 
									
										
										
										
											2019-05-24 14:37:23 -07:00
										 |  |  |     request.window.show_titlebar = m_show_titlebar; | 
					
						
							| 
									
										
										
										
											2019-02-19 01:42:53 +01:00
										 |  |  |     request.window.opacity = m_opacity_when_windowless; | 
					
						
							| 
									
										
										
										
											2019-04-10 14:29:47 +02:00
										 |  |  |     request.window.background_color = m_background_color.value(); | 
					
						
							| 
									
										
										
										
											2019-02-21 00:21:23 +01:00
										 |  |  |     request.window.size_increment = m_size_increment; | 
					
						
							|  |  |  |     request.window.base_size = m_base_size; | 
					
						
							| 
									
										
										
										
											2019-04-03 19:38:44 +02:00
										 |  |  |     request.window.type = (WSAPI_WindowType)m_window_type; | 
					
						
							| 
									
										
										
										
											2019-02-25 22:06:55 +01:00
										 |  |  |     ASSERT(m_title_when_windowless.length() < (ssize_t)sizeof(request.text)); | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     strcpy(request.text, m_title_when_windowless.characters()); | 
					
						
							|  |  |  |     request.text_length = m_title_when_windowless.length(); | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     auto response = GEventLoop::current().connection().sync_request(request, WSAPI_ServerMessage::Type::DidCreateWindow); | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     m_window_id = response.window_id; | 
					
						
							| 
									
										
										
										
											2019-01-20 05:48:43 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     windows().set(m_window_id, this); | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  |     update(); | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  | void GWindow::hide() | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  |     if (!m_window_id) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     windows().remove(m_window_id); | 
					
						
							| 
									
										
										
										
											2019-02-15 09:17:18 +01:00
										 |  |  |     WSAPI_ClientMessage request; | 
					
						
							|  |  |  |     request.type = WSAPI_ClientMessage::Type::DestroyWindow; | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     request.window_id = m_window_id; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     GEventLoop::current().connection().sync_request(request, WSAPI_ServerMessage::Type::DidDestroyWindow); | 
					
						
							| 
									
										
										
										
											2019-04-08 18:58:44 +02:00
										 |  |  |     m_window_id = 0; | 
					
						
							|  |  |  |     m_pending_paint_event_rects.clear(); | 
					
						
							|  |  |  |     m_back_bitmap = nullptr; | 
					
						
							|  |  |  |     m_front_bitmap = nullptr; | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-02 14:58:02 +02:00
										 |  |  | void GWindow::set_title(const StringView& title) | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-03-19 00:01:02 +01:00
										 |  |  |     m_title_when_windowless = title; | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     if (!m_window_id) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-15 09:17:18 +01:00
										 |  |  |     WSAPI_ClientMessage request; | 
					
						
							|  |  |  |     request.type = WSAPI_ClientMessage::Type::SetWindowTitle; | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     request.window_id = m_window_id; | 
					
						
							| 
									
										
										
										
											2019-02-25 22:06:55 +01:00
										 |  |  |     ASSERT(m_title_when_windowless.length() < (ssize_t)sizeof(request.text)); | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     strcpy(request.text, m_title_when_windowless.characters()); | 
					
						
							|  |  |  |     request.text_length = m_title_when_windowless.length(); | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     GEventLoop::current().connection().post_message_to_server(request); | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-01-20 06:02:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-26 05:20:32 +01:00
										 |  |  | String GWindow::title() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     if (!m_window_id) | 
					
						
							|  |  |  |         return m_title_when_windowless; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-15 09:17:18 +01:00
										 |  |  |     WSAPI_ClientMessage request; | 
					
						
							|  |  |  |     request.type = WSAPI_ClientMessage::Type::GetWindowTitle; | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     request.window_id = m_window_id; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     auto response = GEventLoop::current().connection().sync_request(request, WSAPI_ServerMessage::Type::DidGetWindowTitle); | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     return String(response.text, response.text_length); | 
					
						
							| 
									
										
										
										
											2019-01-26 05:20:32 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 00:13:35 +01:00
										 |  |  | Rect GWindow::rect() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     if (!m_window_id) | 
					
						
							|  |  |  |         return m_rect_when_windowless; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-15 09:17:18 +01:00
										 |  |  |     WSAPI_ClientMessage request; | 
					
						
							|  |  |  |     request.type = WSAPI_ClientMessage::Type::GetWindowRect; | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     request.window_id = m_window_id; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     auto response = GEventLoop::current().connection().sync_request(request, WSAPI_ServerMessage::Type::DidGetWindowRect); | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     ASSERT(response.window_id == m_window_id); | 
					
						
							|  |  |  |     return response.window.rect; | 
					
						
							| 
									
										
										
										
											2019-02-08 00:13:35 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-24 23:40:12 +01:00
										 |  |  | void GWindow::set_rect(const Rect& a_rect) | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  |     m_rect_when_windowless = a_rect; | 
					
						
							| 
									
										
										
										
											2019-03-19 00:01:02 +01:00
										 |  |  |     if (!m_window_id) { | 
					
						
							|  |  |  |         if (m_main_widget) | 
					
						
							|  |  |  |             m_main_widget->resize(m_rect_when_windowless.size()); | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-03-19 00:01:02 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-02-15 09:17:18 +01:00
										 |  |  |     WSAPI_ClientMessage request; | 
					
						
							|  |  |  |     request.type = WSAPI_ClientMessage::Type::SetWindowRect; | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |     request.window_id = m_window_id; | 
					
						
							|  |  |  |     request.window.rect = a_rect; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     GEventLoop::current().connection().post_message_to_server(request); | 
					
						
							| 
									
										
										
										
											2019-04-26 20:08:30 +02:00
										 |  |  |     if (m_back_bitmap && m_back_bitmap->size() != a_rect.size()) | 
					
						
							| 
									
										
										
										
											2019-04-10 14:35:13 +02:00
										 |  |  |         m_back_bitmap = nullptr; | 
					
						
							| 
									
										
										
										
											2019-04-26 20:08:30 +02:00
										 |  |  |     if (m_front_bitmap && m_front_bitmap->size() != a_rect.size()) | 
					
						
							| 
									
										
										
										
											2019-04-10 14:35:13 +02:00
										 |  |  |         m_front_bitmap = nullptr; | 
					
						
							| 
									
										
										
										
											2019-04-08 18:58:44 +02:00
										 |  |  |     if (m_main_widget) | 
					
						
							|  |  |  |         m_main_widget->resize(a_rect.size()); | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-03 19:38:44 +02:00
										 |  |  | void GWindow::set_window_type(GWindowType window_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_window_type = window_type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-31 23:52:02 +02:00
										 |  |  | void GWindow::set_override_cursor(GStandardCursor cursor) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_window_id) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     WSAPI_ClientMessage request; | 
					
						
							|  |  |  |     request.type = WSAPI_ClientMessage::Type::SetWindowOverrideCursor; | 
					
						
							|  |  |  |     request.window_id = m_window_id; | 
					
						
							|  |  |  |     request.cursor.cursor = (WSAPI_StandardCursor)cursor; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     GEventLoop::current().connection().post_message_to_server(request); | 
					
						
							| 
									
										
										
										
											2019-03-31 23:52:02 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:56:55 +02:00
										 |  |  | void GWindow::event(CEvent& event) | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-15 22:17:09 +02:00
										 |  |  |     if (event.type() == GEvent::MouseUp || event.type() == GEvent::MouseDown || event.type() == GEvent::MouseDoubleClick || event.type() == GEvent::MouseMove || event.type() == GEvent::MouseWheel) { | 
					
						
							| 
									
										
										
										
											2019-03-24 15:01:56 +01:00
										 |  |  |         auto& mouse_event = static_cast<GMouseEvent&>(event); | 
					
						
							| 
									
										
										
										
											2019-01-27 08:48:34 +01:00
										 |  |  |         if (m_global_cursor_tracking_widget) { | 
					
						
							| 
									
										
										
										
											2019-02-26 10:34:05 +01:00
										 |  |  |             auto window_relative_rect = m_global_cursor_tracking_widget->window_relative_rect(); | 
					
						
							|  |  |  |             Point local_point { mouse_event.x() - window_relative_rect.x(), mouse_event.y() - window_relative_rect.y() }; | 
					
						
							| 
									
										
										
										
											2019-05-13 19:52:57 +02:00
										 |  |  |             auto local_event = make<GMouseEvent>((GEvent::Type)event.type(), local_point, mouse_event.buttons(), mouse_event.button(), mouse_event.modifiers(), mouse_event.wheel_delta()); | 
					
						
							| 
									
										
										
										
											2019-01-27 08:48:34 +01:00
										 |  |  |             m_global_cursor_tracking_widget->event(*local_event); | 
					
						
							| 
									
										
										
										
											2019-03-24 15:01:56 +01:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (m_automatic_cursor_tracking_widget) { | 
					
						
							|  |  |  |             auto window_relative_rect = m_automatic_cursor_tracking_widget->window_relative_rect(); | 
					
						
							|  |  |  |             Point local_point { mouse_event.x() - window_relative_rect.x(), mouse_event.y() - window_relative_rect.y() }; | 
					
						
							| 
									
										
										
										
											2019-05-13 19:52:57 +02:00
										 |  |  |             auto local_event = make<GMouseEvent>((GEvent::Type)event.type(), local_point, mouse_event.buttons(), mouse_event.button(), mouse_event.modifiers(), mouse_event.wheel_delta()); | 
					
						
							| 
									
										
										
										
											2019-03-24 15:01:56 +01:00
										 |  |  |             m_automatic_cursor_tracking_widget->event(*local_event); | 
					
						
							| 
									
										
										
										
											2019-03-25 01:42:15 +01:00
										 |  |  |             if (mouse_event.buttons() == 0) | 
					
						
							| 
									
										
										
										
											2019-03-24 15:01:56 +01:00
										 |  |  |                 m_automatic_cursor_tracking_widget = nullptr; | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2019-01-27 08:48:34 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-01-20 07:03:38 +01:00
										 |  |  |         if (!m_main_widget) | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2019-05-13 19:52:57 +02:00
										 |  |  |         auto result = m_main_widget->hit_test(mouse_event.position()); | 
					
						
							|  |  |  |         auto local_event = make<GMouseEvent>((GEvent::Type)event.type(), result.local_position, mouse_event.buttons(), mouse_event.button(), mouse_event.modifiers(), mouse_event.wheel_delta()); | 
					
						
							|  |  |  |         ASSERT(result.widget); | 
					
						
							|  |  |  |         set_hovered_widget(result.widget); | 
					
						
							|  |  |  |         if (mouse_event.buttons() != 0 && !m_automatic_cursor_tracking_widget) | 
					
						
							|  |  |  |             m_automatic_cursor_tracking_widget = result.widget->make_weak_ptr(); | 
					
						
							|  |  |  |         if (result.widget != m_global_cursor_tracking_widget.ptr()) | 
					
						
							|  |  |  |             return result.widget->event(*local_event); | 
					
						
							| 
									
										
										
										
											2019-01-26 21:58:43 +01:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-01-20 07:03:38 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-20 17:37:28 +02:00
										 |  |  |     if (event.type() == GEvent::MultiPaint) { | 
					
						
							| 
									
										
										
										
											2019-04-08 18:58:44 +02:00
										 |  |  |         if (!m_window_id) | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2019-01-20 07:03:38 +01:00
										 |  |  |         if (!m_main_widget) | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2019-04-20 17:37:28 +02:00
										 |  |  |         auto& paint_event = static_cast<GMultiPaintEvent&>(event); | 
					
						
							|  |  |  |         auto rects = paint_event.rects(); | 
					
						
							|  |  |  |         ASSERT(!rects.is_empty()); | 
					
						
							| 
									
										
										
										
											2019-04-10 15:39:28 +02:00
										 |  |  |         if (m_back_bitmap && m_back_bitmap->size() != paint_event.window_size()) { | 
					
						
							|  |  |  |             // Eagerly discard the backing store if we learn from this paint event that it needs to be bigger.
 | 
					
						
							|  |  |  |             // Otherwise we would have to wait for a resize event to tell us. This way we don't waste the
 | 
					
						
							|  |  |  |             // effort on painting into an undersized bitmap that will be thrown away anyway.
 | 
					
						
							|  |  |  |             m_back_bitmap = nullptr; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  |         bool created_new_backing_store = !m_back_bitmap; | 
					
						
							|  |  |  |         if (!m_back_bitmap) | 
					
						
							|  |  |  |             m_back_bitmap = create_backing_bitmap(paint_event.window_size()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-20 17:37:28 +02:00
										 |  |  |         auto rect = rects.first(); | 
					
						
							|  |  |  |         if (rect.is_empty() || created_new_backing_store) { | 
					
						
							|  |  |  |             rects.clear(); | 
					
						
							| 
									
										
										
										
											2019-06-07 11:46:02 +02:00
										 |  |  |             rects.append({ {}, paint_event.window_size() }); | 
					
						
							| 
									
										
										
										
											2019-04-20 17:37:28 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-21 21:05:10 +02:00
										 |  |  |         for (auto& rect : rects) | 
					
						
							| 
									
										
										
										
											2019-04-20 17:37:28 +02:00
										 |  |  |             m_main_widget->event(*make<GPaintEvent>(rect)); | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |         paint_keybinds(); | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-21 21:05:10 +02:00
										 |  |  |         if (m_double_buffering_enabled) | 
					
						
							|  |  |  |             flip(rects); | 
					
						
							|  |  |  |         else if (created_new_backing_store) | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  |             set_current_backing_bitmap(*m_back_bitmap, true); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  |         if (m_window_id) { | 
					
						
							| 
									
										
										
										
											2019-02-15 09:17:18 +01:00
										 |  |  |             WSAPI_ClientMessage message; | 
					
						
							|  |  |  |             message.type = WSAPI_ClientMessage::Type::DidFinishPainting; | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  |             message.window_id = m_window_id; | 
					
						
							| 
									
										
										
										
											2019-04-22 01:15:47 +02:00
										 |  |  |             message.rect_count = rects.size(); | 
					
						
							|  |  |  |             for (int i = 0; i < min(WSAPI_ClientMessage::max_inline_rect_count, rects.size()); ++i) | 
					
						
							|  |  |  |                 message.rects[i] = rects[i]; | 
					
						
							|  |  |  |             ByteBuffer extra_data; | 
					
						
							|  |  |  |             if (rects.size() > WSAPI_ClientMessage::max_inline_rect_count) | 
					
						
							| 
									
										
										
										
											2019-04-23 13:00:53 +02:00
										 |  |  |                 extra_data = ByteBuffer::wrap(&rects[WSAPI_ClientMessage::max_inline_rect_count], (rects.size() - WSAPI_ClientMessage::max_inline_rect_count) * sizeof(WSAPI_Rect)); | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |             GEventLoop::current().connection().post_message_to_server(message, move(extra_data)); | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-01-26 21:58:43 +01:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-01-20 07:03:38 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:56:55 +02:00
										 |  |  |     if (event.type() == GEvent::KeyUp || event.type() == GEvent::KeyDown) { | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  |         auto keyevent = static_cast<GKeyEvent&>(event); | 
					
						
							|  |  |  |         if (keyevent.logo() && event.type() == GEvent::KeyUp) { | 
					
						
							|  |  |  |             if (m_keybind_mode) { | 
					
						
							|  |  |  |                 m_keybind_mode = false; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 m_keybind_mode = true; | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |                 collect_keyboard_activation_targets(); | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  |                 m_entered_keybind = ""; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             update(); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (m_keybind_mode) { | 
					
						
							| 
									
										
										
										
											2019-06-03 18:13:41 +01:00
										 |  |  |             if (event.type() == GEvent::KeyUp) { | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |                 StringBuilder builder; | 
					
						
							|  |  |  |                 builder.append(m_entered_keybind); | 
					
						
							|  |  |  |                 builder.append(keyevent.text()); | 
					
						
							|  |  |  |                 m_entered_keybind = builder.to_string(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |                 auto found_widget = m_keyboard_activation_targets.find(m_entered_keybind); | 
					
						
							|  |  |  |                 if (found_widget != m_keyboard_activation_targets.end()) { | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |                     m_keybind_mode = false; | 
					
						
							| 
									
										
										
										
											2019-06-16 21:07:55 +01:00
										 |  |  |                     auto event = make<GMouseEvent>(GEvent::MouseDown, Point(), 0, GMouseButton::Left, 0, 0); | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |                     found_widget->value->event(*event); | 
					
						
							| 
									
										
										
										
											2019-06-16 21:07:55 +01:00
										 |  |  |                     event = make<GMouseEvent>(GEvent::MouseUp, Point(), 0, GMouseButton::Left, 0, 0); | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |                     found_widget->value->event(*event); | 
					
						
							|  |  |  |                 } else if (m_entered_keybind.length() >= m_max_keybind_length) { | 
					
						
							|  |  |  |                     m_keybind_mode = false; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 update(); | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             if (m_focused_widget) | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |                 return m_focused_widget->event(event); | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  |             if (m_main_widget) | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |                 return m_main_widget->event(event); | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-02-10 14:28:39 +01:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-01-26 06:39:13 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-26 21:58:43 +01:00
										 |  |  |     if (event.type() == GEvent::WindowBecameActive || event.type() == GEvent::WindowBecameInactive) { | 
					
						
							| 
									
										
										
										
											2019-06-15 21:53:25 +01:00
										 |  |  |         if (event.type() == GEvent::WindowBecameInactive && m_keybind_mode) { | 
					
						
							| 
									
										
										
										
											2019-06-16 21:01:51 +01:00
										 |  |  |             m_keybind_mode = false; | 
					
						
							|  |  |  |             update(); | 
					
						
							| 
									
										
										
										
											2019-06-15 21:53:25 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-26 21:58:43 +01:00
										 |  |  |         m_is_active = event.type() == GEvent::WindowBecameActive; | 
					
						
							| 
									
										
										
										
											2019-02-10 14:28:39 +01:00
										 |  |  |         if (m_main_widget) | 
					
						
							|  |  |  |             m_main_widget->event(event); | 
					
						
							| 
									
										
										
										
											2019-01-26 21:58:43 +01:00
										 |  |  |         if (m_focused_widget) | 
					
						
							|  |  |  |             m_focused_widget->update(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-05 10:31:37 +01:00
										 |  |  |     if (event.type() == GEvent::WindowCloseRequest) { | 
					
						
							|  |  |  |         close(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-20 10:12:19 +01:00
										 |  |  |     if (event.type() == GEvent::WindowLeft) { | 
					
						
							|  |  |  |         set_hovered_widget(nullptr); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-20 15:34:55 +01:00
										 |  |  |     if (event.type() == GEvent::Resize) { | 
					
						
							| 
									
										
										
										
											2019-02-27 18:52:12 +01:00
										 |  |  |         auto new_size = static_cast<GResizeEvent&>(event).size(); | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  |         if (m_back_bitmap && m_back_bitmap->size() != new_size) | 
					
						
							|  |  |  |             m_back_bitmap = nullptr; | 
					
						
							| 
									
										
										
										
											2019-04-22 01:15:47 +02:00
										 |  |  |         if (!m_pending_paint_event_rects.is_empty()) { | 
					
						
							|  |  |  |             m_pending_paint_event_rects.clear_with_capacity(); | 
					
						
							| 
									
										
										
										
											2019-06-07 11:46:02 +02:00
										 |  |  |             m_pending_paint_event_rects.append({ {}, new_size }); | 
					
						
							| 
									
										
										
										
											2019-04-22 01:15:47 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-07 11:46:02 +02:00
										 |  |  |         m_rect_when_windowless = { {}, new_size }; | 
					
						
							|  |  |  |         m_main_widget->set_relative_rect({ {}, new_size }); | 
					
						
							| 
									
										
										
										
											2019-02-20 15:34:55 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-20 14:40:38 +02:00
										 |  |  |     if (event.type() > GEvent::__Begin_WM_Events && event.type() < GEvent::__End_WM_Events) | 
					
						
							| 
									
										
										
										
											2019-04-03 21:03:12 +02:00
										 |  |  |         return wm_event(static_cast<GWMEvent&>(event)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 17:01:54 +02:00
										 |  |  |     CObject::event(event); | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-16 21:01:51 +01:00
										 |  |  | void GWindow::paint_keybinds() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_keybind_mode) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-06-16 16:55:39 +01:00
										 |  |  |     GPainter painter(*m_main_widget); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |     for (auto& keypair : m_keyboard_activation_targets) { | 
					
						
							|  |  |  |         if (!keypair.value) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         auto& widget = *keypair.value; | 
					
						
							| 
									
										
										
										
											2019-06-16 16:55:39 +01:00
										 |  |  |         bool could_be_keybind = true; | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |         for (int i = 0; i < m_entered_keybind.length(); ++i) { | 
					
						
							| 
									
										
										
										
											2019-06-16 16:55:39 +01:00
										 |  |  |             if (keypair.key.characters()[i] != m_entered_keybind.characters()[i]) { | 
					
						
							|  |  |  |                 could_be_keybind = false; | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-06-16 16:55:39 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-09 17:01:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-16 16:55:39 +01:00
										 |  |  |         if (could_be_keybind) { | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |             Rect rect { widget.x() - 5, widget.y() - 5, 4 + Font::default_font().width(keypair.key), 16 }; | 
					
						
							|  |  |  |             Rect highlight_rect { widget.x() - 3, widget.y() - 5, 0, 16 }; | 
					
						
							| 
									
										
										
										
											2019-06-09 17:01:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-30 09:23:16 +02:00
										 |  |  |             painter.fill_rect(rect, Color::WarmGray); | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |             painter.draw_rect(rect, Color::Black); | 
					
						
							| 
									
										
										
										
											2019-06-16 16:55:39 +01:00
										 |  |  |             painter.draw_text(rect, keypair.key.characters(), TextAlignment::Center, Color::Black); | 
					
						
							|  |  |  |             painter.draw_text(highlight_rect, m_entered_keybind.characters(), TextAlignment::CenterLeft, Color::MidGray); | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  | static void collect_keyboard_activation_targets_impl(GWidget& widget, Vector<GWidget*>& targets) | 
					
						
							| 
									
										
										
										
											2019-06-16 21:01:51 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |     widget.for_each_child_widget([&](auto& child) { | 
					
						
							|  |  |  |         if (child.supports_keyboard_activation()) { | 
					
						
							|  |  |  |             targets.append(&child); | 
					
						
							|  |  |  |             collect_keyboard_activation_targets_impl(child, targets); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return IterationDecision::Continue; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GWindow::collect_keyboard_activation_targets() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_keyboard_activation_targets.clear(); | 
					
						
							|  |  |  |     if (!m_main_widget) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Vector<GWidget*> targets; | 
					
						
							|  |  |  |     collect_keyboard_activation_targets_impl(*m_main_widget, targets); | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |     m_max_keybind_length = ceil_div(targets.size(), ('z' - 'a')); | 
					
						
							| 
									
										
										
										
											2019-06-03 18:13:41 +01:00
										 |  |  |     size_t buffer_length = m_max_keybind_length + 1; | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  |     char keybind_buffer[buffer_length]; | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |     for (size_t i = 0; i < buffer_length - 1; ++i) | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |         keybind_buffer[i] = 'a'; | 
					
						
							| 
									
										
										
										
											2019-06-16 21:01:51 +01:00
										 |  |  |     keybind_buffer[buffer_length - 1] = '\0'; | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |     for (auto& widget : targets) { | 
					
						
							|  |  |  |         m_keyboard_activation_targets.set(String(keybind_buffer), widget->make_weak_ptr()); | 
					
						
							|  |  |  |         for (size_t i = 0; i < buffer_length - 1; ++i) { | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |             if (keybind_buffer[i] >= 'z') { | 
					
						
							|  |  |  |                 keybind_buffer[i] = 'a'; | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2019-06-21 10:09:57 +02:00
										 |  |  |                 ++keybind_buffer[i]; | 
					
						
							| 
									
										
										
										
											2019-06-08 15:13:16 +01:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | bool GWindow::is_visible() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-22 10:37:03 +02:00
										 |  |  |     return m_window_id != 0; | 
					
						
							| 
									
										
										
										
											2019-01-20 04:49:48 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-26 05:20:32 +01:00
										 |  |  | void GWindow::update(const Rect& a_rect) | 
					
						
							| 
									
										
										
										
											2019-01-20 05:48:43 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-01-30 20:03:52 +01:00
										 |  |  |     if (!m_window_id) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-06-03 16:02:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-10 14:46:43 +01:00
										 |  |  |     for (auto& pending_rect : m_pending_paint_event_rects) { | 
					
						
							|  |  |  |         if (pending_rect.contains(a_rect)) { | 
					
						
							| 
									
										
										
										
											2019-02-11 08:27:13 +01:00
										 |  |  | #ifdef UPDATE_COALESCING_DEBUG
 | 
					
						
							| 
									
										
										
										
											2019-02-10 14:46:43 +01:00
										 |  |  |             dbgprintf("Ignoring %s since it's contained by pending rect %s\n", a_rect.to_string().characters(), pending_rect.to_string().characters()); | 
					
						
							| 
									
										
										
										
											2019-02-11 08:27:13 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-02-28 13:58:04 +01:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2019-02-10 14:46:43 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-20 15:19:02 +02:00
										 |  |  |     if (m_pending_paint_event_rects.is_empty()) { | 
					
						
							| 
									
										
										
										
											2019-06-07 11:46:02 +02:00
										 |  |  |         deferred_invoke([this](auto&) { | 
					
						
							| 
									
										
										
										
											2019-04-23 20:43:51 +02:00
										 |  |  |             auto rects = move(m_pending_paint_event_rects); | 
					
						
							|  |  |  |             if (rects.is_empty()) | 
					
						
							|  |  |  |                 return; | 
					
						
							| 
									
										
										
										
											2019-04-20 17:19:56 +02:00
										 |  |  |             WSAPI_ClientMessage request; | 
					
						
							|  |  |  |             request.type = WSAPI_ClientMessage::Type::InvalidateRect; | 
					
						
							|  |  |  |             request.window_id = m_window_id; | 
					
						
							| 
									
										
										
										
											2019-04-23 20:43:51 +02:00
										 |  |  |             for (int i = 0; i < min(WSAPI_ClientMessage::max_inline_rect_count, rects.size()); ++i) | 
					
						
							|  |  |  |                 request.rects[i] = rects[i]; | 
					
						
							| 
									
										
										
										
											2019-04-22 01:15:47 +02:00
										 |  |  |             ByteBuffer extra_data; | 
					
						
							| 
									
										
										
										
											2019-04-23 20:43:51 +02:00
										 |  |  |             if (rects.size() > WSAPI_ClientMessage::max_inline_rect_count) | 
					
						
							|  |  |  |                 extra_data = ByteBuffer::wrap(&rects[WSAPI_ClientMessage::max_inline_rect_count], (rects.size() - WSAPI_ClientMessage::max_inline_rect_count) * sizeof(WSAPI_Rect)); | 
					
						
							|  |  |  |             request.rect_count = rects.size(); | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |             GEventLoop::current().connection().post_message_to_server(request, move(extra_data)); | 
					
						
							| 
									
										
										
										
											2019-04-20 15:19:02 +02:00
										 |  |  |         }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     m_pending_paint_event_rects.append(a_rect); | 
					
						
							| 
									
										
										
										
											2019-01-20 05:48:43 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GWindow::set_main_widget(GWidget* widget) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_main_widget == widget) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     m_main_widget = widget; | 
					
						
							| 
									
										
										
										
											2019-02-10 14:28:39 +01:00
										 |  |  |     if (m_main_widget) { | 
					
						
							|  |  |  |         auto new_window_rect = rect(); | 
					
						
							|  |  |  |         if (m_main_widget->horizontal_size_policy() == SizePolicy::Fixed) | 
					
						
							|  |  |  |             new_window_rect.set_width(m_main_widget->preferred_size().width()); | 
					
						
							|  |  |  |         if (m_main_widget->vertical_size_policy() == SizePolicy::Fixed) | 
					
						
							|  |  |  |             new_window_rect.set_height(m_main_widget->preferred_size().height()); | 
					
						
							|  |  |  |         set_rect(new_window_rect); | 
					
						
							| 
									
										
										
										
											2019-06-07 11:46:02 +02:00
										 |  |  |         m_main_widget->set_relative_rect({ {}, new_window_rect.size() }); | 
					
						
							| 
									
										
										
										
											2019-02-10 14:28:39 +01:00
										 |  |  |         m_main_widget->set_window(this); | 
					
						
							|  |  |  |         if (m_main_widget->accepts_focus()) | 
					
						
							|  |  |  |             m_main_widget->set_focus(true); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-01-20 05:48:43 +01:00
										 |  |  |     update(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-01-26 06:39:13 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GWindow::set_focused_widget(GWidget* widget) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_focused_widget == widget) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-01-26 11:24:16 +01:00
										 |  |  |     if (m_focused_widget) { | 
					
						
							| 
									
										
										
										
											2019-03-19 00:01:02 +01:00
										 |  |  |         GEventLoop::current().post_event(*m_focused_widget, make<GEvent>(GEvent::FocusOut)); | 
					
						
							| 
									
										
										
										
											2019-01-26 06:39:13 +01:00
										 |  |  |         m_focused_widget->update(); | 
					
						
							| 
									
										
										
										
											2019-01-26 11:24:16 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-18 23:16:57 +02:00
										 |  |  |     m_focused_widget = widget ? widget->make_weak_ptr() : nullptr; | 
					
						
							| 
									
										
										
										
											2019-01-26 11:24:16 +01:00
										 |  |  |     if (m_focused_widget) { | 
					
						
							| 
									
										
										
										
											2019-03-19 00:01:02 +01:00
										 |  |  |         GEventLoop::current().post_event(*m_focused_widget, make<GEvent>(GEvent::FocusIn)); | 
					
						
							| 
									
										
										
										
											2019-01-26 11:24:16 +01:00
										 |  |  |         m_focused_widget->update(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-01-26 06:39:13 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-01-27 08:48:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GWindow::set_global_cursor_tracking_widget(GWidget* widget) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-14 02:36:06 +02:00
										 |  |  |     if (widget == m_global_cursor_tracking_widget) | 
					
						
							| 
									
										
										
										
											2019-01-27 08:48:34 +01:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-01-31 17:31:23 +01:00
										 |  |  |     m_global_cursor_tracking_widget = widget ? widget->make_weak_ptr() : nullptr; | 
					
						
							| 
									
										
										
										
											2019-03-24 15:01:56 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-02-14 01:21:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 15:01:56 +01:00
										 |  |  | void GWindow::set_automatic_cursor_tracking_widget(GWidget* widget) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-14 02:36:06 +02:00
										 |  |  |     if (widget == m_automatic_cursor_tracking_widget) | 
					
						
							| 
									
										
										
										
											2019-03-24 15:01:56 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  |     m_automatic_cursor_tracking_widget = widget ? widget->make_weak_ptr() : nullptr; | 
					
						
							| 
									
										
										
										
											2019-01-27 08:48:34 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-02-19 01:42:53 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GWindow::set_has_alpha_channel(bool value) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-03 21:07:16 +02:00
										 |  |  |     if (m_has_alpha_channel == value) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-02-19 01:42:53 +01:00
										 |  |  |     m_has_alpha_channel = value; | 
					
						
							| 
									
										
										
										
											2019-05-03 21:07:16 +02:00
										 |  |  |     if (!m_window_id) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_pending_paint_event_rects.clear(); | 
					
						
							|  |  |  |     m_back_bitmap = nullptr; | 
					
						
							|  |  |  |     m_front_bitmap = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     WSAPI_ClientMessage message; | 
					
						
							|  |  |  |     message.type = WSAPI_ClientMessage::Type::SetWindowHasAlphaChannel; | 
					
						
							|  |  |  |     message.window_id = m_window_id; | 
					
						
							|  |  |  |     message.value = value; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     GEventLoop::current().connection().sync_request(message, WSAPI_ServerMessage::DidSetWindowHasAlphaChannel); | 
					
						
							| 
									
										
										
										
											2019-05-03 21:07:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     update(); | 
					
						
							| 
									
										
										
										
											2019-02-19 01:42:53 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  | void GWindow::set_double_buffering_enabled(bool value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASSERT(!m_window_id); | 
					
						
							|  |  |  |     m_double_buffering_enabled = value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-19 01:42:53 +01:00
										 |  |  | void GWindow::set_opacity(float opacity) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_opacity_when_windowless = opacity; | 
					
						
							|  |  |  |     if (!m_window_id) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     WSAPI_ClientMessage request; | 
					
						
							|  |  |  |     request.type = WSAPI_ClientMessage::Type::SetWindowOpacity; | 
					
						
							|  |  |  |     request.window_id = m_window_id; | 
					
						
							|  |  |  |     request.window.opacity = opacity; | 
					
						
							|  |  |  |     m_opacity_when_windowless = opacity; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     GEventLoop::current().connection().post_message_to_server(request); | 
					
						
							| 
									
										
										
										
											2019-02-19 01:42:53 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-02-20 10:12:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GWindow::set_hovered_widget(GWidget* widget) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-14 02:36:06 +02:00
										 |  |  |     if (widget == m_hovered_widget) | 
					
						
							| 
									
										
										
										
											2019-02-20 10:12:19 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_hovered_widget) | 
					
						
							| 
									
										
										
										
											2019-03-19 00:01:02 +01:00
										 |  |  |         GEventLoop::current().post_event(*m_hovered_widget, make<GEvent>(GEvent::Leave)); | 
					
						
							| 
									
										
										
										
											2019-02-20 10:12:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     m_hovered_widget = widget ? widget->make_weak_ptr() : nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_hovered_widget) | 
					
						
							| 
									
										
										
										
											2019-03-19 00:01:02 +01:00
										 |  |  |         GEventLoop::current().post_event(*m_hovered_widget, make<GEvent>(GEvent::Enter)); | 
					
						
							| 
									
										
										
										
											2019-02-20 10:12:19 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GWindow::set_current_backing_bitmap(GraphicsBitmap& bitmap, bool flush_immediately) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     WSAPI_ClientMessage message; | 
					
						
							|  |  |  |     message.type = WSAPI_ClientMessage::Type::SetWindowBackingStore; | 
					
						
							|  |  |  |     message.window_id = m_window_id; | 
					
						
							|  |  |  |     message.backing.bpp = 32; | 
					
						
							|  |  |  |     message.backing.pitch = bitmap.pitch(); | 
					
						
							|  |  |  |     message.backing.shared_buffer_id = bitmap.shared_buffer_id(); | 
					
						
							|  |  |  |     message.backing.has_alpha_channel = bitmap.has_alpha_channel(); | 
					
						
							|  |  |  |     message.backing.size = bitmap.size(); | 
					
						
							|  |  |  |     message.backing.flush_immediately = flush_immediately; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     GEventLoop::current().connection().sync_request(message, WSAPI_ServerMessage::Type::DidSetWindowBackingStore); | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-21 21:05:10 +02:00
										 |  |  | void GWindow::flip(const Vector<Rect, 32>& dirty_rects) | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     swap(m_front_bitmap, m_back_bitmap); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     set_current_backing_bitmap(*m_front_bitmap); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_back_bitmap || m_back_bitmap->size() != m_front_bitmap->size()) { | 
					
						
							|  |  |  |         m_back_bitmap = create_backing_bitmap(m_front_bitmap->size()); | 
					
						
							| 
									
										
										
										
											2019-05-08 03:33:44 +02:00
										 |  |  |         memcpy(m_back_bitmap->scanline(0), m_front_bitmap->scanline(0), m_front_bitmap->size_in_bytes()); | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Copy whatever was painted from the front to the back.
 | 
					
						
							|  |  |  |     Painter painter(*m_back_bitmap); | 
					
						
							| 
									
										
										
										
											2019-05-21 21:05:10 +02:00
										 |  |  |     for (auto& dirty_rect : dirty_rects) | 
					
						
							|  |  |  |         painter.blit(dirty_rect.location(), *m_front_bitmap, dirty_rect); | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-21 18:37:47 +02:00
										 |  |  | NonnullRefPtr<GraphicsBitmap> GWindow::create_backing_bitmap(const Size& size) | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     ASSERT(GEventLoop::current().connection().server_pid()); | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  |     ASSERT(!size.is_empty()); | 
					
						
							| 
									
										
										
										
											2019-05-06 14:04:54 +02:00
										 |  |  |     size_t pitch = round_up_to_power_of_two(size.width() * sizeof(RGBA32), 16); | 
					
						
							|  |  |  |     size_t size_in_bytes = size.height() * pitch; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     auto shared_buffer = SharedBuffer::create(GEventLoop::current().connection().server_pid(), size_in_bytes); | 
					
						
							| 
									
										
										
										
											2019-03-17 04:23:54 +01:00
										 |  |  |     ASSERT(shared_buffer); | 
					
						
							|  |  |  |     auto format = m_has_alpha_channel ? GraphicsBitmap::Format::RGBA32 : GraphicsBitmap::Format::RGB32; | 
					
						
							|  |  |  |     return GraphicsBitmap::create_with_shared_buffer(format, *shared_buffer, size); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-03-19 00:01:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GWindow::set_modal(bool modal) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASSERT(!m_window_id); | 
					
						
							|  |  |  |     m_modal = modal; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-03 21:03:12 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GWindow::wm_event(GWMEvent&) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-13 16:59:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-02 14:58:02 +02:00
										 |  |  | void GWindow::set_icon_path(const StringView& path) | 
					
						
							| 
									
										
										
										
											2019-04-13 16:59:55 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (m_icon_path == path) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     m_icon_path = path; | 
					
						
							|  |  |  |     if (!m_window_id) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     WSAPI_ClientMessage message; | 
					
						
							|  |  |  |     message.type = WSAPI_ClientMessage::Type::SetWindowIcon; | 
					
						
							|  |  |  |     message.window_id = m_window_id; | 
					
						
							| 
									
										
										
										
											2019-06-22 14:41:11 +02:00
										 |  |  |     ASSERT(path.length() < (int)sizeof(message.text)); | 
					
						
							| 
									
										
										
										
											2019-07-08 15:38:44 +02:00
										 |  |  |     strcpy(message.text, String(path).characters()); | 
					
						
							| 
									
										
										
										
											2019-04-13 16:59:55 +02:00
										 |  |  |     message.text_length = path.length(); | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     GEventLoop::current().connection().post_message_to_server(message); | 
					
						
							| 
									
										
										
										
											2019-04-13 16:59:55 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-03 01:38:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GWindow::start_wm_resize() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     WSAPI_ClientMessage message; | 
					
						
							|  |  |  |     message.type = WSAPI_ClientMessage::Type::WM_StartWindowResize; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     message.wm.client_id = GEventLoop::current().connection().my_client_id(); | 
					
						
							| 
									
										
										
										
											2019-05-03 01:38:24 +02:00
										 |  |  |     message.wm.window_id = m_window_id; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |     GEventLoop::current().connection().post_message_to_server(message); | 
					
						
							| 
									
										
										
										
											2019-05-03 01:38:24 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-15 02:39:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | Vector<GWidget*> GWindow::focusable_widgets() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_main_widget) | 
					
						
							| 
									
										
										
										
											2019-06-07 11:46:02 +02:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2019-05-15 02:39:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Vector<GWidget*> collected_widgets; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-07 11:46:02 +02:00
										 |  |  |     Function<void(GWidget&)> collect_focusable_widgets = [&](GWidget& widget) { | 
					
						
							| 
									
										
										
										
											2019-05-15 02:39:58 +02:00
										 |  |  |         if (widget.accepts_focus()) | 
					
						
							|  |  |  |             collected_widgets.append(&widget); | 
					
						
							| 
									
										
										
										
											2019-06-07 11:46:02 +02:00
										 |  |  |         widget.for_each_child_widget([&](auto& child) { | 
					
						
							| 
									
										
										
										
											2019-05-27 03:52:33 +02:00
										 |  |  |             if (!child.is_visible()) | 
					
						
							|  |  |  |                 return IterationDecision::Continue; | 
					
						
							|  |  |  |             if (!child.is_enabled()) | 
					
						
							|  |  |  |                 return IterationDecision::Continue; | 
					
						
							|  |  |  |             collect_focusable_widgets(child); | 
					
						
							|  |  |  |             return IterationDecision::Continue; | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2019-05-15 02:39:58 +02:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     collect_focusable_widgets(*m_main_widget); | 
					
						
							|  |  |  |     return collected_widgets; | 
					
						
							|  |  |  | } |