WSWindow.cpp 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  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. WSWindow::WSWindow(WSMenu& menu)
  8. : m_type(WSWindowType::Menu)
  9. , m_menu(&menu)
  10. {
  11. WSWindowManager::the().add_window(*this);
  12. }
  13. WSWindow::WSWindow(int client_id, int window_id)
  14. : m_client_id(client_id)
  15. , m_type(WSWindowType::Normal)
  16. , m_window_id(window_id)
  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. if (m_title == title)
  27. return;
  28. m_title = move(title);
  29. WSWindowManager::the().notify_title_changed(*this);
  30. }
  31. void WSWindow::set_rect(const Rect& rect)
  32. {
  33. Rect old_rect;
  34. auto* client = WSClientConnection::from_client_id(m_client_id);
  35. if (!client && !m_menu)
  36. return;
  37. if (m_rect == rect)
  38. return;
  39. old_rect = m_rect;
  40. m_rect = rect;
  41. if (!m_backing || old_rect.size() != rect.size()) {
  42. if (m_menu)
  43. m_backing = GraphicsBitmap::create(m_rect.size());
  44. else if (client)
  45. m_backing = client->create_bitmap(m_rect.size());
  46. }
  47. WSWindowManager::the().notify_rect_changed(*this, old_rect, rect);
  48. }
  49. // FIXME: Just use the same types.
  50. static WSAPI_MouseButton to_api(MouseButton button)
  51. {
  52. switch (button) {
  53. case MouseButton::None: return WSAPI_MouseButton::NoButton;
  54. case MouseButton::Left: return WSAPI_MouseButton::Left;
  55. case MouseButton::Right: return WSAPI_MouseButton::Right;
  56. case MouseButton::Middle: return WSAPI_MouseButton::Middle;
  57. }
  58. ASSERT_NOT_REACHED();
  59. }
  60. void WSWindow::on_message(WSMessage& message)
  61. {
  62. if (m_menu) {
  63. m_menu->on_window_message(message);
  64. return;
  65. }
  66. WSAPI_ServerMessage server_message;
  67. server_message.window_id = window_id();
  68. switch (message.type()) {
  69. case WSMessage::MouseMove:
  70. server_message.type = WSAPI_ServerMessage::Type::MouseMove;
  71. server_message.mouse.position = static_cast<WSMouseEvent&>(message).position();
  72. server_message.mouse.button = WSAPI_MouseButton::NoButton;
  73. server_message.mouse.buttons = static_cast<WSMouseEvent&>(message).buttons();
  74. break;
  75. case WSMessage::MouseDown:
  76. server_message.type = WSAPI_ServerMessage::Type::MouseDown;
  77. server_message.mouse.position = static_cast<WSMouseEvent&>(message).position();
  78. server_message.mouse.button = to_api(static_cast<WSMouseEvent&>(message).button());
  79. server_message.mouse.buttons = static_cast<WSMouseEvent&>(message).buttons();
  80. break;
  81. case WSMessage::MouseUp:
  82. server_message.type = WSAPI_ServerMessage::Type::MouseUp;
  83. server_message.mouse.position = static_cast<WSMouseEvent&>(message).position();
  84. server_message.mouse.button = to_api(static_cast<WSMouseEvent&>(message).button());
  85. server_message.mouse.buttons = static_cast<WSMouseEvent&>(message).buttons();
  86. break;
  87. case WSMessage::KeyDown:
  88. server_message.type = WSAPI_ServerMessage::Type::KeyDown;
  89. server_message.key.character = static_cast<WSKeyEvent&>(message).character();
  90. server_message.key.key = static_cast<WSKeyEvent&>(message).key();
  91. server_message.key.alt = static_cast<WSKeyEvent&>(message).alt();
  92. server_message.key.ctrl = static_cast<WSKeyEvent&>(message).ctrl();
  93. server_message.key.shift = static_cast<WSKeyEvent&>(message).shift();
  94. break;
  95. case WSMessage::KeyUp:
  96. server_message.type = WSAPI_ServerMessage::Type::KeyUp;
  97. server_message.key.character = static_cast<WSKeyEvent&>(message).character();
  98. server_message.key.key = static_cast<WSKeyEvent&>(message).key();
  99. server_message.key.alt = static_cast<WSKeyEvent&>(message).alt();
  100. server_message.key.ctrl = static_cast<WSKeyEvent&>(message).ctrl();
  101. server_message.key.shift = static_cast<WSKeyEvent&>(message).shift();
  102. break;
  103. case WSMessage::WindowActivated:
  104. server_message.type = WSAPI_ServerMessage::Type::WindowActivated;
  105. break;
  106. case WSMessage::WindowDeactivated:
  107. server_message.type = WSAPI_ServerMessage::Type::WindowDeactivated;
  108. break;
  109. case WSMessage::WindowCloseRequest:
  110. server_message.type = WSAPI_ServerMessage::Type::WindowCloseRequest;
  111. break;
  112. default:
  113. break;
  114. }
  115. if (server_message.type == WSAPI_ServerMessage::Type::Invalid)
  116. return;
  117. if (auto* client = WSClientConnection::from_client_id(m_client_id))
  118. client->post_message(server_message);
  119. }
  120. void WSWindow::set_global_cursor_tracking_enabled(bool enabled)
  121. {
  122. dbgprintf("WSWindow{%p} global_cursor_tracking <- %u\n", enabled);
  123. m_global_cursor_tracking_enabled = enabled;
  124. }
  125. void WSWindow::set_visible(bool b)
  126. {
  127. if (m_visible == b)
  128. return;
  129. m_visible = b;
  130. invalidate();
  131. }
  132. void WSWindow::invalidate()
  133. {
  134. WSWindowManager::the().invalidate(*this);
  135. }