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
use wasm_bindgen::{closure::WasmClosure, prelude::*, JsCast};
use web_extensions_sys as sys;

// Adapted from https://github.com/rustwasm/gloo/blob/2c9e776701ecb90c53e62dec1abd19c2b70e47c7/crates/events/src/lib.rs#L232-L582
#[must_use = "event listener will never be called after being dropped"]
pub struct EventListener<'a, F: ?Sized> {
    target: &'a sys::EventTarget,
    callback: Option<Closure<F>>,
}

impl<'a, F> EventListener<'a, F>
where
    F: ?Sized + WasmClosure,
{
    #[inline]
    pub(crate) fn raw_new(target: &'a sys::EventTarget, callback: Closure<F>) -> Self {
        target.add_listener(callback.as_ref().unchecked_ref());
        Self {
            target,
            callback: Some(callback),
        }
    }

    /// Keeps the `EventListener` alive forever, so it will never be dropped.
    ///
    /// This should only be used when you want the `EventListener` to last forever, otherwise it will leak memory!
    #[inline]
    pub fn forget(mut self) {
        // take() is necessary because of Rust's restrictions about Drop
        if let Some(callback) = self.callback.take() {
            callback.forget();
        }
    }
}

impl<F: ?Sized> Drop for EventListener<'_, F> {
    #[inline]
    fn drop(&mut self) {
        if let Some(callback) = &self.callback {
            self.target
                .remove_listener(callback.as_ref().unchecked_ref());
        }
    }
}