WSWindow.cpp 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  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. static GraphicsBitmap& default_window_icon()
  8. {
  9. static GraphicsBitmap* s_icon;
  10. if (!s_icon)
  11. s_icon = GraphicsBitmap::load_from_file(GraphicsBitmap::Format::RGBA32, "/res/icons/window16.rgb", { 16, 16 }).leak_ref();
  12. return *s_icon;
  13. }
  14. WSWindow::WSWindow(WSMessageReceiver& internal_owner, WSWindowType type)
  15. : m_internal_owner(&internal_owner)
  16. , m_type(type)
  17. , m_icon(default_window_icon())
  18. {
  19. WSWindowManager::the().add_window(*this);
  20. }
  21. WSWindow::WSWindow(WSClientConnection& client, int window_id)
  22. : m_client(&client)
  23. , m_type(WSWindowType::Normal)
  24. , m_window_id(window_id)
  25. , m_icon(default_window_icon())
  26. {
  27. WSWindowManager::the().add_window(*this);
  28. }
  29. WSWindow::~WSWindow()
  30. {
  31. WSWindowManager::the().remove_window(*this);
  32. }
  33. void WSWindow::set_title(String&& title)
  34. {
  35. if (m_title == title)
  36. return;
  37. m_title = move(title);
  38. WSWindowManager::the().notify_title_changed(*this);
  39. }
  40. void WSWindow::set_rect(const Rect& rect)
  41. {
  42. Rect old_rect;
  43. if (m_rect == rect)
  44. return;
  45. old_rect = m_rect;
  46. m_rect = rect;
  47. if (!m_client && (!m_backing_store || old_rect.size() != rect.size())) {
  48. m_backing_store = GraphicsBitmap::create(GraphicsBitmap::Format::RGB32, m_rect.size());
  49. }
  50. WSWindowManager::the().notify_rect_changed(*this, old_rect, rect);
  51. }
  52. // FIXME: Just use the same types.
  53. static WSAPI_MouseButton to_api(MouseButton button)
  54. {
  55. switch (button) {
  56. case MouseButton::None: return WSAPI_MouseButton::NoButton;
  57. case MouseButton::Left: return WSAPI_MouseButton::Left;
  58. case MouseButton::Right: return WSAPI_MouseButton::Right;
  59. case MouseButton::Middle: return WSAPI_MouseButton::Middle;
  60. }
  61. ASSERT_NOT_REACHED();
  62. }
  63. void WSWindow::on_message(WSMessage& message)
  64. {
  65. if (m_internal_owner)
  66. return m_internal_owner->on_message(message);
  67. WSAPI_ServerMessage server_message;
  68. server_message.window_id = window_id();
  69. switch (message.type()) {
  70. case WSMessage::MouseMove:
  71. server_message.type = WSAPI_ServerMessage::Type::MouseMove;
  72. server_message.mouse.position = static_cast<WSMouseEvent&>(message).position();
  73. server_message.mouse.button = WSAPI_MouseButton::NoButton;
  74. server_message.mouse.buttons = static_cast<WSMouseEvent&>(message).buttons();
  75. server_message.mouse.modifiers = static_cast<WSMouseEvent&>(message).modifiers();
  76. break;
  77. case WSMessage::MouseDown:
  78. server_message.type = WSAPI_ServerMessage::Type::MouseDown;
  79. server_message.mouse.position = static_cast<WSMouseEvent&>(message).position();
  80. server_message.mouse.button = to_api(static_cast<WSMouseEvent&>(message).button());
  81. server_message.mouse.buttons = static_cast<WSMouseEvent&>(message).buttons();
  82. server_message.mouse.modifiers = static_cast<WSMouseEvent&>(message).modifiers();
  83. break;
  84. case WSMessage::MouseUp:
  85. server_message.type = WSAPI_ServerMessage::Type::MouseUp;
  86. server_message.mouse.position = static_cast<WSMouseEvent&>(message).position();
  87. server_message.mouse.button = to_api(static_cast<WSMouseEvent&>(message).button());
  88. server_message.mouse.buttons = static_cast<WSMouseEvent&>(message).buttons();
  89. server_message.mouse.modifiers = static_cast<WSMouseEvent&>(message).modifiers();
  90. break;
  91. case WSMessage::WindowEntered:
  92. server_message.type = WSAPI_ServerMessage::Type::WindowEntered;
  93. break;
  94. case WSMessage::WindowLeft:
  95. server_message.type = WSAPI_ServerMessage::Type::WindowLeft;
  96. break;
  97. case WSMessage::KeyDown:
  98. server_message.type = WSAPI_ServerMessage::Type::KeyDown;
  99. server_message.key.character = static_cast<WSKeyEvent&>(message).character();
  100. server_message.key.key = static_cast<WSKeyEvent&>(message).key();
  101. server_message.key.modifiers = static_cast<WSKeyEvent&>(message).modifiers();
  102. break;
  103. case WSMessage::KeyUp:
  104. server_message.type = WSAPI_ServerMessage::Type::KeyUp;
  105. server_message.key.character = static_cast<WSKeyEvent&>(message).character();
  106. server_message.key.key = static_cast<WSKeyEvent&>(message).key();
  107. server_message.key.modifiers = static_cast<WSKeyEvent&>(message).modifiers();
  108. break;
  109. case WSMessage::WindowActivated:
  110. server_message.type = WSAPI_ServerMessage::Type::WindowActivated;
  111. break;
  112. case WSMessage::WindowDeactivated:
  113. server_message.type = WSAPI_ServerMessage::Type::WindowDeactivated;
  114. break;
  115. case WSMessage::WindowCloseRequest:
  116. server_message.type = WSAPI_ServerMessage::Type::WindowCloseRequest;
  117. break;
  118. case WSMessage::WindowResized:
  119. server_message.type = WSAPI_ServerMessage::Type::WindowResized;
  120. server_message.window.old_rect = static_cast<WSResizeEvent&>(message).old_rect();
  121. server_message.window.rect = static_cast<WSResizeEvent&>(message).rect();
  122. break;
  123. default:
  124. break;
  125. }
  126. if (server_message.type == WSAPI_ServerMessage::Type::Invalid)
  127. return;
  128. ASSERT(m_client);
  129. m_client->post_message(server_message);
  130. }
  131. void WSWindow::set_global_cursor_tracking_enabled(bool enabled)
  132. {
  133. m_global_cursor_tracking_enabled = enabled;
  134. }
  135. void WSWindow::set_visible(bool b)
  136. {
  137. if (m_visible == b)
  138. return;
  139. m_visible = b;
  140. invalidate();
  141. }
  142. void WSWindow::invalidate()
  143. {
  144. WSWindowManager::the().invalidate(*this);
  145. }
  146. bool WSWindow::is_active() const
  147. {
  148. return WSWindowManager::the().active_window() == this;
  149. }