egui/example_wasm/src/lib.rs

113 lines
3.5 KiB
Rust
Raw Normal View History

2019-02-09 22:00:07 +00:00
#![deny(warnings)]
2020-05-10 17:04:10 +00:00
#![warn(clippy::all)]
2020-05-08 20:25:28 +00:00
use std::sync::Arc;
use {
2020-04-12 10:07:51 +00:00
emigui::{
2020-05-10 17:04:10 +00:00
color::srgba, example_app::ExampleApp, label, widgets::Separator, Align, RawInput,
TextStyle, *,
2020-04-12 10:07:51 +00:00
},
emigui_wasm::now_sec,
};
2019-02-09 22:00:07 +00:00
use wasm_bindgen::prelude::*;
2020-04-12 10:07:51 +00:00
2020-05-08 20:25:28 +00:00
#[wasm_bindgen]
2019-02-09 22:00:07 +00:00
pub struct State {
2020-05-10 17:04:10 +00:00
example_app: ExampleApp,
2020-05-08 20:25:28 +00:00
ctx: Arc<Context>,
2019-02-09 22:00:07 +00:00
webgl_painter: emigui_wasm::webgl::Painter,
2020-05-03 11:28:47 +00:00
frame_times: emigui::MovementTracker<f32>,
2019-02-09 22:00:07 +00:00
}
impl State {
fn new(canvas_id: &str, pixels_per_point: f32) -> Result<State, JsValue> {
2020-05-08 20:25:28 +00:00
let ctx = Context::new(pixels_per_point);
emigui_wasm::load_memory(&ctx);
2019-02-09 22:00:07 +00:00
Ok(State {
2020-04-12 10:07:51 +00:00
example_app: Default::default(),
2020-05-08 20:25:28 +00:00
ctx,
2019-02-09 22:00:07 +00:00
webgl_painter: emigui_wasm::webgl::Painter::new(canvas_id)?,
2020-05-03 11:28:47 +00:00
frame_times: emigui::MovementTracker::new(1000, 1.0),
2019-02-09 22:00:07 +00:00
})
}
2020-04-23 17:15:17 +00:00
fn run(&mut self, raw_input: RawInput) -> Result<Output, JsValue> {
let everything_start = now_sec();
2019-02-09 22:00:07 +00:00
let pixels_per_point = raw_input.pixels_per_point;
2020-05-08 20:25:28 +00:00
self.ctx.begin_frame(raw_input);
2019-02-09 22:00:07 +00:00
2020-05-08 20:42:31 +00:00
let mut ui = self.ctx.fullscreen_ui();
let mut ui = ui.centered_column(ui.available_width().min(480.0));
ui.set_align(Align::Min);
ui.add(label!("Emigui!").text_style(TextStyle::Heading));
ui.add_label("Emigui is an immediate mode GUI written in Rust, compiled to WebAssembly, rendered with WebGL.");
ui.add_label(
"Everything you see is rendered as textured triangles. There is no DOM. There are no HTML elements."
2020-04-20 21:33:16 +00:00
);
2020-05-08 20:42:31 +00:00
ui.add_label("This is not JavaScript. This is Rust, running at 60 FPS. This is the web page, reinvented with game tech.");
ui.add_label("This is also work in progress, and not ready for production... yet :)");
ui.horizontal(|ui| {
ui.add_label("Project home page:");
ui.add_hyperlink("https://github.com/emilk/emigui/");
2020-04-23 17:15:17 +00:00
});
2020-05-08 20:42:31 +00:00
ui.add(Separator::new());
2020-04-12 10:07:51 +00:00
2020-05-08 20:42:31 +00:00
ui.set_align(Align::Min);
ui.add_label("WebGl painter info:");
ui.indent("webgl region id", |ui| {
ui.add_label(self.webgl_painter.debug_info());
2019-02-09 22:00:07 +00:00
});
2020-05-08 20:42:31 +00:00
ui.add(
label!(
2020-05-03 11:28:47 +00:00
"CPU usage: {:.2} ms (excludes painting)",
1e3 * self.frame_times.average().unwrap_or_default()
)
.text_style(TextStyle::Monospace),
);
2020-05-08 20:42:31 +00:00
ui.add(
2020-05-03 11:28:47 +00:00
label!(
"FPS: {:.1}",
1.0 / self.frame_times.mean_time_interval().unwrap_or_default()
)
.text_style(TextStyle::Monospace),
2020-04-12 10:07:51 +00:00
);
2019-02-09 22:00:07 +00:00
2020-05-10 17:04:10 +00:00
self.example_app.ui(&self.ctx);
2020-04-23 20:08:42 +00:00
let bg_color = srgba(0, 0, 0, 0); // Use background css color.
2020-05-08 20:25:28 +00:00
let (output, batches) = self.ctx.end_frame();
2020-05-03 11:28:47 +00:00
let now = now_sec();
self.frame_times.add(now, (now - everything_start) as f32);
2020-04-23 17:15:17 +00:00
self.webgl_painter.paint_batches(
2019-03-16 12:37:29 +00:00
bg_color,
batches,
2020-05-08 20:25:28 +00:00
self.ctx.texture(),
pixels_per_point,
2020-04-23 17:15:17 +00:00
)?;
2019-02-09 22:00:07 +00:00
2020-05-08 20:25:28 +00:00
emigui_wasm::save_memory(&self.ctx); // TODO: don't save every frame
2020-04-23 17:15:17 +00:00
Ok(output)
2019-02-09 22:00:07 +00:00
}
}
#[wasm_bindgen]
pub fn new_webgl_gui(canvas_id: &str, pixels_per_point: f32) -> Result<State, JsValue> {
State::new(canvas_id, pixels_per_point)
}
#[wasm_bindgen]
2020-04-23 17:15:17 +00:00
pub fn run_gui(state: &mut State, raw_input_json: &str) -> Result<String, JsValue> {
2019-02-09 22:00:07 +00:00
// TODO: nicer interface than JSON
let raw_input: RawInput = serde_json::from_str(raw_input_json).unwrap();
2020-04-23 17:15:17 +00:00
let output = state.run(raw_input)?;
Ok(serde_json::to_string(&output).unwrap())
2019-02-09 22:00:07 +00:00
}