mirror of
https://github.com/italicsjenga/winit-sonoma-fix.git
synced 2025-01-26 03:36:32 +11:00
Implement web_sys::Canvas
event listeners
This commit is contained in:
parent
c5703eb00a
commit
b79089ea57
9 changed files with 242 additions and 62 deletions
|
@ -55,7 +55,7 @@ impl<T: 'static> Shared<T> {
|
||||||
// It will determine if the event should be immediately sent to the user or buffered for later
|
// It will determine if the event should be immediately sent to the user or buffered for later
|
||||||
pub fn send_event(&self, event: Event<T>) {
|
pub fn send_event(&self, event: Event<T>) {
|
||||||
// If the event loop is closed, it should discard any new events
|
// If the event loop is closed, it should discard any new events
|
||||||
if self.closed() {
|
if self.is_closed() {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -105,7 +105,7 @@ impl<T: 'static> Shared<T> {
|
||||||
self.apply_control_flow(control);
|
self.apply_control_flow(control);
|
||||||
// If the event loop is closed, it has been closed this iteration and now the closing
|
// If the event loop is closed, it has been closed this iteration and now the closing
|
||||||
// event should be emitted
|
// event should be emitted
|
||||||
if self.closed() {
|
if self.is_closed() {
|
||||||
self.handle_event(Event::LoopDestroyed, &mut control);
|
self.handle_event(Event::LoopDestroyed, &mut control);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -114,7 +114,7 @@ impl<T: 'static> Shared<T> {
|
||||||
//
|
//
|
||||||
// It should only ever be called from send_event
|
// It should only ever be called from send_event
|
||||||
fn handle_event(&self, event: Event<T>, control: &mut root::ControlFlow) {
|
fn handle_event(&self, event: Event<T>, control: &mut root::ControlFlow) {
|
||||||
let closed = self.closed();
|
let is_closed = self.is_closed();
|
||||||
|
|
||||||
match *self.0.runner.borrow_mut() {
|
match *self.0.runner.borrow_mut() {
|
||||||
Some(ref mut runner) => {
|
Some(ref mut runner) => {
|
||||||
|
@ -124,7 +124,7 @@ impl<T: 'static> Shared<T> {
|
||||||
(runner.event_handler)(event, control);
|
(runner.event_handler)(event, control);
|
||||||
|
|
||||||
// Maintain closed state, even if the callback changes it
|
// Maintain closed state, even if the callback changes it
|
||||||
if closed {
|
if is_closed {
|
||||||
*control = root::ControlFlow::Exit;
|
*control = root::ControlFlow::Exit;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -138,7 +138,7 @@ impl<T: 'static> Shared<T> {
|
||||||
|
|
||||||
// Don't take events out of the queue if the loop is closed or the runner doesn't exist
|
// Don't take events out of the queue if the loop is closed or the runner doesn't exist
|
||||||
// If the runner doesn't exist and this method recurses, it will recurse infinitely
|
// If the runner doesn't exist and this method recurses, it will recurse infinitely
|
||||||
if !closed && self.0.runner.borrow().is_some() {
|
if !is_closed && self.0.runner.borrow().is_some() {
|
||||||
// Take an event out of the queue and handle it
|
// Take an event out of the queue and handle it
|
||||||
if let Some(event) = self.0.events.borrow_mut().pop_front() {
|
if let Some(event) = self.0.events.borrow_mut().pop_front() {
|
||||||
self.handle_event(event, control);
|
self.handle_event(event, control);
|
||||||
|
@ -149,13 +149,13 @@ impl<T: 'static> Shared<T> {
|
||||||
// Apply the new ControlFlow that has been selected by the user
|
// Apply the new ControlFlow that has been selected by the user
|
||||||
// Start any necessary timeouts etc
|
// Start any necessary timeouts etc
|
||||||
fn apply_control_flow(&self, control_flow: root::ControlFlow) {
|
fn apply_control_flow(&self, control_flow: root::ControlFlow) {
|
||||||
let mut control_flow_status = match control_flow {
|
let new_state = match control_flow {
|
||||||
root::ControlFlow::Poll => {
|
root::ControlFlow::Poll => {
|
||||||
let cloned = self.clone();
|
let cloned = self.clone();
|
||||||
State::Poll {
|
State::Poll {
|
||||||
timeout: backend::Timeout::new(
|
timeout: backend::Timeout::new(
|
||||||
move || cloned.send_event(Event::NewEvents(StartCause::Poll)),
|
move || cloned.send_event(Event::NewEvents(StartCause::Poll)),
|
||||||
Duration::from_millis(1),
|
Duration::from_millis(0),
|
||||||
),
|
),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -163,13 +163,16 @@ impl<T: 'static> Shared<T> {
|
||||||
start: Instant::now(),
|
start: Instant::now(),
|
||||||
},
|
},
|
||||||
root::ControlFlow::WaitUntil(end) => {
|
root::ControlFlow::WaitUntil(end) => {
|
||||||
let cloned = self.clone();
|
|
||||||
let start = Instant::now();
|
let start = Instant::now();
|
||||||
|
|
||||||
let delay = if end <= start {
|
let delay = if end <= start {
|
||||||
Duration::from_millis(0)
|
Duration::from_millis(0)
|
||||||
} else {
|
} else {
|
||||||
end - start
|
end - start
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let cloned = self.clone();
|
||||||
|
|
||||||
State::WaitUntil {
|
State::WaitUntil {
|
||||||
start,
|
start,
|
||||||
end,
|
end,
|
||||||
|
@ -184,22 +187,14 @@ impl<T: 'static> Shared<T> {
|
||||||
|
|
||||||
match *self.0.runner.borrow_mut() {
|
match *self.0.runner.borrow_mut() {
|
||||||
Some(ref mut runner) => {
|
Some(ref mut runner) => {
|
||||||
// Put the new control flow status in the runner, and take out the old one
|
runner.state = new_state;
|
||||||
// This way we can safely take ownership of the TimeoutHandle and clear it,
|
|
||||||
// so that we don't get 'ghost' invocations of Poll or WaitUntil from earlier
|
|
||||||
// set_timeout invocations
|
|
||||||
std::mem::swap(&mut runner.state, &mut control_flow_status);
|
|
||||||
match control_flow_status {
|
|
||||||
State::Poll { timeout } | State::WaitUntil { timeout, .. } => timeout.clear(),
|
|
||||||
_ => (),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
None => (),
|
None => (),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if the event loop is currntly closed
|
// Check if the event loop is currently closed
|
||||||
fn closed(&self) -> bool {
|
fn is_closed(&self) -> bool {
|
||||||
match *self.0.runner.borrow() {
|
match *self.0.runner.borrow() {
|
||||||
Some(ref runner) => runner.state.is_exit(),
|
Some(ref runner) => runner.state.is_exit(),
|
||||||
None => false, // If the event loop is None, it has not been intialised yet, so it cannot be closed
|
None => false, // If the event loop is None, it has not been intialised yet, so it cannot be closed
|
||||||
|
@ -209,7 +204,7 @@ impl<T: 'static> Shared<T> {
|
||||||
// Get the current control flow state
|
// Get the current control flow state
|
||||||
fn current_control_flow(&self) -> root::ControlFlow {
|
fn current_control_flow(&self) -> root::ControlFlow {
|
||||||
match *self.0.runner.borrow() {
|
match *self.0.runner.borrow() {
|
||||||
Some(ref runner) => runner.state.into(),
|
Some(ref runner) => runner.state.control_flow(),
|
||||||
None => root::ControlFlow::Poll,
|
None => root::ControlFlow::Poll,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@ use crate::event_loop::ControlFlow;
|
||||||
|
|
||||||
use instant::Instant;
|
use instant::Instant;
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug)]
|
||||||
pub enum State {
|
pub enum State {
|
||||||
Init,
|
Init,
|
||||||
WaitUntil {
|
WaitUntil {
|
||||||
|
@ -27,13 +27,11 @@ impl State {
|
||||||
_ => false,
|
_ => false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
impl From<State> for ControlFlow {
|
pub fn control_flow(&self) -> ControlFlow {
|
||||||
fn from(state: State) -> ControlFlow {
|
match self {
|
||||||
match state {
|
|
||||||
State::Init => ControlFlow::Poll,
|
State::Init => ControlFlow::Poll,
|
||||||
State::WaitUntil { end, .. } => ControlFlow::WaitUntil(end),
|
State::WaitUntil { end, .. } => ControlFlow::WaitUntil(*end),
|
||||||
State::Wait { .. } => ControlFlow::Wait,
|
State::Wait { .. } => ControlFlow::Wait,
|
||||||
State::Poll { .. } => ControlFlow::Poll,
|
State::Poll { .. } => ControlFlow::Poll,
|
||||||
State::Exit => ControlFlow::Exit,
|
State::Exit => ControlFlow::Exit,
|
||||||
|
|
|
@ -35,10 +35,9 @@ impl<T> WindowTarget<T> {
|
||||||
&self.runner
|
&self.runner
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn register(&self, canvas: &backend::Canvas) {
|
pub fn register(&self, canvas: &mut backend::Canvas) {
|
||||||
let runner = &self.runner;
|
let runner = self.runner.clone();
|
||||||
|
canvas.on_mouse_out(move |pointer_id| {
|
||||||
canvas.on_mouse_out(|pointer_id| {
|
|
||||||
runner.send_event(Event::WindowEvent {
|
runner.send_event(Event::WindowEvent {
|
||||||
window_id: WindowId(window::Id),
|
window_id: WindowId(window::Id),
|
||||||
event: WindowEvent::CursorLeft {
|
event: WindowEvent::CursorLeft {
|
||||||
|
@ -47,7 +46,8 @@ impl<T> WindowTarget<T> {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
canvas.on_mouse_over(|pointer_id| {
|
let runner = self.runner.clone();
|
||||||
|
canvas.on_mouse_over(move |pointer_id| {
|
||||||
runner.send_event(Event::WindowEvent {
|
runner.send_event(Event::WindowEvent {
|
||||||
window_id: WindowId(window::Id),
|
window_id: WindowId(window::Id),
|
||||||
event: WindowEvent::CursorEntered {
|
event: WindowEvent::CursorEntered {
|
||||||
|
@ -56,7 +56,8 @@ impl<T> WindowTarget<T> {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
canvas.on_mouse_move(|pointer_id, position, modifiers| {
|
let runner = self.runner.clone();
|
||||||
|
canvas.on_mouse_move(move |pointer_id, position, modifiers| {
|
||||||
runner.send_event(Event::WindowEvent {
|
runner.send_event(Event::WindowEvent {
|
||||||
window_id: WindowId(window::Id),
|
window_id: WindowId(window::Id),
|
||||||
event: WindowEvent::CursorMoved {
|
event: WindowEvent::CursorMoved {
|
||||||
|
@ -67,7 +68,8 @@ impl<T> WindowTarget<T> {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
canvas.on_mouse_up(|pointer_id, button, modifiers| {
|
let runner = self.runner.clone();
|
||||||
|
canvas.on_mouse_up(move |pointer_id, button, modifiers| {
|
||||||
runner.send_event(Event::WindowEvent {
|
runner.send_event(Event::WindowEvent {
|
||||||
window_id: WindowId(window::Id),
|
window_id: WindowId(window::Id),
|
||||||
event: WindowEvent::MouseInput {
|
event: WindowEvent::MouseInput {
|
||||||
|
@ -79,7 +81,8 @@ impl<T> WindowTarget<T> {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
canvas.on_mouse_down(|pointer_id, button, modifiers| {
|
let runner = self.runner.clone();
|
||||||
|
canvas.on_mouse_down(move |pointer_id, button, modifiers| {
|
||||||
runner.send_event(Event::WindowEvent {
|
runner.send_event(Event::WindowEvent {
|
||||||
window_id: WindowId(window::Id),
|
window_id: WindowId(window::Id),
|
||||||
event: WindowEvent::MouseInput {
|
event: WindowEvent::MouseInput {
|
||||||
|
@ -91,7 +94,8 @@ impl<T> WindowTarget<T> {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
canvas.on_mouse_scroll(|pointer_id, delta, modifiers| {
|
let runner = self.runner.clone();
|
||||||
|
canvas.on_mouse_scroll(move |pointer_id, delta, modifiers| {
|
||||||
runner.send_event(Event::WindowEvent {
|
runner.send_event(Event::WindowEvent {
|
||||||
window_id: WindowId(window::Id),
|
window_id: WindowId(window::Id),
|
||||||
event: WindowEvent::MouseWheel {
|
event: WindowEvent::MouseWheel {
|
||||||
|
|
|
@ -0,0 +1,6 @@
|
||||||
|
#[cfg(feature = "stdweb")]
|
||||||
|
impl WindowExtStdweb for RootWindow {
|
||||||
|
fn canvas(&self) -> CanvasElement {
|
||||||
|
self.window.canvas.clone()
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,12 +1,20 @@
|
||||||
use crate::dpi::LogicalSize;
|
use super::event;
|
||||||
|
use crate::dpi::{LogicalPosition, LogicalSize};
|
||||||
use crate::error::OsError as RootOE;
|
use crate::error::OsError as RootOE;
|
||||||
|
use crate::event::{ModifiersState, MouseButton, MouseScrollDelta};
|
||||||
use crate::platform_impl::OsError;
|
use crate::platform_impl::OsError;
|
||||||
|
|
||||||
use wasm_bindgen::JsCast;
|
use wasm_bindgen::{closure::Closure, JsCast};
|
||||||
use web_sys::HtmlCanvasElement;
|
use web_sys::{HtmlCanvasElement, PointerEvent, WheelEvent};
|
||||||
|
|
||||||
pub struct Canvas {
|
pub struct Canvas {
|
||||||
raw: HtmlCanvasElement,
|
raw: HtmlCanvasElement,
|
||||||
|
on_mouse_out: Option<Closure<dyn FnMut(PointerEvent)>>,
|
||||||
|
on_mouse_over: Option<Closure<dyn FnMut(PointerEvent)>>,
|
||||||
|
on_mouse_up: Option<Closure<dyn FnMut(PointerEvent)>>,
|
||||||
|
on_mouse_down: Option<Closure<dyn FnMut(PointerEvent)>>,
|
||||||
|
on_mouse_move: Option<Closure<dyn FnMut(PointerEvent)>>,
|
||||||
|
on_mouse_scroll: Option<Closure<dyn FnMut(WheelEvent)>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Canvas {
|
impl Canvas {
|
||||||
|
@ -25,7 +33,15 @@ impl Canvas {
|
||||||
.append_child(&canvas)
|
.append_child(&canvas)
|
||||||
.map_err(|_| os_error!(OsError("Failed to append canvas".to_owned())))?;
|
.map_err(|_| os_error!(OsError("Failed to append canvas".to_owned())))?;
|
||||||
|
|
||||||
Ok(Canvas { raw: canvas })
|
Ok(Canvas {
|
||||||
|
raw: canvas,
|
||||||
|
on_mouse_out: None,
|
||||||
|
on_mouse_over: None,
|
||||||
|
on_mouse_up: None,
|
||||||
|
on_mouse_down: None,
|
||||||
|
on_mouse_move: None,
|
||||||
|
on_mouse_scroll: None,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_attribute(&self, attribute: &str, value: &str) {
|
pub fn set_attribute(&self, attribute: &str, value: &str) {
|
||||||
|
@ -53,14 +69,98 @@ impl Canvas {
|
||||||
self.raw.set_height(size.height as u32);
|
self.raw.set_height(size.height as u32);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn raw(&self) -> HtmlCanvasElement {
|
pub fn raw(&self) -> &HtmlCanvasElement {
|
||||||
self.raw.clone()
|
&self.raw
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn on_mouse_out<F>(&self, f: F) {}
|
pub fn on_mouse_out<F>(&mut self, mut handler: F)
|
||||||
pub fn on_mouse_over<F>(&self, f: F) {}
|
where
|
||||||
pub fn on_mouse_up<F>(&self, f: F) {}
|
F: 'static + FnMut(i32),
|
||||||
pub fn on_mouse_down<F>(&self, f: F) {}
|
{
|
||||||
pub fn on_mouse_move<F>(&self, f: F) {}
|
self.on_mouse_out = Some(self.add_event("pointerout", move |event: PointerEvent| {
|
||||||
pub fn on_mouse_scroll<F>(&self, f: F) {}
|
handler(event.pointer_id());
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn on_mouse_over<F>(&mut self, mut handler: F)
|
||||||
|
where
|
||||||
|
F: 'static + FnMut(i32),
|
||||||
|
{
|
||||||
|
self.on_mouse_over = Some(self.add_event("pointerover", move |event: PointerEvent| {
|
||||||
|
handler(event.pointer_id());
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn on_mouse_up<F>(&mut self, mut handler: F)
|
||||||
|
where
|
||||||
|
F: 'static + FnMut(i32, MouseButton, ModifiersState),
|
||||||
|
{
|
||||||
|
self.on_mouse_up = Some(self.add_event("pointerup", move |event: PointerEvent| {
|
||||||
|
handler(
|
||||||
|
event.pointer_id(),
|
||||||
|
event::mouse_button(&event),
|
||||||
|
event::mouse_modifiers(&event),
|
||||||
|
);
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn on_mouse_down<F>(&mut self, mut handler: F)
|
||||||
|
where
|
||||||
|
F: 'static + FnMut(i32, MouseButton, ModifiersState),
|
||||||
|
{
|
||||||
|
self.on_mouse_down = Some(self.add_event("pointerdown", move |event: PointerEvent| {
|
||||||
|
handler(
|
||||||
|
event.pointer_id(),
|
||||||
|
event::mouse_button(&event),
|
||||||
|
event::mouse_modifiers(&event),
|
||||||
|
);
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn on_mouse_move<F>(&mut self, mut handler: F)
|
||||||
|
where
|
||||||
|
F: 'static + FnMut(i32, LogicalPosition, ModifiersState),
|
||||||
|
{
|
||||||
|
self.on_mouse_move = Some(self.add_event("pointermove", move |event: PointerEvent| {
|
||||||
|
handler(
|
||||||
|
event.pointer_id(),
|
||||||
|
event::mouse_position(&event),
|
||||||
|
event::mouse_modifiers(&event),
|
||||||
|
);
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn on_mouse_scroll<F>(&mut self, mut handler: F)
|
||||||
|
where
|
||||||
|
F: 'static + FnMut(i32, MouseScrollDelta, ModifiersState),
|
||||||
|
{
|
||||||
|
self.on_mouse_scroll = Some(self.add_event("wheel", move |event: WheelEvent| {
|
||||||
|
if let Some(delta) = event::mouse_scroll_delta(&event) {
|
||||||
|
handler(0, delta, event::mouse_modifiers(&event));
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_event<E, F>(&self, event_name: &str, mut handler: F) -> Closure<FnMut(E)>
|
||||||
|
where
|
||||||
|
E: 'static + AsRef<web_sys::Event> + wasm_bindgen::convert::FromWasmAbi,
|
||||||
|
F: 'static + FnMut(E),
|
||||||
|
{
|
||||||
|
let closure = Closure::wrap(Box::new(move |event: E| {
|
||||||
|
{
|
||||||
|
let event_ref = event.as_ref();
|
||||||
|
event_ref.prevent_default();
|
||||||
|
event_ref.stop_propagation();
|
||||||
|
event_ref.cancel_bubble();
|
||||||
|
}
|
||||||
|
|
||||||
|
handler(event);
|
||||||
|
}) as Box<dyn FnMut(E)>);
|
||||||
|
|
||||||
|
self.raw
|
||||||
|
.add_event_listener_with_callback(event_name, &closure.as_ref().unchecked_ref())
|
||||||
|
.expect("Failed to add event listener with callback");
|
||||||
|
|
||||||
|
closure
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
41
src/platform_impl/web/web_sys/event.rs
Normal file
41
src/platform_impl/web/web_sys/event.rs
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
use crate::dpi::LogicalPosition;
|
||||||
|
use crate::event::{ModifiersState, MouseButton, MouseScrollDelta};
|
||||||
|
|
||||||
|
use std::convert::TryInto;
|
||||||
|
use web_sys::{MouseEvent, WheelEvent};
|
||||||
|
|
||||||
|
pub fn mouse_button(event: &MouseEvent) -> MouseButton {
|
||||||
|
match event.button() {
|
||||||
|
0 => MouseButton::Left,
|
||||||
|
1 => MouseButton::Middle,
|
||||||
|
2 => MouseButton::Right,
|
||||||
|
i => MouseButton::Other((i - 3).try_into().expect("very large mouse button value")),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn mouse_modifiers(event: &MouseEvent) -> ModifiersState {
|
||||||
|
ModifiersState {
|
||||||
|
shift: event.shift_key(),
|
||||||
|
ctrl: event.ctrl_key(),
|
||||||
|
alt: event.alt_key(),
|
||||||
|
logo: event.meta_key(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn mouse_position(event: &MouseEvent) -> LogicalPosition {
|
||||||
|
LogicalPosition {
|
||||||
|
x: event.offset_x() as f64,
|
||||||
|
y: event.offset_y() as f64,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn mouse_scroll_delta(event: &WheelEvent) -> Option<MouseScrollDelta> {
|
||||||
|
let x = event.delta_x();
|
||||||
|
let y = event.delta_y();
|
||||||
|
|
||||||
|
match event.delta_mode() {
|
||||||
|
WheelEvent::DOM_DELTA_LINE => Some(MouseScrollDelta::LineDelta(x as f32, y as f32)),
|
||||||
|
WheelEvent::DOM_DELTA_PIXEL => Some(MouseScrollDelta::PixelDelta(LogicalPosition { x, y })),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,11 +1,16 @@
|
||||||
mod canvas;
|
mod canvas;
|
||||||
mod document;
|
mod document;
|
||||||
|
mod event;
|
||||||
mod timeout;
|
mod timeout;
|
||||||
|
|
||||||
pub use self::canvas::Canvas;
|
pub use self::canvas::Canvas;
|
||||||
pub use self::document::Document;
|
pub use self::document::Document;
|
||||||
pub use self::timeout::Timeout;
|
pub use self::timeout::Timeout;
|
||||||
|
|
||||||
|
use crate::platform::web::WindowExtWebSys;
|
||||||
|
use crate::window::Window;
|
||||||
|
use web_sys::HtmlCanvasElement;
|
||||||
|
|
||||||
pub fn request_animation_frame<F>(f: F)
|
pub fn request_animation_frame<F>(f: F)
|
||||||
where
|
where
|
||||||
F: Fn(),
|
F: Fn(),
|
||||||
|
@ -15,3 +20,9 @@ where
|
||||||
pub fn throw(msg: &str) {
|
pub fn throw(msg: &str) {
|
||||||
wasm_bindgen::throw_str(msg);
|
wasm_bindgen::throw_str(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl WindowExtWebSys for Window {
|
||||||
|
fn canvas(&self) -> HtmlCanvasElement {
|
||||||
|
self.window.canvas().raw().clone()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -1,12 +1,40 @@
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
use wasm_bindgen::closure::Closure;
|
||||||
|
use wasm_bindgen::JsCast;
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug)]
|
||||||
pub struct Timeout {}
|
pub struct Timeout {
|
||||||
|
handle: i32,
|
||||||
|
_closure: Closure<FnMut()>,
|
||||||
|
}
|
||||||
|
|
||||||
impl Timeout {
|
impl Timeout {
|
||||||
pub fn new<F>(f: F, duration: Duration) -> Timeout {
|
pub fn new<F>(f: F, duration: Duration) -> Timeout
|
||||||
Timeout {}
|
where
|
||||||
}
|
F: 'static + FnMut(),
|
||||||
|
{
|
||||||
|
let window = web_sys::window().expect("Failed to obtain window");
|
||||||
|
|
||||||
pub fn clear(self) {}
|
let closure = Closure::wrap(Box::new(f) as Box<dyn FnMut()>);
|
||||||
|
|
||||||
|
let handle = window
|
||||||
|
.set_timeout_with_callback_and_timeout_and_arguments_0(
|
||||||
|
&closure.as_ref().unchecked_ref(),
|
||||||
|
duration.as_millis() as i32,
|
||||||
|
)
|
||||||
|
.expect("Failed to set timeout");
|
||||||
|
|
||||||
|
Timeout {
|
||||||
|
handle,
|
||||||
|
_closure: closure,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for Timeout {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
let window = web_sys::window().expect("Failed to obtain window");
|
||||||
|
|
||||||
|
window.clear_timeout_with_handle(self.handle);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,9 +24,9 @@ impl Window {
|
||||||
attr: WindowAttributes,
|
attr: WindowAttributes,
|
||||||
_: PlatformSpecificBuilderAttributes,
|
_: PlatformSpecificBuilderAttributes,
|
||||||
) -> Result<Self, RootOE> {
|
) -> Result<Self, RootOE> {
|
||||||
let canvas = backend::Canvas::create()?;
|
let mut canvas = backend::Canvas::create()?;
|
||||||
|
|
||||||
target.register(&canvas);
|
target.register(&mut canvas);
|
||||||
|
|
||||||
let runner = target.runner.clone();
|
let runner = target.runner.clone();
|
||||||
let redraw = Box::new(move || {
|
let redraw = Box::new(move || {
|
||||||
|
@ -58,6 +58,10 @@ impl Window {
|
||||||
Ok(window)
|
Ok(window)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn canvas(&self) -> &backend::Canvas {
|
||||||
|
&self.canvas
|
||||||
|
}
|
||||||
|
|
||||||
pub fn set_title(&self, title: &str) {
|
pub fn set_title(&self, title: &str) {
|
||||||
backend::Document::set_title(title);
|
backend::Document::set_title(title);
|
||||||
}
|
}
|
||||||
|
@ -256,13 +260,6 @@ impl Window {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "stdweb")]
|
|
||||||
impl WindowExtStdweb for RootWindow {
|
|
||||||
fn canvas(&self) -> CanvasElement {
|
|
||||||
self.window.canvas.clone()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
||||||
pub struct Id;
|
pub struct Id;
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue