Add a facility for delaying quit until critical tasks finish
Co-Authored-By: Antonio Scandurra <me@as-cii.com>
This commit is contained in:
parent
6e5ec2a00d
commit
b8994c2a89
7 changed files with 70 additions and 4 deletions
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -2309,6 +2309,7 @@ dependencies = [
|
||||||
"etagere",
|
"etagere",
|
||||||
"font-kit",
|
"font-kit",
|
||||||
"foreign-types",
|
"foreign-types",
|
||||||
|
"futures",
|
||||||
"gpui_macros",
|
"gpui_macros",
|
||||||
"image 0.23.14",
|
"image 0.23.14",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
|
|
|
@ -15,6 +15,7 @@ backtrace = "0.3"
|
||||||
ctor = "0.1"
|
ctor = "0.1"
|
||||||
env_logger = { version = "0.8", optional = true }
|
env_logger = { version = "0.8", optional = true }
|
||||||
etagere = "0.2"
|
etagere = "0.2"
|
||||||
|
futures = "0.3"
|
||||||
image = "0.23"
|
image = "0.23"
|
||||||
lazy_static = "1.4.0"
|
lazy_static = "1.4.0"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
|
|
|
@ -265,6 +265,18 @@ impl App {
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn on_quit<F>(self, mut callback: F) -> Self
|
||||||
|
where
|
||||||
|
F: 'static + FnMut(&mut MutableAppContext),
|
||||||
|
{
|
||||||
|
let cx = self.0.clone();
|
||||||
|
self.0
|
||||||
|
.borrow_mut()
|
||||||
|
.foreground_platform
|
||||||
|
.on_quit(Box::new(move || callback(&mut *cx.borrow_mut())));
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
pub fn on_event<F>(self, mut callback: F) -> Self
|
pub fn on_event<F>(self, mut callback: F) -> Self
|
||||||
where
|
where
|
||||||
F: 'static + FnMut(Event, &mut MutableAppContext) -> bool,
|
F: 'static + FnMut(Event, &mut MutableAppContext) -> bool,
|
||||||
|
|
|
@ -38,9 +38,13 @@ pub enum Foreground {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub enum Background {
|
pub enum Background {
|
||||||
Deterministic(Arc<Deterministic>),
|
Deterministic {
|
||||||
|
executor: Arc<Deterministic>,
|
||||||
|
critical_tasks: Mutex<Vec<Task<()>>>,
|
||||||
|
},
|
||||||
Production {
|
Production {
|
||||||
executor: Arc<smol::Executor<'static>>,
|
executor: Arc<smol::Executor<'static>>,
|
||||||
|
critical_tasks: Mutex<Vec<Task<()>>>,
|
||||||
_stop: channel::Sender<()>,
|
_stop: channel::Sender<()>,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -500,6 +504,7 @@ impl Background {
|
||||||
|
|
||||||
Self::Production {
|
Self::Production {
|
||||||
executor,
|
executor,
|
||||||
|
critical_tasks: Default::default(),
|
||||||
_stop: stop.0,
|
_stop: stop.0,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -516,11 +521,36 @@ impl Background {
|
||||||
let future = any_future(future);
|
let future = any_future(future);
|
||||||
let any_task = match self {
|
let any_task = match self {
|
||||||
Self::Production { executor, .. } => executor.spawn(future),
|
Self::Production { executor, .. } => executor.spawn(future),
|
||||||
Self::Deterministic(executor) => executor.spawn(future),
|
Self::Deterministic { executor, .. } => executor.spawn(future),
|
||||||
};
|
};
|
||||||
Task::send(any_task)
|
Task::send(any_task)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn spawn_critical<T, F>(&self, future: F)
|
||||||
|
where
|
||||||
|
T: 'static + Send,
|
||||||
|
F: Send + Future<Output = T> + 'static,
|
||||||
|
{
|
||||||
|
let task = self.spawn(async move {
|
||||||
|
future.await;
|
||||||
|
});
|
||||||
|
match self {
|
||||||
|
Self::Production { critical_tasks, .. }
|
||||||
|
| Self::Deterministic { critical_tasks, .. } => critical_tasks.lock().push(task),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn block_on_critical_tasks(&self, timeout: Duration) -> bool {
|
||||||
|
match self {
|
||||||
|
Background::Production { critical_tasks, .. }
|
||||||
|
| Self::Deterministic { critical_tasks, .. } => {
|
||||||
|
let tasks = mem::take(&mut *critical_tasks.lock());
|
||||||
|
self.block_with_timeout(timeout, futures::future::join_all(tasks))
|
||||||
|
.is_ok()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn block_with_timeout<F, T>(
|
pub fn block_with_timeout<F, T>(
|
||||||
&self,
|
&self,
|
||||||
timeout: Duration,
|
timeout: Duration,
|
||||||
|
@ -534,7 +564,7 @@ impl Background {
|
||||||
if !timeout.is_zero() {
|
if !timeout.is_zero() {
|
||||||
let output = match self {
|
let output = match self {
|
||||||
Self::Production { .. } => smol::block_on(util::timeout(timeout, &mut future)).ok(),
|
Self::Production { .. } => smol::block_on(util::timeout(timeout, &mut future)).ok(),
|
||||||
Self::Deterministic(executor) => executor.block_on(&mut future),
|
Self::Deterministic { executor, .. } => executor.block_on(&mut future),
|
||||||
};
|
};
|
||||||
if let Some(output) = output {
|
if let Some(output) = output {
|
||||||
return Ok(*output.downcast().unwrap());
|
return Ok(*output.downcast().unwrap());
|
||||||
|
@ -587,7 +617,10 @@ pub fn deterministic(seed: u64) -> (Rc<Foreground>, Arc<Background>) {
|
||||||
let executor = Arc::new(Deterministic::new(seed));
|
let executor = Arc::new(Deterministic::new(seed));
|
||||||
(
|
(
|
||||||
Rc::new(Foreground::Deterministic(executor.clone())),
|
Rc::new(Foreground::Deterministic(executor.clone())),
|
||||||
Arc::new(Background::Deterministic(executor)),
|
Arc::new(Background::Deterministic {
|
||||||
|
executor,
|
||||||
|
critical_tasks: Default::default(),
|
||||||
|
}),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -60,6 +60,7 @@ pub trait Platform: Send + Sync {
|
||||||
pub(crate) trait ForegroundPlatform {
|
pub(crate) trait ForegroundPlatform {
|
||||||
fn on_become_active(&self, callback: Box<dyn FnMut()>);
|
fn on_become_active(&self, callback: Box<dyn FnMut()>);
|
||||||
fn on_resign_active(&self, callback: Box<dyn FnMut()>);
|
fn on_resign_active(&self, callback: Box<dyn FnMut()>);
|
||||||
|
fn on_quit(&self, callback: Box<dyn FnMut()>);
|
||||||
fn on_event(&self, callback: Box<dyn FnMut(Event) -> bool>);
|
fn on_event(&self, callback: Box<dyn FnMut(Event) -> bool>);
|
||||||
fn on_open_files(&self, callback: Box<dyn FnMut(Vec<PathBuf>)>);
|
fn on_open_files(&self, callback: Box<dyn FnMut(Vec<PathBuf>)>);
|
||||||
fn run(&self, on_finish_launching: Box<dyn FnOnce() -> ()>);
|
fn run(&self, on_finish_launching: Box<dyn FnOnce() -> ()>);
|
||||||
|
|
|
@ -81,6 +81,10 @@ unsafe fn build_classes() {
|
||||||
sel!(applicationDidResignActive:),
|
sel!(applicationDidResignActive:),
|
||||||
did_resign_active as extern "C" fn(&mut Object, Sel, id),
|
did_resign_active as extern "C" fn(&mut Object, Sel, id),
|
||||||
);
|
);
|
||||||
|
decl.add_method(
|
||||||
|
sel!(applicationWillTerminate:),
|
||||||
|
will_terminate as extern "C" fn(&mut Object, Sel, id),
|
||||||
|
);
|
||||||
decl.add_method(
|
decl.add_method(
|
||||||
sel!(handleGPUIMenuItem:),
|
sel!(handleGPUIMenuItem:),
|
||||||
handle_menu_item as extern "C" fn(&mut Object, Sel, id),
|
handle_menu_item as extern "C" fn(&mut Object, Sel, id),
|
||||||
|
@ -100,6 +104,7 @@ pub struct MacForegroundPlatform(RefCell<MacForegroundPlatformState>);
|
||||||
pub struct MacForegroundPlatformState {
|
pub struct MacForegroundPlatformState {
|
||||||
become_active: Option<Box<dyn FnMut()>>,
|
become_active: Option<Box<dyn FnMut()>>,
|
||||||
resign_active: Option<Box<dyn FnMut()>>,
|
resign_active: Option<Box<dyn FnMut()>>,
|
||||||
|
quit: Option<Box<dyn FnMut()>>,
|
||||||
event: Option<Box<dyn FnMut(crate::Event) -> bool>>,
|
event: Option<Box<dyn FnMut(crate::Event) -> bool>>,
|
||||||
menu_command: Option<Box<dyn FnMut(&dyn AnyAction)>>,
|
menu_command: Option<Box<dyn FnMut(&dyn AnyAction)>>,
|
||||||
open_files: Option<Box<dyn FnMut(Vec<PathBuf>)>>,
|
open_files: Option<Box<dyn FnMut(Vec<PathBuf>)>>,
|
||||||
|
@ -196,6 +201,10 @@ impl platform::ForegroundPlatform for MacForegroundPlatform {
|
||||||
self.0.borrow_mut().resign_active = Some(callback);
|
self.0.borrow_mut().resign_active = Some(callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn on_quit(&self, callback: Box<dyn FnMut()>) {
|
||||||
|
self.0.borrow_mut().quit = Some(callback);
|
||||||
|
}
|
||||||
|
|
||||||
fn on_event(&self, callback: Box<dyn FnMut(crate::Event) -> bool>) {
|
fn on_event(&self, callback: Box<dyn FnMut(crate::Event) -> bool>) {
|
||||||
self.0.borrow_mut().event = Some(callback);
|
self.0.borrow_mut().event = Some(callback);
|
||||||
}
|
}
|
||||||
|
@ -664,6 +673,13 @@ extern "C" fn did_resign_active(this: &mut Object, _: Sel, _: id) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extern "C" fn will_terminate(this: &mut Object, _: Sel, _: id) {
|
||||||
|
let platform = unsafe { get_foreground_platform(this) };
|
||||||
|
if let Some(callback) = platform.0.borrow_mut().quit.as_mut() {
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
extern "C" fn open_files(this: &mut Object, _: Sel, _: id, paths: id) {
|
extern "C" fn open_files(this: &mut Object, _: Sel, _: id, paths: id) {
|
||||||
let paths = unsafe {
|
let paths = unsafe {
|
||||||
(0..paths.count())
|
(0..paths.count())
|
||||||
|
|
|
@ -58,6 +58,8 @@ impl super::ForegroundPlatform for ForegroundPlatform {
|
||||||
|
|
||||||
fn on_resign_active(&self, _: Box<dyn FnMut()>) {}
|
fn on_resign_active(&self, _: Box<dyn FnMut()>) {}
|
||||||
|
|
||||||
|
fn on_quit(&self, _: Box<dyn FnMut()>) {}
|
||||||
|
|
||||||
fn on_event(&self, _: Box<dyn FnMut(crate::Event) -> bool>) {}
|
fn on_event(&self, _: Box<dyn FnMut(crate::Event) -> bool>) {}
|
||||||
|
|
||||||
fn on_open_files(&self, _: Box<dyn FnMut(Vec<std::path::PathBuf>)>) {}
|
fn on_open_files(&self, _: Box<dyn FnMut(Vec<std::path::PathBuf>)>) {}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue