Skip to content

Commit d69c549

Browse files
committed
Allow aborting an exit event
1 parent 87ac744 commit d69c549

File tree

4 files changed

+79
-4
lines changed

4 files changed

+79
-4
lines changed

eframe/CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ NOTE: [`egui_web`](egui_web/CHANGELOG.md), [`egui-winit`](egui-winit/CHANGELOG.m
99
* The default native backend is now `egui_glow` (instead of `egui_glium`) ([#1020](https://github.com/emilk/egui/pull/1020)).
1010
* The default web painter is now `egui_glow` (instead of WebGL) ([#1020](https://github.com/emilk/egui/pull/1020)).
1111
* Fix horizontal scrolling direction on Linux.
12+
* Added `App::on_exit_event` ([#1038](https://github.com/emilk/egui/pull/1038))
1213

1314

1415
## 0.16.0 - 2021-12-29

eframe/examples/confirm_exit.rs

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")] // hide console window on Windows in release
2+
3+
use eframe::{egui, epi};
4+
5+
#[derive(Default)]
6+
struct MyApp {
7+
can_exit: bool,
8+
is_exiting: bool,
9+
}
10+
11+
impl epi::App for MyApp {
12+
fn name(&self) -> &str {
13+
"Confirm exit"
14+
}
15+
16+
fn on_exit_event(&mut self) -> bool {
17+
self.is_exiting = true;
18+
self.can_exit
19+
}
20+
21+
fn update(&mut self, ctx: &egui::Context, frame: &epi::Frame) {
22+
egui::CentralPanel::default().show(ctx, |ui| {
23+
ui.heading("Try to close the window");
24+
});
25+
26+
if self.is_exiting {
27+
egui::Window::new("Do you want to quit?")
28+
.collapsible(false)
29+
.resizable(false)
30+
.show(ctx, |ui| {
31+
ui.horizontal(|ui| {
32+
if ui.button("Yes!").clicked() {
33+
self.can_exit = true;
34+
frame.quit();
35+
}
36+
37+
if ui.button("Not yet").clicked() {
38+
self.is_exiting = false;
39+
}
40+
});
41+
});
42+
}
43+
}
44+
}
45+
46+
fn main() {
47+
let options = eframe::NativeOptions::default();
48+
eframe::run_native(Box::new(MyApp::default()), options);
49+
}

egui-winit/src/epi.rs

+15-3
Original file line numberDiff line numberDiff line change
@@ -240,7 +240,11 @@ impl EpiIntegration {
240240
self.app
241241
.setup(&self.egui_ctx, &self.frame, self.persistence.storage());
242242
let app_output = self.frame.take_app_output();
243-
self.quit |= app_output.quit;
243+
244+
if app_output.quit {
245+
self.quit = self.app.on_exit_event();
246+
}
247+
244248
crate::epi::handle_app_output(window, self.egui_ctx.pixels_per_point(), app_output);
245249
}
246250

@@ -260,7 +264,12 @@ impl EpiIntegration {
260264

261265
pub fn on_event(&mut self, event: &winit::event::WindowEvent<'_>) {
262266
use winit::event::WindowEvent;
263-
self.quit |= matches!(event, WindowEvent::CloseRequested | WindowEvent::Destroyed);
267+
if *event == WindowEvent::CloseRequested {
268+
self.quit = self.app.on_exit_event();
269+
} else if *event == WindowEvent::Destroyed {
270+
self.quit = true;
271+
}
272+
264273
self.egui_winit.on_event(&self.egui_ctx, event);
265274
}
266275

@@ -282,7 +291,10 @@ impl EpiIntegration {
282291
.handle_output(window, &self.egui_ctx, egui_output);
283292

284293
let app_output = self.frame.take_app_output();
285-
self.quit |= app_output.quit;
294+
295+
if app_output.quit {
296+
self.quit = self.app.on_exit_event();
297+
}
286298

287299
crate::epi::handle_app_output(window, self.egui_ctx.pixels_per_point(), app_output);
288300

epi/src/lib.rs

+14-1
Original file line numberDiff line numberDiff line change
@@ -144,7 +144,20 @@ pub trait App {
144144
/// where `APPNAME` is what is returned by [`Self::name()`].
145145
fn save(&mut self, _storage: &mut dyn Storage) {}
146146

147-
/// Called once on shutdown (before or after [`Self::save`])
147+
/// Called before an exit that can be aborted.
148+
/// By returning `false` the exit will be aborted. To continue the exit return `true`.
149+
///
150+
/// A scenario where this method will be run is after pressing the close button on a native
151+
/// window, which allows you to ask the user whether they want to do something before exiting.
152+
/// See the example `eframe/examples/confirm_exit.rs` for practical usage.
153+
///
154+
/// It will _not_ be called on the web or when the window is forcefully closed.
155+
fn on_exit_event(&mut self) -> bool {
156+
true
157+
}
158+
159+
/// Called once on shutdown (before or after [`Self::save`]). If you need to abort an exit use
160+
/// [`Self::on_exit_event`]
148161
fn on_exit(&mut self) {}
149162

150163
// ---------

0 commit comments

Comments
 (0)