From 05fdcb5b270062b4bc85d3433ed12551f8a6741d Mon Sep 17 00:00:00 2001 From: josh65536 Date: Tue, 4 Aug 2020 21:39:09 -0400 Subject: [PATCH] Web: Use mouse events instead of pointer events if the latter isn't supported (#1630) * Fixed Safari not getting mouse events * Edited changelog * Addressed compiler warnings Co-authored-by: Ryan G --- CHANGELOG.md | 1 + src/platform_impl/web/web_sys/canvas.rs | 146 ++++++++++++++++++------ 2 files changed, 109 insertions(+), 38 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index dc9a6b38..d9257f7a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -10,6 +10,7 @@ - On android added support for `run_return`. - On MacOS, Fixed fullscreen and dialog support for `run_return`. - On Windows, fix bug where we'd try to emit `MainEventsCleared` events during nested win32 event loops. +- On Web, use mouse events if pointer events aren't supported. This affects Safari. - On Windows, `set_ime_position` is now a no-op instead of a runtime crash. - On Android, `set_fullscreen` is now a no-op instead of a runtime crash. - On iOS and Android, `set_inner_size` is now a no-op instead of a runtime crash. diff --git a/src/platform_impl/web/web_sys/canvas.rs b/src/platform_impl/web/web_sys/canvas.rs index e8fd3a2b..617f6ed6 100644 --- a/src/platform_impl/web/web_sys/canvas.rs +++ b/src/platform_impl/web/web_sys/canvas.rs @@ -9,8 +9,8 @@ use std::rc::Rc; use wasm_bindgen::{closure::Closure, JsCast}; use web_sys::{ - Event, FocusEvent, HtmlCanvasElement, KeyboardEvent, MediaQueryListEvent, PointerEvent, - WheelEvent, + Event, FocusEvent, HtmlCanvasElement, KeyboardEvent, MediaQueryListEvent, MouseEvent, + PointerEvent, WheelEvent, }; pub struct Canvas { @@ -24,8 +24,14 @@ pub struct Canvas { on_cursor_leave: Option>, on_cursor_enter: Option>, on_cursor_move: Option>, - on_mouse_press: Option>, - on_mouse_release: Option>, + on_pointer_press: Option>, + on_pointer_release: Option>, + // Fallback events when pointer event support is missing + on_mouse_leave: Option>, + on_mouse_enter: Option>, + on_mouse_move: Option>, + on_mouse_press: Option>, + on_mouse_release: Option>, on_mouse_wheel: Option>, on_fullscreen_change: Option>, wants_fullscreen: Rc>, @@ -76,8 +82,13 @@ impl Canvas { on_cursor_leave: None, on_cursor_enter: None, on_cursor_move: None, - on_mouse_release: None, + on_pointer_release: None, + on_pointer_press: None, + on_mouse_leave: None, + on_mouse_enter: None, + on_mouse_move: None, on_mouse_press: None, + on_mouse_release: None, on_mouse_wheel: None, on_fullscreen_change: None, wants_fullscreen: Rc::new(RefCell::new(false)), @@ -180,63 +191,110 @@ impl Canvas { where F: 'static + FnMut(i32), { - self.on_cursor_leave = Some(self.add_event("pointerout", move |event: PointerEvent| { - handler(event.pointer_id()); - })); + if has_pointer_event() { + self.on_cursor_leave = + Some(self.add_event("pointerout", move |event: PointerEvent| { + handler(event.pointer_id()); + })); + } else { + self.on_mouse_leave = Some(self.add_event("mouseout", move |_: MouseEvent| { + handler(0); + })); + } } pub fn on_cursor_enter(&mut self, mut handler: F) where F: 'static + FnMut(i32), { - self.on_cursor_enter = Some(self.add_event("pointerover", move |event: PointerEvent| { - handler(event.pointer_id()); - })); + if has_pointer_event() { + self.on_cursor_enter = + Some(self.add_event("pointerover", move |event: PointerEvent| { + handler(event.pointer_id()); + })); + } else { + self.on_mouse_enter = Some(self.add_event("mouseover", move |_: MouseEvent| { + handler(0); + })); + } } pub fn on_mouse_release(&mut self, mut handler: F) where F: 'static + FnMut(i32, MouseButton, ModifiersState), { - self.on_mouse_release = Some(self.add_user_event( - "pointerup", - move |event: PointerEvent| { - handler( - event.pointer_id(), - event::mouse_button(&event), - event::mouse_modifiers(&event), - ); - }, - )); + if has_pointer_event() { + self.on_pointer_release = Some(self.add_user_event( + "pointerup", + move |event: PointerEvent| { + handler( + event.pointer_id(), + event::mouse_button(&event), + event::mouse_modifiers(&event), + ); + }, + )); + } else { + self.on_mouse_release = + Some(self.add_user_event("mouseup", move |event: MouseEvent| { + handler( + 0, + event::mouse_button(&event), + event::mouse_modifiers(&event), + ); + })); + } } pub fn on_mouse_press(&mut self, mut handler: F) where F: 'static + FnMut(i32, MouseButton, ModifiersState), { - self.on_mouse_press = Some(self.add_user_event( - "pointerdown", - move |event: PointerEvent| { - handler( - event.pointer_id(), - event::mouse_button(&event), - event::mouse_modifiers(&event), - ); - }, - )); + if has_pointer_event() { + self.on_pointer_press = Some(self.add_user_event( + "pointerdown", + move |event: PointerEvent| { + handler( + event.pointer_id(), + event::mouse_button(&event), + event::mouse_modifiers(&event), + ); + }, + )); + } else { + self.on_mouse_press = + Some(self.add_user_event("mousedown", move |event: MouseEvent| { + handler( + 0, + event::mouse_button(&event), + event::mouse_modifiers(&event), + ); + })); + } } pub fn on_cursor_move(&mut self, mut handler: F) where F: 'static + FnMut(i32, PhysicalPosition, ModifiersState), { - self.on_cursor_move = Some(self.add_event("pointermove", move |event: PointerEvent| { - handler( - event.pointer_id(), - event::mouse_position(&event).to_physical(super::scale_factor()), - event::mouse_modifiers(&event), - ); - })); + if has_pointer_event() { + self.on_cursor_move = + Some(self.add_event("pointermove", move |event: PointerEvent| { + handler( + event.pointer_id(), + event::mouse_position(&event).to_physical(super::scale_factor()), + event::mouse_modifiers(&event), + ); + })); + } else { + self.on_mouse_move = Some(self.add_event("mousemove", move |event: MouseEvent| { + handler( + 0, + event::mouse_position(&event).to_physical(super::scale_factor()), + event::mouse_modifiers(&event), + ); + })); + } } pub fn on_mouse_wheel(&mut self, mut handler: F) @@ -334,3 +392,15 @@ impl Canvas { super::is_fullscreen(&self.raw) } } + +/// Returns whether pointer events are supported. +/// Used to decide whether to use pointer events +/// or plain mouse events. Note that Safari +/// doesn't support pointer events now. +fn has_pointer_event() -> bool { + if let Some(window) = web_sys::window() { + window.get("PointerEvent").is_some() + } else { + false + } +}