Skip to content

Commit 657ddc5

Browse files
committed
Break up run_glow into three functions
This will allow us to try out another event loop, one that continues execution after the window closes.
1 parent a827c3e commit 657ddc5

File tree

1 file changed

+193
-105
lines changed

1 file changed

+193
-105
lines changed

eframe/src/native/run.rs

+193-105
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,13 @@
1+
//! Note that this file contains two similar paths - one for [`glow`], one for [`wgpu`].
2+
//! When making changes to one you often also want to apply it to the other.
3+
4+
use std::sync::Arc;
5+
6+
use egui_winit::winit;
7+
use winit::event_loop::ControlFlow;
8+
19
use super::epi_integration;
210
use crate::epi;
3-
use egui_winit::winit;
411

512
struct RequestRepaintEvent;
613

@@ -47,73 +54,107 @@ fn create_display(
4754

4855
pub use epi::NativeOptions;
4956

57+
enum EventResult {
58+
Repaint,
59+
Exit,
60+
Continue,
61+
}
62+
5063
/// Run an egui app
5164
#[cfg(feature = "glow")]
52-
pub fn run_glow(
53-
app_name: &str,
54-
native_options: &epi::NativeOptions,
55-
app_creator: epi::AppCreator,
56-
) -> ! {
57-
let storage = epi_integration::create_storage(app_name);
58-
let window_settings = epi_integration::load_window_settings(storage.as_deref());
59-
let event_loop = winit::event_loop::EventLoop::with_user_event();
60-
61-
let window_builder =
62-
epi_integration::window_builder(native_options, &window_settings).with_title(app_name);
63-
let (gl_window, gl) = create_display(native_options, window_builder, &event_loop);
64-
let gl = std::sync::Arc::new(gl);
65-
66-
let mut painter = egui_glow::Painter::new(gl.clone(), None, "")
67-
.unwrap_or_else(|error| panic!("some OpenGL error occurred {}\n", error));
68-
69-
let system_theme = native_options.system_theme();
70-
let mut integration = epi_integration::EpiIntegration::new(
71-
&event_loop,
72-
painter.max_texture_side(),
73-
gl_window.window(),
74-
system_theme,
75-
storage,
76-
Some(gl.clone()),
77-
#[cfg(feature = "wgpu")]
78-
None,
79-
);
80-
let theme = system_theme.unwrap_or(native_options.default_theme);
81-
integration.egui_ctx.set_visuals(theme.egui_visuals());
82-
83-
{
84-
let event_loop_proxy = egui::mutex::Mutex::new(event_loop.create_proxy());
85-
integration.egui_ctx.set_request_repaint_callback(move || {
86-
event_loop_proxy.lock().send_event(RequestRepaintEvent).ok();
87-
});
65+
mod glow_integration {
66+
use super::*;
67+
68+
struct GlowEframe {
69+
gl_window: glutin::WindowedContext<glutin::PossiblyCurrent>,
70+
gl: Arc<glow::Context>,
71+
painter: egui_glow::Painter,
72+
integration: epi_integration::EpiIntegration,
73+
app: Box<dyn epi::App>,
74+
is_focused: bool,
8875
}
8976

90-
let mut app = app_creator(&epi::CreationContext {
91-
egui_ctx: integration.egui_ctx.clone(),
92-
integration_info: integration.frame.info(),
93-
storage: integration.frame.storage(),
94-
gl: Some(gl.clone()),
95-
#[cfg(feature = "wgpu")]
96-
render_state: None,
97-
});
77+
impl GlowEframe {
78+
fn new(
79+
event_loop: &winit::event_loop::EventLoop<RequestRepaintEvent>,
80+
app_name: &str,
81+
native_options: &epi::NativeOptions,
82+
app_creator: epi::AppCreator,
83+
) -> Self {
84+
let storage = epi_integration::create_storage(app_name);
85+
let window_settings = epi_integration::load_window_settings(storage.as_deref());
86+
87+
let window_builder = epi_integration::window_builder(native_options, &window_settings)
88+
.with_title(app_name);
89+
let (gl_window, gl) = create_display(native_options, window_builder, event_loop);
90+
let gl = Arc::new(gl);
91+
92+
let painter = egui_glow::Painter::new(gl.clone(), None, "")
93+
.unwrap_or_else(|error| panic!("some OpenGL error occurred {}\n", error));
94+
95+
let system_theme = native_options.system_theme();
96+
let mut integration = epi_integration::EpiIntegration::new(
97+
event_loop,
98+
painter.max_texture_side(),
99+
gl_window.window(),
100+
system_theme,
101+
storage,
102+
Some(gl.clone()),
103+
#[cfg(feature = "wgpu")]
104+
None,
105+
);
106+
let theme = system_theme.unwrap_or(native_options.default_theme);
107+
integration.egui_ctx.set_visuals(theme.egui_visuals());
98108

99-
if app.warm_up_enabled() {
100-
integration.warm_up(app.as_mut(), gl_window.window());
101-
}
109+
{
110+
let event_loop_proxy = egui::mutex::Mutex::new(event_loop.create_proxy());
111+
integration.egui_ctx.set_request_repaint_callback(move || {
112+
event_loop_proxy.lock().send_event(RequestRepaintEvent).ok();
113+
});
114+
}
102115

103-
let mut is_focused = true;
116+
let mut app = app_creator(&epi::CreationContext {
117+
egui_ctx: integration.egui_ctx.clone(),
118+
integration_info: integration.frame.info(),
119+
storage: integration.frame.storage(),
120+
gl: Some(gl.clone()),
121+
#[cfg(feature = "wgpu")]
122+
render_state: None,
123+
});
124+
125+
if app.warm_up_enabled() {
126+
integration.warm_up(app.as_mut(), gl_window.window());
127+
}
104128

105-
event_loop.run(move |event, _, control_flow| {
106-
let window = gl_window.window();
129+
Self {
130+
gl_window,
131+
gl,
132+
painter,
133+
integration,
134+
app,
135+
is_focused: true,
136+
}
137+
}
107138

108-
let mut redraw = || {
139+
fn paint(&mut self) -> ControlFlow {
109140
#[cfg(feature = "puffin")]
110141
puffin::GlobalProfiler::lock().new_frame();
111142
crate::profile_scope!("frame");
112143

144+
let Self {
145+
gl_window,
146+
gl,
147+
app,
148+
integration,
149+
painter,
150+
..
151+
} = self;
152+
let window = gl_window.window();
153+
113154
let screen_size_in_pixels: [u32; 2] = window.inner_size().into();
114155

115156
egui_glow::painter::clear(
116-
&gl,
157+
gl,
117158
screen_size_in_pixels,
118159
app.clear_color(&integration.egui_ctx.style().visuals),
119160
);
@@ -146,11 +187,11 @@ pub fn run_glow(
146187
gl_window.swap_buffers().unwrap();
147188
}
148189

149-
*control_flow = if integration.should_quit() {
150-
winit::event_loop::ControlFlow::Exit
190+
let control_flow = if integration.should_quit() {
191+
ControlFlow::Exit
151192
} else if repaint_after.is_zero() {
152193
window.request_redraw();
153-
winit::event_loop::ControlFlow::Poll
194+
ControlFlow::Poll
154195
} else if let Some(repaint_after_instant) =
155196
std::time::Instant::now().checked_add(repaint_after)
156197
{
@@ -159,14 +200,14 @@ pub fn run_glow(
159200
// technically, this might lead to some weird corner cases where the user *WANTS*
160201
// winit to use `WaitUntil(MAX_INSTANT)` explicitly. they can roll their own
161202
// egui backend impl i guess.
162-
winit::event_loop::ControlFlow::WaitUntil(repaint_after_instant)
203+
ControlFlow::WaitUntil(repaint_after_instant)
163204
} else {
164-
winit::event_loop::ControlFlow::Wait
205+
ControlFlow::Wait
165206
};
166207

167208
integration.maybe_autosave(app.as_mut(), window);
168209

169-
if !is_focused {
210+
if !self.is_focused {
170211
// On Mac, a minimized Window uses up all CPU: https://github.com/emilk/egui/issues/325
171212
// We can't know if we are minimized: https://github.com/rust-windowing/winit/issues/208
172213
// But we know if we are focused (in foreground). When minimized, we are not focused.
@@ -175,58 +216,105 @@ pub fn run_glow(
175216
crate::profile_scope!("bg_sleep");
176217
std::thread::sleep(std::time::Duration::from_millis(10));
177218
}
178-
};
179-
match event {
180-
// Platform-dependent event handlers to workaround a winit bug
181-
// See: https://github.com/rust-windowing/winit/issues/987
182-
// See: https://github.com/rust-windowing/winit/issues/1619
183-
winit::event::Event::RedrawEventsCleared if cfg!(windows) => redraw(),
184-
winit::event::Event::RedrawRequested(_) if !cfg!(windows) => redraw(),
185219

186-
winit::event::Event::WindowEvent { event, .. } => {
187-
match &event {
188-
winit::event::WindowEvent::Focused(new_focused) => {
189-
is_focused = *new_focused;
190-
}
191-
winit::event::WindowEvent::Resized(physical_size) => {
192-
// Resize with 0 width and height is used by winit to signal a minimize event on Windows.
193-
// See: https://github.com/rust-windowing/winit/issues/208
194-
// This solves an issue where the app would panic when minimizing on Windows.
195-
if physical_size.width > 0 && physical_size.height > 0 {
196-
gl_window.resize(*physical_size);
220+
control_flow
221+
}
222+
223+
fn on_event(&mut self, event: winit::event::Event<'_, RequestRepaintEvent>) -> EventResult {
224+
let Self {
225+
gl_window,
226+
gl,
227+
integration,
228+
painter,
229+
..
230+
} = self;
231+
let window = gl_window.window();
232+
233+
match event {
234+
// Platform-dependent event handlers to workaround a winit bug
235+
// See: https://github.com/rust-windowing/winit/issues/987
236+
// See: https://github.com/rust-windowing/winit/issues/1619
237+
winit::event::Event::RedrawEventsCleared if cfg!(windows) => EventResult::Repaint,
238+
winit::event::Event::RedrawRequested(_) if !cfg!(windows) => EventResult::Repaint,
239+
240+
winit::event::Event::WindowEvent { event, .. } => {
241+
match &event {
242+
winit::event::WindowEvent::Focused(new_focused) => {
243+
self.is_focused = *new_focused;
197244
}
245+
winit::event::WindowEvent::Resized(physical_size) => {
246+
// Resize with 0 width and height is used by winit to signal a minimize event on Windows.
247+
// See: https://github.com/rust-windowing/winit/issues/208
248+
// This solves an issue where the app would panic when minimizing on Windows.
249+
if physical_size.width > 0 && physical_size.height > 0 {
250+
gl_window.resize(*physical_size);
251+
}
252+
}
253+
winit::event::WindowEvent::ScaleFactorChanged {
254+
new_inner_size, ..
255+
} => {
256+
gl_window.resize(**new_inner_size);
257+
}
258+
winit::event::WindowEvent::CloseRequested if integration.should_quit() => {
259+
return EventResult::Exit
260+
}
261+
_ => {}
198262
}
199-
winit::event::WindowEvent::ScaleFactorChanged { new_inner_size, .. } => {
200-
gl_window.resize(**new_inner_size);
201-
}
202-
winit::event::WindowEvent::CloseRequested if integration.should_quit() => {
203-
*control_flow = winit::event_loop::ControlFlow::Exit;
263+
264+
integration.on_event(self.app.as_mut(), &event);
265+
if integration.should_quit() {
266+
EventResult::Exit
267+
} else {
268+
window.request_redraw(); // TODO(emilk): ask egui if the events warrants a repaint instead
269+
EventResult::Continue
204270
}
205-
_ => {}
206271
}
207-
208-
integration.on_event(app.as_mut(), &event);
209-
if integration.should_quit() {
210-
*control_flow = winit::event_loop::ControlFlow::Exit;
272+
winit::event::Event::LoopDestroyed => {
273+
integration.save(&mut *self.app, window);
274+
self.app.on_exit(Some(gl));
275+
painter.destroy();
276+
EventResult::Continue
211277
}
212-
window.request_redraw(); // TODO(emilk): ask egui if the events warrants a repaint instead
213-
}
214-
winit::event::Event::LoopDestroyed => {
215-
integration.save(&mut *app, window);
216-
app.on_exit(Some(&gl));
217-
painter.destroy();
218-
}
219-
winit::event::Event::UserEvent(RequestRepaintEvent) => window.request_redraw(),
220-
winit::event::Event::NewEvents(winit::event::StartCause::ResumeTimeReached {
221-
..
222-
}) => {
223-
window.request_redraw();
278+
winit::event::Event::UserEvent(RequestRepaintEvent)
279+
| winit::event::Event::NewEvents(winit::event::StartCause::ResumeTimeReached {
280+
..
281+
}) => {
282+
window.request_redraw();
283+
EventResult::Continue
284+
}
285+
_ => EventResult::Continue,
224286
}
225-
_ => {}
226287
}
227-
});
288+
}
289+
290+
pub fn run_glow(
291+
app_name: &str,
292+
native_options: &epi::NativeOptions,
293+
app_creator: epi::AppCreator,
294+
) -> ! {
295+
let event_loop = winit::event_loop::EventLoop::with_user_event();
296+
let mut glow_eframe = GlowEframe::new(&event_loop, app_name, native_options, app_creator);
297+
298+
event_loop.run(move |event, _, control_flow| {
299+
let event_result = glow_eframe.on_event(event);
300+
match event_result {
301+
EventResult::Continue => {}
302+
EventResult::Repaint => {
303+
*control_flow = glow_eframe.paint();
304+
}
305+
EventResult::Exit => {
306+
*control_flow = ControlFlow::Exit;
307+
}
308+
}
309+
})
310+
}
228311
}
229312

313+
#[cfg(feature = "glow")]
314+
pub use glow_integration::run_glow;
315+
316+
// ----------------------------------------------------------------------------
317+
230318
// TODO(emilk): merge with with the clone above
231319
/// Run an egui app
232320
#[cfg(feature = "wgpu")]
@@ -331,10 +419,10 @@ pub fn run_wgpu(
331419
);
332420

333421
*control_flow = if integration.should_quit() {
334-
winit::event_loop::ControlFlow::Exit
422+
ControlFlow::Exit
335423
} else if repaint_after.is_zero() {
336424
window.request_redraw();
337-
winit::event_loop::ControlFlow::Poll
425+
ControlFlow::Poll
338426
} else if let Some(repaint_after_instant) =
339427
std::time::Instant::now().checked_add(repaint_after)
340428
{
@@ -343,9 +431,9 @@ pub fn run_wgpu(
343431
// technically, this might lead to some weird corner cases where the user *WANTS*
344432
// winit to use `WaitUntil(MAX_INSTANT)` explicitly. they can roll their own
345433
// egui backend impl i guess.
346-
winit::event_loop::ControlFlow::WaitUntil(repaint_after_instant)
434+
ControlFlow::WaitUntil(repaint_after_instant)
347435
} else {
348-
winit::event_loop::ControlFlow::Wait
436+
ControlFlow::Wait
349437
};
350438

351439
integration.maybe_autosave(app.as_mut(), window);

0 commit comments

Comments
 (0)