WMClientConnection.cpp 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. /*
  2. * Copyright (c) 2021, the SerenityOS developers.
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <WindowServer/AppletManager.h>
  7. #include <WindowServer/ClientConnection.h>
  8. #include <WindowServer/Screen.h>
  9. #include <WindowServer/WMClientConnection.h>
  10. namespace WindowServer {
  11. HashMap<int, NonnullRefPtr<WMClientConnection>> WMClientConnection::s_connections {};
  12. WMClientConnection::WMClientConnection(NonnullRefPtr<Core::LocalSocket> client_socket, int client_id)
  13. : IPC::ClientConnection<WindowManagerClientEndpoint, WindowManagerServerEndpoint>(*this, move(client_socket), client_id)
  14. {
  15. s_connections.set(client_id, *this);
  16. }
  17. WMClientConnection::~WMClientConnection()
  18. {
  19. // The WM has gone away, so take away the applet manager (cause there's nowhere
  20. // to draw it in).
  21. AppletManager::the().set_position({});
  22. }
  23. void WMClientConnection::die()
  24. {
  25. deferred_invoke([this] {
  26. s_connections.remove(client_id());
  27. });
  28. }
  29. void WMClientConnection::set_applet_area_position(Gfx::IntPoint const& position)
  30. {
  31. if (m_window_id < 0) {
  32. did_misbehave("SetAppletAreaPosition: WM didn't assign window as manager yet");
  33. // FIXME: return ok boolean?
  34. return;
  35. }
  36. AppletManager::the().set_position(position);
  37. WindowServer::ClientConnection::for_each_client([](auto& connection) {
  38. connection.post_message(Messages::WindowClient::AppletAreaRectChanged(AppletManager::the().window()->rect()));
  39. });
  40. }
  41. void WMClientConnection::set_active_window(i32 client_id, i32 window_id)
  42. {
  43. auto* client = WindowServer::ClientConnection::from_client_id(client_id);
  44. if (!client) {
  45. did_misbehave("SetActiveWindow: Bad client ID");
  46. return;
  47. }
  48. auto it = client->m_windows.find(window_id);
  49. if (it == client->m_windows.end()) {
  50. did_misbehave("SetActiveWindow: Bad window ID");
  51. return;
  52. }
  53. auto& window = *(*it).value;
  54. WindowManager::the().minimize_windows(window, false);
  55. WindowManager::the().move_to_front_and_make_active(window);
  56. }
  57. void WMClientConnection::popup_window_menu(i32 client_id, i32 window_id, Gfx::IntPoint const& screen_position)
  58. {
  59. auto* client = WindowServer::ClientConnection::from_client_id(client_id);
  60. if (!client) {
  61. did_misbehave("PopupWindowMenu: Bad client ID");
  62. return;
  63. }
  64. auto it = client->m_windows.find(window_id);
  65. if (it == client->m_windows.end()) {
  66. did_misbehave("PopupWindowMenu: Bad window ID");
  67. return;
  68. }
  69. auto& window = *(*it).value;
  70. if (auto* modal_window = window.blocking_modal_window()) {
  71. modal_window->popup_window_menu(screen_position, WindowMenuDefaultAction::BasedOnWindowState);
  72. } else {
  73. window.popup_window_menu(screen_position, WindowMenuDefaultAction::BasedOnWindowState);
  74. }
  75. }
  76. void WMClientConnection::start_window_resize(i32 client_id, i32 window_id)
  77. {
  78. auto* client = WindowServer::ClientConnection::from_client_id(client_id);
  79. if (!client) {
  80. did_misbehave("WM_StartWindowResize: Bad client ID");
  81. return;
  82. }
  83. auto it = client->m_windows.find(window_id);
  84. if (it == client->m_windows.end()) {
  85. did_misbehave("WM_StartWindowResize: Bad window ID");
  86. return;
  87. }
  88. auto& window = *(*it).value;
  89. // FIXME: We are cheating a bit here by using the current cursor location and hard-coding the left button.
  90. // Maybe the client should be allowed to specify what initiated this request?
  91. WindowManager::the().start_window_resize(window, ScreenInput::the().cursor_location(), MouseButton::Primary);
  92. }
  93. void WMClientConnection::set_window_minimized(i32 client_id, i32 window_id, bool minimized)
  94. {
  95. auto* client = WindowServer::ClientConnection::from_client_id(client_id);
  96. if (!client) {
  97. did_misbehave("WM_SetWindowMinimized: Bad client ID");
  98. return;
  99. }
  100. auto it = client->m_windows.find(window_id);
  101. if (it == client->m_windows.end()) {
  102. did_misbehave("WM_SetWindowMinimized: Bad window ID");
  103. return;
  104. }
  105. auto& window = *(*it).value;
  106. WindowManager::the().minimize_windows(window, minimized);
  107. }
  108. void WMClientConnection::toggle_show_desktop()
  109. {
  110. bool should_hide = false;
  111. auto& current_window_stack = WindowManager::the().current_window_stack();
  112. current_window_stack.for_each_window([&](auto& window) {
  113. if (window.type() == WindowType::Normal && window.is_minimizable()) {
  114. if (!window.is_hidden() && !window.is_minimized()) {
  115. should_hide = true;
  116. return IterationDecision::Break;
  117. }
  118. }
  119. return IterationDecision::Continue;
  120. });
  121. current_window_stack.for_each_window([&](auto& window) {
  122. if (window.type() == WindowType::Normal && window.is_minimizable()) {
  123. auto state = window.minimized_state();
  124. if (state == WindowMinimizedState::None || state == WindowMinimizedState::Hidden) {
  125. WindowManager::the().hide_windows(window, should_hide);
  126. }
  127. }
  128. return IterationDecision::Continue;
  129. });
  130. }
  131. void WMClientConnection::set_event_mask(u32 event_mask)
  132. {
  133. m_event_mask = event_mask;
  134. }
  135. void WMClientConnection::set_manager_window(i32 window_id)
  136. {
  137. m_window_id = window_id;
  138. // Let the window manager know that we obtained a manager window, and should
  139. // receive information about other windows.
  140. WindowManager::the().greet_window_manager(*this);
  141. }
  142. void WMClientConnection::set_workspace(u32 row, u32 col)
  143. {
  144. WindowManager::the().switch_to_window_stack(row, col);
  145. }
  146. void WMClientConnection::set_window_taskbar_rect(i32 client_id, i32 window_id, Gfx::IntRect const& rect)
  147. {
  148. // Because the Taskbar (which should be the only user of this API) does not own the
  149. // window or the client id, there is a possibility that it may send this message for
  150. // a window or client that may have been destroyed already. This is not an error,
  151. // and we should not call did_misbehave() for either.
  152. auto* client = WindowServer::ClientConnection::from_client_id(client_id);
  153. if (!client)
  154. return;
  155. auto it = client->m_windows.find(window_id);
  156. if (it == client->m_windows.end())
  157. return;
  158. auto& window = *(*it).value;
  159. window.set_taskbar_rect(rect);
  160. }
  161. }