macos: Implement run_return (#1108)

* macos: Implement run_return

* Update comments

* Fix CHANGELOG.md
This commit is contained in:
Kirill Chibisov 2019-08-23 12:30:53 +03:00 committed by Hal Gentz
parent 31110be396
commit 7b707e7d75
3 changed files with 35 additions and 28 deletions

View file

@ -1,5 +1,7 @@
# Unreleased # Unreleased
- On macOS, implement `run_return`.
# 0.20.0 Alpha 3 (2019-08-14) # 0.20.0 Alpha 3 (2019-08-14)
- On macOS, drop the run closure on exit. - On macOS, drop the run closure on exit.

View file

@ -3,6 +3,7 @@ use std::{
fmt::{self, Debug}, fmt::{self, Debug},
hint::unreachable_unchecked, hint::unreachable_unchecked,
mem, mem,
rc::Rc,
sync::{ sync::{
atomic::{AtomicBool, Ordering}, atomic::{AtomicBool, Ordering},
Mutex, MutexGuard, Mutex, MutexGuard,
@ -37,13 +38,13 @@ pub trait EventHandler: Debug {
fn handle_user_events(&mut self, control_flow: &mut ControlFlow); fn handle_user_events(&mut self, control_flow: &mut ControlFlow);
} }
struct EventLoopHandler<F, T: 'static> { struct EventLoopHandler<T: 'static> {
callback: F, callback: Box<dyn FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow)>,
will_exit: bool, will_exit: bool,
window_target: RootWindowTarget<T>, window_target: Rc<RootWindowTarget<T>>,
} }
impl<F, T> Debug for EventLoopHandler<F, T> { impl<T> Debug for EventLoopHandler<T> {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result { fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
formatter formatter
.debug_struct("EventLoopHandler") .debug_struct("EventLoopHandler")
@ -52,11 +53,7 @@ impl<F, T> Debug for EventLoopHandler<F, T> {
} }
} }
impl<F, T> EventHandler for EventLoopHandler<F, T> impl<T> EventHandler for EventLoopHandler<T> {
where
F: 'static + FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow),
T: 'static,
{
fn handle_nonuser_event(&mut self, event: Event<Never>, control_flow: &mut ControlFlow) { fn handle_nonuser_event(&mut self, event: Event<Never>, control_flow: &mut ControlFlow) {
(self.callback)(event.userify(), &self.window_target, control_flow); (self.callback)(event.userify(), &self.window_target, control_flow);
self.will_exit |= *control_flow == ControlFlow::Exit; self.will_exit |= *control_flow == ControlFlow::Exit;
@ -180,13 +177,20 @@ impl Handler {
pub enum AppState {} pub enum AppState {}
impl AppState { impl AppState {
pub fn set_callback<F, T>(callback: F, window_target: RootWindowTarget<T>) // This function extends lifetime of `callback` to 'static as its side effect
pub unsafe fn set_callback<F, T>(callback: F, window_target: Rc<RootWindowTarget<T>>)
where where
F: 'static + FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow), F: FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow),
T: 'static,
{ {
*HANDLER.callback.lock().unwrap() = Some(Box::new(EventLoopHandler { *HANDLER.callback.lock().unwrap() = Some(Box::new(EventLoopHandler {
callback, // This transmute is always safe, in case it was reached through `run`, since our
// lifetime will be already 'static. In other cases caller should ensure that all data
// they passed to callback will actually outlive it, some apps just can't move
// everything to event loop, so this is something that they should care about.
callback: mem::transmute::<
Box<dyn FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow)>,
Box<dyn FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow)>,
>(Box::new(callback)),
will_exit: false, will_exit: false,
window_target, window_target,
})); }));
@ -299,7 +303,7 @@ impl AppState {
} }
HANDLER.update_start_time(); HANDLER.update_start_time();
match HANDLER.get_old_and_new_control_flow() { match HANDLER.get_old_and_new_control_flow() {
(ControlFlow::Exit, _) | (_, ControlFlow::Exit) => unreachable!(), (ControlFlow::Exit, _) | (_, ControlFlow::Exit) => (),
(old, new) if old == new => (), (old, new) if old == new => (),
(_, ControlFlow::Wait) => HANDLER.waker().stop(), (_, ControlFlow::Wait) => HANDLER.waker().stop(),
(_, ControlFlow::WaitUntil(instant)) => HANDLER.waker().start_at(instant), (_, ControlFlow::WaitUntil(instant)) => HANDLER.waker().start_at(instant),

View file

@ -1,5 +1,6 @@
use std::{ use std::{
collections::VecDeque, marker::PhantomData, mem, os::raw::c_void, process, ptr, sync::mpsc, collections::VecDeque, marker::PhantomData, mem, os::raw::c_void, process, ptr, rc::Rc,
sync::mpsc,
}; };
use cocoa::{ use cocoa::{
@ -34,7 +35,7 @@ impl<T> Default for EventLoopWindowTarget<T> {
} }
pub struct EventLoop<T: 'static> { pub struct EventLoop<T: 'static> {
window_target: RootWindowTarget<T>, window_target: Rc<RootWindowTarget<T>>,
_delegate: IdRef, _delegate: IdRef,
} }
@ -59,10 +60,10 @@ impl<T> EventLoop<T> {
}; };
setup_control_flow_observers(); setup_control_flow_observers();
EventLoop { EventLoop {
window_target: RootWindowTarget { window_target: Rc::new(RootWindowTarget {
p: Default::default(), p: Default::default(),
_marker: PhantomData, _marker: PhantomData,
}, }),
_delegate: delegate, _delegate: delegate,
} }
} }
@ -81,28 +82,28 @@ impl<T> EventLoop<T> {
&self.window_target &self.window_target
} }
pub fn run<F>(self, callback: F) -> ! pub fn run<F>(mut self, callback: F) -> !
where where
F: 'static + FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow), F: 'static + FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow),
{
self.run_return(callback);
process::exit(0);
}
pub fn run_return<F>(&mut self, callback: F)
where
F: FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow),
{ {
unsafe { unsafe {
let _pool = NSAutoreleasePool::new(nil); let _pool = NSAutoreleasePool::new(nil);
let app = NSApp(); let app = NSApp();
assert_ne!(app, nil); assert_ne!(app, nil);
AppState::set_callback(callback, self.window_target); AppState::set_callback(callback, Rc::clone(&self.window_target));
let _: () = msg_send![app, run]; let _: () = msg_send![app, run];
AppState::exit(); AppState::exit();
process::exit(0)
} }
} }
pub fn run_return<F>(&mut self, _callback: F)
where
F: FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow),
{
unimplemented!();
}
pub fn create_proxy(&self) -> Proxy<T> { pub fn create_proxy(&self) -> Proxy<T> {
Proxy::new(self.window_target.p.sender.clone()) Proxy::new(self.window_target.p.sender.clone())
} }