mirror of
https://github.com/italicsjenga/winit-sonoma-fix.git
synced 2025-01-14 22:40:38 +11:00
c10c820311
* Reimplement NativeDisplayMode on iOS for #1310 * Type annotations from code review. Co-Authored-By: Aleksi Juvani <3168386+aleksijuvani@users.noreply.github.com> Co-authored-by: Aleksi Juvani <3168386+aleksijuvani@users.noreply.github.com>
589 lines
22 KiB
Rust
589 lines
22 KiB
Rust
use std::collections::HashMap;
|
|
|
|
use objc::{
|
|
declare::ClassDecl,
|
|
runtime::{Class, Object, Sel, BOOL, NO, YES},
|
|
};
|
|
|
|
use crate::{
|
|
event::{DeviceId as RootDeviceId, Event, Force, Touch, TouchPhase, WindowEvent},
|
|
platform::ios::MonitorHandleExtIOS,
|
|
platform_impl::platform::{
|
|
app_state::{self, OSCapabilities},
|
|
event_loop,
|
|
ffi::{
|
|
id, nil, CGFloat, CGPoint, CGRect, UIForceTouchCapability, UIInterfaceOrientationMask,
|
|
UIRectEdge, UITouchPhase, UITouchType,
|
|
},
|
|
window::PlatformSpecificWindowBuilderAttributes,
|
|
DeviceId,
|
|
},
|
|
window::{Fullscreen, WindowAttributes, WindowId as RootWindowId},
|
|
};
|
|
|
|
macro_rules! add_property {
|
|
(
|
|
$decl:ident,
|
|
$name:ident: $t:ty,
|
|
$setter_name:ident: |$object:ident| $after_set:expr,
|
|
$getter_name:ident,
|
|
) => {
|
|
add_property!(
|
|
$decl,
|
|
$name: $t,
|
|
$setter_name: true, |_, _|{}; |$object| $after_set,
|
|
$getter_name,
|
|
)
|
|
};
|
|
(
|
|
$decl:ident,
|
|
$name:ident: $t:ty,
|
|
$setter_name:ident: $capability:expr, $err:expr; |$object:ident| $after_set:expr,
|
|
$getter_name:ident,
|
|
) => {
|
|
{
|
|
const VAR_NAME: &'static str = concat!("_", stringify!($name));
|
|
$decl.add_ivar::<$t>(VAR_NAME);
|
|
let setter = if $capability {
|
|
#[allow(non_snake_case)]
|
|
extern "C" fn $setter_name($object: &mut Object, _: Sel, value: $t) {
|
|
unsafe {
|
|
$object.set_ivar::<$t>(VAR_NAME, value);
|
|
}
|
|
$after_set
|
|
}
|
|
$setter_name
|
|
} else {
|
|
#[allow(non_snake_case)]
|
|
extern "C" fn $setter_name($object: &mut Object, _: Sel, value: $t) {
|
|
unsafe {
|
|
$object.set_ivar::<$t>(VAR_NAME, value);
|
|
}
|
|
$err(&app_state::os_capabilities(), "ignoring")
|
|
}
|
|
$setter_name
|
|
};
|
|
#[allow(non_snake_case)]
|
|
extern "C" fn $getter_name($object: &Object, _: Sel) -> $t {
|
|
unsafe { *$object.get_ivar::<$t>(VAR_NAME) }
|
|
}
|
|
$decl.add_method(
|
|
sel!($setter_name:),
|
|
setter as extern "C" fn(&mut Object, Sel, $t),
|
|
);
|
|
$decl.add_method(
|
|
sel!($getter_name),
|
|
$getter_name as extern "C" fn(&Object, Sel) -> $t,
|
|
);
|
|
}
|
|
};
|
|
}
|
|
|
|
// requires main thread
|
|
unsafe fn get_view_class(root_view_class: &'static Class) -> &'static Class {
|
|
static mut CLASSES: Option<HashMap<*const Class, &'static Class>> = None;
|
|
static mut ID: usize = 0;
|
|
|
|
if CLASSES.is_none() {
|
|
CLASSES = Some(HashMap::default());
|
|
}
|
|
|
|
let classes = CLASSES.as_mut().unwrap();
|
|
|
|
classes.entry(root_view_class).or_insert_with(move || {
|
|
let uiview_class = class!(UIView);
|
|
let is_uiview: BOOL = msg_send![root_view_class, isSubclassOfClass: uiview_class];
|
|
assert_eq!(
|
|
is_uiview, YES,
|
|
"`root_view_class` must inherit from `UIView`"
|
|
);
|
|
|
|
extern "C" fn draw_rect(object: &Object, _: Sel, rect: CGRect) {
|
|
unsafe {
|
|
let window: id = msg_send![object, window];
|
|
assert!(!window.is_null());
|
|
app_state::handle_nonuser_event(Event::WindowEvent {
|
|
window_id: RootWindowId(window.into()),
|
|
event: WindowEvent::RedrawRequested,
|
|
});
|
|
let superclass: &'static Class = msg_send![object, superclass];
|
|
let () = msg_send![super(object, superclass), drawRect: rect];
|
|
}
|
|
}
|
|
|
|
extern "C" fn layout_subviews(object: &Object, _: Sel) {
|
|
unsafe {
|
|
let superclass: &'static Class = msg_send![object, superclass];
|
|
let () = msg_send![super(object, superclass), layoutSubviews];
|
|
|
|
let window: id = msg_send![object, window];
|
|
assert!(!window.is_null());
|
|
let bounds: CGRect = msg_send![window, bounds];
|
|
let screen: id = msg_send![window, screen];
|
|
let screen_space: id = msg_send![screen, coordinateSpace];
|
|
let screen_frame: CGRect =
|
|
msg_send![object, convertRect:bounds toCoordinateSpace:screen_space];
|
|
let size = crate::dpi::LogicalSize {
|
|
width: screen_frame.size.width as _,
|
|
height: screen_frame.size.height as _,
|
|
};
|
|
app_state::handle_nonuser_event(Event::WindowEvent {
|
|
window_id: RootWindowId(window.into()),
|
|
event: WindowEvent::Resized(size),
|
|
});
|
|
}
|
|
}
|
|
|
|
extern "C" fn set_content_scale_factor(
|
|
object: &mut Object,
|
|
_: Sel,
|
|
untrusted_hidpi_factor: CGFloat,
|
|
) {
|
|
unsafe {
|
|
let superclass: &'static Class = msg_send![object, superclass];
|
|
let () = msg_send![
|
|
super(object, superclass),
|
|
setContentScaleFactor: untrusted_hidpi_factor
|
|
];
|
|
|
|
let window: id = msg_send![object, window];
|
|
// `window` is null when `setContentScaleFactor` is invoked prior to `[UIWindow
|
|
// makeKeyAndVisible]` at window creation time (either manually or internally by
|
|
// UIKit when the `UIView` is first created), in which case we send no events here
|
|
if window.is_null() {
|
|
return;
|
|
}
|
|
// `setContentScaleFactor` may be called with a value of 0, which means "reset the
|
|
// content scale factor to a device-specific default value", so we can't use the
|
|
// parameter here. We can query the actual factor using the getter
|
|
let hidpi_factor: CGFloat = msg_send![object, contentScaleFactor];
|
|
assert!(
|
|
!hidpi_factor.is_nan()
|
|
&& hidpi_factor.is_finite()
|
|
&& hidpi_factor.is_sign_positive()
|
|
&& hidpi_factor > 0.0,
|
|
"invalid hidpi_factor set on UIView",
|
|
);
|
|
let bounds: CGRect = msg_send![object, bounds];
|
|
let screen: id = msg_send![window, screen];
|
|
let screen_space: id = msg_send![screen, coordinateSpace];
|
|
let screen_frame: CGRect =
|
|
msg_send![object, convertRect:bounds toCoordinateSpace:screen_space];
|
|
let size = crate::dpi::LogicalSize {
|
|
width: screen_frame.size.width as _,
|
|
height: screen_frame.size.height as _,
|
|
};
|
|
app_state::handle_nonuser_events(
|
|
std::iter::once(Event::WindowEvent {
|
|
window_id: RootWindowId(window.into()),
|
|
event: WindowEvent::HiDpiFactorChanged(hidpi_factor as _),
|
|
})
|
|
.chain(std::iter::once(Event::WindowEvent {
|
|
window_id: RootWindowId(window.into()),
|
|
event: WindowEvent::Resized(size),
|
|
})),
|
|
);
|
|
}
|
|
}
|
|
|
|
extern "C" fn handle_touches(object: &Object, _: Sel, touches: id, _: id) {
|
|
unsafe {
|
|
let window: id = msg_send![object, window];
|
|
assert!(!window.is_null());
|
|
let uiscreen: id = msg_send![window, screen];
|
|
let touches_enum: id = msg_send![touches, objectEnumerator];
|
|
let mut touch_events = Vec::new();
|
|
let os_supports_force = app_state::os_capabilities().force_touch;
|
|
loop {
|
|
let touch: id = msg_send![touches_enum, nextObject];
|
|
if touch == nil {
|
|
break;
|
|
}
|
|
let location: CGPoint = msg_send![touch, locationInView: nil];
|
|
let touch_type: UITouchType = msg_send![touch, type];
|
|
let force = if os_supports_force {
|
|
let trait_collection: id = msg_send![object, traitCollection];
|
|
let touch_capability: UIForceTouchCapability =
|
|
msg_send![trait_collection, forceTouchCapability];
|
|
// Both the OS _and_ the device need to be checked for force touch support.
|
|
if touch_capability == UIForceTouchCapability::Available {
|
|
let force: CGFloat = msg_send![touch, force];
|
|
let max_possible_force: CGFloat =
|
|
msg_send![touch, maximumPossibleForce];
|
|
let altitude_angle: Option<f64> = if touch_type == UITouchType::Pencil {
|
|
let angle: CGFloat = msg_send![touch, altitudeAngle];
|
|
Some(angle as _)
|
|
} else {
|
|
None
|
|
};
|
|
Some(Force::Calibrated {
|
|
force: force as _,
|
|
max_possible_force: max_possible_force as _,
|
|
altitude_angle,
|
|
})
|
|
} else {
|
|
None
|
|
}
|
|
} else {
|
|
None
|
|
};
|
|
let touch_id = touch as u64;
|
|
let phase: UITouchPhase = msg_send![touch, phase];
|
|
let phase = match phase {
|
|
UITouchPhase::Began => TouchPhase::Started,
|
|
UITouchPhase::Moved => TouchPhase::Moved,
|
|
// 2 is UITouchPhase::Stationary and is not expected here
|
|
UITouchPhase::Ended => TouchPhase::Ended,
|
|
UITouchPhase::Cancelled => TouchPhase::Cancelled,
|
|
_ => panic!("unexpected touch phase: {:?}", phase as i32),
|
|
};
|
|
|
|
touch_events.push(Event::WindowEvent {
|
|
window_id: RootWindowId(window.into()),
|
|
event: WindowEvent::Touch(Touch {
|
|
device_id: RootDeviceId(DeviceId { uiscreen }),
|
|
id: touch_id,
|
|
location: (location.x as f64, location.y as f64).into(),
|
|
force,
|
|
phase,
|
|
}),
|
|
});
|
|
}
|
|
app_state::handle_nonuser_events(touch_events);
|
|
}
|
|
}
|
|
|
|
let mut decl = ClassDecl::new(&format!("WinitUIView{}", ID), root_view_class)
|
|
.expect("Failed to declare class `WinitUIView`");
|
|
ID += 1;
|
|
decl.add_method(
|
|
sel!(drawRect:),
|
|
draw_rect as extern "C" fn(&Object, Sel, CGRect),
|
|
);
|
|
decl.add_method(
|
|
sel!(layoutSubviews),
|
|
layout_subviews as extern "C" fn(&Object, Sel),
|
|
);
|
|
decl.add_method(
|
|
sel!(setContentScaleFactor:),
|
|
set_content_scale_factor as extern "C" fn(&mut Object, Sel, CGFloat),
|
|
);
|
|
|
|
decl.add_method(
|
|
sel!(touchesBegan:withEvent:),
|
|
handle_touches as extern "C" fn(this: &Object, _: Sel, _: id, _: id),
|
|
);
|
|
decl.add_method(
|
|
sel!(touchesMoved:withEvent:),
|
|
handle_touches as extern "C" fn(this: &Object, _: Sel, _: id, _: id),
|
|
);
|
|
decl.add_method(
|
|
sel!(touchesEnded:withEvent:),
|
|
handle_touches as extern "C" fn(this: &Object, _: Sel, _: id, _: id),
|
|
);
|
|
decl.add_method(
|
|
sel!(touchesCancelled:withEvent:),
|
|
handle_touches as extern "C" fn(this: &Object, _: Sel, _: id, _: id),
|
|
);
|
|
|
|
decl.register()
|
|
})
|
|
}
|
|
|
|
// requires main thread
|
|
unsafe fn get_view_controller_class() -> &'static Class {
|
|
static mut CLASS: Option<&'static Class> = None;
|
|
if CLASS.is_none() {
|
|
let os_capabilities = app_state::os_capabilities();
|
|
|
|
let uiviewcontroller_class = class!(UIViewController);
|
|
|
|
extern "C" fn should_autorotate(_: &Object, _: Sel) -> BOOL {
|
|
YES
|
|
}
|
|
|
|
let mut decl = ClassDecl::new("WinitUIViewController", uiviewcontroller_class)
|
|
.expect("Failed to declare class `WinitUIViewController`");
|
|
decl.add_method(
|
|
sel!(shouldAutorotate),
|
|
should_autorotate as extern "C" fn(&Object, Sel) -> BOOL,
|
|
);
|
|
add_property! {
|
|
decl,
|
|
prefers_status_bar_hidden: BOOL,
|
|
setPrefersStatusBarHidden: |object| {
|
|
unsafe {
|
|
let () = msg_send![object, setNeedsStatusBarAppearanceUpdate];
|
|
}
|
|
},
|
|
prefersStatusBarHidden,
|
|
}
|
|
add_property! {
|
|
decl,
|
|
prefers_home_indicator_auto_hidden: BOOL,
|
|
setPrefersHomeIndicatorAutoHidden:
|
|
os_capabilities.home_indicator_hidden,
|
|
OSCapabilities::home_indicator_hidden_err_msg;
|
|
|object| {
|
|
unsafe {
|
|
let () = msg_send![object, setNeedsUpdateOfHomeIndicatorAutoHidden];
|
|
}
|
|
},
|
|
prefersHomeIndicatorAutoHidden,
|
|
}
|
|
add_property! {
|
|
decl,
|
|
supported_orientations: UIInterfaceOrientationMask,
|
|
setSupportedInterfaceOrientations: |object| {
|
|
unsafe {
|
|
let () = msg_send![class!(UIViewController), attemptRotationToDeviceOrientation];
|
|
}
|
|
},
|
|
supportedInterfaceOrientations,
|
|
}
|
|
add_property! {
|
|
decl,
|
|
preferred_screen_edges_deferring_system_gestures: UIRectEdge,
|
|
setPreferredScreenEdgesDeferringSystemGestures:
|
|
os_capabilities.defer_system_gestures,
|
|
OSCapabilities::defer_system_gestures_err_msg;
|
|
|object| {
|
|
unsafe {
|
|
let () = msg_send![object, setNeedsUpdateOfScreenEdgesDeferringSystemGestures];
|
|
}
|
|
},
|
|
preferredScreenEdgesDeferringSystemGestures,
|
|
}
|
|
CLASS = Some(decl.register());
|
|
}
|
|
CLASS.unwrap()
|
|
}
|
|
|
|
// requires main thread
|
|
unsafe fn get_window_class() -> &'static Class {
|
|
static mut CLASS: Option<&'static Class> = None;
|
|
if CLASS.is_none() {
|
|
let uiwindow_class = class!(UIWindow);
|
|
|
|
extern "C" fn become_key_window(object: &Object, _: Sel) {
|
|
unsafe {
|
|
app_state::handle_nonuser_event(Event::WindowEvent {
|
|
window_id: RootWindowId(object.into()),
|
|
event: WindowEvent::Focused(true),
|
|
});
|
|
let () = msg_send![super(object, class!(UIWindow)), becomeKeyWindow];
|
|
}
|
|
}
|
|
|
|
extern "C" fn resign_key_window(object: &Object, _: Sel) {
|
|
unsafe {
|
|
app_state::handle_nonuser_event(Event::WindowEvent {
|
|
window_id: RootWindowId(object.into()),
|
|
event: WindowEvent::Focused(false),
|
|
});
|
|
let () = msg_send![super(object, class!(UIWindow)), resignKeyWindow];
|
|
}
|
|
}
|
|
|
|
let mut decl = ClassDecl::new("WinitUIWindow", uiwindow_class)
|
|
.expect("Failed to declare class `WinitUIWindow`");
|
|
decl.add_method(
|
|
sel!(becomeKeyWindow),
|
|
become_key_window as extern "C" fn(&Object, Sel),
|
|
);
|
|
decl.add_method(
|
|
sel!(resignKeyWindow),
|
|
resign_key_window as extern "C" fn(&Object, Sel),
|
|
);
|
|
|
|
CLASS = Some(decl.register());
|
|
}
|
|
CLASS.unwrap()
|
|
}
|
|
|
|
// requires main thread
|
|
pub unsafe fn create_view(
|
|
_window_attributes: &WindowAttributes,
|
|
platform_attributes: &PlatformSpecificWindowBuilderAttributes,
|
|
frame: CGRect,
|
|
) -> id {
|
|
let class = get_view_class(platform_attributes.root_view_class);
|
|
|
|
let view: id = msg_send![class, alloc];
|
|
assert!(!view.is_null(), "Failed to create `UIView` instance");
|
|
let view: id = msg_send![view, initWithFrame: frame];
|
|
assert!(!view.is_null(), "Failed to initialize `UIView` instance");
|
|
let () = msg_send![view, setMultipleTouchEnabled: YES];
|
|
if let Some(hidpi_factor) = platform_attributes.hidpi_factor {
|
|
let () = msg_send![view, setContentScaleFactor: hidpi_factor as CGFloat];
|
|
}
|
|
|
|
view
|
|
}
|
|
|
|
// requires main thread
|
|
pub unsafe fn create_view_controller(
|
|
_window_attributes: &WindowAttributes,
|
|
platform_attributes: &PlatformSpecificWindowBuilderAttributes,
|
|
view: id,
|
|
) -> id {
|
|
let class = get_view_controller_class();
|
|
|
|
let view_controller: id = msg_send![class, alloc];
|
|
assert!(
|
|
!view_controller.is_null(),
|
|
"Failed to create `UIViewController` instance"
|
|
);
|
|
let view_controller: id = msg_send![view_controller, init];
|
|
assert!(
|
|
!view_controller.is_null(),
|
|
"Failed to initialize `UIViewController` instance"
|
|
);
|
|
let status_bar_hidden = if platform_attributes.prefers_status_bar_hidden {
|
|
YES
|
|
} else {
|
|
NO
|
|
};
|
|
let idiom = event_loop::get_idiom();
|
|
let supported_orientations = UIInterfaceOrientationMask::from_valid_orientations_idiom(
|
|
platform_attributes.valid_orientations,
|
|
idiom,
|
|
);
|
|
let prefers_home_indicator_hidden = if platform_attributes.prefers_home_indicator_hidden {
|
|
YES
|
|
} else {
|
|
NO
|
|
};
|
|
let edges: UIRectEdge = platform_attributes
|
|
.preferred_screen_edges_deferring_system_gestures
|
|
.into();
|
|
let () = msg_send![
|
|
view_controller,
|
|
setPrefersStatusBarHidden: status_bar_hidden
|
|
];
|
|
let () = msg_send![
|
|
view_controller,
|
|
setSupportedInterfaceOrientations: supported_orientations
|
|
];
|
|
let () = msg_send![
|
|
view_controller,
|
|
setPrefersHomeIndicatorAutoHidden: prefers_home_indicator_hidden
|
|
];
|
|
let () = msg_send![
|
|
view_controller,
|
|
setPreferredScreenEdgesDeferringSystemGestures: edges
|
|
];
|
|
let () = msg_send![view_controller, setView: view];
|
|
view_controller
|
|
}
|
|
|
|
// requires main thread
|
|
pub unsafe fn create_window(
|
|
window_attributes: &WindowAttributes,
|
|
_platform_attributes: &PlatformSpecificWindowBuilderAttributes,
|
|
frame: CGRect,
|
|
view_controller: id,
|
|
) -> id {
|
|
let class = get_window_class();
|
|
|
|
let window: id = msg_send![class, alloc];
|
|
assert!(!window.is_null(), "Failed to create `UIWindow` instance");
|
|
let window: id = msg_send![window, initWithFrame: frame];
|
|
assert!(
|
|
!window.is_null(),
|
|
"Failed to initialize `UIWindow` instance"
|
|
);
|
|
let () = msg_send![window, setRootViewController: view_controller];
|
|
match window_attributes.fullscreen {
|
|
Some(Fullscreen::Exclusive(ref video_mode)) => {
|
|
let uiscreen = video_mode.monitor().ui_screen() as id;
|
|
let () = msg_send![uiscreen, setCurrentMode: video_mode.video_mode.screen_mode.0];
|
|
msg_send![window, setScreen:video_mode.monitor().ui_screen()]
|
|
}
|
|
Some(Fullscreen::Borderless(ref monitor)) => {
|
|
msg_send![window, setScreen:monitor.ui_screen()]
|
|
}
|
|
None => (),
|
|
}
|
|
|
|
window
|
|
}
|
|
|
|
pub fn create_delegate_class() {
|
|
extern "C" fn did_finish_launching(_: &mut Object, _: Sel, _: id, _: id) -> BOOL {
|
|
unsafe {
|
|
app_state::did_finish_launching();
|
|
}
|
|
YES
|
|
}
|
|
|
|
extern "C" fn did_become_active(_: &Object, _: Sel, _: id) {
|
|
unsafe { app_state::handle_nonuser_event(Event::Resumed) }
|
|
}
|
|
|
|
extern "C" fn will_resign_active(_: &Object, _: Sel, _: id) {
|
|
unsafe { app_state::handle_nonuser_event(Event::Suspended) }
|
|
}
|
|
|
|
extern "C" fn will_enter_foreground(_: &Object, _: Sel, _: id) {}
|
|
extern "C" fn did_enter_background(_: &Object, _: Sel, _: id) {}
|
|
|
|
extern "C" fn will_terminate(_: &Object, _: Sel, _: id) {
|
|
unsafe {
|
|
let app: id = msg_send![class!(UIApplication), sharedApplication];
|
|
let windows: id = msg_send![app, windows];
|
|
let windows_enum: id = msg_send![windows, objectEnumerator];
|
|
let mut events = Vec::new();
|
|
loop {
|
|
let window: id = msg_send![windows_enum, nextObject];
|
|
if window == nil {
|
|
break;
|
|
}
|
|
let is_winit_window: BOOL = msg_send![window, isKindOfClass: class!(WinitUIWindow)];
|
|
if is_winit_window == YES {
|
|
events.push(Event::WindowEvent {
|
|
window_id: RootWindowId(window.into()),
|
|
event: WindowEvent::Destroyed,
|
|
});
|
|
}
|
|
}
|
|
app_state::handle_nonuser_events(events);
|
|
app_state::terminated();
|
|
}
|
|
}
|
|
|
|
let ui_responder = class!(UIResponder);
|
|
let mut decl =
|
|
ClassDecl::new("AppDelegate", ui_responder).expect("Failed to declare class `AppDelegate`");
|
|
|
|
unsafe {
|
|
decl.add_method(
|
|
sel!(application:didFinishLaunchingWithOptions:),
|
|
did_finish_launching as extern "C" fn(&mut Object, Sel, id, id) -> BOOL,
|
|
);
|
|
|
|
decl.add_method(
|
|
sel!(applicationDidBecomeActive:),
|
|
did_become_active as extern "C" fn(&Object, Sel, id),
|
|
);
|
|
decl.add_method(
|
|
sel!(applicationWillResignActive:),
|
|
will_resign_active as extern "C" fn(&Object, Sel, id),
|
|
);
|
|
decl.add_method(
|
|
sel!(applicationWillEnterForeground:),
|
|
will_enter_foreground as extern "C" fn(&Object, Sel, id),
|
|
);
|
|
decl.add_method(
|
|
sel!(applicationDidEnterBackground:),
|
|
did_enter_background as extern "C" fn(&Object, Sel, id),
|
|
);
|
|
|
|
decl.add_method(
|
|
sel!(applicationWillTerminate:),
|
|
will_terminate as extern "C" fn(&Object, Sel, id),
|
|
);
|
|
|
|
decl.register();
|
|
}
|
|
}
|