WSWindow.cpp 9.0 KB

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