ClientConnection.cpp 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/Badge.h>
  7. #include <LibGfx/Bitmap.h>
  8. #include <LibGfx/StandardCursor.h>
  9. #include <LibGfx/SystemTheme.h>
  10. #include <WindowServer/AppletManager.h>
  11. #include <WindowServer/ClientConnection.h>
  12. #include <WindowServer/Compositor.h>
  13. #include <WindowServer/Menu.h>
  14. #include <WindowServer/MenuItem.h>
  15. #include <WindowServer/Menubar.h>
  16. #include <WindowServer/Screen.h>
  17. #include <WindowServer/Window.h>
  18. #include <WindowServer/WindowClientEndpoint.h>
  19. #include <WindowServer/WindowManager.h>
  20. #include <WindowServer/WindowSwitcher.h>
  21. #include <errno.h>
  22. #include <stdio.h>
  23. #include <unistd.h>
  24. namespace WindowServer {
  25. HashMap<int, NonnullRefPtr<ClientConnection>>* s_connections;
  26. void ClientConnection::for_each_client(Function<void(ClientConnection&)> callback)
  27. {
  28. if (!s_connections)
  29. return;
  30. for (auto& it : *s_connections) {
  31. callback(*it.value);
  32. }
  33. }
  34. ClientConnection* ClientConnection::from_client_id(int client_id)
  35. {
  36. if (!s_connections)
  37. return nullptr;
  38. auto it = s_connections->find(client_id);
  39. if (it == s_connections->end())
  40. return nullptr;
  41. return (*it).value.ptr();
  42. }
  43. ClientConnection::ClientConnection(NonnullRefPtr<Core::LocalSocket> client_socket, int client_id)
  44. : IPC::ClientConnection<WindowClientEndpoint, WindowServerEndpoint>(*this, move(client_socket), client_id)
  45. {
  46. if (!s_connections)
  47. s_connections = new HashMap<int, NonnullRefPtr<ClientConnection>>;
  48. s_connections->set(client_id, *this);
  49. async_fast_greet(Screen::the().rect(), Gfx::current_system_theme_buffer(), Gfx::FontDatabase::default_font_query(), Gfx::FontDatabase::fixed_width_font_query());
  50. }
  51. ClientConnection::~ClientConnection()
  52. {
  53. if (m_has_display_link)
  54. Compositor::the().decrement_display_link_count({});
  55. MenuManager::the().close_all_menus_from_client({}, *this);
  56. auto windows = move(m_windows);
  57. for (auto& window : windows) {
  58. window.value->detach_client({});
  59. if (window.value->type() == WindowType::Applet)
  60. AppletManager::the().remove_applet(window.value);
  61. }
  62. }
  63. void ClientConnection::die()
  64. {
  65. deferred_invoke([this](auto&) {
  66. s_connections->remove(client_id());
  67. });
  68. }
  69. void ClientConnection::notify_about_new_screen_rect(Gfx::IntRect const& rect)
  70. {
  71. async_screen_rect_changed(rect);
  72. }
  73. void ClientConnection::create_menubar(i32 menubar_id)
  74. {
  75. auto menubar = Menubar::create(*this, menubar_id);
  76. m_menubars.set(menubar_id, move(menubar));
  77. }
  78. void ClientConnection::destroy_menubar(i32 menubar_id)
  79. {
  80. auto it = m_menubars.find(menubar_id);
  81. if (it == m_menubars.end()) {
  82. did_misbehave("DestroyMenubar: Bad menubar ID");
  83. return;
  84. }
  85. m_menubars.remove(it);
  86. }
  87. void ClientConnection::create_menu(i32 menu_id, String const& menu_title)
  88. {
  89. auto menu = Menu::construct(this, menu_id, menu_title);
  90. m_menus.set(menu_id, move(menu));
  91. }
  92. void ClientConnection::destroy_menu(i32 menu_id)
  93. {
  94. auto it = m_menus.find(menu_id);
  95. if (it == m_menus.end()) {
  96. did_misbehave("DestroyMenu: Bad menu ID");
  97. return;
  98. }
  99. auto& menu = *(*it).value;
  100. menu.close();
  101. m_menus.remove(it);
  102. remove_child(menu);
  103. }
  104. void ClientConnection::set_window_menubar(i32 window_id, i32 menubar_id)
  105. {
  106. RefPtr<Window> window;
  107. {
  108. auto it = m_windows.find(window_id);
  109. if (it == m_windows.end()) {
  110. did_misbehave("SetWindowMenubar: Bad window ID");
  111. return;
  112. }
  113. window = it->value;
  114. }
  115. RefPtr<Menubar> menubar;
  116. if (menubar_id != -1) {
  117. auto it = m_menubars.find(menubar_id);
  118. if (it == m_menubars.end()) {
  119. did_misbehave("SetWindowMenubar: Bad menubar ID");
  120. return;
  121. }
  122. menubar = *(*it).value;
  123. }
  124. window->set_menubar(menubar);
  125. }
  126. void ClientConnection::add_menu_to_menubar(i32 menubar_id, i32 menu_id)
  127. {
  128. auto it = m_menubars.find(menubar_id);
  129. auto jt = m_menus.find(menu_id);
  130. if (it == m_menubars.end()) {
  131. did_misbehave("AddMenuToMenubar: Bad menubar ID");
  132. return;
  133. }
  134. if (jt == m_menus.end()) {
  135. did_misbehave("AddMenuToMenubar: Bad menu ID");
  136. return;
  137. }
  138. auto& menubar = *(*it).value;
  139. auto& menu = *(*jt).value;
  140. menubar.add_menu(menu);
  141. }
  142. void ClientConnection::add_menu_item(i32 menu_id, i32 identifier, i32 submenu_id,
  143. String const& text, bool enabled, bool checkable, bool checked, bool is_default,
  144. String const& shortcut, Gfx::ShareableBitmap const& icon, bool exclusive)
  145. {
  146. auto it = m_menus.find(menu_id);
  147. if (it == m_menus.end()) {
  148. dbgln("AddMenuItem: Bad menu ID: {}", menu_id);
  149. return;
  150. }
  151. auto& menu = *(*it).value;
  152. auto menu_item = make<MenuItem>(menu, identifier, text, shortcut, enabled, checkable, checked);
  153. if (is_default)
  154. menu_item->set_default(true);
  155. menu_item->set_icon(icon.bitmap());
  156. menu_item->set_submenu_id(submenu_id);
  157. menu_item->set_exclusive(exclusive);
  158. menu.add_item(move(menu_item));
  159. }
  160. void ClientConnection::popup_menu(i32 menu_id, Gfx::IntPoint const& screen_position)
  161. {
  162. auto position = screen_position;
  163. auto it = m_menus.find(menu_id);
  164. if (it == m_menus.end()) {
  165. did_misbehave("PopupMenu: Bad menu ID");
  166. return;
  167. }
  168. auto& menu = *(*it).value;
  169. menu.popup(position);
  170. }
  171. void ClientConnection::dismiss_menu(i32 menu_id)
  172. {
  173. auto it = m_menus.find(menu_id);
  174. if (it == m_menus.end()) {
  175. did_misbehave("DismissMenu: Bad menu ID");
  176. return;
  177. }
  178. auto& menu = *(*it).value;
  179. menu.close();
  180. }
  181. void ClientConnection::update_menu_item(i32 menu_id, i32 identifier, [[maybe_unused]] i32 submenu_id,
  182. String const& text, bool enabled, bool checkable, bool checked, bool is_default,
  183. String const& shortcut)
  184. {
  185. auto it = m_menus.find(menu_id);
  186. if (it == m_menus.end()) {
  187. did_misbehave("UpdateMenuItem: Bad menu ID");
  188. return;
  189. }
  190. auto& menu = *(*it).value;
  191. auto* menu_item = menu.item_with_identifier(identifier);
  192. if (!menu_item) {
  193. did_misbehave("UpdateMenuItem: Bad menu item identifier");
  194. return;
  195. }
  196. menu_item->set_text(text);
  197. menu_item->set_shortcut_text(shortcut);
  198. menu_item->set_enabled(enabled);
  199. menu_item->set_checkable(checkable);
  200. menu_item->set_default(is_default);
  201. if (checkable)
  202. menu_item->set_checked(checked);
  203. }
  204. void ClientConnection::add_menu_separator(i32 menu_id)
  205. {
  206. auto it = m_menus.find(menu_id);
  207. if (it == m_menus.end()) {
  208. did_misbehave("AddMenuSeparator: Bad menu ID");
  209. return;
  210. }
  211. auto& menu = *(*it).value;
  212. menu.add_item(make<MenuItem>(menu, MenuItem::Separator));
  213. }
  214. void ClientConnection::move_window_to_front(i32 window_id)
  215. {
  216. auto it = m_windows.find(window_id);
  217. if (it == m_windows.end()) {
  218. did_misbehave("MoveWindowToFront: Bad window ID");
  219. return;
  220. }
  221. WindowManager::the().move_to_front_and_make_active(*(*it).value);
  222. }
  223. void ClientConnection::set_fullscreen(i32 window_id, bool fullscreen)
  224. {
  225. auto it = m_windows.find(window_id);
  226. if (it == m_windows.end()) {
  227. did_misbehave("SetFullscreen: Bad window ID");
  228. return;
  229. }
  230. it->value->set_fullscreen(fullscreen);
  231. }
  232. void ClientConnection::set_frameless(i32 window_id, bool frameless)
  233. {
  234. auto it = m_windows.find(window_id);
  235. if (it == m_windows.end()) {
  236. did_misbehave("SetFrameless: Bad window ID");
  237. return;
  238. }
  239. it->value->set_frameless(frameless);
  240. WindowManager::the().tell_wms_window_state_changed(*it->value);
  241. }
  242. void ClientConnection::set_window_opacity(i32 window_id, float opacity)
  243. {
  244. auto it = m_windows.find(window_id);
  245. if (it == m_windows.end()) {
  246. did_misbehave("SetWindowOpacity: Bad window ID");
  247. return;
  248. }
  249. it->value->set_opacity(opacity);
  250. }
  251. void ClientConnection::set_wallpaper(String const& path)
  252. {
  253. Compositor::the().set_wallpaper(path, [&](bool success) {
  254. async_set_wallpaper_finished(success);
  255. });
  256. }
  257. void ClientConnection::set_background_color(String const& background_color)
  258. {
  259. Compositor::the().set_background_color(background_color);
  260. }
  261. void ClientConnection::set_wallpaper_mode(String const& mode)
  262. {
  263. Compositor::the().set_wallpaper_mode(mode);
  264. }
  265. Messages::WindowServer::GetWallpaperResponse ClientConnection::get_wallpaper()
  266. {
  267. return Compositor::the().wallpaper_path();
  268. }
  269. Messages::WindowServer::SetResolutionResponse ClientConnection::set_resolution(Gfx::IntSize const& resolution, int scale_factor)
  270. {
  271. return { WindowManager::the().set_resolution(resolution.width(), resolution.height(), scale_factor), WindowManager::the().resolution(), WindowManager::the().scale_factor() };
  272. }
  273. void ClientConnection::set_window_title(i32 window_id, String const& title)
  274. {
  275. auto it = m_windows.find(window_id);
  276. if (it == m_windows.end()) {
  277. did_misbehave("SetWindowTitle: Bad window ID");
  278. return;
  279. }
  280. it->value->set_title(title);
  281. }
  282. Messages::WindowServer::GetWindowTitleResponse ClientConnection::get_window_title(i32 window_id)
  283. {
  284. auto it = m_windows.find(window_id);
  285. if (it == m_windows.end()) {
  286. did_misbehave("GetWindowTitle: Bad window ID");
  287. return nullptr;
  288. }
  289. return it->value->title();
  290. }
  291. Messages::WindowServer::IsMaximizedResponse ClientConnection::is_maximized(i32 window_id)
  292. {
  293. auto it = m_windows.find(window_id);
  294. if (it == m_windows.end()) {
  295. did_misbehave("IsMaximized: Bad window ID");
  296. return nullptr;
  297. }
  298. return it->value->is_maximized();
  299. }
  300. void ClientConnection::set_window_icon_bitmap(i32 window_id, Gfx::ShareableBitmap const& icon)
  301. {
  302. auto it = m_windows.find(window_id);
  303. if (it == m_windows.end()) {
  304. did_misbehave("SetWindowIconBitmap: Bad window ID");
  305. return;
  306. }
  307. auto& window = *(*it).value;
  308. if (icon.is_valid()) {
  309. window.set_icon(*icon.bitmap());
  310. } else {
  311. window.set_default_icon();
  312. }
  313. window.frame().invalidate_titlebar();
  314. WindowManager::the().tell_wms_window_icon_changed(window);
  315. }
  316. Messages::WindowServer::SetWindowRectResponse ClientConnection::set_window_rect(i32 window_id, Gfx::IntRect const& rect)
  317. {
  318. auto it = m_windows.find(window_id);
  319. if (it == m_windows.end()) {
  320. did_misbehave("SetWindowRect: Bad window ID");
  321. return nullptr;
  322. }
  323. auto& window = *(*it).value;
  324. if (window.is_fullscreen()) {
  325. dbgln("ClientConnection: Ignoring SetWindowRect request for fullscreen window");
  326. return nullptr;
  327. }
  328. if (rect.location() != window.rect().location()) {
  329. window.set_default_positioned(false);
  330. }
  331. auto new_rect = rect;
  332. window.apply_minimum_size(new_rect);
  333. window.set_rect(new_rect);
  334. window.nudge_into_desktop();
  335. window.request_update(window.rect());
  336. return window.rect();
  337. }
  338. Messages::WindowServer::GetWindowRectResponse ClientConnection::get_window_rect(i32 window_id)
  339. {
  340. auto it = m_windows.find(window_id);
  341. if (it == m_windows.end()) {
  342. did_misbehave("GetWindowRect: Bad window ID");
  343. return nullptr;
  344. }
  345. return it->value->rect();
  346. }
  347. void ClientConnection::set_window_minimum_size(i32 window_id, Gfx::IntSize const& size)
  348. {
  349. auto it = m_windows.find(window_id);
  350. if (it == m_windows.end()) {
  351. did_misbehave("SetWindowMinimumSize: Bad window ID");
  352. return;
  353. }
  354. auto& window = *(*it).value;
  355. if (window.is_fullscreen()) {
  356. dbgln("ClientConnection: Ignoring SetWindowMinimumSize request for fullscreen window");
  357. return;
  358. }
  359. window.set_minimum_size(size);
  360. if (window.width() < window.minimum_size().width() || window.height() < window.minimum_size().height()) {
  361. // New minimum size is larger than the current window size, resize accordingly.
  362. auto new_rect = window.rect();
  363. bool did_size_clamp = window.apply_minimum_size(new_rect);
  364. window.set_rect(new_rect);
  365. window.nudge_into_desktop();
  366. window.request_update(window.rect());
  367. if (did_size_clamp)
  368. window.refresh_client_size();
  369. }
  370. }
  371. Messages::WindowServer::GetWindowMinimumSizeResponse ClientConnection::get_window_minimum_size(i32 window_id)
  372. {
  373. auto it = m_windows.find(window_id);
  374. if (it == m_windows.end()) {
  375. did_misbehave("GetWindowMinimumSize: Bad window ID");
  376. return nullptr;
  377. }
  378. return it->value->minimum_size();
  379. }
  380. Messages::WindowServer::GetAppletRectOnScreenResponse ClientConnection::get_applet_rect_on_screen(i32 window_id)
  381. {
  382. auto it = m_windows.find(window_id);
  383. if (it == m_windows.end()) {
  384. did_misbehave("GetAppletRectOnScreen: Bad window ID");
  385. return nullptr;
  386. }
  387. Gfx::IntRect applet_area_rect;
  388. if (auto* applet_area_window = AppletManager::the().window())
  389. applet_area_rect = applet_area_window->rect();
  390. return it->value->rect_in_applet_area().translated(applet_area_rect.location());
  391. }
  392. Window* ClientConnection::window_from_id(i32 window_id)
  393. {
  394. auto it = m_windows.find(window_id);
  395. if (it == m_windows.end())
  396. return nullptr;
  397. return it->value.ptr();
  398. }
  399. Messages::WindowServer::CreateWindowResponse ClientConnection::create_window(Gfx::IntRect const& rect,
  400. bool auto_position, bool has_alpha_channel, bool modal, bool minimizable, bool resizable,
  401. bool fullscreen, bool frameless, bool accessory, float opacity, float alpha_hit_threshold,
  402. Gfx::IntSize const& base_size, Gfx::IntSize const& size_increment, Gfx::IntSize const& minimum_size,
  403. Optional<Gfx::IntSize> const& resize_aspect_ratio, i32 type, String const& title, i32 parent_window_id)
  404. {
  405. Window* parent_window = nullptr;
  406. if (parent_window_id) {
  407. parent_window = window_from_id(parent_window_id);
  408. if (!parent_window) {
  409. did_misbehave("CreateWindow with bad parent_window_id");
  410. return nullptr;
  411. }
  412. }
  413. int window_id = m_next_window_id++;
  414. auto window = Window::construct(*this, (WindowType)type, window_id, modal, minimizable, frameless, resizable, fullscreen, accessory, parent_window);
  415. window->set_has_alpha_channel(has_alpha_channel);
  416. window->set_title(title);
  417. if (!fullscreen) {
  418. Gfx::IntRect new_rect = rect;
  419. if (auto_position && window->is_movable()) {
  420. new_rect = { WindowManager::the().get_recommended_window_position({ 100, 100 }), rect.size() };
  421. window->set_default_positioned(true);
  422. }
  423. window->set_minimum_size(minimum_size);
  424. bool did_size_clamp = window->apply_minimum_size(new_rect);
  425. window->set_rect(new_rect);
  426. window->nudge_into_desktop();
  427. if (did_size_clamp)
  428. window->refresh_client_size();
  429. }
  430. if (window->type() == WindowType::Desktop) {
  431. window->set_rect(WindowManager::the().desktop_rect());
  432. window->recalculate_rect();
  433. }
  434. window->set_opacity(opacity);
  435. window->set_alpha_hit_threshold(alpha_hit_threshold);
  436. window->set_size_increment(size_increment);
  437. window->set_base_size(base_size);
  438. window->set_resize_aspect_ratio(resize_aspect_ratio);
  439. window->invalidate(true, true);
  440. if (window->type() == WindowType::Applet)
  441. AppletManager::the().add_applet(*window);
  442. m_windows.set(window_id, move(window));
  443. return window_id;
  444. }
  445. void ClientConnection::destroy_window(Window& window, Vector<i32>& destroyed_window_ids)
  446. {
  447. for (auto& child_window : window.child_windows()) {
  448. if (!child_window)
  449. continue;
  450. VERIFY(child_window->window_id() != window.window_id());
  451. destroy_window(*child_window, destroyed_window_ids);
  452. }
  453. for (auto& accessory_window : window.accessory_windows()) {
  454. if (!accessory_window)
  455. continue;
  456. VERIFY(accessory_window->window_id() != window.window_id());
  457. destroy_window(*accessory_window, destroyed_window_ids);
  458. }
  459. destroyed_window_ids.append(window.window_id());
  460. if (window.type() == WindowType::Applet)
  461. AppletManager::the().remove_applet(window);
  462. window.destroy();
  463. remove_child(window);
  464. m_windows.remove(window.window_id());
  465. }
  466. Messages::WindowServer::DestroyWindowResponse ClientConnection::destroy_window(i32 window_id)
  467. {
  468. auto it = m_windows.find(window_id);
  469. if (it == m_windows.end()) {
  470. did_misbehave("DestroyWindow: Bad window ID");
  471. return nullptr;
  472. }
  473. auto& window = *(*it).value;
  474. Vector<i32> destroyed_window_ids;
  475. destroy_window(window, destroyed_window_ids);
  476. return destroyed_window_ids;
  477. }
  478. void ClientConnection::post_paint_message(Window& window, bool ignore_occlusion)
  479. {
  480. auto rect_set = window.take_pending_paint_rects();
  481. if (window.is_minimized() || (!ignore_occlusion && window.is_occluded()))
  482. return;
  483. async_paint(window.window_id(), window.size(), rect_set.rects());
  484. }
  485. void ClientConnection::invalidate_rect(i32 window_id, Vector<Gfx::IntRect> const& rects, bool ignore_occlusion)
  486. {
  487. auto it = m_windows.find(window_id);
  488. if (it == m_windows.end()) {
  489. did_misbehave("InvalidateRect: Bad window ID");
  490. return;
  491. }
  492. auto& window = *(*it).value;
  493. for (size_t i = 0; i < rects.size(); ++i)
  494. window.request_update(rects[i].intersected({ {}, window.size() }), ignore_occlusion);
  495. }
  496. void ClientConnection::did_finish_painting(i32 window_id, Vector<Gfx::IntRect> const& rects)
  497. {
  498. auto it = m_windows.find(window_id);
  499. if (it == m_windows.end()) {
  500. did_misbehave("DidFinishPainting: Bad window ID");
  501. return;
  502. }
  503. auto& window = *(*it).value;
  504. for (auto& rect : rects)
  505. window.invalidate(rect);
  506. if (window.has_alpha_channel() && window.alpha_hit_threshold() > 0.0f)
  507. WindowManager::the().reevaluate_hovered_window(&window);
  508. WindowSwitcher::the().refresh_if_needed();
  509. }
  510. void ClientConnection::set_window_backing_store(i32 window_id, [[maybe_unused]] i32 bpp,
  511. [[maybe_unused]] i32 pitch, IPC::File const& anon_file, i32 serial, bool has_alpha_channel,
  512. Gfx::IntSize const& size, bool flush_immediately)
  513. {
  514. auto it = m_windows.find(window_id);
  515. if (it == m_windows.end()) {
  516. did_misbehave("SetWindowBackingStore: Bad window ID");
  517. return;
  518. }
  519. auto& window = *(*it).value;
  520. if (window.last_backing_store() && window.last_backing_store_serial() == serial) {
  521. window.swap_backing_stores();
  522. } else {
  523. // FIXME: Plumb scale factor here eventually.
  524. Core::AnonymousBuffer buffer = Core::AnonymousBuffer::create_from_anon_fd(anon_file.take_fd(), pitch * size.height());
  525. if (!buffer.is_valid()) {
  526. did_misbehave("SetWindowBackingStore: Failed to create anonymous buffer for window backing store");
  527. return;
  528. }
  529. auto backing_store = Gfx::Bitmap::create_with_anonymous_buffer(
  530. has_alpha_channel ? Gfx::BitmapFormat::BGRA8888 : Gfx::BitmapFormat::BGRx8888,
  531. buffer,
  532. size,
  533. 1,
  534. {});
  535. window.set_backing_store(move(backing_store), serial);
  536. }
  537. if (flush_immediately)
  538. window.invalidate(false);
  539. }
  540. void ClientConnection::set_global_cursor_tracking(i32 window_id, bool enabled)
  541. {
  542. auto it = m_windows.find(window_id);
  543. if (it == m_windows.end()) {
  544. did_misbehave("SetGlobalCursorTracking: Bad window ID");
  545. return;
  546. }
  547. it->value->set_global_cursor_tracking_enabled(enabled);
  548. }
  549. void ClientConnection::set_window_cursor(i32 window_id, i32 cursor_type)
  550. {
  551. auto it = m_windows.find(window_id);
  552. if (it == m_windows.end()) {
  553. did_misbehave("SetWindowCursor: Bad window ID");
  554. return;
  555. }
  556. auto& window = *(*it).value;
  557. if (cursor_type < 0 || cursor_type >= (i32)Gfx::StandardCursor::__Count) {
  558. did_misbehave("SetWindowCursor: Bad cursor type");
  559. return;
  560. }
  561. window.set_cursor(Cursor::create((Gfx::StandardCursor)cursor_type));
  562. if (&window == WindowManager::the().hovered_window())
  563. Compositor::the().invalidate_cursor();
  564. }
  565. void ClientConnection::set_window_custom_cursor(i32 window_id, Gfx::ShareableBitmap const& cursor)
  566. {
  567. auto it = m_windows.find(window_id);
  568. if (it == m_windows.end()) {
  569. did_misbehave("SetWindowCustomCursor: Bad window ID");
  570. return;
  571. }
  572. auto& window = *(*it).value;
  573. if (!cursor.is_valid()) {
  574. did_misbehave("SetWindowCustomCursor: Bad cursor");
  575. return;
  576. }
  577. window.set_cursor(Cursor::create(*cursor.bitmap()));
  578. Compositor::the().invalidate_cursor();
  579. }
  580. void ClientConnection::set_window_has_alpha_channel(i32 window_id, bool has_alpha_channel)
  581. {
  582. auto it = m_windows.find(window_id);
  583. if (it == m_windows.end()) {
  584. did_misbehave("SetWindowHasAlphaChannel: Bad window ID");
  585. return;
  586. }
  587. it->value->set_has_alpha_channel(has_alpha_channel);
  588. }
  589. void ClientConnection::set_window_alpha_hit_threshold(i32 window_id, float threshold)
  590. {
  591. auto it = m_windows.find(window_id);
  592. if (it == m_windows.end()) {
  593. did_misbehave("SetWindowAlphaHitThreshold: Bad window ID");
  594. return;
  595. }
  596. it->value->set_alpha_hit_threshold(threshold);
  597. }
  598. void ClientConnection::start_window_resize(i32 window_id)
  599. {
  600. auto it = m_windows.find(window_id);
  601. if (it == m_windows.end()) {
  602. did_misbehave("WM_StartWindowResize: Bad window ID");
  603. return;
  604. }
  605. auto& window = *(*it).value;
  606. if (!window.is_resizable()) {
  607. dbgln("Client wants to start resizing a non-resizable window");
  608. return;
  609. }
  610. // FIXME: We are cheating a bit here by using the current cursor location and hard-coding the left button.
  611. // Maybe the client should be allowed to specify what initiated this request?
  612. WindowManager::the().start_window_resize(window, Screen::the().cursor_location(), MouseButton::Left);
  613. }
  614. Messages::WindowServer::StartDragResponse ClientConnection::start_drag(String const& text, HashMap<String, ByteBuffer> const& mime_data, Gfx::ShareableBitmap const& drag_bitmap)
  615. {
  616. auto& wm = WindowManager::the();
  617. if (wm.dnd_client())
  618. return false;
  619. wm.start_dnd_drag(*this, text, drag_bitmap.bitmap(), Core::MimeData::construct(mime_data));
  620. return true;
  621. }
  622. Messages::WindowServer::SetSystemThemeResponse ClientConnection::set_system_theme(String const& theme_path, String const& theme_name)
  623. {
  624. bool success = WindowManager::the().update_theme(theme_path, theme_name);
  625. return success;
  626. }
  627. Messages::WindowServer::GetSystemThemeResponse ClientConnection::get_system_theme()
  628. {
  629. auto wm_config = Core::ConfigFile::open("/etc/WindowServer.ini");
  630. auto name = wm_config->read_entry("Theme", "Name");
  631. return name;
  632. }
  633. Messages::WindowServer::SetSystemFontsResponse ClientConnection::set_system_fonts(String const& default_font_query, String const& fixed_width_font_query)
  634. {
  635. if (!Gfx::FontDatabase::the().get_by_name(default_font_query)
  636. || !Gfx::FontDatabase::the().get_by_name(fixed_width_font_query)) {
  637. dbgln("Received unusable font queries: '{}' and '{}'", default_font_query, fixed_width_font_query);
  638. return false;
  639. }
  640. dbgln("Updating fonts: '{}' and '{}'", default_font_query, fixed_width_font_query);
  641. Gfx::FontDatabase::set_default_font_query(default_font_query);
  642. Gfx::FontDatabase::set_fixed_width_font_query(fixed_width_font_query);
  643. ClientConnection::for_each_client([&](auto& client) {
  644. client.async_update_system_fonts(default_font_query, fixed_width_font_query);
  645. });
  646. WindowManager::the().invalidate_after_theme_or_font_change();
  647. auto wm_config = Core::ConfigFile::open("/etc/WindowServer.ini");
  648. wm_config->write_entry("Fonts", "Default", default_font_query);
  649. wm_config->write_entry("Fonts", "FixedWidth", fixed_width_font_query);
  650. return true;
  651. }
  652. void ClientConnection::set_window_base_size_and_size_increment(i32 window_id, Gfx::IntSize const& base_size, Gfx::IntSize const& size_increment)
  653. {
  654. auto it = m_windows.find(window_id);
  655. if (it == m_windows.end()) {
  656. did_misbehave("SetWindowBaseSizeAndSizeIncrementResponse: Bad window ID");
  657. return;
  658. }
  659. auto& window = *it->value;
  660. window.set_base_size(base_size);
  661. window.set_size_increment(size_increment);
  662. }
  663. void ClientConnection::set_window_resize_aspect_ratio(i32 window_id, Optional<Gfx::IntSize> const& resize_aspect_ratio)
  664. {
  665. auto it = m_windows.find(window_id);
  666. if (it == m_windows.end()) {
  667. did_misbehave("SetWindowResizeAspectRatioResponse: Bad window ID");
  668. return;
  669. }
  670. auto& window = *it->value;
  671. window.set_resize_aspect_ratio(resize_aspect_ratio);
  672. }
  673. void ClientConnection::enable_display_link()
  674. {
  675. if (m_has_display_link)
  676. return;
  677. m_has_display_link = true;
  678. Compositor::the().increment_display_link_count({});
  679. }
  680. void ClientConnection::disable_display_link()
  681. {
  682. if (!m_has_display_link)
  683. return;
  684. m_has_display_link = false;
  685. Compositor::the().decrement_display_link_count({});
  686. }
  687. void ClientConnection::notify_display_link(Badge<Compositor>)
  688. {
  689. if (!m_has_display_link)
  690. return;
  691. async_display_link_notification();
  692. }
  693. void ClientConnection::set_window_progress(i32 window_id, Optional<i32> const& progress)
  694. {
  695. auto it = m_windows.find(window_id);
  696. if (it == m_windows.end()) {
  697. did_misbehave("SetWindowProgress with bad window ID");
  698. return;
  699. }
  700. it->value->set_progress(progress);
  701. }
  702. void ClientConnection::refresh_system_theme()
  703. {
  704. // Post the client an UpdateSystemTheme message to refresh its theme.
  705. async_update_system_theme(Gfx::current_system_theme_buffer());
  706. }
  707. void ClientConnection::pong()
  708. {
  709. m_ping_timer = nullptr;
  710. set_unresponsive(false);
  711. }
  712. Messages::WindowServer::GetGlobalCursorPositionResponse ClientConnection::get_global_cursor_position()
  713. {
  714. return Screen::the().cursor_location();
  715. }
  716. void ClientConnection::set_mouse_acceleration(float factor)
  717. {
  718. double dbl_factor = (double)factor;
  719. if (dbl_factor < mouse_accel_min || dbl_factor > mouse_accel_max) {
  720. did_misbehave("SetMouseAcceleration with bad acceleration factor");
  721. return;
  722. }
  723. WindowManager::the().set_acceleration_factor(dbl_factor);
  724. }
  725. Messages::WindowServer::GetMouseAccelerationResponse ClientConnection::get_mouse_acceleration()
  726. {
  727. return Screen::the().acceleration_factor();
  728. }
  729. void ClientConnection::set_scroll_step_size(u32 step_size)
  730. {
  731. if (step_size < scroll_step_size_min) {
  732. did_misbehave("SetScrollStepSize with bad scroll step size");
  733. return;
  734. }
  735. WindowManager::the().set_scroll_step_size(step_size);
  736. }
  737. Messages::WindowServer::GetScrollStepSizeResponse ClientConnection::get_scroll_step_size()
  738. {
  739. return Screen::the().scroll_step_size();
  740. }
  741. void ClientConnection::set_double_click_speed(i32 speed)
  742. {
  743. if (speed < double_click_speed_min || speed > double_click_speed_max) {
  744. did_misbehave("SetDoubleClickSpeed with bad speed");
  745. return;
  746. }
  747. WindowManager::the().set_double_click_speed(speed);
  748. }
  749. Messages::WindowServer::GetDoubleClickSpeedResponse ClientConnection::get_double_click_speed()
  750. {
  751. return WindowManager::the().double_click_speed();
  752. }
  753. void ClientConnection::set_unresponsive(bool unresponsive)
  754. {
  755. if (m_unresponsive == unresponsive)
  756. return;
  757. m_unresponsive = unresponsive;
  758. for (auto& it : m_windows) {
  759. auto& window = *it.value;
  760. window.invalidate(true, true);
  761. if (unresponsive) {
  762. window.set_cursor_override(WindowManager::the().wait_cursor());
  763. } else {
  764. window.remove_cursor_override();
  765. }
  766. }
  767. Compositor::the().invalidate_cursor();
  768. }
  769. void ClientConnection::may_have_become_unresponsive()
  770. {
  771. async_ping();
  772. m_ping_timer = Core::Timer::create_single_shot(1000, [this] {
  773. set_unresponsive(true);
  774. });
  775. m_ping_timer->start();
  776. }
  777. void ClientConnection::did_become_responsive()
  778. {
  779. set_unresponsive(false);
  780. }
  781. Messages::WindowServer::GetScreenBitmapResponse ClientConnection::get_screen_bitmap(Optional<Gfx::IntRect> const& rect)
  782. {
  783. if (rect.has_value()) {
  784. auto bitmap = Compositor::the().front_bitmap_for_screenshot({}).cropped(rect.value());
  785. return bitmap->to_shareable_bitmap();
  786. }
  787. auto& bitmap = Compositor::the().front_bitmap_for_screenshot({});
  788. return bitmap.to_shareable_bitmap();
  789. }
  790. Messages::WindowServer::IsWindowModifiedResponse ClientConnection::is_window_modified(i32 window_id)
  791. {
  792. auto it = m_windows.find(window_id);
  793. if (it == m_windows.end()) {
  794. did_misbehave("IsWindowModified: Bad window ID");
  795. return nullptr;
  796. }
  797. auto& window = *it->value;
  798. return window.is_modified();
  799. }
  800. void ClientConnection::set_window_modified(i32 window_id, bool modified)
  801. {
  802. auto it = m_windows.find(window_id);
  803. if (it == m_windows.end()) {
  804. did_misbehave("SetWindowModified: Bad window ID");
  805. return;
  806. }
  807. auto& window = *it->value;
  808. window.set_modified(modified);
  809. }
  810. }