WSWindow.cpp 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. #include "WSWindow.h"
  2. #include "WSWindowManager.h"
  3. #include "WSMessage.h"
  4. #include "WSMessageLoop.h"
  5. #include <WindowServer/WSAPITypes.h>
  6. #include <WindowServer/WSClientConnection.h>
  7. static GraphicsBitmap& default_window_icon()
  8. {
  9. static GraphicsBitmap* s_icon;
  10. if (!s_icon)
  11. s_icon = GraphicsBitmap::load_from_file("/res/icons/16x16/window.png").leak_ref();
  12. return *s_icon;
  13. }
  14. WSWindow::WSWindow(WSMessageReceiver& internal_owner, WSWindowType type)
  15. : m_internal_owner(&internal_owner)
  16. , m_type(type)
  17. , m_icon(default_window_icon())
  18. {
  19. WSWindowManager::the().add_window(*this);
  20. }
  21. WSWindow::WSWindow(WSClientConnection& client, WSWindowType window_type, int window_id, bool modal)
  22. : m_client(&client)
  23. , m_type(window_type)
  24. , m_modal(modal)
  25. , m_window_id(window_id)
  26. , m_icon(default_window_icon())
  27. {
  28. // FIXME: This should not be hard-coded here.
  29. if (m_type == WSWindowType::Taskbar)
  30. m_listens_to_wm_events = true;
  31. WSWindowManager::the().add_window(*this);
  32. }
  33. WSWindow::~WSWindow()
  34. {
  35. WSWindowManager::the().remove_window(*this);
  36. }
  37. void WSWindow::set_title(String&& title)
  38. {
  39. if (m_title == title)
  40. return;
  41. m_title = move(title);
  42. WSWindowManager::the().notify_title_changed(*this);
  43. }
  44. void WSWindow::set_rect(const Rect& rect)
  45. {
  46. Rect old_rect;
  47. if (m_rect == rect)
  48. return;
  49. old_rect = m_rect;
  50. m_rect = rect;
  51. if (!m_client && (!m_backing_store || old_rect.size() != rect.size())) {
  52. m_backing_store = GraphicsBitmap::create(GraphicsBitmap::Format::RGB32, m_rect.size());
  53. }
  54. WSWindowManager::the().notify_rect_changed(*this, old_rect, rect);
  55. }
  56. // FIXME: Just use the same types.
  57. static WSAPI_MouseButton to_api(MouseButton button)
  58. {
  59. switch (button) {
  60. case MouseButton::None: return WSAPI_MouseButton::NoButton;
  61. case MouseButton::Left: return WSAPI_MouseButton::Left;
  62. case MouseButton::Right: return WSAPI_MouseButton::Right;
  63. case MouseButton::Middle: return WSAPI_MouseButton::Middle;
  64. }
  65. ASSERT_NOT_REACHED();
  66. }
  67. void WSWindow::handle_mouse_event(const WSMouseEvent& event)
  68. {
  69. set_automatic_cursor_tracking_enabled(event.buttons() != 0);
  70. WSAPI_ServerMessage server_message;
  71. server_message.window_id = window_id();
  72. switch (event.type()) {
  73. case WSMessage::MouseMove: server_message.type = WSAPI_ServerMessage::Type::MouseMove; break;
  74. case WSMessage::MouseDown: server_message.type = WSAPI_ServerMessage::Type::MouseDown; break;
  75. case WSMessage::MouseUp: server_message.type = WSAPI_ServerMessage::Type::MouseUp; break;
  76. default: ASSERT_NOT_REACHED();
  77. }
  78. server_message.mouse.position = event.position();
  79. server_message.mouse.button = to_api(event.button());
  80. server_message.mouse.buttons = event.buttons();
  81. server_message.mouse.modifiers = event.modifiers();
  82. m_client->post_message(server_message);
  83. }
  84. void WSWindow::on_message(const WSMessage& message)
  85. {
  86. if (m_internal_owner)
  87. return m_internal_owner->on_message(message);
  88. if (is_blocked_by_modal_window())
  89. return;
  90. WSAPI_ServerMessage server_message;
  91. server_message.window_id = window_id();
  92. if (message.is_mouse_event())
  93. return handle_mouse_event(static_cast<const WSMouseEvent&>(message));
  94. switch (message.type()) {
  95. case WSMessage::WindowEntered:
  96. server_message.type = WSAPI_ServerMessage::Type::WindowEntered;
  97. break;
  98. case WSMessage::WindowLeft:
  99. server_message.type = WSAPI_ServerMessage::Type::WindowLeft;
  100. break;
  101. case WSMessage::KeyDown:
  102. server_message.type = WSAPI_ServerMessage::Type::KeyDown;
  103. server_message.key.character = static_cast<const WSKeyEvent&>(message).character();
  104. server_message.key.key = static_cast<const WSKeyEvent&>(message).key();
  105. server_message.key.modifiers = static_cast<const WSKeyEvent&>(message).modifiers();
  106. break;
  107. case WSMessage::KeyUp:
  108. server_message.type = WSAPI_ServerMessage::Type::KeyUp;
  109. server_message.key.character = static_cast<const WSKeyEvent&>(message).character();
  110. server_message.key.key = static_cast<const WSKeyEvent&>(message).key();
  111. server_message.key.modifiers = static_cast<const WSKeyEvent&>(message).modifiers();
  112. break;
  113. case WSMessage::WindowActivated:
  114. server_message.type = WSAPI_ServerMessage::Type::WindowActivated;
  115. break;
  116. case WSMessage::WindowDeactivated:
  117. server_message.type = WSAPI_ServerMessage::Type::WindowDeactivated;
  118. break;
  119. case WSMessage::WindowCloseRequest:
  120. server_message.type = WSAPI_ServerMessage::Type::WindowCloseRequest;
  121. break;
  122. case WSMessage::WindowResized:
  123. server_message.type = WSAPI_ServerMessage::Type::WindowResized;
  124. server_message.window.old_rect = static_cast<const WSResizeEvent&>(message).old_rect();
  125. server_message.window.rect = static_cast<const WSResizeEvent&>(message).rect();
  126. break;
  127. case WSMessage::WM_WindowAdded: {
  128. auto& added_event = static_cast<const WSWMWindowAddedEvent&>(message);
  129. server_message.type = WSAPI_ServerMessage::Type::WM_WindowAdded;
  130. server_message.wm.client_id = added_event.client_id();
  131. server_message.wm.window_id = added_event.window_id();
  132. server_message.wm.is_active = added_event.is_active();
  133. ASSERT(added_event.title().length() < sizeof(server_message.text));
  134. memcpy(server_message.text, added_event.title().characters(), added_event.title().length());
  135. server_message.text_length = added_event.title().length();
  136. server_message.wm.rect = added_event.rect();
  137. break;
  138. }
  139. case WSMessage::WM_WindowRemoved: {
  140. auto& removed_event = static_cast<const WSWMWindowRemovedEvent&>(message);
  141. server_message.type = WSAPI_ServerMessage::Type::WM_WindowRemoved;
  142. server_message.wm.client_id = removed_event.client_id();
  143. server_message.wm.window_id = removed_event.window_id();
  144. break;
  145. }
  146. case WSMessage::WM_WindowStateChanged: {
  147. auto& changed_event = static_cast<const WSWMWindowStateChangedEvent&>(message);
  148. server_message.type = WSAPI_ServerMessage::Type::WM_WindowStateChanged;
  149. server_message.wm.client_id = changed_event.client_id();
  150. server_message.wm.window_id = changed_event.window_id();
  151. server_message.wm.is_active = changed_event.is_active();
  152. ASSERT(changed_event.title().length() < sizeof(server_message.text));
  153. memcpy(server_message.text, changed_event.title().characters(), changed_event.title().length());
  154. server_message.text_length = changed_event.title().length();
  155. server_message.wm.rect = changed_event.rect();
  156. break;
  157. }
  158. default:
  159. break;
  160. }
  161. if (server_message.type == WSAPI_ServerMessage::Type::Invalid)
  162. return;
  163. m_client->post_message(server_message);
  164. }
  165. void WSWindow::set_global_cursor_tracking_enabled(bool enabled)
  166. {
  167. m_global_cursor_tracking_enabled = enabled;
  168. }
  169. void WSWindow::set_visible(bool b)
  170. {
  171. if (m_visible == b)
  172. return;
  173. m_visible = b;
  174. invalidate();
  175. }
  176. void WSWindow::set_resizable(bool resizable)
  177. {
  178. if (m_resizable == resizable)
  179. return;
  180. m_resizable = resizable;
  181. }
  182. void WSWindow::invalidate()
  183. {
  184. WSWindowManager::the().invalidate(*this);
  185. }
  186. bool WSWindow::is_active() const
  187. {
  188. return WSWindowManager::the().active_window() == this;
  189. }
  190. bool WSWindow::is_blocked_by_modal_window() const
  191. {
  192. return !is_modal() && client() && client()->is_showing_modal_window();
  193. }