From ce890c34551d9fb542f10dcb644d22d382e0c921 Mon Sep 17 00:00:00 2001 From: Kirill Chibisov Date: Sun, 24 Apr 2022 23:35:18 +0300 Subject: [PATCH] Unify behavior of `resizable` across platforms This makes X11 and Wayland follow Windows and macOS, so the size of the window could be set even though it has resizable attribute set to false. Fixes #2242. --- CHANGELOG.md | 1 + src/platform_impl/linux/wayland/window/mod.rs | 3 +++ src/platform_impl/linux/wayland/window/shim.rs | 13 ++++++++++++- src/platform_impl/linux/x11/window.rs | 11 +++++++++-- src/window.rs | 5 +++-- 5 files changed, 28 insertions(+), 5 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 86c3a83c..760439ab 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -38,6 +38,7 @@ And please only add new entries to the top of this list, right below the `# Unre sent a cancel or frame event. - On iOS, send `RedrawEventsCleared` even if there are no redraw events, consistent with other platforms. - **Breaking:** Replaced `Window::with_app_id` and `Window::with_class` with `Window::with_name` on `WindowBuilderExtUnix`. +- On Wayland and X11, fix window not resizing with `Window::set_inner_size` after calling `Window:set_resizable(false)`. # 0.26.1 (2022-01-05) diff --git a/src/platform_impl/linux/wayland/window/mod.rs b/src/platform_impl/linux/wayland/window/mod.rs index 4ef8bd2f..4cff918f 100644 --- a/src/platform_impl/linux/wayland/window/mod.rs +++ b/src/platform_impl/linux/wayland/window/mod.rs @@ -220,6 +220,9 @@ impl Window { window_requests.clone(), ); + // Set resizable state, so we can determine how to handle `Window::set_inner_size`. + window_handle.is_resizable.set(attributes.resizable); + let mut winit_state = event_loop_window_target.state.borrow_mut(); winit_state.window_map.insert(window_id, window_handle); diff --git a/src/platform_impl/linux/wayland/window/shim.rs b/src/platform_impl/linux/wayland/window/shim.rs index 60247b31..90695be5 100644 --- a/src/platform_impl/linux/wayland/window/shim.rs +++ b/src/platform_impl/linux/wayland/window/shim.rs @@ -154,6 +154,9 @@ pub struct WindowHandle { /// Current cursor icon. pub cursor_icon: Cell, + /// Whether the window is resizable. + pub is_resizable: Cell, + /// Visible cursor or not. cursor_visible: Cell, @@ -193,6 +196,7 @@ impl WindowHandle { size, pending_window_requests, cursor_icon: Cell::new(CursorIcon::Default), + is_resizable: Cell::new(true), confined: Cell::new(false), cursor_visible: Cell::new(true), pointers: Vec::new(), @@ -443,7 +447,14 @@ pub fn handle_window_requests(winit_state: &mut WinitState) { window_update.redraw_requested = true; } WindowRequest::FrameSize(size) => { - // Set new size. + if !window_handle.is_resizable.get() { + // On Wayland non-resizable window is achieved by setting both min and max + // size of the window to the same value. + let size = Some((size.width, size.height)); + window_handle.window.set_max_size(size); + window_handle.window.set_min_size(size); + } + window_handle.window.resize(size.width, size.height); // We should refresh the frame after resize. diff --git a/src/platform_impl/linux/x11/window.rs b/src/platform_impl/linux/x11/window.rs index 1be2f7cc..e3707960 100644 --- a/src/platform_impl/linux/x11/window.rs +++ b/src/platform_impl/linux/x11/window.rs @@ -1115,8 +1115,15 @@ impl UnownedWindow { #[inline] pub fn set_inner_size(&self, size: Size) { let scale_factor = self.scale_factor(); - let (width, height) = size.to_physical::(scale_factor).into(); - self.set_inner_size_physical(width, height); + let size = size.to_physical::(scale_factor).into(); + if !self.shared_state.lock().is_resizable { + self.update_normal_hints(|normal_hints| { + normal_hints.set_min_size(Some(size)); + normal_hints.set_max_size(Some(size)); + }) + .expect("Failed to call `XSetWMNormalHints`"); + } + self.set_inner_size_physical(size.0, size.1); } fn update_normal_hints(&self, callback: F) -> Result<(), XError> diff --git a/src/window.rs b/src/window.rs index f0303325..a1f11070 100644 --- a/src/window.rs +++ b/src/window.rs @@ -665,8 +665,9 @@ impl Window { /// Sets whether the window is resizable or not. /// - /// Note that making the window unresizable doesn't exempt you from handling `Resized`, as that event can still be - /// triggered by DPI scaling, entering fullscreen mode, etc. + /// Note that making the window unresizable doesn't exempt you from handling `Resized`, as that + /// event can still be triggered by DPI scaling, entering fullscreen mode, etc. Also, the + /// window could still be resized by calling `[Window::set_inner_size]`. /// /// ## Platform-specific ///