WSWindow.cpp 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. #include "WSWindow.h"
  2. #include "WSWindowManager.h"
  3. #include "WSMessage.h"
  4. #include "WSMessageLoop.h"
  5. #include "Process.h"
  6. WSWindow::WSWindow(WSMenu& menu)
  7. : m_lock("WSWindow (menu)")
  8. , m_type(WSWindowType::Menu)
  9. , m_menu(&menu)
  10. {
  11. WSWindowManager::the().add_window(*this);
  12. }
  13. WSWindow::WSWindow(Process& process, int window_id)
  14. : m_lock("WSWindow (normal)")
  15. , m_type(WSWindowType::Normal)
  16. , m_process(&process)
  17. , m_window_id(window_id)
  18. , m_pid(process.pid())
  19. {
  20. WSWindowManager::the().add_window(*this);
  21. }
  22. WSWindow::~WSWindow()
  23. {
  24. WSWindowManager::the().remove_window(*this);
  25. }
  26. void WSWindow::set_title(String&& title)
  27. {
  28. {
  29. WSWindowLocker locker(*this);
  30. if (m_title == title)
  31. return;
  32. m_title = move(title);
  33. }
  34. WSWindowManager::the().notify_title_changed(*this);
  35. }
  36. void WSWindow::set_rect(const Rect& rect)
  37. {
  38. Rect old_rect;
  39. {
  40. WSWindowLocker locker(*this);
  41. if (!m_process && !m_menu)
  42. return;
  43. if (m_rect == rect)
  44. return;
  45. old_rect = m_rect;
  46. m_rect = rect;
  47. if (!m_backing || old_rect.size() != rect.size()) {
  48. if (m_process)
  49. m_backing = GraphicsBitmap::create(*m_process, m_rect.size());
  50. if (m_menu)
  51. m_backing = GraphicsBitmap::create_kernel_only(m_rect.size());
  52. }
  53. }
  54. WSWindowManager::the().notify_rect_changed(*this, old_rect, rect);
  55. }
  56. // FIXME: Just use the same types.
  57. static GUI_MouseButton to_api(MouseButton button)
  58. {
  59. switch (button) {
  60. case MouseButton::None: return GUI_MouseButton::NoButton;
  61. case MouseButton::Left: return GUI_MouseButton::Left;
  62. case MouseButton::Right: return GUI_MouseButton::Right;
  63. case MouseButton::Middle: return GUI_MouseButton::Middle;
  64. }
  65. }
  66. void WSWindow::on_message(WSMessage& message)
  67. {
  68. if (m_menu) {
  69. m_menu->on_window_message(message);
  70. return;
  71. }
  72. GUI_Event gui_event;
  73. gui_event.window_id = window_id();
  74. switch (message.type()) {
  75. case WSMessage::WM_ClientWantsToPaint:
  76. gui_event.type = GUI_Event::Type::Paint;
  77. gui_event.paint.rect = static_cast<WSClientWantsToPaintMessage&>(message).rect();
  78. break;
  79. case WSMessage::MouseMove:
  80. gui_event.type = GUI_Event::Type::MouseMove;
  81. gui_event.mouse.position = static_cast<WSMouseEvent&>(message).position();
  82. gui_event.mouse.button = GUI_MouseButton::NoButton;
  83. gui_event.mouse.buttons = static_cast<WSMouseEvent&>(message).buttons();
  84. break;
  85. case WSMessage::MouseDown:
  86. gui_event.type = GUI_Event::Type::MouseDown;
  87. gui_event.mouse.position = static_cast<WSMouseEvent&>(message).position();
  88. gui_event.mouse.button = to_api(static_cast<WSMouseEvent&>(message).button());
  89. gui_event.mouse.buttons = static_cast<WSMouseEvent&>(message).buttons();
  90. break;
  91. case WSMessage::MouseUp:
  92. gui_event.type = GUI_Event::Type::MouseUp;
  93. gui_event.mouse.position = static_cast<WSMouseEvent&>(message).position();
  94. gui_event.mouse.button = to_api(static_cast<WSMouseEvent&>(message).button());
  95. gui_event.mouse.buttons = static_cast<WSMouseEvent&>(message).buttons();
  96. break;
  97. case WSMessage::KeyDown:
  98. gui_event.type = GUI_Event::Type::KeyDown;
  99. gui_event.key.character = static_cast<WSKeyEvent&>(message).character();
  100. gui_event.key.key = static_cast<WSKeyEvent&>(message).key();
  101. gui_event.key.alt = static_cast<WSKeyEvent&>(message).alt();
  102. gui_event.key.ctrl = static_cast<WSKeyEvent&>(message).ctrl();
  103. gui_event.key.shift = static_cast<WSKeyEvent&>(message).shift();
  104. break;
  105. case WSMessage::KeyUp:
  106. gui_event.type = GUI_Event::Type::KeyUp;
  107. gui_event.key.character = static_cast<WSKeyEvent&>(message).character();
  108. gui_event.key.key = static_cast<WSKeyEvent&>(message).key();
  109. gui_event.key.alt = static_cast<WSKeyEvent&>(message).alt();
  110. gui_event.key.ctrl = static_cast<WSKeyEvent&>(message).ctrl();
  111. gui_event.key.shift = static_cast<WSKeyEvent&>(message).shift();
  112. break;
  113. case WSMessage::WM_ClientFinishedPaint:
  114. WSWindowManager::the().invalidate(*this, static_cast<WSClientFinishedPaintMessage&>(message).rect());
  115. return;
  116. case WSMessage::WM_SetWindowRect:
  117. set_rect(static_cast<WSSetWindowRectMessage&>(message).rect());
  118. return;
  119. case WSMessage::WM_SetWindowTitle:
  120. set_title(static_cast<WSSetWindowTitleMessage&>(message).title());
  121. return;
  122. case WSMessage::WM_DestroyWindow:
  123. delete this;
  124. return;
  125. case WSMessage::WindowActivated:
  126. gui_event.type = GUI_Event::Type::WindowActivated;
  127. break;
  128. case WSMessage::WindowDeactivated:
  129. gui_event.type = GUI_Event::Type::WindowDeactivated;
  130. break;
  131. case WSMessage::WindowCloseRequest:
  132. gui_event.type = GUI_Event::Type::WindowCloseRequest;
  133. break;
  134. default:
  135. break;
  136. }
  137. if (gui_event.type == GUI_Event::Type::Invalid)
  138. return;
  139. {
  140. WSWindowLocker window_locker(*this);
  141. if (!m_process)
  142. return;
  143. LOCKER(m_process->gui_events_lock());
  144. m_process->gui_events().append(move(gui_event));
  145. }
  146. }
  147. void WSWindow::notify_process_died(Badge<Process>)
  148. {
  149. WSWindowLocker locker(*this);
  150. m_process = nullptr;
  151. }
  152. void WSWindow::set_global_cursor_tracking_enabled(bool enabled)
  153. {
  154. dbgprintf("WSWindow{%p} global_cursor_tracking <- %u\n", enabled);
  155. m_global_cursor_tracking_enabled = enabled;
  156. }
  157. void WSWindow::set_visible(bool b)
  158. {
  159. if (m_visible == b)
  160. return;
  161. m_visible = b;
  162. invalidate();
  163. }
  164. void WSWindow::invalidate()
  165. {
  166. WSWindowManager::the().invalidate(*this);
  167. }