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
+
1
9
use super :: epi_integration;
2
10
use crate :: epi;
3
- use egui_winit:: winit;
4
11
5
12
struct RequestRepaintEvent ;
6
13
@@ -47,73 +54,107 @@ fn create_display(
47
54
48
55
pub use epi:: NativeOptions ;
49
56
57
+ enum EventResult {
58
+ Repaint ,
59
+ Exit ,
60
+ Continue ,
61
+ }
62
+
50
63
/// Run an egui app
51
64
#[ 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 ,
88
75
}
89
76
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 ( ) ) ;
98
108
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
+ }
102
115
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
+ }
104
128
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
+ }
107
138
108
- let mut redraw = || {
139
+ fn paint ( & mut self ) -> ControlFlow {
109
140
#[ cfg( feature = "puffin" ) ]
110
141
puffin:: GlobalProfiler :: lock ( ) . new_frame ( ) ;
111
142
crate :: profile_scope!( "frame" ) ;
112
143
144
+ let Self {
145
+ gl_window,
146
+ gl,
147
+ app,
148
+ integration,
149
+ painter,
150
+ ..
151
+ } = self ;
152
+ let window = gl_window. window ( ) ;
153
+
113
154
let screen_size_in_pixels: [ u32 ; 2 ] = window. inner_size ( ) . into ( ) ;
114
155
115
156
egui_glow:: painter:: clear (
116
- & gl,
157
+ gl,
117
158
screen_size_in_pixels,
118
159
app. clear_color ( & integration. egui_ctx . style ( ) . visuals ) ,
119
160
) ;
@@ -146,11 +187,11 @@ pub fn run_glow(
146
187
gl_window. swap_buffers ( ) . unwrap ( ) ;
147
188
}
148
189
149
- * control_flow = if integration. should_quit ( ) {
150
- winit :: event_loop :: ControlFlow :: Exit
190
+ let control_flow = if integration. should_quit ( ) {
191
+ ControlFlow :: Exit
151
192
} else if repaint_after. is_zero ( ) {
152
193
window. request_redraw ( ) ;
153
- winit :: event_loop :: ControlFlow :: Poll
194
+ ControlFlow :: Poll
154
195
} else if let Some ( repaint_after_instant) =
155
196
std:: time:: Instant :: now ( ) . checked_add ( repaint_after)
156
197
{
@@ -159,14 +200,14 @@ pub fn run_glow(
159
200
// technically, this might lead to some weird corner cases where the user *WANTS*
160
201
// winit to use `WaitUntil(MAX_INSTANT)` explicitly. they can roll their own
161
202
// egui backend impl i guess.
162
- winit :: event_loop :: ControlFlow :: WaitUntil ( repaint_after_instant)
203
+ ControlFlow :: WaitUntil ( repaint_after_instant)
163
204
} else {
164
- winit :: event_loop :: ControlFlow :: Wait
205
+ ControlFlow :: Wait
165
206
} ;
166
207
167
208
integration. maybe_autosave ( app. as_mut ( ) , window) ;
168
209
169
- if !is_focused {
210
+ if !self . is_focused {
170
211
// On Mac, a minimized Window uses up all CPU: https://github.com/emilk/egui/issues/325
171
212
// We can't know if we are minimized: https://github.com/rust-windowing/winit/issues/208
172
213
// But we know if we are focused (in foreground). When minimized, we are not focused.
@@ -175,58 +216,105 @@ pub fn run_glow(
175
216
crate :: profile_scope!( "bg_sleep" ) ;
176
217
std:: thread:: sleep ( std:: time:: Duration :: from_millis ( 10 ) ) ;
177
218
}
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 ( ) ,
185
219
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;
197
244
}
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
+ _ => { }
198
262
}
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
204
270
}
205
- _ => { }
206
271
}
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
211
277
}
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 ,
224
286
}
225
- _ => { }
226
287
}
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
+ }
228
311
}
229
312
313
+ #[ cfg( feature = "glow" ) ]
314
+ pub use glow_integration:: run_glow;
315
+
316
+ // ----------------------------------------------------------------------------
317
+
230
318
// TODO(emilk): merge with with the clone above
231
319
/// Run an egui app
232
320
#[ cfg( feature = "wgpu" ) ]
@@ -331,10 +419,10 @@ pub fn run_wgpu(
331
419
) ;
332
420
333
421
* control_flow = if integration. should_quit ( ) {
334
- winit :: event_loop :: ControlFlow :: Exit
422
+ ControlFlow :: Exit
335
423
} else if repaint_after. is_zero ( ) {
336
424
window. request_redraw ( ) ;
337
- winit :: event_loop :: ControlFlow :: Poll
425
+ ControlFlow :: Poll
338
426
} else if let Some ( repaint_after_instant) =
339
427
std:: time:: Instant :: now ( ) . checked_add ( repaint_after)
340
428
{
@@ -343,9 +431,9 @@ pub fn run_wgpu(
343
431
// technically, this might lead to some weird corner cases where the user *WANTS*
344
432
// winit to use `WaitUntil(MAX_INSTANT)` explicitly. they can roll their own
345
433
// egui backend impl i guess.
346
- winit :: event_loop :: ControlFlow :: WaitUntil ( repaint_after_instant)
434
+ ControlFlow :: WaitUntil ( repaint_after_instant)
347
435
} else {
348
- winit :: event_loop :: ControlFlow :: Wait
436
+ ControlFlow :: Wait
349
437
} ;
350
438
351
439
integration. maybe_autosave ( app. as_mut ( ) , window) ;
0 commit comments