aboutsummaryrefslogtreecommitdiffstats
path: root/src/lib.rs
blob: 0c767e75a97edf8e0fa51afa922c6bcd53d19300 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
use std::cell::RefCell;
use std::sync::{Mutex};
use std::rc::Rc;

use log::{Level, info, debug};
use wasm_bindgen::prelude::*;

use state::State;

mod line;
mod point;
mod state;

struct RenderLoop {
    closure: Option<Closure<dyn FnMut(f64)>>,
}

impl RenderLoop {
    pub fn new() -> Self {
        Self {
            closure: None,
        }
    }
}

fn window() -> web_sys::Window {
    web_sys::window().expect("no window")
}

fn document() -> web_sys::Document {
    window().document().expect("no document")
}

fn canvas() -> Result<web_sys::HtmlCanvasElement, JsValue> {
    let x = document()
        .query_selector("canvas")?
        .expect("canvas element");
    Ok(x.dyn_into::<web_sys::HtmlCanvasElement>()?)
}

fn fps() -> Result<web_sys::HtmlElement, JsValue> {
    let x = document()
        .query_selector("#fps")?
        .expect("fps counter exists");
    Ok(x.dyn_into::<web_sys::HtmlElement>()?)
}

fn go() -> Result<web_sys::HtmlElement, JsValue> {
    let x = document()
        .query_selector("button")?
        .expect("go button exists");
    Ok(x.dyn_into::<web_sys::HtmlElement>()?)
}

fn request_animation_frame(f: &Closure<dyn FnMut(f64)>) {
    window()
        .request_animation_frame(f.as_ref().unchecked_ref())
        .expect("should register `requestAnimationFrame` OK");
}

fn rloop2<T: FnMut(f64) -> bool + 'static>(mut fun: T) -> Rc<RefCell<RenderLoop>> {
    let res = Rc::new(RefCell::new(RenderLoop::new()));
    let rloop = res.clone();

    (*res).borrow_mut().closure = Some(Closure::new(move |t| {
        if fun(t) {
            request_animation_frame(rloop.borrow().closure.as_ref().expect("can borrow rloop"));
        }
    }));

    res
}

#[wasm_bindgen(start)]
pub fn init() -> Result<(), JsValue> {
    console_log::init_with_level(Level::Debug).expect("couldn't init console log");
    info!("wasm init");

    let paused = Rc::new(Mutex::new(true));
    let p1 = paused.clone();

    let mut s = State::new(canvas()?, fps()?)?;
    s.render_frame(0.0)?;

    let render_loop = rloop2(move |t| {
        s.render_frame(t).expect("should render frame");
        !*paused.lock().unwrap()
    });
    request_animation_frame(render_loop.borrow().closure.as_ref().expect("can borrow res"));

    let a = Closure::<dyn FnMut()>::new(move || {
        let mut p = p1.lock().unwrap();
        debug!("go clicked: {}", *p);
        *p = !*p;
        let text = if *p {
            "go"
        } else {
            request_animation_frame(render_loop.borrow().closure.as_ref().expect("render closure exists"));
            "pause"
        };
        go().expect("go button works").set_text_content(Some(text));
    });
    go()?.set_onclick(Some(a.as_ref().unchecked_ref()));
    // otherwise it gets dropped after we're done, invalidating the
    // handler.
    a.forget();

    Ok(())
}