Clean up macOS class declaration (#2458)

* Begin abstraction over AppKit

* Clean up NSApplication delegate declaration

* Clean up NSApplication override declaration

* Clean up NSWindow delegate declaration

* Clean up NSWindow override declaration

* Clean up NSView delegate declaration
This commit is contained in:
Mads Marquart 2022-09-02 18:46:18 +02:00 committed by GitHub
parent 112965b4ff
commit d67c928120
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
15 changed files with 873 additions and 1013 deletions

View file

@ -148,6 +148,8 @@ extern crate bitflags;
#[cfg(any(target_os = "macos", target_os = "ios"))] #[cfg(any(target_os = "macos", target_os = "ios"))]
#[macro_use] #[macro_use]
extern crate objc; extern crate objc;
#[cfg(target_os = "macos")]
extern crate objc as objc2;
pub mod dpi; pub mod dpi;
#[macro_use] #[macro_use]

View file

@ -4,32 +4,28 @@ use cocoa::{
appkit::{self, NSEvent}, appkit::{self, NSEvent},
base::id, base::id,
}; };
use objc::{ use objc2::foundation::NSObject;
declare::ClassBuilder, use objc2::{declare_class, ClassType};
runtime::{Class, Object, Sel},
};
use once_cell::sync::Lazy;
use super::appkit::{NSApplication, NSResponder};
use super::{app_state::AppState, event::EventWrapper, util, DEVICE_ID}; use super::{app_state::AppState, event::EventWrapper, util, DEVICE_ID};
use crate::event::{DeviceEvent, ElementState, Event}; use crate::event::{DeviceEvent, ElementState, Event};
pub struct AppClass(pub *const Class); declare_class!(
unsafe impl Send for AppClass {} #[derive(Debug, PartialEq, Eq, Hash)]
unsafe impl Sync for AppClass {} pub(super) struct WinitApplication {}
pub static APP_CLASS: Lazy<AppClass> = Lazy::new(|| unsafe { unsafe impl ClassType for WinitApplication {
let superclass = class!(NSApplication); #[inherits(NSResponder, NSObject)]
let mut decl = ClassBuilder::new("WinitApp", superclass).unwrap(); type Super = NSApplication;
}
decl.add_method(sel!(sendEvent:), send_event as extern "C" fn(_, _, _));
AppClass(decl.register())
});
unsafe impl WinitApplication {
// Normally, holding Cmd + any key never sends us a `keyUp` event for that key. // Normally, holding Cmd + any key never sends us a `keyUp` event for that key.
// Overriding `sendEvent:` like this fixes that. (https://stackoverflow.com/a/15294196) // Overriding `sendEvent:` like this fixes that. (https://stackoverflow.com/a/15294196)
// Fun fact: Firefox still has this bug! (https://bugzilla.mozilla.org/show_bug.cgi?id=1299553) // Fun fact: Firefox still has this bug! (https://bugzilla.mozilla.org/show_bug.cgi?id=1299553)
extern "C" fn send_event(this: &Object, _sel: Sel, event: id) { #[sel(sendEvent:)]
fn send_event(&self, event: id) {
unsafe { unsafe {
// For posterity, there are some undocumented event types // For posterity, there are some undocumented event types
// (https://github.com/servo/cocoa-rs/issues/155) // (https://github.com/servo/cocoa-rs/issues/155)
@ -42,15 +38,16 @@ extern "C" fn send_event(this: &Object, _sel: Sel, event: id) {
appkit::NSEventModifierFlags::NSCommandKeyMask, appkit::NSEventModifierFlags::NSCommandKeyMask,
) )
{ {
let key_window: id = msg_send![this, keyWindow]; let key_window: id = msg_send![self, keyWindow];
let _: () = msg_send![key_window, sendEvent: event]; let _: () = msg_send![key_window, sendEvent: event];
} else { } else {
maybe_dispatch_device_event(event); maybe_dispatch_device_event(event);
let superclass = util::superclass(this); let _: () = msg_send![super(self), sendEvent: event];
let _: () = msg_send![super(this, superclass), sendEvent: event];
} }
} }
} }
}
);
unsafe fn maybe_dispatch_device_event(event: id) { unsafe fn maybe_dispatch_device_event(event: id) {
let event_type = event.eventType(); let event_type = event.eventType();

View file

@ -1,89 +1,64 @@
use std::{ use cocoa::appkit::NSApplicationActivationPolicy;
cell::{RefCell, RefMut}, use objc2::foundation::NSObject;
os::raw::c_void, use objc2::rc::{Id, Shared};
}; use objc2::runtime::Object;
use objc2::{declare_class, ClassType};
use cocoa::base::id; use super::app_state::AppState;
use objc::{
declare::ClassBuilder,
runtime::{Class, Object, Sel},
};
use once_cell::sync::Lazy;
use crate::{platform::macos::ActivationPolicy, platform_impl::platform::app_state::AppState}; declare_class!(
#[derive(Debug)]
static AUX_DELEGATE_STATE_NAME: &str = "auxState"; pub(super) struct ApplicationDelegate {
activation_policy: NSApplicationActivationPolicy,
pub struct AuxDelegateState { default_menu: bool,
pub activation_policy: ActivationPolicy,
pub default_menu: bool,
} }
pub struct AppDelegateClass(pub *const Class); unsafe impl ClassType for ApplicationDelegate {
unsafe impl Send for AppDelegateClass {} type Super = NSObject;
unsafe impl Sync for AppDelegateClass {} const NAME: &'static str = "WinitApplicationDelegate";
pub static APP_DELEGATE_CLASS: Lazy<AppDelegateClass> = Lazy::new(|| unsafe {
let superclass = class!(NSResponder);
let mut decl = ClassBuilder::new("WinitAppDelegate", superclass).unwrap();
decl.add_class_method(sel!(new), new as extern "C" fn(_, _) -> _);
decl.add_method(sel!(dealloc), dealloc as extern "C" fn(_, _));
decl.add_method(
sel!(applicationDidFinishLaunching:),
did_finish_launching as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(applicationWillTerminate:),
will_terminate as extern "C" fn(_, _, _),
);
decl.add_ivar::<*mut c_void>(AUX_DELEGATE_STATE_NAME);
AppDelegateClass(decl.register())
});
/// Safety: Assumes that Object is an instance of APP_DELEGATE_CLASS
pub unsafe fn get_aux_state_mut(this: &Object) -> RefMut<'_, AuxDelegateState> {
let ptr: *mut c_void = *this.ivar(AUX_DELEGATE_STATE_NAME);
// Watch out that this needs to be the correct type
(*(ptr as *mut RefCell<AuxDelegateState>)).borrow_mut()
} }
extern "C" fn new(class: &Class, _: Sel) -> id { unsafe impl ApplicationDelegate {
unsafe { #[sel(initWithActivationPolicy:defaultMenu:)]
let this: id = msg_send![class, alloc]; fn init(
let this: id = msg_send![this, init]; &mut self,
// TODO: Remove the need for this initialization here activation_policy: NSApplicationActivationPolicy,
(*this).set_ivar( default_menu: bool,
AUX_DELEGATE_STATE_NAME, ) -> Option<&mut Self> {
Box::into_raw(Box::new(RefCell::new(AuxDelegateState { let this: Option<&mut Self> = unsafe { msg_send![super(self), init] };
activation_policy: ActivationPolicy::Regular, this.map(|this| {
default_menu: true, *this.activation_policy = activation_policy;
}))) as *mut c_void, *this.default_menu = default_menu;
);
this this
} })
} }
extern "C" fn dealloc(this: &Object, _: Sel) { #[sel(applicationDidFinishLaunching:)]
unsafe { fn did_finish_launching(&self, _sender: *const Object) {
let state_ptr: *mut c_void = *(this.ivar(AUX_DELEGATE_STATE_NAME));
// As soon as the box is constructed it is immediately dropped, releasing the underlying
// memory
drop(Box::from_raw(state_ptr as *mut RefCell<AuxDelegateState>));
}
}
extern "C" fn did_finish_launching(this: &Object, _: Sel, _: id) {
trace_scope!("applicationDidFinishLaunching:"); trace_scope!("applicationDidFinishLaunching:");
AppState::launched(this); AppState::launched(*self.activation_policy, *self.default_menu);
} }
extern "C" fn will_terminate(_this: &Object, _: Sel, _: id) { #[sel(applicationWillTerminate:)]
trace!("Triggered `applicationWillTerminate`"); fn will_terminate(&self, _sender: *const Object) {
trace_scope!("applicationWillTerminate:");
// TODO: Notify every window that it will be destroyed, like done in iOS? // TODO: Notify every window that it will be destroyed, like done in iOS?
AppState::exit(); AppState::exit();
trace!("Completed `applicationWillTerminate`"); }
}
);
impl ApplicationDelegate {
pub(super) fn new(
activation_policy: NSApplicationActivationPolicy,
default_menu: bool,
) -> Id<Self, Shared> {
unsafe {
msg_send_id![
msg_send_id![Self::class(), alloc],
initWithActivationPolicy: activation_policy,
defaultMenu: default_menu,
]
}
}
} }

View file

@ -13,25 +13,20 @@ use std::{
}; };
use cocoa::{ use cocoa::{
appkit::{NSApp, NSApplication, NSWindow}, appkit::{NSApp, NSApplication, NSApplicationActivationPolicy, NSWindow},
base::{id, nil}, base::{id, nil},
foundation::NSSize, foundation::NSSize,
}; };
use objc::{ use objc::foundation::is_main_thread;
foundation::is_main_thread, use objc::rc::autoreleasepool;
rc::autoreleasepool, use objc::runtime::Bool;
runtime::{Bool, Object},
};
use once_cell::sync::Lazy; use once_cell::sync::Lazy;
use crate::{ use crate::{
dpi::LogicalSize, dpi::LogicalSize,
event::{Event, StartCause, WindowEvent}, event::{Event, StartCause, WindowEvent},
event_loop::{ControlFlow, EventLoopWindowTarget as RootWindowTarget}, event_loop::{ControlFlow, EventLoopWindowTarget as RootWindowTarget},
platform::macos::ActivationPolicy, platform_impl::platform::{
platform_impl::{
get_aux_state_mut,
platform::{
event::{EventProxy, EventWrapper}, event::{EventProxy, EventWrapper},
event_loop::{post_dummy_event, PanicInfo}, event_loop::{post_dummy_event, PanicInfo},
menu, menu,
@ -39,7 +34,6 @@ use crate::{
util::{IdRef, Never}, util::{IdRef, Never},
window::get_window_id, window::get_window_id,
}, },
},
window::WindowId, window::WindowId,
}; };
@ -283,17 +277,21 @@ impl AppState {
} }
} }
pub fn launched(app_delegate: &Object) { pub fn launched(activation_policy: NSApplicationActivationPolicy, create_default_menu: bool) {
apply_activation_policy(app_delegate);
unsafe { unsafe {
let ns_app = NSApp(); let ns_app = NSApp();
// We need to delay setting the activation policy and activating the app
// until `applicationDidFinishLaunching` has been called. Otherwise the
// menu bar is initially unresponsive on macOS 10.15.
ns_app.setActivationPolicy_(activation_policy);
window_activation_hack(ns_app); window_activation_hack(ns_app);
// TODO: Consider allowing the user to specify they don't want their application activated // TODO: Consider allowing the user to specify they don't want their application activated
ns_app.activateIgnoringOtherApps_(Bool::YES.as_raw()); ns_app.activateIgnoringOtherApps_(Bool::YES.as_raw());
}; };
HANDLER.set_ready(); HANDLER.set_ready();
HANDLER.waker().start(); HANDLER.waker().start();
let create_default_menu = unsafe { get_aux_state_mut(app_delegate).default_menu };
if create_default_menu { if create_default_menu {
// The menubar initialization should be before the `NewEvents` event, to allow // The menubar initialization should be before the `NewEvents` event, to allow
// overriding of the default menu even if it's created // overriding of the default menu even if it's created
@ -450,18 +448,3 @@ unsafe fn window_activation_hack(ns_app: id) {
} }
} }
} }
fn apply_activation_policy(app_delegate: &Object) {
unsafe {
use cocoa::appkit::NSApplicationActivationPolicy::*;
let ns_app = NSApp();
// We need to delay setting the activation policy and activating the app
// until `applicationDidFinishLaunching` has been called. Otherwise the
// menu bar is initially unresponsive on macOS 10.15.
let act_pol = get_aux_state_mut(app_delegate).activation_policy;
ns_app.setActivationPolicy_(match act_pol {
ActivationPolicy::Regular => NSApplicationActivationPolicyRegular,
ActivationPolicy::Accessory => NSApplicationActivationPolicyAccessory,
ActivationPolicy::Prohibited => NSApplicationActivationPolicyProhibited,
});
}
}

