From 47445083852851a7ac7ab3285906f140ec48351c Mon Sep 17 00:00:00 2001 From: Bogaevsky Date: Fri, 30 Aug 2019 20:26:35 +0200 Subject: [PATCH] macOS: Dpi overhaul (#997) * WIP - Make EL2 DPI changes and implement on Windows (#895) * Modify DPI API publicly and on Windows * Add generic Position and make dpi creation functions const * Make examples work * Fix fullscreen windows not appearing * Replace Logical coordinates in window events with Physical coordinates * Update HiDpiFactorChanged * Document to_static * fix app_state errors * fixes hidpi related errors in window_delegate * fix bad merge * dpi_factor edits in window_delegate * fixes type and lifetime errors in window and window_delegate * applies fmt * complies with @aleksijuvani requested changes * modifies Handler lifetimes * fixes lifetime isues, adds propper handling for HiDpiChanged * applies fmt * restore original lifetimes * solution is somewhere out there * applies fmt * pass as references * resolves issue with HANDLER * crate visible type error * fixes visibility issues * applies fmt * deals with warnings * simplifies new_inner_size setting algorthm * moves proxy instead of referencing it and removes double deref from proxy.ns_window * makes @Osspial tests (https://github.com/rust-windowing/winit/pull/997\#discussion_r301852354) pass * complies with @aleksijuvani suggested changes * makes max window size std::f32::MAX --- Cargo.toml | 2 +- src/platform_impl/macos/app_state.rs | 123 ++++++++++++++++++-- src/platform_impl/macos/event.rs | 25 +++- src/platform_impl/macos/event_loop.rs | 12 +- src/platform_impl/macos/util/mod.rs | 1 + src/platform_impl/macos/window.rs | 126 ++++++++++++++------- src/platform_impl/macos/window_delegate.rs | 73 +++++++----- 7 files changed, 273 insertions(+), 89 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 8dc3e2e5dd..89a9f0db3b 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -36,7 +36,7 @@ cocoa = "0.18.4" core-foundation = "0.6" core-graphics = "0.17.3" dispatch = "0.1.4" -objc = "0.2.3" +objc = "0.2.6" [target.'cfg(target_os = "macos")'.dependencies.core-video-sys] version = "0.1.3" diff --git a/src/platform_impl/macos/app_state.rs b/src/platform_impl/macos/app_state.rs index d063fe8484..8034e26af9 100644 --- a/src/platform_impl/macos/app_state.rs +++ b/src/platform_impl/macos/app_state.rs @@ -11,12 +11,22 @@ use std::{ time::Instant, }; -use cocoa::{appkit::NSApp, base::nil}; +use cocoa::{ + appkit::{NSApp, NSWindow}, + base::nil, + foundation::NSSize, +}; use crate::{ + dpi::LogicalSize, event::{Event, StartCause, WindowEvent}, event_loop::{ControlFlow, EventLoopWindowTarget as RootWindowTarget}, - platform_impl::platform::{observer::EventLoopWaker, util::Never}, + platform_impl::platform::{ + event::{EventProxy, EventWrapper}, + observer::EventLoopWaker, + util::{IdRef, Never}, + window::get_window_id, + }, window::WindowId, }; @@ -24,8 +34,8 @@ lazy_static! { static ref HANDLER: Handler = Default::default(); } -impl Event { - fn userify(self) -> Event { +impl<'a, Never> Event<'a, Never> { + fn userify(self) -> Event<'a, T> { self.map_nonuser_event() // `Never` can't be constructed, so the `UserEvent` variant can't // be present here. @@ -34,7 +44,8 @@ impl Event { } pub trait EventHandler: Debug { - fn handle_nonuser_event(&mut self, event: Event, control_flow: &mut ControlFlow); + // Not sure probably it should accept Event<'static, Never> + fn handle_nonuser_event(&mut self, event: Event<'_, Never>, control_flow: &mut ControlFlow); fn handle_user_events(&mut self, control_flow: &mut ControlFlow); } @@ -53,8 +64,17 @@ impl Debug for EventLoopHandler { } } +<<<<<<< HEAD impl EventHandler for EventLoopHandler { fn handle_nonuser_event(&mut self, event: Event, control_flow: &mut ControlFlow) { +======= +impl EventHandler for EventLoopHandler +where + F: 'static + FnMut(Event<'_, T>, &RootWindowTarget, &mut ControlFlow), + T: 'static, +{ + fn handle_nonuser_event(&mut self, event: Event<'_, Never>, control_flow: &mut ControlFlow) { +>>>>>>> macOS: Dpi overhaul (#997) (self.callback)(event.userify(), &self.window_target, control_flow); self.will_exit |= *control_flow == ControlFlow::Exit; if self.will_exit { @@ -83,7 +103,12 @@ struct Handler { control_flow_prev: Mutex, start_time: Mutex>, callback: Mutex>>, +<<<<<<< HEAD pending_events: Mutex>>, +======= + pending_events: Mutex>>, + deferred_events: Mutex>, +>>>>>>> macOS: Dpi overhaul (#997) pending_redraw: Mutex>, waker: Mutex, } @@ -92,15 +117,23 @@ unsafe impl Send for Handler {} unsafe impl Sync for Handler {} impl Handler { - fn events<'a>(&'a self) -> MutexGuard<'a, VecDeque>> { + fn events(&self) -> MutexGuard<'_, VecDeque>> { self.pending_events.lock().unwrap() } +<<<<<<< HEAD fn redraw<'a>(&'a self) -> MutexGuard<'a, Vec> { +======= + fn deferred(&self) -> MutexGuard<'_, VecDeque> { + self.deferred_events.lock().unwrap() + } + + fn redraw(&self) -> MutexGuard<'_, Vec> { +>>>>>>> macOS: Dpi overhaul (#997) self.pending_redraw.lock().unwrap() } - fn waker<'a>(&'a self) -> MutexGuard<'a, EventLoopWaker> { + fn waker(&self) -> MutexGuard<'_, EventLoopWaker> { self.waker.lock().unwrap() } @@ -136,10 +169,17 @@ impl Handler { *self.start_time.lock().unwrap() = Some(Instant::now()); } - fn take_events(&self) -> VecDeque> { + fn take_events(&self) -> VecDeque> { mem::replace(&mut *self.events(), Default::default()) } +<<<<<<< HEAD +======= + fn take_deferred(&self) -> VecDeque { + mem::replace(&mut *self.deferred(), Default::default()) + } + +>>>>>>> macOS: Dpi overhaul (#997) fn should_redraw(&self) -> Vec { mem::replace(&mut *self.redraw(), Default::default()) } @@ -152,7 +192,7 @@ impl Handler { self.in_callback.store(in_callback, Ordering::Release); } - fn handle_nonuser_event(&self, event: Event) { + fn handle_nonuser_event(&self, event: Event<'_, Never>) { if let Some(ref mut callback) = *self.callback.lock().unwrap() { callback.handle_nonuser_event(event, &mut *self.control_flow.lock().unwrap()); } @@ -163,6 +203,42 @@ impl Handler { callback.handle_user_events(&mut *self.control_flow.lock().unwrap()); } } + + fn handle_hidpi_factor_changed_event( + &self, + ns_window: IdRef, + suggested_size: LogicalSize, + hidpi_factor: f64, + ) { + let size = suggested_size.to_physical(hidpi_factor); + let new_inner_size = &mut Some(size); + let event = Event::WindowEvent { + window_id: WindowId(get_window_id(*ns_window)), + event: WindowEvent::HiDpiFactorChanged { + hidpi_factor, + new_inner_size, + }, + }; + + self.handle_nonuser_event(event); + + // let origin = unsafe { NSWindow::frame(*ns_window).origin }; + let physical_size = new_inner_size.unwrap_or(size); + let logical_size = physical_size.to_logical(hidpi_factor); + let size = NSSize::new(logical_size.width, logical_size.height); + // let rect = NSRect::new(origin, size); + unsafe { NSWindow::setContentSize_(*ns_window, size) }; + } + + fn handle_event(&self, proxy: EventProxy) { + match proxy { + EventProxy::HiDpiFactorChangedProxy { + ns_window, + suggested_size, + hidpi_factor, + } => self.handle_hidpi_factor_changed_event(ns_window, suggested_size, hidpi_factor), + } + } } pub enum AppState {} @@ -171,7 +247,12 @@ impl AppState { // This function extends lifetime of `callback` to 'static as its side effect pub unsafe fn set_callback(callback: F, window_target: Rc>) where +<<<<<<< HEAD F: FnMut(Event, &RootWindowTarget, &mut ControlFlow), +======= + F: 'static + FnMut(Event<'_, T>, &RootWindowTarget, &mut ControlFlow), + T: 'static, +>>>>>>> macOS: Dpi overhaul (#997) { *HANDLER.callback.lock().unwrap() = Some(Box::new(EventLoopHandler { // This transmute is always safe, in case it was reached through `run`, since our @@ -241,20 +322,40 @@ impl AppState { } } - pub fn queue_event(event: Event) { + pub fn queue_event(event: Event<'static, Never>) { if !unsafe { msg_send![class!(NSThread), isMainThread] } { panic!("Event queued from different thread: {:#?}", event); } HANDLER.events().push_back(event); } - pub fn queue_events(mut events: VecDeque>) { + pub fn queue_events(mut events: VecDeque>) { if !unsafe { msg_send![class!(NSThread), isMainThread] } { panic!("Events queued from different thread: {:#?}", events); } HANDLER.events().append(&mut events); } +<<<<<<< HEAD +======= + pub fn send_event_immediately(wrapper: EventWrapper) { + if !unsafe { msg_send![class!(NSThread), isMainThread] } { + panic!("Event sent from different thread: {:#?}", wrapper); + } + HANDLER.deferred().push_back(wrapper); + if !HANDLER.get_in_callback() { + HANDLER.set_in_callback(true); + for wrapper in HANDLER.take_deferred() { + match wrapper { + EventWrapper::StaticEvent(event) => HANDLER.handle_nonuser_event(event), + EventWrapper::EventProxy(proxy) => HANDLER.handle_event(proxy), + }; + } + HANDLER.set_in_callback(false); + } + } + +>>>>>>> macOS: Dpi overhaul (#997) pub fn cleared() { if !HANDLER.is_ready() { return; diff --git a/src/platform_impl/macos/event.rs b/src/platform_impl/macos/event.rs index 02d6b9c266..29b3df7c10 100644 --- a/src/platform_impl/macos/event.rs +++ b/src/platform_impl/macos/event.rs @@ -6,10 +6,29 @@ use cocoa::{ }; use crate::{ - event::{ElementState, KeyboardInput, ModifiersState, VirtualKeyCode, WindowEvent}, - platform_impl::platform::DEVICE_ID, + dpi::LogicalSize, + event::{ElementState, Event, KeyboardInput, ModifiersState, VirtualKeyCode, WindowEvent}, + platform_impl::platform::{ + util::{IdRef, Never}, + DEVICE_ID, + }, }; +#[derive(Debug)] +pub enum EventWrapper { + StaticEvent(Event<'static, Never>), + EventProxy(EventProxy), +} + +#[derive(Debug, PartialEq)] +pub enum EventProxy { + HiDpiFactorChangedProxy { + ns_window: IdRef, + suggested_size: LogicalSize, + hidpi_factor: f64, + }, +} + pub fn char_to_keycode(c: char) -> Option { // We only translate keys that are affected by keyboard layout. // @@ -244,7 +263,7 @@ pub unsafe fn modifier_event( ns_event: id, keymask: NSEventModifierFlags, was_key_pressed: bool, -) -> Option { +) -> Option> { if !was_key_pressed && NSEvent::modifierFlags(ns_event).contains(keymask) || was_key_pressed && !NSEvent::modifierFlags(ns_event).contains(keymask) { diff --git a/src/platform_impl/macos/event_loop.rs b/src/platform_impl/macos/event_loop.rs index f3df1b5248..305edc7e36 100644 --- a/src/platform_impl/macos/event_loop.rs +++ b/src/platform_impl/macos/event_loop.rs @@ -84,7 +84,7 @@ impl EventLoop { pub fn run(mut self, callback: F) -> ! where - F: 'static + FnMut(Event, &RootWindowTarget, &mut ControlFlow), + F: 'static + FnMut(Event<'_, T>, &RootWindowTarget, &mut ControlFlow), { self.run_return(callback); process::exit(0); @@ -104,6 +104,16 @@ impl EventLoop { } } +<<<<<<< HEAD +======= + pub fn run_return(&mut self, _callback: F) + where + F: FnMut(Event<'_, T>, &RootWindowTarget, &mut ControlFlow), + { + unimplemented!(); + } + +>>>>>>> macOS: Dpi overhaul (#997) pub fn create_proxy(&self) -> Proxy { Proxy::new(self.window_target.p.sender.clone()) } diff --git a/src/platform_impl/macos/util/mod.rs b/src/platform_impl/macos/util/mod.rs index 238c9e5ddc..3a454cbba4 100644 --- a/src/platform_impl/macos/util/mod.rs +++ b/src/platform_impl/macos/util/mod.rs @@ -31,6 +31,7 @@ pub const EMPTY_RANGE: ffi::NSRange = ffi::NSRange { length: 0, }; +#[derive(Debug, PartialEq)] pub struct IdRef(id); impl IdRef { diff --git a/src/platform_impl/macos/window.rs b/src/platform_impl/macos/window.rs index 4c9e3f9b32..3934c820c1 100644 --- a/src/platform_impl/macos/window.rs +++ b/src/platform_impl/macos/window.rs @@ -10,7 +10,9 @@ use std::{ }; use crate::{ - dpi::{LogicalPosition, LogicalSize}, + dpi::{ + LogicalPosition, LogicalSize, PhysicalPosition, PhysicalSize, Position, Size, Size::Logical, + }, error::{ExternalError, NotSupportedError, OsError as RootOsError}, icon::Icon, monitor::{MonitorHandle as RootMonitorHandle, VideoMode as RootVideoMode}, @@ -119,22 +121,33 @@ fn create_window( unsafe { let pool = NSAutoreleasePool::new(nil); let screen = match attrs.fullscreen { +<<<<<<< HEAD Some(Fullscreen::Borderless(RootMonitorHandle { inner: ref monitor })) | Some(Fullscreen::Exclusive(RootVideoMode { video_mode: VideoMode { ref monitor, .. }, })) => { let monitor_screen = monitor.ns_screen(); Some(monitor_screen.unwrap_or(appkit::NSScreen::mainScreen(nil))) +======= + Some(ref monitor_id) => { + let monitor_screen = monitor_id.inner.ns_screen(); + Some(monitor_screen.unwrap_or(NSScreen::mainScreen(nil))) +>>>>>>> macOS: Dpi overhaul (#997) } None => None, }; let frame = match screen { - Some(screen) => appkit::NSScreen::frame(screen), + Some(screen) => NSScreen::frame(screen), None => { - let (width, height) = attrs - .inner_size - .map(|logical| (logical.width, logical.height)) - .unwrap_or_else(|| (800.0, 600.0)); + let screen = NSScreen::mainScreen(nil); + let hidpi_factor = NSScreen::backingScaleFactor(screen) as f64; + let (width, height) = match attrs.inner_size { + Some(size) => { + let logical = size.to_logical(hidpi_factor); + (logical.width, logical.height) + } + None => (800.0, 600.0), + }; NSRect::new(NSPoint::new(0.0, 0.0), NSSize::new(width, height)) } }; @@ -285,6 +298,7 @@ pub struct UnownedWindow { decorations: AtomicBool, cursor: Weak>, cursor_visible: AtomicBool, + pub inner_rect: Option, } unsafe impl Send for UnownedWindow {} @@ -321,6 +335,8 @@ impl UnownedWindow { let input_context = unsafe { util::create_input_context(*ns_view) }; + let dpi_factor = unsafe { NSWindow::backingScaleFactor(*ns_window) as f64 }; + unsafe { if win_attribs.transparent { ns_window.setOpaque_(NO); @@ -328,13 +344,14 @@ impl UnownedWindow { } ns_app.activateIgnoringOtherApps_(YES); - - win_attribs - .min_inner_size - .map(|dim| set_min_inner_size(*ns_window, dim)); - win_attribs - .max_inner_size - .map(|dim| set_max_inner_size(*ns_window, dim)); + win_attribs.min_inner_size.map(|dim| { + let logical_dim = dim.to_logical(dpi_factor); + set_min_inner_size(*ns_window, logical_dim) + }); + win_attribs.max_inner_size.map(|dim| { + let logical_dim = dim.to_logical(dpi_factor); + set_max_inner_size(*ns_window, logical_dim) + }); use cocoa::foundation::NSArray; // register for drag and drop operations. @@ -354,6 +371,9 @@ impl UnownedWindow { let maximized = win_attribs.maximized; let visible = win_attribs.visible; let decorations = win_attribs.decorations; + let inner_rect = win_attribs + .inner_size + .map(|size| size.to_physical(dpi_factor)); let window = Arc::new(UnownedWindow { ns_view, @@ -363,6 +383,7 @@ impl UnownedWindow { decorations: AtomicBool::new(decorations), cursor, cursor_visible: AtomicBool::new(true), + inner_rect, }); let delegate = new_delegate(&window, fullscreen.is_some()); @@ -419,27 +440,31 @@ impl UnownedWindow { AppState::queue_redraw(RootWindowId(self.id())); } - pub fn outer_position(&self) -> Result { + pub fn outer_position(&self) -> Result { let frame_rect = unsafe { NSWindow::frame(*self.ns_window) }; - Ok(( + let position = LogicalPosition::new( frame_rect.origin.x as f64, util::bottom_left_to_top_left(frame_rect), - ) - .into()) + ); + let dpi_factor = self.hidpi_factor(); + Ok(position.to_physical(dpi_factor)) } - pub fn inner_position(&self) -> Result { + pub fn inner_position(&self) -> Result { let content_rect = unsafe { NSWindow::contentRectForFrameRect_(*self.ns_window, NSWindow::frame(*self.ns_window)) }; - Ok(( + let position = LogicalPosition::new( content_rect.origin.x as f64, util::bottom_left_to_top_left(content_rect), - ) - .into()) + ); + let dpi_factor = self.hidpi_factor(); + Ok(position.to_physical(dpi_factor)) } - pub fn set_outer_position(&self, position: LogicalPosition) { + pub fn set_outer_position(&self, position: Position) { + let dpi_factor = self.hidpi_factor(); + let position = position.to_logical(dpi_factor); let dummy = NSRect::new( NSPoint::new( position.x, @@ -455,35 +480,50 @@ impl UnownedWindow { } #[inline] - pub fn inner_size(&self) -> LogicalSize { + pub fn inner_size(&self) -> PhysicalSize { let view_frame = unsafe { NSView::frame(*self.ns_view) }; - (view_frame.size.width as f64, view_frame.size.height as f64).into() + let logical: LogicalSize = + (view_frame.size.width as f64, view_frame.size.height as f64).into(); + let dpi_factor = self.hidpi_factor(); + logical.to_physical(dpi_factor) } #[inline] - pub fn outer_size(&self) -> LogicalSize { + pub fn outer_size(&self) -> PhysicalSize { let view_frame = unsafe { NSWindow::frame(*self.ns_window) }; - (view_frame.size.width as f64, view_frame.size.height as f64).into() + let logical: LogicalSize = + (view_frame.size.width as f64, view_frame.size.height as f64).into(); + let dpi_factor = self.hidpi_factor(); + logical.to_physical(dpi_factor) } #[inline] - pub fn set_inner_size(&self, size: LogicalSize) { + pub fn set_inner_size(&self, size: Size) { unsafe { - util::set_content_size_async(*self.ns_window, size); + let dpi_factor = self.hidpi_factor(); + util::set_content_size_async(*self.ns_window, size.to_logical(dpi_factor)); } } - pub fn set_min_inner_size(&self, dimensions: Option) { + pub fn set_min_inner_size(&self, dimensions: Option) { unsafe { - let dimensions = dimensions.unwrap_or_else(|| (0, 0).into()); - set_min_inner_size(*self.ns_window, dimensions); + let dimensions = dimensions.unwrap_or(Logical(LogicalSize { + width: 0.0, + height: 0.0, + })); + let dpi_factor = self.hidpi_factor(); + set_min_inner_size(*self.ns_window, dimensions.to_logical(dpi_factor)); } } - pub fn set_max_inner_size(&self, dimensions: Option) { + pub fn set_max_inner_size(&self, dimensions: Option) { unsafe { - let dimensions = dimensions.unwrap_or_else(|| (!0, !0).into()); - set_max_inner_size(*self.ns_window, dimensions); + let dimensions = dimensions.unwrap_or(Logical(LogicalSize { + width: std::f32::MAX as f64, + height: std::f32::MAX as f64, + })); + let dpi_factor = self.hidpi_factor(); + set_max_inner_size(*self.ns_window, dimensions.to_logical(dpi_factor)); } } @@ -547,14 +587,14 @@ impl UnownedWindow { } #[inline] - pub fn set_cursor_position( - &self, - cursor_position: LogicalPosition, - ) -> Result<(), ExternalError> { - let window_position = self.inner_position().unwrap(); + pub fn set_cursor_position(&self, cursor_position: Position) -> Result<(), ExternalError> { + let physical_window_position = self.inner_position().unwrap(); + let dpi_factor = self.hidpi_factor(); + let window_position = physical_window_position.to_logical(dpi_factor); + let logical_cursor_position = cursor_position.to_logical(dpi_factor); let point = appkit::CGPoint { - x: (cursor_position.x + window_position.x) as CGFloat, - y: (cursor_position.y + window_position.y) as CGFloat, + x: (logical_cursor_position.x + window_position.x) as CGFloat, + y: (logical_cursor_position.y + window_position.y) as CGFloat, }; CGDisplay::warp_mouse_cursor_position(point) .map_err(|e| ExternalError::Os(os_error!(OsError::CGError(e))))?; @@ -861,7 +901,9 @@ impl UnownedWindow { } #[inline] - pub fn set_ime_position(&self, logical_spot: LogicalPosition) { + pub fn set_ime_position(&self, spot: Position) { + let dpi_factor = self.hidpi_factor(); + let logical_spot = spot.to_logical(dpi_factor); unsafe { view::set_ime_position( *self.ns_view, diff --git a/src/platform_impl/macos/window_delegate.rs b/src/platform_impl/macos/window_delegate.rs index cb1bf3c26e..f4591d17e9 100644 --- a/src/platform_impl/macos/window_delegate.rs +++ b/src/platform_impl/macos/window_delegate.rs @@ -19,6 +19,7 @@ use crate::{ event::{Event, WindowEvent}, platform_impl::platform::{ app_state::AppState, + event::{EventProxy, EventWrapper}, util::{self, IdRef}, window::{get_window_id, UnownedWindow}, }, @@ -47,20 +48,18 @@ pub struct WindowDelegateState { impl WindowDelegateState { pub fn new(window: &Arc, initial_fullscreen: bool) -> Self { - let dpi_factor = window.hidpi_factor(); - + let hidpi_factor = window.hidpi_factor(); let mut delegate_state = WindowDelegateState { ns_window: window.ns_window.clone(), ns_view: window.ns_view.clone(), window: Arc::downgrade(&window), initial_fullscreen, previous_position: None, - previous_dpi_factor: dpi_factor, + previous_dpi_factor: hidpi_factor, }; - if dpi_factor != 1.0 { - delegate_state.emit_event(WindowEvent::HiDpiFactorChanged(dpi_factor)); - delegate_state.emit_resize_event(); + if hidpi_factor != 1.0 { + delegate_state.emit_static_hidpi_factor_changed_event(); } delegate_state @@ -73,7 +72,7 @@ impl WindowDelegateState { self.window.upgrade().map(|ref window| callback(window)) } - pub fn emit_event(&mut self, event: WindowEvent) { + pub fn emit_event(&mut self, event: WindowEvent<'static>) { let event = Event::WindowEvent { window_id: WindowId(get_window_id(*self.ns_window)), event, @@ -81,10 +80,36 @@ impl WindowDelegateState { AppState::queue_event(event); } + pub fn emit_static_hidpi_factor_changed_event(&mut self) { + let hidpi_factor = self.get_hidpi_factor(); + if hidpi_factor == self.previous_dpi_factor { + return (); + }; + + self.previous_dpi_factor = hidpi_factor; + let wrapper = EventWrapper::EventProxy(EventProxy::HiDpiFactorChangedProxy { + ns_window: IdRef::retain(*self.ns_window), + suggested_size: self.view_size(), + hidpi_factor, + }); + AppState::send_event_immediately(wrapper); + } + pub fn emit_resize_event(&mut self) { +<<<<<<< HEAD let rect = unsafe { NSView::frame(*self.ns_view) }; let size = LogicalSize::new(rect.size.width as f64, rect.size.height as f64); self.emit_event(WindowEvent::Resized(size)); +======= + let hidpi_factor = self.get_hidpi_factor(); + let physical_size = self.view_size().to_physical(hidpi_factor); + let event = Event::WindowEvent { + window_id: WindowId(get_window_id(*self.ns_window)), + event: WindowEvent::Resized(physical_size), + }; + let wrapper = EventWrapper::StaticEvent(event); + AppState::send_event_immediately(wrapper); +>>>>>>> macOS: Dpi overhaul (#997) } fn emit_move_event(&mut self) { @@ -97,6 +122,15 @@ impl WindowDelegateState { self.emit_event(WindowEvent::Moved((x, y).into())); } } + + fn get_hidpi_factor(&self) -> f64 { + (unsafe { NSWindow::backingScaleFactor(*self.ns_window) }) as f64 + } + + fn view_size(&self) -> LogicalSize { + let ns_size = unsafe { NSView::frame(*self.ns_view).size }; + LogicalSize::new(ns_size.width as f64, ns_size.height as f64) + } } pub fn new_delegate(window: &Arc, initial_fullscreen: bool) -> IdRef { @@ -140,10 +174,6 @@ lazy_static! { sel!(windowDidMove:), window_did_move as extern "C" fn(&Object, Sel, id), ); - decl.add_method( - sel!(windowDidChangeScreen:), - window_did_change_screen as extern "C" fn(&Object, Sel, id), - ); decl.add_method( sel!(windowDidChangeBackingProperties:), window_did_change_backing_properties as extern "C" fn(&Object, Sel, id), @@ -273,29 +303,10 @@ extern "C" fn window_did_move(this: &Object, _: Sel, _: id) { trace!("Completed `windowDidMove:`"); } -extern "C" fn window_did_change_screen(this: &Object, _: Sel, _: id) { - trace!("Triggered `windowDidChangeScreen:`"); - with_state(this, |state| { - let dpi_factor = unsafe { NSWindow::backingScaleFactor(*state.ns_window) } as f64; - if state.previous_dpi_factor != dpi_factor { - state.previous_dpi_factor = dpi_factor; - state.emit_event(WindowEvent::HiDpiFactorChanged(dpi_factor)); - state.emit_resize_event(); - } - }); - trace!("Completed `windowDidChangeScreen:`"); -} - -// This will always be called before `window_did_change_screen`. extern "C" fn window_did_change_backing_properties(this: &Object, _: Sel, _: id) { trace!("Triggered `windowDidChangeBackingProperties:`"); with_state(this, |state| { - let dpi_factor = unsafe { NSWindow::backingScaleFactor(*state.ns_window) } as f64; - if state.previous_dpi_factor != dpi_factor { - state.previous_dpi_factor = dpi_factor; - state.emit_event(WindowEvent::HiDpiFactorChanged(dpi_factor)); - state.emit_resize_event(); - } + state.emit_static_hidpi_factor_changed_event(); }); trace!("Completed `windowDidChangeBackingProperties:`"); }