Skip to main content

truce_gui/
platform.rs

1//! Platform window bridging for baseview.
2//!
3//! Bridges truce's `RawWindowHandle` to baseview's `HasRawWindowHandle`
4//! (raw-window-handle 0.5), and provides scale factor querying and
5//! wgpu surface creation.
6
7// `HasRawDisplayHandle` / `RwhRawDisplayHandle` are only touched on
8// the Linux (X11) arm of `HasRawWindowHandle for ParentWindow`;
9// silence the macOS/Windows dead-import warning.
10#[allow(unused_imports)]
11use raw_window_handle::{
12    HasRawDisplayHandle, HasRawWindowHandle, RawDisplayHandle as RwhRawDisplayHandle,
13    RawWindowHandle as RwhRawWindowHandle,
14};
15use truce_core::editor::RawWindowHandle;
16
17use std::sync::Arc;
18use std::sync::atomic::{AtomicU64, Ordering};
19
20/// Newtype bridging truce's `RawWindowHandle` to baseview's
21/// `HasRawWindowHandle` (raw-window-handle 0.5).
22pub struct ParentWindow(pub RawWindowHandle);
23
24unsafe impl HasRawWindowHandle for ParentWindow {
25    fn raw_window_handle(&self) -> RwhRawWindowHandle {
26        match self.0 {
27            RawWindowHandle::AppKit(ptr) => {
28                let mut handle = raw_window_handle::AppKitWindowHandle::empty();
29                handle.ns_view = ptr;
30                RwhRawWindowHandle::AppKit(handle)
31            }
32            RawWindowHandle::UiKit(ptr) => {
33                // baseview doesn't host on iOS - the iOS editor
34                // path attaches a UIView directly without going
35                // through this bridge. We surface the handle for
36                // completeness (and so future iOS-aware backends
37                // can read it) but in practice no caller on iOS
38                // reaches this arm.
39                let mut handle = raw_window_handle::UiKitWindowHandle::empty();
40                handle.ui_view = ptr;
41                RwhRawWindowHandle::UiKit(handle)
42            }
43            RawWindowHandle::Win32(ptr) => {
44                let mut handle = raw_window_handle::Win32WindowHandle::empty();
45                handle.hwnd = ptr;
46                RwhRawWindowHandle::Win32(handle)
47            }
48            RawWindowHandle::X11(window_id) => {
49                let mut handle = raw_window_handle::XlibWindowHandle::empty();
50                // rwh 0.5 field type is c_ulong: u64 on Linux/macOS, u32 on Windows.
51                // The Windows narrowing is the lossy edge - `XID` is 32-bit there.
52                #[allow(clippy::cast_possible_truncation)]
53                {
54                    handle.window = window_id as _;
55                }
56                RwhRawWindowHandle::Xlib(handle)
57            }
58        }
59    }
60}
61
62/// Query the backing scale factor from the parent `NSView`'s window.
63#[cfg(target_os = "macos")]
64#[must_use]
65pub fn query_backing_scale(parent: &RawWindowHandle) -> f64 {
66    use objc::{msg_send, sel, sel_impl};
67
68    let ns_view_ptr = match parent {
69        RawWindowHandle::AppKit(ptr) => *ptr,
70        _ => return 1.0,
71    };
72
73    if ns_view_ptr.is_null() {
74        return 1.0;
75    }
76
77    unsafe {
78        let ns_view = ns_view_ptr.cast::<objc::runtime::Object>();
79        let window: *mut objc::runtime::Object = msg_send![ns_view, window];
80        let scale: f64 = if window.is_null() {
81            let screen: *mut objc::runtime::Object = msg_send![objc::class!(NSScreen), mainScreen];
82            if screen.is_null() {
83                2.0
84            } else {
85                msg_send![screen, backingScaleFactor]
86            }
87        } else {
88            msg_send![window, backingScaleFactor]
89        };
90        if scale < 1.0 { 1.0 } else { scale }
91    }
92}
93
94#[cfg(target_os = "windows")]
95#[must_use]
96pub fn query_backing_scale(parent: &RawWindowHandle) -> f64 {
97    let hwnd = match parent {
98        RawWindowHandle::Win32(ptr) => *ptr,
99        _ => return 1.0,
100    };
101    win32_dpi_scale(hwnd)
102}
103
104#[cfg(target_os = "linux")]
105#[must_use]
106pub fn query_backing_scale(_parent: &RawWindowHandle) -> f64 {
107    main_screen_scale()
108}
109
110#[cfg(target_os = "ios")]
111#[must_use]
112pub fn query_backing_scale(parent: &RawWindowHandle) -> f64 {
113    use objc2::msg_send;
114    use objc2::runtime::AnyObject;
115
116    let ui_view_ptr = match parent {
117        RawWindowHandle::UiKit(ptr) => *ptr,
118        _ => return 1.0,
119    };
120    if ui_view_ptr.is_null() {
121        return main_screen_scale();
122    }
123    // SAFETY: UIView is a UIKit class; `contentScaleFactor` is a
124    // public Objective-C property returning CGFloat (= f64 on
125    // arm64). Called on the main thread per UIKit's threading
126    // rule, which is also where AUv3 view controllers live.
127    unsafe {
128        let ui_view: *mut AnyObject = ui_view_ptr.cast();
129        let scale: f64 = msg_send![ui_view, contentScaleFactor];
130        if scale > 0.0 { scale } else { 1.0 }
131    }
132}
133
134#[cfg(target_os = "ios")]
135#[must_use]
136pub fn main_screen_scale() -> f64 {
137    use objc2::msg_send;
138    use objc2::runtime::{AnyClass, AnyObject};
139    // SAFETY: `+[UIScreen mainScreen]` is documented to return the
140    // process's primary screen on the main thread.
141    unsafe {
142        let Some(cls) = AnyClass::get(c"UIScreen") else {
143            return 1.0;
144        };
145        let screen: *mut AnyObject = msg_send![cls, mainScreen];
146        if screen.is_null() {
147            return 1.0;
148        }
149        let scale: f64 = msg_send![screen, scale];
150        if scale > 0.0 { scale } else { 1.0 }
151    }
152}
153
154/// Query the main screen's backing scale factor (no parent window needed).
155#[cfg(target_os = "macos")]
156#[must_use]
157pub fn main_screen_scale() -> f64 {
158    use objc::{msg_send, sel, sel_impl};
159    unsafe {
160        let screen: *mut objc::runtime::Object = msg_send![objc::class!(NSScreen), mainScreen];
161        if screen.is_null() {
162            1.0
163        } else {
164            let scale: f64 = msg_send![screen, backingScaleFactor];
165            if scale < 1.0 { 1.0 } else { scale }
166        }
167    }
168}
169
170#[cfg(target_os = "windows")]
171#[must_use]
172pub fn main_screen_scale() -> f64 {
173    win32_dpi_scale(std::ptr::null_mut())
174}
175
176/// Shared, mutable editor scale factor.
177///
178/// Single source of truth for the live content-scale of an open plugin
179/// window. Each GUI backend (egui / iced / slint) constructs one in
180/// `Editor::open`, stores it on the editor for `set_scale_factor` to
181/// write through, and hands a clone to its baseview `WindowHandler` so
182/// the render thread can pick up changes between frames.
183///
184/// Two writers, one reader-per-frame:
185/// - `Editor::set_scale_factor` (host → editor, e.g. CLAP `set_scale`,
186///   VST3 Windows `IPlugViewContentScaleSupport`).
187/// - `WindowEvent::Resized` (baseview → handler, fired when the OS
188///   reports a new content scale, e.g. dragging the window across
189///   monitors with different DPIs).
190///
191/// Most-recent-write wins. The handler tracks a `last_applied_scale`
192/// alongside its `EditorScale` clone and, when it observes a divergence
193/// at frame start, recomputes physical sizes and reconfigures its
194/// surface / renderer.
195#[derive(Clone)]
196pub struct EditorScale {
197    inner: Arc<AtomicU64>,
198}
199
200impl EditorScale {
201    /// Construct with an initial scale. Non-finite or non-positive
202    /// values clamp to 1.0 so callers never have to defend against
203    /// `0.0 * size` collapsing the surface.
204    #[must_use]
205    pub fn new(initial: f64) -> Self {
206        let v = if initial.is_finite() && initial > 0.0 {
207            initial
208        } else {
209            1.0
210        };
211        Self {
212            inner: Arc::new(AtomicU64::new(v.to_bits())),
213        }
214    }
215
216    /// Read the current scale.
217    #[must_use]
218    pub fn get(&self) -> f64 {
219        f64::from_bits(self.inner.load(Ordering::Relaxed))
220    }
221
222    /// Read the current scale, narrowed to `f32` for renderer / DSP
223    /// use. Display scales never exceed 4.0 in practice, so the f64
224    /// → f32 narrowing is invisible.
225    #[allow(clippy::cast_possible_truncation)]
226    #[must_use]
227    pub fn get_f32(&self) -> f32 {
228        self.get() as f32
229    }
230
231    /// Update the current scale. Non-finite or non-positive values are
232    /// silently dropped - callers are forwarding numbers from hosts /
233    /// `info.scale()` where a bad value is a host bug, not something
234    /// to propagate into the surface config.
235    pub fn set(&self, scale: f64) {
236        if scale.is_finite() && scale > 0.0 {
237            self.inner.store(scale.to_bits(), Ordering::Relaxed);
238        } else {
239            // Surface the upstream bug at least in debug builds so a
240            // host that's emitting bad scales doesn't get silently
241            // ignored. Production builds drop quietly to keep the
242            // editor running.
243            log::warn!(
244                "EditorScale::set ignored a bad value ({scale}); \
245                 expected finite, positive f64",
246            );
247        }
248    }
249
250    /// Pick up a host-driven scale change since the last frame.
251    ///
252    /// Reads the current scale (narrowed to `f32`) and compares it
253    /// bit-identically against `last`. When the value moved, updates
254    /// `last` and returns `Some(cur)`; otherwise returns `None`.
255    ///
256    /// Used by every editor backend's per-frame loop to gate surface /
257    /// renderer reconfiguration on actual host scale events. Bit-equality
258    /// is the correct semantics - the cell is written verbatim from
259    /// host callbacks, never through accumulating arithmetic, so an
260    /// epsilon-based check would either thrash on noise (there is
261    /// none) or miss a legitimate `1.0 → 1.0001` host signal.
262    #[allow(clippy::cast_possible_truncation, clippy::float_cmp)]
263    pub fn take_change(&self, last: &mut f32) -> Option<f32> {
264        let cur = self.get() as f32;
265        if cur == *last {
266            None
267        } else {
268            *last = cur;
269            Some(cur)
270        }
271    }
272}
273
274/// Convert a logical extent (in points) to physical pixels.
275///
276/// Standardised rounding policy across every truce GUI backend:
277/// round to nearest, then clamp the result to `1` so a degenerate
278/// `0 × scale` doesn't collapse a wgpu surface (`width: 0` is a
279/// validation error). The `logical.max(1)` guard handles the
280/// converse - a zero-logical caller can't multiply through to `0`
281/// before the round.
282///
283/// Canonical definition lives in `truce-gui-types` so `truce-gpu`'s
284/// `WgpuBackend` can call it without a `truce-gui` dep (cycle); the
285/// re-export below preserves the historical `truce_gui::to_physical_px`
286/// path.
287pub use truce_gui_types::to_physical_px;
288
289/// Cached display scale factor on Linux, stored as f64 bits. Zero means unset.
290///
291/// Linux has no safe synchronous DPI query from plugin code - the authoritative
292/// value is read by baseview internally (from `Xft.dpi` with a screen-geometry
293/// fallback) and delivered via `WindowEvent::Resized::info.scale()` once the
294/// window is live. We cache the first value an editor sees there so that later
295/// pre-window `main_screen_scale()` calls (e.g. the next editor's `::new`)
296/// return something useful instead of 1.0.
297#[cfg(target_os = "linux")]
298static LINUX_SCALE_BITS: AtomicU64 = AtomicU64::new(0);
299
300/// Record the display scale factor observed from baseview on Linux. Editors
301/// should call this from their `WindowEvent::Resized` handlers so subsequent
302/// pre-window queries match what baseview is delivering. No-op on non-Linux.
303pub fn note_linux_scale_factor(scale: f64) {
304    #[cfg(target_os = "linux")]
305    {
306        if scale.is_finite() && scale > 0.0 {
307            LINUX_SCALE_BITS.store(scale.to_bits(), Ordering::Relaxed);
308        }
309    }
310    #[cfg(not(target_os = "linux"))]
311    {
312        let _ = scale;
313    }
314}
315
316#[cfg(target_os = "linux")]
317pub fn main_screen_scale() -> f64 {
318    // Priority: TRUCE_SCALE env var (dev/test override) → cached scale
319    // observed from baseview → 1.0 fallback. No side-channel Xlib calls -
320    // those crashed inside NVIDIA's Vulkan driver when invoked from the
321    // render thread.
322    if let Ok(s) = std::env::var("TRUCE_SCALE")
323        && let Ok(v) = s.parse::<f64>()
324        && v.is_finite()
325        && v > 0.0
326    {
327        return v;
328    }
329    let bits = LINUX_SCALE_BITS.load(Ordering::Relaxed);
330    if bits == 0 {
331        return 1.0;
332    }
333    let v = f64::from_bits(bits);
334    if v.is_finite() && v > 0.0 { v } else { 1.0 }
335}
336
337/// Query the DPI scale factor on Windows.
338/// If `hwnd` is non-null, queries per-window DPI; otherwise queries the system DPI.
339#[cfg(target_os = "windows")]
340fn win32_dpi_scale(hwnd: *mut std::ffi::c_void) -> f64 {
341    // Default DPI is 96; scale = actual_dpi / 96.
342    const DEFAULT_DPI: u32 = 96;
343
344    unsafe extern "system" {
345        fn GetDpiForWindow(hwnd: *mut std::ffi::c_void) -> u32;
346        fn GetDpiForSystem() -> u32;
347    }
348
349    let dpi = if hwnd.is_null() {
350        unsafe { GetDpiForSystem() }
351    } else {
352        let d = unsafe { GetDpiForWindow(hwnd) };
353        if d == 0 {
354            unsafe { GetDpiForSystem() }
355        } else {
356            d
357        }
358    };
359
360    if dpi == 0 {
361        1.0
362    } else {
363        f64::from(dpi) / f64::from(DEFAULT_DPI)
364    }
365}
366
367#[cfg(target_os = "windows")]
368fn current_module_hinstance() -> Option<std::num::NonZeroIsize> {
369    unsafe extern "system" {
370        fn GetModuleHandleW(lpModuleName: *const u16) -> isize;
371    }
372    // SAFETY: `GetModuleHandleW(NULL)` is documented to return the running
373    // EXE's HMODULE without acquiring a refcount; no threading or aliasing
374    // concerns. Returns 0 only in pathological cases (kernel32 missing).
375    let hmodule = unsafe { GetModuleHandleW(std::ptr::null()) };
376    std::num::NonZeroIsize::new(hmodule)
377}
378
379/// Bridge a baseview raw-window-handle 0.5 to a wgpu-compatible
380/// `SurfaceTargetUnsafe` using rwh 0.6 types.
381///
382/// Both `truce-gui`'s blit pipeline (cpu mode) and
383/// `truce_gpu::WgpuBackend::from_window` (gpu mode, used by
384/// `GpuEditor`) need this bridge; the two crates can't share a
385/// canonical copy without forming a dep cycle, so each carries its
386/// own ~100 LOC version. The two are kept in sync by inspection.
387///
388/// # Safety
389/// The window handle must be valid for the lifetime of the returned surface.
390#[cfg(not(target_os = "ios"))]
391#[must_use]
392pub unsafe fn create_wgpu_surface(
393    instance: &wgpu::Instance,
394    window: &baseview::Window,
395) -> Option<wgpu::Surface<'static>> {
396    unsafe {
397        let rwh = window.raw_window_handle();
398        let surface_target = match rwh {
399            #[cfg(target_os = "macos")]
400            RwhRawWindowHandle::AppKit(handle) => {
401                let ns_view = handle.ns_view;
402                if ns_view.is_null() {
403                    return None;
404                }
405                let rwh6_window = wgpu::rwh::RawWindowHandle::AppKit(
406                    wgpu::rwh::AppKitWindowHandle::new(std::ptr::NonNull::new(ns_view)?),
407                );
408                let rwh6_display =
409                    wgpu::rwh::RawDisplayHandle::AppKit(wgpu::rwh::AppKitDisplayHandle::new());
410                wgpu::SurfaceTargetUnsafe::RawHandle {
411                    raw_display_handle: Some(rwh6_display),
412                    raw_window_handle: rwh6_window,
413                }
414            }
415            #[cfg(target_os = "windows")]
416            RwhRawWindowHandle::Win32(handle) => {
417                let hwnd = handle.hwnd;
418                if hwnd.is_null() {
419                    return None;
420                }
421                let mut win32 =
422                    wgpu::rwh::Win32WindowHandle::new(std::num::NonZeroIsize::new(hwnd as isize)?);
423                // wgpu's Vulkan backend requires `hinstance` to be set
424                // (`vkCreateWin32SurfaceKHR` rejects a null HINSTANCE).
425                // baseview leaves the rwh 0.5 `hinstance` field at null,
426                // so populate it here with the running module's HMODULE.
427                win32.hinstance = current_module_hinstance();
428                let rwh6_window = wgpu::rwh::RawWindowHandle::Win32(win32);
429                let rwh6_display =
430                    wgpu::rwh::RawDisplayHandle::Windows(wgpu::rwh::WindowsDisplayHandle::new());
431                wgpu::SurfaceTargetUnsafe::RawHandle {
432                    raw_display_handle: Some(rwh6_display),
433                    raw_window_handle: rwh6_window,
434                }
435            }
436            #[cfg(target_os = "linux")]
437            RwhRawWindowHandle::Xlib(handle) => {
438                let RwhRawDisplayHandle::Xlib(display_handle) = window.raw_display_handle() else {
439                    return None;
440                };
441                let display_ptr = std::ptr::NonNull::new(display_handle.display);
442                let rwh6_window = wgpu::rwh::RawWindowHandle::Xlib(
443                    wgpu::rwh::XlibWindowHandle::new(handle.window),
444                );
445                let rwh6_display = wgpu::rwh::RawDisplayHandle::Xlib(
446                    wgpu::rwh::XlibDisplayHandle::new(display_ptr, display_handle.screen),
447                );
448                wgpu::SurfaceTargetUnsafe::RawHandle {
449                    raw_display_handle: Some(rwh6_display),
450                    raw_window_handle: rwh6_window,
451                }
452            }
453            _ => return None,
454        };
455
456        instance.create_surface_unsafe(surface_target).ok()
457    }
458}