View file

@ -0,0 +1,14 @@
use objc2::foundation::NSObject;
use objc2::{extern_class, ClassType};
use super::NSResponder;
extern_class!(
#[derive(Debug, PartialEq, Eq, Hash)]
pub(crate) struct NSApplication;
unsafe impl ClassType for NSApplication {
#[inherits(NSObject)]
type Super = NSResponder;
}
);

View file

@ -0,0 +1,11 @@
#![deny(unsafe_op_in_unsafe_fn)]
mod application;
mod responder;
mod view;
mod window;
pub(crate) use self::application::NSApplication;
pub(crate) use self::responder::NSResponder;
pub(crate) use self::view::NSView;
pub(crate) use self::window::NSWindow;

View file

@ -0,0 +1,11 @@
use objc2::foundation::NSObject;
use objc2::{extern_class, ClassType};
extern_class!(
#[derive(Debug, PartialEq, Eq, Hash)]
pub(crate) struct NSResponder;
unsafe impl ClassType for NSResponder {
type Super = NSObject;
}
);

View file

@ -0,0 +1,14 @@
use objc2::foundation::NSObject;
use objc2::{extern_class, ClassType};
use super::NSResponder;
extern_class!(
#[derive(Debug, PartialEq, Eq, Hash)]
pub(crate) struct NSView;
unsafe impl ClassType for NSView {
#[inherits(NSObject)]
type Super = NSResponder;
}
);

