egui/example_wasm/src/lib.rs

110 lines
3.5 KiB
Rust
Raw Normal View History

2019-02-09 22:00:07 +00:00
#![deny(warnings)]
extern crate serde_json;
extern crate wasm_bindgen;
extern crate emigui;
extern crate emigui_wasm;
use {
2020-04-12 10:07:51 +00:00
emigui::{
color::srgba,
example_app::ExampleApp,
label,
widgets::{Label, Separator},
Align, Emigui, RawInput, TextStyle, Window, *,
2020-04-12 10:07:51 +00:00
},
emigui_wasm::now_sec,
};
2019-02-09 22:00:07 +00:00
use wasm_bindgen::prelude::*;
#[wasm_bindgen]
2020-04-12 10:07:51 +00:00
2019-02-09 22:00:07 +00:00
pub struct State {
2020-04-12 10:07:51 +00:00
example_app: ExampleApp,
2019-02-09 22:00:07 +00:00
emigui: Emigui,
webgl_painter: emigui_wasm::webgl::Painter,
everything_ms: f64,
}
impl State {
fn new(canvas_id: &str, pixels_per_point: f32) -> Result<State, JsValue> {
Ok(State {
2020-04-12 10:07:51 +00:00
example_app: Default::default(),
2019-02-09 22:00:07 +00:00
emigui: Emigui::new(pixels_per_point),
webgl_painter: emigui_wasm::webgl::Painter::new(canvas_id)?,
everything_ms: 0.0,
})
}
fn run(&mut self, raw_input: RawInput) -> Result<(), JsValue> {
let everything_start = now_sec();
2019-02-09 22:00:07 +00:00
self.emigui.new_frame(raw_input);
let mut region = self.emigui.background_region();
let mut region = region.centered_column(region.available_width().min(480.0));
region.set_align(Align::Min);
2020-04-12 10:07:51 +00:00
region.add(label!("Emigui!").text_style(TextStyle::Heading));
region.add(label!("Emigui is an immediate mode GUI written in Rust, compiled to WebAssembly, rendered with WebGL."));
region.add(label!(
"Everything you see is rendered as textured triangles. There is no DOM. There are no HTML elements."
));
region.add(label!("This not JavaScript. This is Rust code, running at 60 Hz. This is the web page, reinvented with game tech."));
region.add(label!(
"This is also work in progress, and not ready for production... yet :)"
2020-04-12 10:07:51 +00:00
));
region.add(Separator::new());
self.example_app.ui(&mut region);
self.emigui.ui(&mut region);
region.set_align(Align::Min);
2019-02-09 22:00:07 +00:00
region.add(label!("WebGl painter info:"));
region.indent(|region| {
region.add(label!(self.webgl_painter.debug_info()));
});
2020-04-12 10:07:51 +00:00
region.add(
label!("Everything: {:.1} ms", self.everything_ms).text_style(TextStyle::Monospace),
);
2019-02-09 22:00:07 +00:00
Window::new("Test window").show(region.ctx(), |region| {
region.add(label!("Grab the window and move it around!"));
region.add(label!(
"This window can be reisized, but not smaller than the contents."
));
});
Window::new("Another test window")
.default_pos(pos2(400.0, 100.0))
.show(region.ctx(), |region| {
region.add(label!("This might be on top of the other window?"));
region.add(label!("Second line of text"));
});
2019-03-16 12:37:29 +00:00
let bg_color = srgba(16, 16, 16, 255);
let batches = self.emigui.paint();
let result = self.webgl_painter.paint_batches(
2019-03-16 12:37:29 +00:00
bg_color,
batches,
2019-03-16 12:37:29 +00:00
self.emigui.texture(),
raw_input.pixels_per_point,
);
2019-02-09 22:00:07 +00:00
self.everything_ms = 1000.0 * (now_sec() - everything_start);
2019-02-09 22:00:07 +00:00
result
}
}
#[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]
pub fn run_gui(state: &mut State, raw_input_json: &str) -> Result<(), JsValue> {
// TODO: nicer interface than JSON
let raw_input: RawInput = serde_json::from_str(raw_input_json).unwrap();
state.run(raw_input)
}