mirror of
https://github.com/emilk/egui.git
synced 2026-06-27 23:13:13 -04:00
I love egui! Thank you Emil <3 This request specifically enables an `eframe::App` which stores a lifetime. In general, I believe this is necessary because `eframe::App` currently does not seem to provide a good place to allocate and then borrow from long-lived data between `update()` calls. To attempt to borrow such long-lived data from a field of the `App` itself would be to create a self-referential struct. A hacky alternative is to allocate long-lived data with `Box::leak`, but that's a code smell and would cause problems if a program ever creates multiple Apps. As a more specific motivating example, I am developing with the [inkwell](https://github.com/TheDan64/inkwell/) crate which requires creating a `inkwell::context::Context` instance which is then borrowed from by a bazillion things with a dedicated `'ctx` lifetime. I need such a `inkwell::context::Context` for the duration of my `eframe::App` but I can't store it as a field of the app. The most natural solution to me is to simply to lift the inkwell context outside of the App and borrow from it, but that currently fails because the AppCreator implicitly has a `'static` lifetime requirement due to the use of `dyn` trait objects. Here is a simpler, self-contained motivating example adapted from the current [hello world example](https://docs.rs/eframe/latest/eframe/): ```rust use eframe::egui; struct LongLivedThing { message: String, } fn main() { let long_lived_thing = LongLivedThing { message: "Hello World!".to_string(), }; let native_options = eframe::NativeOptions::default(); eframe::run_native( "My egui App", native_options, Box::new(|cc| Ok(Box::new(MyEguiApp::new(cc, &long_lived_thing)))), // ^^^^^^^^^^^^^^^^^ // BORROWING from long_lived_thing in App ); } struct MyEguiApp<'a> { long_lived_thing: &'a LongLivedThing, } impl<'a> MyEguiApp<'a> { fn new(cc: &eframe::CreationContext<'_>, long_lived_thing: &'a LongLivedThing) -> Self { // Customize egui here with cc.egui_ctx.set_fonts and cc.egui_ctx.set_visuals. // Restore app state using cc.storage (requires the "persistence" feature). // Use the cc.gl (a glow::Context) to create graphics shaders and buffers that you can use // for e.g. egui::PaintCallback. MyEguiApp { long_lived_thing } } } impl<'a> eframe::App for MyEguiApp<'a> { fn update(&mut self, ctx: &egui::Context, frame: &mut eframe::Frame) { egui::CentralPanel::default().show(ctx, |ui| { ui.heading(&self.long_lived_thing.message); }); } } ``` This currently fails to compile with: ```plaintext error[E0597]: `long_lived_thing` does not live long enough --> src/main.rs:16:55 | 16 | Box::new(|cc| Ok(Box::new(MyEguiApp::new(cc, &long_lived_thing)))), | ----------------------------------------------^^^^^^^^^^^^^^^^---- | | | | | | | borrowed value does not live long enough | | value captured here | cast requires that `long_lived_thing` is borrowed for `'static` 17 | ); 18 | } | - `long_lived_thing` dropped here while still borrowed | = note: due to object lifetime defaults, `Box<dyn for<'a, 'b> FnOnce(&'a CreationContext<'b>) -> Result<Box<dyn App>, Box<dyn std::error::Error + Send + Sync>>>` actually means `Box<(dyn for<'a, 'b> FnOnce(&'a CreationContext<'b>) -> Result<Box<dyn App>, Box<dyn std::error::Error + Send + Sync>> + 'static)>` ``` With the added lifetimes in this request, I'm able to compile and run this as expected on Ubuntu + Wayland. I see the CI has been emailing me about some build failures and I'll do what I can to address those. Currently running the check.sh script as well. This is intended to resolve https://github.com/emilk/egui/issues/2152 <!-- Please read the "Making a PR" section of [`CONTRIBUTING.md`](https://github.com/emilk/egui/blob/master/CONTRIBUTING.md) before opening a Pull Request! * Keep your PR:s small and focused. * The PR title is what ends up in the changelog, so make it descriptive! * If applicable, add a screenshot or gif. * If it is a non-trivial addition, consider adding a demo for it to `egui_demo_lib`, or a new example. * Do NOT open PR:s from your `master` branch, as that makes it hard for maintainers to test and add commits to your PR. * Remember to run `cargo fmt` and `cargo clippy`. * Open the PR as a draft until you have self-reviewed it and run `./scripts/check.sh`. * When you have addressed a PR comment, mark it as resolved. Please be patient! I will review your PR, but my time is limited! --> * Closes https://github.com/emilk/egui/issues/2152 * [x] I have followed the instructions in the PR template
323 lines
10 KiB
Rust
323 lines
10 KiB
Rust
use egui::TexturesDelta;
|
|
|
|
use crate::{epi, App};
|
|
|
|
use super::{now_sec, text_agent::TextAgent, web_painter::WebPainter, NeedRepaint};
|
|
|
|
pub struct AppRunner {
|
|
#[allow(dead_code)]
|
|
web_options: crate::WebOptions,
|
|
pub(crate) frame: epi::Frame,
|
|
egui_ctx: egui::Context,
|
|
painter: super::ActiveWebPainter,
|
|
pub(crate) input: super::WebInput,
|
|
app: Box<dyn epi::App>,
|
|
pub(crate) needs_repaint: std::sync::Arc<NeedRepaint>,
|
|
last_save_time: f64,
|
|
pub(crate) text_agent: TextAgent,
|
|
|
|
// Output for the last run:
|
|
textures_delta: TexturesDelta,
|
|
clipped_primitives: Option<Vec<egui::ClippedPrimitive>>,
|
|
}
|
|
|
|
impl Drop for AppRunner {
|
|
fn drop(&mut self) {
|
|
log::debug!("AppRunner has fully dropped");
|
|
}
|
|
}
|
|
|
|
impl AppRunner {
|
|
/// # Errors
|
|
/// Failure to initialize WebGL renderer, or failure to create app.
|
|
pub async fn new(
|
|
canvas: web_sys::HtmlCanvasElement,
|
|
web_options: crate::WebOptions,
|
|
app_creator: epi::AppCreator<'static>,
|
|
text_agent: TextAgent,
|
|
) -> Result<Self, String> {
|
|
let painter = super::ActiveWebPainter::new(canvas, &web_options).await?;
|
|
|
|
let info = epi::IntegrationInfo {
|
|
web_info: epi::WebInfo {
|
|
user_agent: super::user_agent().unwrap_or_default(),
|
|
location: super::web_location(),
|
|
},
|
|
cpu_usage: None,
|
|
};
|
|
let storage = LocalStorage::default();
|
|
|
|
let egui_ctx = egui::Context::default();
|
|
egui_ctx.set_os(egui::os::OperatingSystem::from_user_agent(
|
|
&super::user_agent().unwrap_or_default(),
|
|
));
|
|
super::storage::load_memory(&egui_ctx);
|
|
|
|
egui_ctx.options_mut(|o| {
|
|
// On web by default egui follows the zoom factor of the browser,
|
|
// and lets the browser handle the zoom shortscuts.
|
|
// A user can still zoom egui separately by calling [`egui::Context::set_zoom_factor`].
|
|
o.zoom_with_keyboard = false;
|
|
o.zoom_factor = 1.0;
|
|
});
|
|
|
|
let cc = epi::CreationContext {
|
|
egui_ctx: egui_ctx.clone(),
|
|
integration_info: info.clone(),
|
|
storage: Some(&storage),
|
|
|
|
#[cfg(feature = "glow")]
|
|
gl: Some(painter.gl().clone()),
|
|
|
|
#[cfg(feature = "glow")]
|
|
get_proc_address: None,
|
|
|
|
#[cfg(all(feature = "wgpu", not(feature = "glow")))]
|
|
wgpu_render_state: painter.render_state(),
|
|
#[cfg(all(feature = "wgpu", feature = "glow"))]
|
|
wgpu_render_state: None,
|
|
};
|
|
let app = app_creator(&cc).map_err(|err| err.to_string())?;
|
|
|
|
let frame = epi::Frame {
|
|
info,
|
|
storage: Some(Box::new(storage)),
|
|
|
|
#[cfg(feature = "glow")]
|
|
gl: Some(painter.gl().clone()),
|
|
|
|
#[cfg(all(feature = "wgpu", not(feature = "glow")))]
|
|
wgpu_render_state: painter.render_state(),
|
|
#[cfg(all(feature = "wgpu", feature = "glow"))]
|
|
wgpu_render_state: None,
|
|
};
|
|
|
|
let needs_repaint: std::sync::Arc<NeedRepaint> = Default::default();
|
|
{
|
|
let needs_repaint = needs_repaint.clone();
|
|
egui_ctx.set_request_repaint_callback(move |info| {
|
|
needs_repaint.repaint_after(info.delay.as_secs_f64());
|
|
});
|
|
}
|
|
|
|
let mut runner = Self {
|
|
web_options,
|
|
frame,
|
|
egui_ctx,
|
|
painter,
|
|
input: Default::default(),
|
|
app,
|
|
needs_repaint,
|
|
last_save_time: now_sec(),
|
|
text_agent,
|
|
textures_delta: Default::default(),
|
|
clipped_primitives: None,
|
|
};
|
|
|
|
runner.input.raw.max_texture_side = Some(runner.painter.max_texture_side());
|
|
runner
|
|
.input
|
|
.raw
|
|
.viewports
|
|
.entry(egui::ViewportId::ROOT)
|
|
.or_default()
|
|
.native_pixels_per_point = Some(super::native_pixels_per_point());
|
|
runner.input.raw.system_theme = super::system_theme();
|
|
|
|
Ok(runner)
|
|
}
|
|
|
|
pub fn egui_ctx(&self) -> &egui::Context {
|
|
&self.egui_ctx
|
|
}
|
|
|
|
/// Get mutable access to the concrete [`App`] we enclose.
|
|
///
|
|
/// This will panic if your app does not implement [`App::as_any_mut`].
|
|
pub fn app_mut<ConcreteApp: 'static + App>(&mut self) -> &mut ConcreteApp {
|
|
self.app
|
|
.as_any_mut()
|
|
.expect("Your app must implement `as_any_mut`, but it doesn't")
|
|
.downcast_mut::<ConcreteApp>()
|
|
.expect("app_mut got the wrong type of App")
|
|
}
|
|
|
|
pub fn auto_save_if_needed(&mut self) {
|
|
let time_since_last_save = now_sec() - self.last_save_time;
|
|
if time_since_last_save > self.app.auto_save_interval().as_secs_f64() {
|
|
self.save();
|
|
}
|
|
}
|
|
|
|
pub fn save(&mut self) {
|
|
if self.app.persist_egui_memory() {
|
|
super::storage::save_memory(&self.egui_ctx);
|
|
}
|
|
if let Some(storage) = self.frame.storage_mut() {
|
|
self.app.save(storage);
|
|
}
|
|
self.last_save_time = now_sec();
|
|
}
|
|
|
|
pub fn canvas(&self) -> &web_sys::HtmlCanvasElement {
|
|
self.painter.canvas()
|
|
}
|
|
|
|
pub fn destroy(mut self) {
|
|
log::debug!("Destroying AppRunner");
|
|
self.painter.destroy();
|
|
}
|
|
|
|
pub fn has_outstanding_paint_data(&self) -> bool {
|
|
self.clipped_primitives.is_some()
|
|
}
|
|
|
|
/// Does the eframe app have focus?
|
|
///
|
|
/// Technically: does either the canvas or the [`TextAgent`] have focus?
|
|
pub fn has_focus(&self) -> bool {
|
|
super::has_focus(self.canvas()) || self.text_agent.has_focus()
|
|
}
|
|
|
|
pub fn update_focus(&mut self) {
|
|
let has_focus = self.has_focus();
|
|
if self.input.raw.focused != has_focus {
|
|
log::trace!("{} Focus changed to {has_focus}", self.canvas().id());
|
|
self.input.set_focus(has_focus);
|
|
|
|
if !has_focus {
|
|
// We lost focus - good idea to save
|
|
self.save();
|
|
}
|
|
self.egui_ctx().request_repaint();
|
|
}
|
|
}
|
|
|
|
/// Runs the logic, but doesn't paint the result.
|
|
///
|
|
/// The result can be painted later with a call to [`Self::run_and_paint`] or [`Self::paint`].
|
|
pub fn logic(&mut self) {
|
|
// We sometimes miss blur/focus events due to the text agent, so let's just poll each frame:
|
|
self.update_focus();
|
|
|
|
let canvas_size = super::canvas_size_in_points(self.canvas(), self.egui_ctx());
|
|
let mut raw_input = self.input.new_frame(canvas_size);
|
|
|
|
self.app.raw_input_hook(&self.egui_ctx, &mut raw_input);
|
|
|
|
let full_output = self.egui_ctx.run(raw_input, |egui_ctx| {
|
|
self.app.update(egui_ctx, &mut self.frame);
|
|
});
|
|
let egui::FullOutput {
|
|
platform_output,
|
|
textures_delta,
|
|
shapes,
|
|
pixels_per_point,
|
|
viewport_output,
|
|
} = full_output;
|
|
|
|
if viewport_output.len() > 1 {
|
|
log::warn!("Multiple viewports not yet supported on the web");
|
|
}
|
|
for viewport_output in viewport_output.values() {
|
|
for command in &viewport_output.commands {
|
|
// TODO(emilk): handle some of the commands
|
|
log::warn!(
|
|
"Unhandled egui viewport command: {command:?} - not implemented in web backend"
|
|
);
|
|
}
|
|
}
|
|
|
|
self.handle_platform_output(platform_output);
|
|
self.textures_delta.append(textures_delta);
|
|
self.clipped_primitives = Some(self.egui_ctx.tessellate(shapes, pixels_per_point));
|
|
}
|
|
|
|
/// Paint the results of the last call to [`Self::logic`].
|
|
pub fn paint(&mut self) {
|
|
let textures_delta = std::mem::take(&mut self.textures_delta);
|
|
let clipped_primitives = std::mem::take(&mut self.clipped_primitives);
|
|
|
|
if let Some(clipped_primitives) = clipped_primitives {
|
|
if let Err(err) = self.painter.paint_and_update_textures(
|
|
self.app.clear_color(&self.egui_ctx.style().visuals),
|
|
&clipped_primitives,
|
|
self.egui_ctx.pixels_per_point(),
|
|
&textures_delta,
|
|
) {
|
|
log::error!("Failed to paint: {}", super::string_from_js_value(&err));
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn report_frame_time(&mut self, cpu_usage_seconds: f32) {
|
|
self.frame.info.cpu_usage = Some(cpu_usage_seconds);
|
|
}
|
|
|
|
fn handle_platform_output(&mut self, platform_output: egui::PlatformOutput) {
|
|
#[cfg(feature = "web_screen_reader")]
|
|
if self.egui_ctx.options(|o| o.screen_reader) {
|
|
super::screen_reader::speak(&platform_output.events_description());
|
|
}
|
|
|
|
let egui::PlatformOutput {
|
|
cursor_icon,
|
|
open_url,
|
|
copied_text,
|
|
events: _, // already handled
|
|
mutable_text_under_cursor: _, // TODO(#4569): https://github.com/emilk/egui/issues/4569
|
|
ime,
|
|
#[cfg(feature = "accesskit")]
|
|
accesskit_update: _, // not currently implemented
|
|
} = platform_output;
|
|
|
|
super::set_cursor_icon(cursor_icon);
|
|
if let Some(open) = open_url {
|
|
super::open_url(&open.url, open.new_tab);
|
|
}
|
|
|
|
if !copied_text.is_empty() {
|
|
super::set_clipboard_text(&copied_text);
|
|
}
|
|
|
|
if self.has_focus() {
|
|
// The eframe app has focus.
|
|
if ime.is_some() {
|
|
// We are editing text: give the focus to the text agent.
|
|
self.text_agent.focus();
|
|
} else {
|
|
// We are not editing text - give the focus to the canvas.
|
|
self.text_agent.blur();
|
|
self.canvas().focus().ok();
|
|
}
|
|
}
|
|
|
|
if let Err(err) = self
|
|
.text_agent
|
|
.move_to(ime, self.canvas(), self.egui_ctx.zoom_factor())
|
|
{
|
|
log::error!(
|
|
"failed to update text agent position: {}",
|
|
super::string_from_js_value(&err)
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
#[derive(Default)]
|
|
struct LocalStorage {}
|
|
|
|
impl epi::Storage for LocalStorage {
|
|
fn get_string(&self, key: &str) -> Option<String> {
|
|
super::storage::local_storage_get(key)
|
|
}
|
|
|
|
fn set_string(&mut self, key: &str, value: String) {
|
|
super::storage::local_storage_set(key, &value);
|
|
}
|
|
|
|
fn flush(&mut self) {}
|
|
}
|