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}