WSWindow.cpp 5.6 KB

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