1
0
mirror of https://github.com/emilk/egui.git synced 2026-06-28 07:23:13 -04:00

Make same simplification for the glow integration

This commit is contained in:
Emil Ernerfeldt
2023-11-14 07:06:12 +01:00
parent 2c9426b0ec
commit 274cdea620

View File

@@ -657,7 +657,7 @@ mod glow_integration {
std::thread::sleep(std::time::Duration::from_millis(10));
}
glutin.process_viewport_builders(viewports_out);
glutin.process_viewport_updates(viewports_out, focused_viewport);
for (viewport_id, command) in viewport_commands {
if let Some(viewport) = glutin.viewports.get(&viewport_id) {
@@ -707,29 +707,18 @@ mod glow_integration {
}
struct Viewport {
gl_surface: Option<glutin::surface::Surface<glutin::surface::WindowSurface>>,
window: Option<Rc<Window>>,
ids: ViewportIdPair,
builder: ViewportBuilder,
/// The user-callback that shows the ui.
/// None for immediate viewports.
viewport_ui_cb: Option<Arc<ViewportUiCallback>>,
gl_surface: Option<glutin::surface::Surface<glutin::surface::WindowSurface>>,
window: Option<Rc<Window>>,
egui_winit: Option<egui_winit::State>,
}
impl Viewport {
pub fn new(ids: ViewportIdPair) -> Self {
Self {
gl_surface: None,
window: None,
ids,
viewport_ui_cb: None,
egui_winit: None,
}
}
}
/// This struct will contain both persistent and temporary glutin state.
///
/// Platform Quirks:
@@ -755,14 +744,12 @@ mod glow_integration {
viewports: ViewportIdMap<Viewport>,
viewport_maps: HashMap<WindowId, ViewportId>,
window_maps: ViewportIdMap<WindowId>,
builders: ViewportIdMap<ViewportBuilder>,
}
impl GlutinWindowContext {
#[allow(unsafe_code)]
unsafe fn new(
window_builder: ViewportBuilder,
viewport_builder: ViewportBuilder,
native_options: &epi::NativeOptions,
event_loop: &EventLoopWindowTarget<UserEvent>,
) -> Result<Self> {
@@ -813,7 +800,7 @@ mod glow_integration {
let display_builder = glutin_winit::DisplayBuilder::new()
// we might want to expose this option to users in the future. maybe using an env var or using native_options.
.with_preference(glutin_winit::ApiPrefence::FallbackEgl) // https://github.com/emilk/egui/issues/2520#issuecomment-1367841150
.with_window_builder(Some(create_winit_window_builder(&window_builder)));
.with_window_builder(Some(create_winit_window_builder(&viewport_builder)));
let (window, gl_config) = {
crate::profile_scope!("DisplayBuilder::build");
@@ -883,17 +870,15 @@ mod glow_integration {
viewports.insert(
ViewportId::ROOT,
Viewport {
ids: ViewportIdPair::ROOT,
builder: viewport_builder,
viewport_ui_cb: None,
gl_surface: None,
window: window.map(Rc::new),
egui_winit: None,
viewport_ui_cb: None,
ids: ViewportIdPair::ROOT,
},
);
let mut builders = ViewportIdMap::default();
builders.insert(ViewportId::ROOT, window_builder);
// the fun part with opengl gl is that we never know whether there is an error. the context creation might have failed, but
// it could keep working until we try to make surface current or swap buffers or something else. future glutin improvements might
// help us start from scratch again if we fail context creation and go back to preferEgl or try with different config etc..
@@ -905,7 +890,6 @@ mod glow_integration {
current_gl_context: None,
not_current_gl_context,
viewports,
builders,
viewport_maps,
max_texture_side: None,
window_maps,
@@ -953,14 +937,13 @@ mod glow_integration {
.get_mut(&viewport_id)
.expect("viewport doesn't exist");
let builder = &self.builders[&viewport.ids.this];
// make sure we have a window or create one.
let window = viewport.window.take().unwrap_or_else(|| {
log::debug!("window doesn't exist yet. creating one now with finalize_window");
Rc::new(
glutin_winit::finalize_window(
event_loop,
create_winit_window_builder(builder),
create_winit_window_builder(&viewport.builder),
&self.gl_config,
)
.expect("failed to finalize glutin window"),
@@ -1091,70 +1074,34 @@ mod glow_integration {
self.gl_config.display().get_proc_address(addr)
}
fn process_viewport_builders(&mut self, mut viewports: Vec<ViewportOutput>) {
fn process_viewport_updates(
&mut self,
viewport_output: Vec<ViewportOutput>,
focused_viewport: Option<ViewportId>,
) {
let mut active_viewports_ids = ViewportIdSet::default();
active_viewports_ids.insert(ViewportId::ROOT);
// Process existing viewpors:
viewports.retain_mut(
|ViewportOutput {
builder: new_builder,
ids,
viewport_ui_cb,
}| {
let builder = self.builders.entry(ids.this).or_insert(new_builder.clone());
let (commands, recreate) = builder.patch(new_builder);
if let Some(viewport) = self.viewports.get_mut(&ids.this) {
viewport.ids.parent = ids.parent;
viewport.viewport_ui_cb = viewport_ui_cb.clone(); // always update the latest callback
if recreate {
viewport.window = None;
viewport.gl_surface = None;
} else if let Some(window) = &viewport.window {
let is_viewport_focused = false; // TODO
process_viewport_commands(commands, window, is_viewport_focused);
}
active_viewports_ids.insert(ids.this);
false
} else {
true
}
},
);
// Process new viewports:
for ViewportOutput {
mut builder,
ids,
builder,
viewport_ui_cb,
} in viewports
} in viewport_output
{
if builder.icon.is_none() {
// Inherit from parent as fallback.
builder.icon = self.builders.get(&ids.parent).and_then(|b| b.icon.clone());
}
{
self.viewports.insert(
ids.this,
Viewport {
gl_surface: None,
window: None,
egui_winit: None,
viewport_ui_cb,
ids,
},
);
self.builders.insert(ids.this, builder);
}
active_viewports_ids.insert(ids.this);
initialize_or_update_viewport(
&mut self.viewports,
ids,
builder,
viewport_ui_cb,
focused_viewport,
);
}
// GC old viewports
self.viewports
.retain(|id, _| active_viewports_ids.contains(id));
self.builders
.retain(|id, _| active_viewports_ids.contains(id));
self.viewport_maps
.retain(|_, id| active_viewports_ids.contains(id));
self.window_maps
@@ -1162,6 +1109,61 @@ mod glow_integration {
}
}
fn initialize_or_update_viewport(
viewports: &mut ViewportIdMap<Viewport>,
ids: ViewportIdPair,
mut builder: ViewportBuilder,
viewport_ui_cb: Option<Arc<dyn Fn(&egui::Context) + Send + Sync>>,
focused_viewport: Option<ViewportId>,
) -> &mut Viewport {
if builder.icon.is_none() {
// Inherit icon from parent
builder.icon = viewports
.get_mut(&ids.parent)
.and_then(|vp| vp.builder.icon.clone());
}
match viewports.entry(ids.this) {
std::collections::hash_map::Entry::Vacant(entry) => {
// New viewport:
log::debug!("Creating new viewport {:?} ({:?})", ids.this, builder.title);
entry.insert(Viewport {
ids,
builder,
viewport_ui_cb,
window: None,
egui_winit: None,
gl_surface: None,
})
}
std::collections::hash_map::Entry::Occupied(mut entry) => {
// Patch an existing viewport:
let viewport = entry.get_mut();
viewport.ids.parent = ids.parent;
viewport.viewport_ui_cb = viewport_ui_cb;
let (commands, recreate) = viewport.builder.patch(&builder);
if recreate {
log::debug!(
"Recreating window for viewport {:?} ({:?})",
ids.this,
builder.title
);
viewport.window = None;
viewport.egui_winit = None;
} else if let Some(window) = &viewport.window {
let is_viewport_focused = focused_viewport == Some(ids.this);
process_viewport_commands(commands, window, is_viewport_focused);
}
entry.into_mut()
}
}
}
struct GlowWinitApp {
repaint_proxy: Arc<egui::mutex::Mutex<EventLoopProxy<UserEvent>>>,
app_name: String,
@@ -1410,33 +1412,21 @@ mod glow_integration {
let ImmediateViewport {
ids,
mut builder,
builder,
viewport_ui_cb,
} = immediate_viewport;
if !glutin.borrow().viewports.contains_key(&ids.this) {
// A new viewport - create a window for it!
{
let mut glutin = glutin.borrow_mut();
// Inherit parent icon if none
if builder.icon.is_none() && glutin.builders.get(&ids.this).is_none() {
builder.icon = glutin
.builders
.get(&ids.parent)
.and_then(|b| b.icon.clone());
let viewport =
initialize_or_update_viewport(&mut glutin.viewports, ids, builder, None, None);
if viewport.gl_surface.is_none() {
glutin.init_viewport(ids.this, event_loop).expect(
"Failed to initialize window in egui::Context::show_viewport_immediate",
);
}
glutin
.viewports
.entry(ids.this)
.or_insert(Viewport::new(ids));
glutin.builders.entry(ids.this).or_insert(builder);
glutin
.init_viewport(ids.this, event_loop)
.expect("Failed to initialize window in egui::Context::show_viewport_immediate");
}
let input = {
@@ -1472,7 +1462,6 @@ mod glow_integration {
let GlutinWindowContext {
current_gl_context,
builders,
viewports,
..
} = &mut *glutin;
@@ -1512,8 +1501,7 @@ mod glow_integration {
.unwrap()
.is_current(current_gl_context)
{
let builder = &builders[&viewport.ids.this];
log::error!("egui::show_viewport_immediate with title: `{:?}` is not created in main thread, try to use wgpu!", builder.title.clone().unwrap_or_default());
log::error!("egui::show_viewport_immediate: viewport {:?} ({:?}) is not created in main thread, try to use wgpu!", viewport.ids.this, viewport.builder.title);
}
let gl = &painter.gl().clone();