View file

@ -0,0 +1,14 @@
use objc2::foundation::NSObject;
use objc2::{extern_class, ClassType};
use super::NSResponder;
extern_class!(
#[derive(Debug, PartialEq, Eq, Hash)]
pub(crate) struct NSWindow;
unsafe impl ClassType for NSWindow {
#[inherits(NSObject)]
type Super = NSResponder;
}
);

View file

@ -16,8 +16,9 @@ use cocoa::{
base::{id, nil}, base::{id, nil},
foundation::{NSPoint, NSTimeInterval}, foundation::{NSPoint, NSTimeInterval},
}; };
use objc::foundation::is_main_thread; use objc2::foundation::is_main_thread;
use objc::rc::autoreleasepool; use objc2::rc::{autoreleasepool, Id, Shared};
use objc2::ClassType;
use raw_window_handle::{AppKitDisplayHandle, RawDisplayHandle}; use raw_window_handle::{AppKitDisplayHandle, RawDisplayHandle};
use crate::{ use crate::{
@ -25,16 +26,12 @@ use crate::{
event_loop::{ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootWindowTarget}, event_loop::{ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootWindowTarget},
monitor::MonitorHandle as RootMonitorHandle, monitor::MonitorHandle as RootMonitorHandle,
platform::macos::ActivationPolicy, platform::macos::ActivationPolicy,
platform_impl::{ platform_impl::platform::{
get_aux_state_mut, app::WinitApplication,
platform::{ app_delegate::ApplicationDelegate,
app::APP_CLASS,
app_delegate::APP_DELEGATE_CLASS,
app_state::{AppState, Callback}, app_state::{AppState, Callback},
monitor::{self, MonitorHandle}, monitor::{self, MonitorHandle},
observer::*, observer::*,
util::IdRef,
},
}, },
}; };
@ -113,7 +110,7 @@ impl<T> EventLoopWindowTarget<T> {
pub struct EventLoop<T: 'static> { pub struct EventLoop<T: 'static> {
/// The delegate is only weakly referenced by NSApplication, so we keep /// The delegate is only weakly referenced by NSApplication, so we keep
/// it around here as well. /// it around here as well.
_delegate: IdRef, _delegate: Id<ApplicationDelegate, Shared>,
window_target: Rc<RootWindowTarget<T>>, window_target: Rc<RootWindowTarget<T>>,
panic_info: Rc<PanicInfo>, panic_info: Rc<PanicInfo>,
@ -153,16 +150,18 @@ impl<T> EventLoop<T> {
// `sharedApplication`) is called anywhere else, or we'll end up // `sharedApplication`) is called anywhere else, or we'll end up
// with the wrong `NSApplication` class and the wrong thread could // with the wrong `NSApplication` class and the wrong thread could
// be marked as main. // be marked as main.
let app: id = msg_send![APP_CLASS.0, sharedApplication]; let app: id = msg_send![WinitApplication::class(), sharedApplication];
let delegate = IdRef::new(msg_send![APP_DELEGATE_CLASS.0, new]); use cocoa::appkit::NSApplicationActivationPolicy::*;
let activation_policy = match attributes.activation_policy {
let mut aux_state = get_aux_state_mut(&**delegate); ActivationPolicy::Regular => NSApplicationActivationPolicyRegular,
aux_state.activation_policy = attributes.activation_policy; ActivationPolicy::Accessory => NSApplicationActivationPolicyAccessory,
aux_state.default_menu = attributes.default_menu; ActivationPolicy::Prohibited => NSApplicationActivationPolicyProhibited,
};
let delegate = ApplicationDelegate::new(activation_policy, attributes.default_menu);
autoreleasepool(|_| { autoreleasepool(|_| {
let _: () = msg_send![app, setDelegate:*delegate]; let _: () = msg_send![app, setDelegate: &*delegate];
}); });
delegate delegate

View file

@ -7,6 +7,7 @@ mod util;
mod app; mod app;
mod app_delegate; mod app_delegate;
mod app_state; mod app_state;
mod appkit;
mod event; mod event;
mod event_loop; mod event_loop;
mod ffi; mod ffi;
@ -20,7 +21,6 @@ mod window_delegate;
use std::{fmt, ops::Deref, sync::Arc}; use std::{fmt, ops::Deref, sync::Arc};
pub(crate) use self::{ pub(crate) use self::{
app_delegate::get_aux_state_mut,
event_loop::{ event_loop::{
EventLoop, EventLoopProxy, EventLoopWindowTarget, PlatformSpecificEventLoopAttributes, EventLoop, EventLoopProxy, EventLoopWindowTarget, PlatformSpecificEventLoopAttributes,
}, },

View file

@ -12,8 +12,7 @@ use cocoa::{
foundation::{NSPoint, NSRect, NSString}, foundation::{NSPoint, NSRect, NSString},
}; };
use core_graphics::display::CGDisplay; use core_graphics::display::CGDisplay;
use objc::foundation::{NSRange, NSUInteger}; use objc2::foundation::{NSRange, NSUInteger};
use objc::runtime::{Class, Object};
use crate::dpi::LogicalPosition; use crate::dpi::LogicalPosition;
use crate::platform_impl::platform::ffi; use crate::platform_impl::platform::ffi;
@ -144,11 +143,6 @@ pub unsafe fn app_name() -> Option<id> {
} }
} }
pub unsafe fn superclass(this: &Object) -> &Class {
let superclass: *const Class = msg_send![this, superclass];
&*superclass
}
#[allow(dead_code)] #[allow(dead_code)]
pub unsafe fn open_emoji_picker() { pub unsafe fn open_emoji_picker() {
let _: () = msg_send![NSApp(), orderFrontCharacterPalette: nil]; let _: () = msg_send![NSApp(), orderFrontCharacterPalette: nil];

File diff suppressed because it is too large Load diff

View file

@ -45,13 +45,12 @@ use cocoa::{
foundation::{NSDictionary, NSPoint, NSRect, NSSize}, foundation::{NSDictionary, NSPoint, NSRect, NSSize},
}; };
use core_graphics::display::{CGDisplay, CGDisplayMode}; use core_graphics::display::{CGDisplay, CGDisplayMode};
use objc::{ use objc2::foundation::{is_main_thread, NSObject, NSUInteger};
declare::ClassBuilder, use objc2::rc::autoreleasepool;
foundation::{is_main_thread, NSUInteger}, use objc2::runtime::{Bool, Object};
rc::autoreleasepool, use objc2::{declare_class, ClassType};
runtime::{Bool, Class, Object, Sel},
}; use super::appkit::{NSResponder, NSWindow as NSWindowClass};
use once_cell::sync::Lazy;
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] #[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct WindowId(pub usize); pub struct WindowId(pub usize);
@ -204,7 +203,7 @@ fn create_window(
masks |= NSWindowStyleMask::NSFullSizeContentViewWindowMask; masks |= NSWindowStyleMask::NSFullSizeContentViewWindowMask;
} }
let ns_window: id = msg_send![WINDOW_CLASS.0, alloc]; let ns_window: id = msg_send![WinitWindow::class(), alloc];
let ns_window = IdRef::new(ns_window.initWithContentRect_styleMask_backing_defer_( let ns_window = IdRef::new(ns_window.initWithContentRect_styleMask_backing_defer_(
frame, frame,
masks, masks,
@ -262,34 +261,28 @@ fn create_window(
}) })
} }
struct WindowClass(*const Class); declare_class!(
unsafe impl Send for WindowClass {} struct WinitWindow {}
unsafe impl Sync for WindowClass {}
static WINDOW_CLASS: Lazy<WindowClass> = Lazy::new(|| unsafe { unsafe impl ClassType for WinitWindow {
let window_superclass = class!(NSWindow); #[inherits(NSResponder, NSObject)]
let mut decl = ClassBuilder::new("WinitWindow", window_superclass).unwrap(); type Super = NSWindowClass;
}
pub extern "C" fn can_become_main_window(_: &Object, _: Sel) -> Bool { unsafe impl WinitWindow {
#[sel(canBecomeMainWindow)]
fn can_become_main_window(&self) -> bool {
trace_scope!("canBecomeMainWindow"); trace_scope!("canBecomeMainWindow");
Bool::YES true
} }
pub extern "C" fn can_become_key_window(_: &Object, _: Sel) -> Bool { #[sel(canBecomeKeyWindow)]
fn can_become_key_window(&self) -> bool {
trace_scope!("canBecomeKeyWindow"); trace_scope!("canBecomeKeyWindow");
Bool::YES true
}
} }
decl.add_method(
sel!(canBecomeMainWindow),
can_become_main_window as extern "C" fn(_, _) -> _,
); );
decl.add_method(
sel!(canBecomeKeyWindow),
can_become_key_window as extern "C" fn(_, _) -> _,
);
WindowClass(decl.register())
});
#[derive(Default)] #[derive(Default)]
pub struct SharedState { pub struct SharedState {

View file

@ -8,13 +8,10 @@ use cocoa::{
appkit::{self, NSApplicationPresentationOptions, NSView, NSWindow, NSWindowOcclusionState}, appkit::{self, NSApplicationPresentationOptions, NSView, NSWindow, NSWindowOcclusionState},
base::{id, nil}, base::{id, nil},
}; };
use objc::{ use objc2::foundation::{NSObject, NSUInteger};
declare::ClassBuilder, use objc2::rc::autoreleasepool;
foundation::NSUInteger, use objc2::runtime::Object;
rc::autoreleasepool, use objc2::{declare_class, ClassType};
runtime::{Bool, Class, Object, Sel},
};
use once_cell::sync::Lazy;
use crate::{ use crate::{
dpi::{LogicalPosition, LogicalSize}, dpi::{LogicalPosition, LogicalSize},
@ -29,7 +26,7 @@ use crate::{
window::{Fullscreen, WindowId}, window::{Fullscreen, WindowId},
}; };
pub struct WindowDelegateState { struct WindowDelegateState {
ns_window: IdRef, // never changes ns_window: IdRef, // never changes
ns_view: IdRef, // never changes ns_view: IdRef, // never changes
@ -50,7 +47,7 @@ pub struct WindowDelegateState {
} }
impl WindowDelegateState { impl WindowDelegateState {
pub fn new(window: &Arc<UnownedWindow>, initial_fullscreen: bool) -> Self { fn new(window: &Arc<UnownedWindow>, initial_fullscreen: bool) -> Self {
let scale_factor = window.scale_factor(); let scale_factor = window.scale_factor();
let mut delegate_state = WindowDelegateState { let mut delegate_state = WindowDelegateState {
ns_window: window.ns_window.clone(), ns_window: window.ns_window.clone(),
@ -75,7 +72,7 @@ impl WindowDelegateState {
self.window.upgrade().map(|ref window| callback(window)) self.window.upgrade().map(|ref window| callback(window))
} }
pub fn emit_event(&mut self, event: WindowEvent<'static>) { fn emit_event(&mut self, event: WindowEvent<'static>) {
let event = Event::WindowEvent { let event = Event::WindowEvent {
window_id: WindowId(get_window_id(*self.ns_window)), window_id: WindowId(get_window_id(*self.ns_window)),
event, event,
@ -83,7 +80,7 @@ impl WindowDelegateState {
AppState::queue_event(EventWrapper::StaticEvent(event)); AppState::queue_event(EventWrapper::StaticEvent(event));
} }
pub fn emit_static_scale_factor_changed_event(&mut self) { fn emit_static_scale_factor_changed_event(&mut self) {
let scale_factor = self.get_scale_factor(); let scale_factor = self.get_scale_factor();
if scale_factor == self.previous_scale_factor { if scale_factor == self.previous_scale_factor {
return; return;
@ -126,146 +123,55 @@ pub fn new_delegate(window: &Arc<UnownedWindow>, initial_fullscreen: bool) -> Id
unsafe { unsafe {
// This is free'd in `dealloc` // This is free'd in `dealloc`
let state_ptr = Box::into_raw(Box::new(state)) as *mut c_void; let state_ptr = Box::into_raw(Box::new(state)) as *mut c_void;
let delegate: id = msg_send![WINDOW_DELEGATE_CLASS.0, alloc]; let delegate: id = msg_send![WinitWindowDelegate::class(), alloc];
IdRef::new(msg_send![delegate, initWithWinit: state_ptr]) IdRef::new(msg_send![delegate, initWithWinit: state_ptr])
} }
} }
struct WindowDelegateClass(*const Class); declare_class! {
unsafe impl Send for WindowDelegateClass {} #[derive(Debug)]
unsafe impl Sync for WindowDelegateClass {} struct WinitWindowDelegate {
state: *mut c_void,
static WINDOW_DELEGATE_CLASS: Lazy<WindowDelegateClass> = Lazy::new(|| unsafe {
let superclass = class!(NSResponder);
let mut decl = ClassBuilder::new("WinitWindowDelegate", superclass).unwrap();
decl.add_method(sel!(dealloc), dealloc as extern "C" fn(_, _));
decl.add_method(
sel!(initWithWinit:),
init_with_winit as extern "C" fn(_, _, _) -> _,
);
decl.add_method(
sel!(windowShouldClose:),
window_should_close as extern "C" fn(_, _, _) -> _,
);
decl.add_method(
sel!(windowWillClose:),
window_will_close as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(windowDidResize:),
window_did_resize as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(windowDidMove:),
window_did_move as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(windowDidChangeBackingProperties:),
window_did_change_backing_properties as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(windowDidBecomeKey:),
window_did_become_key as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(windowDidResignKey:),
window_did_resign_key as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(draggingEntered:),
dragging_entered as extern "C" fn(_, _, _) -> _,
);
decl.add_method(
sel!(prepareForDragOperation:),
prepare_for_drag_operation as extern "C" fn(_, _, _) -> _,
);
decl.add_method(
sel!(performDragOperation:),
perform_drag_operation as extern "C" fn(_, _, _) -> _,
);
decl.add_method(
sel!(concludeDragOperation:),
conclude_drag_operation as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(draggingExited:),
dragging_exited as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(window:willUseFullScreenPresentationOptions:),
window_will_use_fullscreen_presentation_options as extern "C" fn(_, _, _, _) -> _,
);
decl.add_method(
sel!(windowDidEnterFullScreen:),
window_did_enter_fullscreen as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(windowWillEnterFullScreen:),
window_will_enter_fullscreen as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(windowDidExitFullScreen:),
window_did_exit_fullscreen as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(windowWillExitFullScreen:),
window_will_exit_fullscreen as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(windowDidFailToEnterFullScreen:),
window_did_fail_to_enter_fullscreen as extern "C" fn(_, _, _),
);
decl.add_method(
sel!(windowDidChangeOcclusionState:),
window_did_change_occlusion_state as extern "C" fn(_, _, _),
);
decl.add_ivar::<*mut c_void>("winitState");
WindowDelegateClass(decl.register())
});
// This function is definitely unsafe, but labeling that would increase
// boilerplate and wouldn't really clarify anything...
fn with_state<F: FnOnce(&mut WindowDelegateState) -> T, T>(this: &Object, callback: F) {
let state_ptr = unsafe {
let state_ptr: *mut c_void = *this.ivar("winitState");
&mut *(state_ptr as *mut WindowDelegateState)
};
callback(state_ptr);
} }
extern "C" fn dealloc(this: &Object, _sel: Sel) { unsafe impl ClassType for WinitWindowDelegate {
with_state(this, |state| unsafe { type Super = NSObject;
}
unsafe impl WinitWindowDelegate {
#[sel(dealloc)]
fn dealloc(&mut self) {
self.with_state(|state| unsafe {
drop(Box::from_raw(state as *mut WindowDelegateState)); drop(Box::from_raw(state as *mut WindowDelegateState));
}); });
} }
extern "C" fn init_with_winit(this: &Object, _sel: Sel, state: *mut c_void) -> id { #[sel(initWithWinit:)]
unsafe { fn init_with_winit(&mut self, state: *mut c_void) -> Option<&mut Self> {
let this: id = msg_send![this, init]; let this: Option<&mut Self> = unsafe { msg_send![self, init] };
if this != nil { this.map(|this| {
(*this).set_ivar("winitState", state); *this.state = state;
with_state(&*this, |state| { this.with_state(|state| {
let _: () = msg_send![*state.ns_window, setDelegate: this]; let _: () = unsafe { msg_send![*state.ns_window, setDelegate: &*this] };
}); });
}
this this
})
} }
} }
extern "C" fn window_should_close(this: &Object, _: Sel, _: id) -> Bool { // NSWindowDelegate + NSDraggingDestination protocols
unsafe impl WinitWindowDelegate {
#[sel(windowShouldClose:)]
fn window_should_close(&self, _: id) -> bool {
trace_scope!("windowShouldClose:"); trace_scope!("windowShouldClose:");
with_state(this, |state| state.emit_event(WindowEvent::CloseRequested)); self.with_state(|state| state.emit_event(WindowEvent::CloseRequested));
Bool::NO false
} }
extern "C" fn window_will_close(this: &Object, _: Sel, _: id) { #[sel(windowWillClose:)]
fn window_will_close(&self, _: id) {
trace_scope!("windowWillClose:"); trace_scope!("windowWillClose:");
with_state(this, |state| unsafe { self.with_state(|state| unsafe {
// `setDelegate:` retains the previous value and then autoreleases it // `setDelegate:` retains the previous value and then autoreleases it
autoreleasepool(|_| { autoreleasepool(|_| {
// Since El Capitan, we need to be careful that delegate methods can't // Since El Capitan, we need to be careful that delegate methods can't
@ -276,41 +182,46 @@ extern "C" fn window_will_close(this: &Object, _: Sel, _: id) {
}); });
} }
extern "C" fn window_did_resize(this: &Object, _: Sel, _: id) { #[sel(windowDidResize:)]
fn window_did_resize(&self, _: id) {
trace_scope!("windowDidResize:"); trace_scope!("windowDidResize:");
with_state(this, |state| { self.with_state(|state| {
// NOTE: WindowEvent::Resized is reported in frameDidChange. // NOTE: WindowEvent::Resized is reported in frameDidChange.
state.emit_move_event(); state.emit_move_event();
}); });
} }
// This won't be triggered if the move was part of a resize. // This won't be triggered if the move was part of a resize.
extern "C" fn window_did_move(this: &Object, _: Sel, _: id) { #[sel(windowDidMove:)]
fn window_did_move(&self, _: id) {
trace_scope!("windowDidMove:"); trace_scope!("windowDidMove:");
with_state(this, |state| { self.with_state(|state| {
state.emit_move_event(); state.emit_move_event();
}); });
} }
extern "C" fn window_did_change_backing_properties(this: &Object, _: Sel, _: id) { #[sel(windowDidChangeBackingProperties:)]
fn window_did_change_backing_properties(&self, _: id) {
trace_scope!("windowDidChangeBackingProperties:"); trace_scope!("windowDidChangeBackingProperties:");
with_state(this, |state| { self.with_state(|state| {
state.emit_static_scale_factor_changed_event(); state.emit_static_scale_factor_changed_event();
}); });
} }
extern "C" fn window_did_become_key(this: &Object, _: Sel, _: id) { #[sel(windowDidBecomeKey:)]
fn window_did_become_key(&self, _: id) {
trace_scope!("windowDidBecomeKey:"); trace_scope!("windowDidBecomeKey:");
with_state(this, |state| { self.with_state(|state| {
// TODO: center the cursor if the window had mouse grab when it // TODO: center the cursor if the window had mouse grab when it
// lost focus // lost focus
state.emit_event(WindowEvent::Focused(true)); state.emit_event(WindowEvent::Focused(true));
}); });
} }
extern "C" fn window_did_resign_key(this: &Object, _: Sel, _: id) { #[sel(windowDidResignKey:)]
fn window_did_resign_key(&self, _: id) {
trace_scope!("windowDidResignKey:"); trace_scope!("windowDidResignKey:");
with_state(this, |state| { self.with_state(|state| {
// It happens rather often, e.g. when the user is Cmd+Tabbing, that the // It happens rather often, e.g. when the user is Cmd+Tabbing, that the
// NSWindowDelegate will receive a didResignKey event despite no event // NSWindowDelegate will receive a didResignKey event despite no event
// being received when the modifiers are released. This is because // being received when the modifiers are released. This is because
@ -339,7 +250,8 @@ extern "C" fn window_did_resign_key(this: &Object, _: Sel, _: id) {
} }
/// Invoked when the dragged image enters destination bounds or frame /// Invoked when the dragged image enters destination bounds or frame
extern "C" fn dragging_entered(this: &Object, _: Sel, sender: id) -> Bool { #[sel(draggingEntered:)]
fn dragging_entered(&self, sender: id) -> bool {
trace_scope!("draggingEntered:"); trace_scope!("draggingEntered:");
use cocoa::{appkit::NSPasteboard, foundation::NSFastEnumeration}; use cocoa::{appkit::NSPasteboard, foundation::NSFastEnumeration};
@ -356,23 +268,25 @@ extern "C" fn dragging_entered(this: &Object, _: Sel, sender: id) -> Bool {
let f = NSString::UTF8String(file); let f = NSString::UTF8String(file);
let path = CStr::from_ptr(f).to_string_lossy().into_owned(); let path = CStr::from_ptr(f).to_string_lossy().into_owned();
with_state(this, |state| { self.with_state(|state| {
state.emit_event(WindowEvent::HoveredFile(PathBuf::from(path))); state.emit_event(WindowEvent::HoveredFile(PathBuf::from(path)));
}); });
} }
} }
Bool::YES true
} }
/// Invoked when the image is released /// Invoked when the image is released
extern "C" fn prepare_for_drag_operation(_: &Object, _: Sel, _: id) -> Bool { #[sel(prepareForDragOperation:)]
fn prepare_for_drag_operation(&self, _: id) -> bool {
trace_scope!("prepareForDragOperation:"); trace_scope!("prepareForDragOperation:");
Bool::YES true
} }
/// Invoked after the released image has been removed from the screen /// Invoked after the released image has been removed from the screen
extern "C" fn perform_drag_operation(this: &Object, _: Sel, sender: id) -> Bool { #[sel(performDragOperation:)]
fn perform_drag_operation(&self, sender: id) -> bool {
trace_scope!("performDragOperation:"); trace_scope!("performDragOperation:");
use cocoa::{appkit::NSPasteboard, foundation::NSFastEnumeration}; use cocoa::{appkit::NSPasteboard, foundation::NSFastEnumeration};
@ -389,33 +303,36 @@ extern "C" fn perform_drag_operation(this: &Object, _: Sel, sender: id) -> Bool
let f = NSString::UTF8String(file); let f = NSString::UTF8String(file);
let path = CStr::from_ptr(f).to_string_lossy().into_owned(); let path = CStr::from_ptr(f).to_string_lossy().into_owned();
with_state(this, |state| { self.with_state(|state| {
state.emit_event(WindowEvent::DroppedFile(PathBuf::from(path))); state.emit_event(WindowEvent::DroppedFile(PathBuf::from(path)));
}); });
} }
} }
Bool::YES true
} }
/// Invoked when the dragging operation is complete /// Invoked when the dragging operation is complete
extern "C" fn conclude_drag_operation(_: &Object, _: Sel, _: id) { #[sel(concludeDragOperation:)]
fn conclude_drag_operation(&self, _: id) {
trace_scope!("concludeDragOperation:"); trace_scope!("concludeDragOperation:");
} }
/// Invoked when the dragging operation is cancelled /// Invoked when the dragging operation is cancelled
extern "C" fn dragging_exited(this: &Object, _: Sel, _: id) { #[sel(draggingExited:)]
fn dragging_exited(&self, _: id) {
trace_scope!("draggingExited:"); trace_scope!("draggingExited:");
with_state(this, |state| { self.with_state(|state| {
state.emit_event(WindowEvent::HoveredFileCancelled) state.emit_event(WindowEvent::HoveredFileCancelled)
}); });
} }
/// Invoked when before enter fullscreen /// Invoked when before enter fullscreen
extern "C" fn window_will_enter_fullscreen(this: &Object, _: Sel, _: id) { #[sel(windowWillEnterFullscreen:)]
fn window_will_enter_fullscreen(&self, _: id) {
trace_scope!("windowWillEnterFullscreen:"); trace_scope!("windowWillEnterFullscreen:");
with_state(this, |state| { self.with_state(|state| {
state.with_window(|window| { state.with_window(|window| {
let mut shared_state = window.lock_shared_state("window_will_enter_fullscreen"); let mut shared_state = window.lock_shared_state("window_will_enter_fullscreen");
shared_state.maximized = window.is_zoomed(); shared_state.maximized = window.is_zoomed();
@ -442,10 +359,11 @@ extern "C" fn window_will_enter_fullscreen(this: &Object, _: Sel, _: id) {
} }
/// Invoked when before exit fullscreen /// Invoked when before exit fullscreen
extern "C" fn window_will_exit_fullscreen(this: &Object, _: Sel, _: id) { #[sel(windowWillExitFullScreen:)]
fn window_will_exit_fullscreen(&self, _: id) {
trace_scope!("windowWillExitFullScreen:"); trace_scope!("windowWillExitFullScreen:");
with_state(this, |state| { self.with_state(|state| {
state.with_window(|window| { state.with_window(|window| {
let mut shared_state = window.lock_shared_state("window_will_exit_fullscreen"); let mut shared_state = window.lock_shared_state("window_will_exit_fullscreen");
shared_state.in_fullscreen_transition = true; shared_state.in_fullscreen_transition = true;
@ -453,9 +371,9 @@ extern "C" fn window_will_exit_fullscreen(this: &Object, _: Sel, _: id) {
}); });
} }
extern "C" fn window_will_use_fullscreen_presentation_options( #[sel(window:willUseFullScreenPresentationOptions:)]
this: &Object, fn window_will_use_fullscreen_presentation_options(
_: Sel, &self,
_: id, _: id,
proposed_options: NSUInteger, proposed_options: NSUInteger,
) -> NSUInteger { ) -> NSUInteger {
@ -469,7 +387,7 @@ extern "C" fn window_will_use_fullscreen_presentation_options(
// we don't, for consistency. If we do, it should be documented that the // we don't, for consistency. If we do, it should be documented that the
// user-provided options are ignored in exclusive fullscreen. // user-provided options are ignored in exclusive fullscreen.
let mut options: NSUInteger = proposed_options; let mut options: NSUInteger = proposed_options;
with_state(this, |state| { self.with_state(|state| {
state.with_window(|window| { state.with_window(|window| {
let shared_state = let shared_state =
window.lock_shared_state("window_will_use_fullscreen_presentation_options"); window.lock_shared_state("window_will_use_fullscreen_presentation_options");
@ -486,9 +404,10 @@ extern "C" fn window_will_use_fullscreen_presentation_options(
} }
/// Invoked when entered fullscreen /// Invoked when entered fullscreen
extern "C" fn window_did_enter_fullscreen(this: &Object, _: Sel, _: id) { #[sel(windowDidEnterFullscreen:)]
fn window_did_enter_fullscreen(&self, _: id) {
trace_scope!("windowDidEnterFullscreen:"); trace_scope!("windowDidEnterFullscreen:");
with_state(this, |state| { self.with_state(|state| {
state.initial_fullscreen = false; state.initial_fullscreen = false;
state.with_window(|window| { state.with_window(|window| {
let mut shared_state = window.lock_shared_state("window_did_enter_fullscreen"); let mut shared_state = window.lock_shared_state("window_did_enter_fullscreen");
@ -503,10 +422,11 @@ extern "C" fn window_did_enter_fullscreen(this: &Object, _: Sel, _: id) {
} }
/// Invoked when exited fullscreen /// Invoked when exited fullscreen
extern "C" fn window_did_exit_fullscreen(this: &Object, _: Sel, _: id) { #[sel(windowDidExitFullscreen:)]
fn window_did_exit_fullscreen(&self, _: id) {
trace_scope!("windowDidExitFullscreen:"); trace_scope!("windowDidExitFullscreen:");
with_state(this, |state| { self.with_state(|state| {
state.with_window(|window| { state.with_window(|window| {
window.restore_state_from_fullscreen(); window.restore_state_from_fullscreen();
let mut shared_state = window.lock_shared_state("window_did_exit_fullscreen"); let mut shared_state = window.lock_shared_state("window_did_exit_fullscreen");
@ -536,9 +456,10 @@ extern "C" fn window_did_exit_fullscreen(this: &Object, _: Sel, _: id) {
/// due to being in the midst of handling some other animation or user gesture. /// due to being in the midst of handling some other animation or user gesture.
/// This method indicates that there was an error, and you should clean up any /// This method indicates that there was an error, and you should clean up any
/// work you may have done to prepare to enter full-screen mode. /// work you may have done to prepare to enter full-screen mode.
extern "C" fn window_did_fail_to_enter_fullscreen(this: &Object, _: Sel, _: id) { #[sel(windowDidFailToEnterFullscreen:)]
fn window_did_fail_to_enter_fullscreen(&self, _: id) {
trace_scope!("windowDidFailToEnterFullscreen:"); trace_scope!("windowDidFailToEnterFullscreen:");
with_state(this, |state| { self.with_state(|state| {
state.with_window(|window| { state.with_window(|window| {
let mut shared_state = window.lock_shared_state("window_did_fail_to_enter_fullscreen"); let mut shared_state = window.lock_shared_state("window_did_fail_to_enter_fullscreen");
shared_state.in_fullscreen_transition = false; shared_state.in_fullscreen_transition = false;
@ -559,10 +480,11 @@ extern "C" fn window_did_fail_to_enter_fullscreen(this: &Object, _: Sel, _: id)
} }
// Invoked when the occlusion state of the window changes // Invoked when the occlusion state of the window changes
extern "C" fn window_did_change_occlusion_state(this: &Object, _: Sel, _: id) { #[sel(windowDidChangeOcclusionState:)]
fn window_did_change_occlusion_state(&self, _: id) {
trace_scope!("windowDidChangeOcclusionState:"); trace_scope!("windowDidChangeOcclusionState:");
unsafe { unsafe {
with_state(this, |state| { self.with_state(|state| {
state.emit_event(WindowEvent::Occluded( state.emit_event(WindowEvent::Occluded(
!state !state
.ns_window .ns_window
@ -571,4 +493,18 @@ extern "C" fn window_did_change_occlusion_state(this: &Object, _: Sel, _: id) {
)) ))
}); });
} }
}
}
}
impl WinitWindowDelegate {
// This function is definitely unsafe (&self -> &mut state), but labeling that
// would increase boilerplate and wouldn't really clarify anything...
fn with_state<F: FnOnce(&mut WindowDelegateState) -> T, T>(&self, callback: F) {
let state_ptr = unsafe {
let state_ptr: *mut c_void = *self.state;
&mut *(state_ptr as *mut WindowDelegateState)
};
callback(state_ptr);
}
} }