WSWindow.cpp 10.0 KB

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