From 4f7e75daa76eafdfc389044522230d97ac4ab659 Mon Sep 17 00:00:00 2001 From: Ben Merritt Date: Sat, 8 Jun 2019 16:17:59 -0700 Subject: [PATCH] Start implementing a "callback registry" so closures aren't leaked --- src/platform_impl/web_sys/event_loop.rs | 84 ++++++++++++++++--------- src/platform_impl/web_sys/mod.rs | 2 +- src/platform_impl/web_sys/window.rs | 10 ++- 3 files changed, 64 insertions(+), 32 deletions(-) diff --git a/src/platform_impl/web_sys/event_loop.rs b/src/platform_impl/web_sys/event_loop.rs index d9f43c8858..d34f9ae682 100644 --- a/src/platform_impl/web_sys/event_loop.rs +++ b/src/platform_impl/web_sys/event_loop.rs @@ -13,7 +13,7 @@ use std::collections::VecDeque; use std::marker::PhantomData; use std::rc::Rc; use wasm_bindgen::{prelude::*, JsCast}; -use web_sys::{EventTarget, FocusEvent, HtmlCanvasElement, KeyboardEvent, PointerEvent, WheelEvent}; +use web_sys::{EventTarget, FocusEvent, KeyboardEvent, PointerEvent, WheelEvent}; use window::WindowId as RootWI; #[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] @@ -27,6 +27,7 @@ impl DeviceId { pub struct EventLoop { elw: RootELW, + callbacks: CallbackRegistry, } pub struct EventLoopWindowTarget { @@ -71,12 +72,13 @@ struct EventLoopRunner { impl EventLoop { pub fn new() -> Self { - EventLoop { - elw: RootELW { - p: EventLoopWindowTarget::new(), - _marker: PhantomData, - }, - } + let elw = RootELW { + p: EventLoopWindowTarget::new(), + _marker: PhantomData, + }; + let document = document(); + let callbacks = CallbackRegistry::new(elw.p.runner.clone(), document.unchecked_into()); + EventLoop { elw, callbacks } } pub fn available_monitors(&self) -> VecDequeIter { @@ -87,7 +89,7 @@ impl EventLoop { MonitorHandle } - pub fn run(self, mut event_handler: F) -> ! + pub fn run(mut self, mut event_handler: F) -> ! where F: 'static + FnMut(Event, &RootELW, &mut ControlFlow), { @@ -99,20 +101,19 @@ impl EventLoop { }; runner.set_listener(Box::new(move |evt, ctrl| event_handler(evt, &relw, ctrl))); - let document = &document(); - add_event(&runner, document, "blur", |elrs, _: FocusEvent| { + self.callbacks.add_event("blur", |elrs, _: FocusEvent| { elrs.send_event(Event::WindowEvent { window_id: RootWI(WindowId), event: WindowEvent::Focused(false), }); }); - add_event(&runner, document, "focus", |elrs, _: FocusEvent| { + self.callbacks.add_event("focus", |elrs, _: FocusEvent| { elrs.send_event(Event::WindowEvent { window_id: RootWI(WindowId), event: WindowEvent::Focused(true), }); }); - add_event(&runner, document, "keydown", |elrs, event: KeyboardEvent| { + self.callbacks.add_event("keydown", |elrs, event: KeyboardEvent| { let key = event.key(); let mut characters = key.chars(); let first = characters.next(); @@ -136,7 +137,7 @@ impl EventLoop { }, }); }); - add_event(&runner, document, "keyup", |elrs, event: KeyboardEvent| { + self.callbacks.add_event("keyup", |elrs, event: KeyboardEvent| { elrs.send_event(Event::WindowEvent { window_id: RootWI(WindowId), event: WindowEvent::KeyboardInput { @@ -169,8 +170,27 @@ impl EventLoop { } } -pub fn register(elrs: &EventLoopRunnerShared, canvas: &HtmlCanvasElement) { - add_event(elrs, canvas, "pointerout", |elrs, event: PointerEvent| { +struct SavedCallback { + pub event: &'static str, + pub callback: Box>, +} + +/** + * Manages the lifetime of the closures that are used to bind Web events to `winit` events + */ +pub struct CallbackRegistry { + runner: EventLoopRunnerShared, + event_target: EventTarget, + callbacks: Vec, +} + +impl CallbackRegistry { + pub fn new(runner: EventLoopRunnerShared, event_target: EventTarget) -> Self { + CallbackRegistry { runner, event_target, callbacks: Vec::new() } + } + + pub fn register_window_events(&mut self) { + self.add_event("pointerout", |elrs, event: PointerEvent| { elrs.send_event(Event::WindowEvent { window_id: RootWI(WindowId), event: WindowEvent::CursorLeft { @@ -178,7 +198,7 @@ pub fn register(elrs: &EventLoopRunnerShared, canvas: &HtmlCanvas }, }); }); - add_event(elrs, canvas, "pointerover", |elrs, event: PointerEvent| { + self.add_event("pointerover", |elrs, event: PointerEvent| { elrs.send_event(Event::WindowEvent { window_id: RootWI(WindowId), event: WindowEvent::CursorEntered { @@ -186,7 +206,7 @@ pub fn register(elrs: &EventLoopRunnerShared, canvas: &HtmlCanvas }, }); }); - add_event(elrs, canvas, "pointermove", |elrs, event: PointerEvent| { + self.add_event("pointermove", |elrs, event: PointerEvent| { elrs.send_event(Event::WindowEvent { window_id: RootWI(WindowId), event: WindowEvent::CursorMoved { @@ -199,7 +219,7 @@ pub fn register(elrs: &EventLoopRunnerShared, canvas: &HtmlCanvas }, }); }); - add_event(elrs, canvas, "pointerup", |elrs, event: PointerEvent| { + self.add_event("pointerup", |elrs, event: PointerEvent| { elrs.send_event(Event::WindowEvent { window_id: RootWI(WindowId), event: WindowEvent::MouseInput { @@ -210,7 +230,7 @@ pub fn register(elrs: &EventLoopRunnerShared, canvas: &HtmlCanvas }, }); }); - add_event(elrs, canvas, "pointerdown", |elrs, event: PointerEvent| { + self.add_event("pointerdown", |elrs, event: PointerEvent| { elrs.send_event(Event::WindowEvent { window_id: RootWI(WindowId), event: WindowEvent::MouseInput { @@ -221,7 +241,7 @@ pub fn register(elrs: &EventLoopRunnerShared, canvas: &HtmlCanvas }, }); }); - add_event(elrs, canvas, "wheel", |elrs, event: WheelEvent| { + self.add_event("wheel", |elrs, event: WheelEvent| { let x = event.delta_x(); let y = event.delta_y(); let delta = match event.delta_mode() { @@ -241,18 +261,17 @@ pub fn register(elrs: &EventLoopRunnerShared, canvas: &HtmlCanvas }); } -fn add_event( - elrs: &EventLoopRunnerShared, - target: &EventTarget, - event: &str, + fn add_event( + &mut self, + event: &'static str, mut handler: F, ) where E: AsRef + wasm_bindgen::convert::FromWasmAbi + 'static, F: FnMut(&EventLoopRunnerShared, E) + 'static, { - let elrs = elrs.clone(); + let elrs = self.runner.clone(); - let closure = Closure::wrap(Box::new(move |event: E| { + let callback = Closure::wrap(Box::new(move |event: E| { // Don't capture the event if the events loop has been destroyed match &*elrs.runner.borrow() { Some(ref runner) if runner.control == ControlFlow::Exit => return, @@ -267,8 +286,17 @@ fn add_event( handler(&elrs, event); }) as Box); - target.add_event_listener_with_callback(event, &closure.as_ref().unchecked_ref()); - closure.forget(); // TODO: don't leak this. + self.event_target.add_event_listener_with_callback(event, &callback.as_ref().unchecked_ref()); + self.callbacks.push(SavedCallback { event, callback: Box::new(callback) }); + } +} + +impl Drop for CallbackRegistry { + fn drop(&mut self) { + for callback in self.callbacks.iter() { + self.event_target.remove_event_listener_with_callback(callback.event, callback.callback.as_ref().as_ref().unchecked_ref()); + } + } } impl ELRShared { diff --git a/src/platform_impl/web_sys/mod.rs b/src/platform_impl/web_sys/mod.rs index 1b0591b15d..a8a4d508f3 100644 --- a/src/platform_impl/web_sys/mod.rs +++ b/src/platform_impl/web_sys/mod.rs @@ -5,7 +5,7 @@ mod events; mod window; pub use self::event_loop::{ - register, DeviceId, EventLoop, EventLoopProxy, EventLoopRunnerShared, EventLoopWindowTarget, + CallbackRegistry, DeviceId, EventLoop, EventLoopProxy, EventLoopRunnerShared, EventLoopWindowTarget, }; pub use self::events::{ button_mapping, keyboard_modifiers_state, mouse_button, mouse_modifiers_state, scancode, diff --git a/src/platform_impl/web_sys/window.rs b/src/platform_impl/web_sys/window.rs index c5c0f88ed2..85eea88daf 100644 --- a/src/platform_impl/web_sys/window.rs +++ b/src/platform_impl/web_sys/window.rs @@ -6,7 +6,8 @@ use platform::web_sys::WindowExtWebSys; use platform_impl::platform::{document, window}; use monitor::{MonitorHandle as RootMH}; use window::{CursorIcon, Window as RootWindow, WindowAttributes, WindowId as RootWI}; -use super::{EventLoopWindowTarget, OsError, register}; +use super::{CallbackRegistry, EventLoopWindowTarget, OsError}; +use std::any::Any; use std::collections::VecDeque; use std::collections::vec_deque::IntoIter as VecDequeIter; use std::cell::RefCell; @@ -51,6 +52,7 @@ pub struct Window { pub(crate) redraw: Box, previous_pointer: RefCell<&'static str>, position: RefCell, + callbacks: Box, } impl Window { @@ -64,7 +66,8 @@ impl Window { .ok_or_else(|| os_error!(OsError("Failed to find body node".to_owned())))? .append_child(&canvas).map_err(|_| os_error!(OsError("Failed to append canvas".to_owned())))?; - register(&target.runner, &canvas); + let mut callbacks = CallbackRegistry::new(target.runner.clone(), canvas.clone().unchecked_into()); + callbacks.register_window_events(); let runner = target.runner.clone(); let redraw = Box::new(move || { @@ -85,7 +88,8 @@ impl Window { position: RefCell::new(LogicalPosition { x: 0.0, y: 0.0 - }) + }), + callbacks: Box::new(callbacks), }; if let Some(inner_size) = attr.inner_size {