pub struct Closure<T>where
T: ?Sized,{ /* private fields */ }
dep_web_sys
only.Expand description
A handle to both a closure in Rust as well as JS closure which will invoke the Rust closure.
A Closure
is the primary way that a 'static
lifetime closure is
transferred from Rust to JS. Closure
currently requires that the closures
it’s created with have the 'static
lifetime in Rust for soundness reasons.
This type is a “handle” in the sense that whenever it is dropped it will
invalidate the JS closure that it refers to. Any usage of the closure in JS
after the Closure
has been dropped will raise an exception. It’s then up
to you to arrange for Closure
to be properly deallocate at an appropriate
location in your program.
The type parameter on Closure
is the type of closure that this represents.
Currently this can only be the Fn
and FnMut
traits with up to 7
arguments (and an optional return value).
§Examples
Here are a number of examples of using Closure
.
§Using the setInterval
API
Sample usage of Closure
to invoke the setInterval
API.
use wasm_bindgen::prelude::*;
#[wasm_bindgen]
extern "C" {
fn setInterval(closure: &Closure<dyn FnMut()>, time: u32) -> i32;
fn clearInterval(id: i32);
#[wasm_bindgen(js_namespace = console)]
fn log(s: &str);
}
#[wasm_bindgen]
pub struct IntervalHandle {
interval_id: i32,
_closure: Closure<dyn FnMut()>,
}
impl Drop for IntervalHandle {
fn drop(&mut self) {
clearInterval(self.interval_id);
}
}
#[wasm_bindgen]
pub fn run() -> IntervalHandle {
// First up we use `Closure::new` to wrap up a Rust closure and create
// a JS closure.
let cb = Closure::new(|| {
log("interval elapsed!");
});
// Next we pass this via reference to the `setInterval` function, and
// `setInterval` gets a handle to the corresponding JS closure.
let interval_id = setInterval(&cb, 1_000);
// If we were to drop `cb` here it would cause an exception to be raised
// whenever the interval elapses. Instead we *return* our handle back to JS
// so JS can decide when to cancel the interval and deallocate the closure.
IntervalHandle {
interval_id,
_closure: cb,
}
}
§Casting a Closure
to a js_sys::Function
This is the same setInterval
example as above, except it is using
web_sys
(which uses js_sys::Function
for callbacks) instead of manually
writing bindings to setInterval
and other Web APIs.
use wasm_bindgen::JsCast;
#[wasm_bindgen]
pub struct IntervalHandle {
interval_id: i32,
_closure: Closure<dyn FnMut()>,
}
impl Drop for IntervalHandle {
fn drop(&mut self) {
let window = web_sys::window().unwrap();
window.clear_interval_with_handle(self.interval_id);
}
}
#[wasm_bindgen]
pub fn run() -> Result<IntervalHandle, JsValue> {
let cb = Closure::new(|| {
web_sys::console::log_1(&"interval elapsed!".into());
});
let window = web_sys::window().unwrap();
let interval_id = window.set_interval_with_callback_and_timeout_and_arguments_0(
// Note this method call, which uses `as_ref()` to get a `JsValue`
// from our `Closure` which is then converted to a `&Function`
// using the `JsCast::unchecked_ref` function.
cb.as_ref().unchecked_ref(),
1_000,
)?;
// Same as above.
Ok(IntervalHandle {
interval_id,
_closure: cb,
})
}
§Using FnOnce
and Closure::once
with requestAnimationFrame
Because requestAnimationFrame
only calls its callback once, we can use
FnOnce
and Closure::once
with it.
use wasm_bindgen::prelude::*;
#[wasm_bindgen]
extern "C" {
fn requestAnimationFrame(closure: &Closure<dyn FnMut()>) -> u32;
fn cancelAnimationFrame(id: u32);
#[wasm_bindgen(js_namespace = console)]
fn log(s: &str);
}
#[wasm_bindgen]
pub struct AnimationFrameHandle {
animation_id: u32,
_closure: Closure<dyn FnMut()>,
}
impl Drop for AnimationFrameHandle {
fn drop(&mut self) {
cancelAnimationFrame(self.animation_id);
}
}
// A type that will log a message when it is dropped.
struct LogOnDrop(&'static str);
impl Drop for LogOnDrop {
fn drop(&mut self) {
log(self.0);
}
}
#[wasm_bindgen]
pub fn run() -> AnimationFrameHandle {
// We are using `Closure::once` which takes a `FnOnce`, so the function
// can drop and/or move things that it closes over.
let fired = LogOnDrop("animation frame fired or canceled");
let cb = Closure::once(move || drop(fired));
// Schedule the animation frame!
let animation_id = requestAnimationFrame(&cb);
// Again, return a handle to JS, so that the closure is not dropped
// immediately and JS can decide whether to cancel the animation frame.
AnimationFrameHandle {
animation_id,
_closure: cb,
}
}
§Converting FnOnce
s directly into JavaScript Functions with Closure::once_into_js
If we don’t want to allow a FnOnce
to be eagerly dropped (maybe because we
just want it to drop after it is called and don’t care about cancellation)
then we can use the Closure::once_into_js
function.
This is the same requestAnimationFrame
example as above, but without
supporting early cancellation.
use wasm_bindgen::prelude::*;
#[wasm_bindgen]
extern "C" {
// We modify the binding to take an untyped `JsValue` since that is what
// is returned by `Closure::once_into_js`.
//
// If we were using the `web_sys` binding for `requestAnimationFrame`,
// then the call sites would cast the `JsValue` into a `&js_sys::Function`
// using `f.unchecked_ref::<js_sys::Function>()`. See the `web_sys`
// example above for details.
fn requestAnimationFrame(callback: JsValue);
#[wasm_bindgen(js_namespace = console)]
fn log(s: &str);
}
// A type that will log a message when it is dropped.
struct LogOnDrop(&'static str);
impl Drop for LogOnDrop {
fn drop(&mut self) {
log(self.0);
}
}
#[wasm_bindgen]
pub fn run() {
// We are using `Closure::once_into_js` which takes a `FnOnce` and
// converts it into a JavaScript function, which is returned as a
// `JsValue`.
let fired = LogOnDrop("animation frame fired");
let cb = Closure::once_into_js(move || drop(fired));
// Schedule the animation frame!
requestAnimationFrame(cb);
// No need to worry about whether or not we drop a `Closure`
// here or return some sort of handle to JS!
}
Implementations§
Source§impl<T> Closure<T>where
T: WasmClosure + ?Sized,
impl<T> Closure<T>where
T: WasmClosure + ?Sized,
Sourcepub fn new<F>(t: F) -> Closure<T>where
F: IntoWasmClosure<T> + 'static,
Available on crate feature dep_js_sys
only.
pub fn new<F>(t: F) -> Closure<T>where
F: IntoWasmClosure<T> + 'static,
dep_js_sys
only.Creates a new instance of Closure
from the provided Rust function.
Note that the closure provided here, F
, has a few requirements
associated with it:
-
It must implement
Fn
orFnMut
(forFnOnce
functions seeClosure::once
andClosure::once_into_js
). -
It must be
'static
, aka no stack references (use themove
keyword). -
It can have at most 7 arguments.
-
Its arguments and return values are all types that can be shared with JS (i.e. have
#[wasm_bindgen]
annotations or are simple numbers, etc.)
Sourcepub fn wrap(data: Box<T>) -> Closure<T>
Available on crate feature dep_js_sys
only.
pub fn wrap(data: Box<T>) -> Closure<T>
dep_js_sys
only.A more direct version of Closure::new
which creates a Closure
from
a Box<dyn Fn>
/Box<dyn FnMut>
, which is how it’s kept internally.
Sourcepub fn into_js_value(self) -> JsValue
Available on crate feature dep_js_sys
only.
pub fn into_js_value(self) -> JsValue
dep_js_sys
only.Release memory management of this closure from Rust to the JS GC.
When a Closure
is dropped it will release the Rust memory and
invalidate the associated JS closure, but this isn’t always desired.
Some callbacks are alive for the entire duration of the program or for a
lifetime dynamically managed by the JS GC. This function can be used
to drop this Closure
while keeping the associated JS function still
valid.
If the platform supports weak references, the Rust memory will be reclaimed when the JS closure is GC’d. If weak references is not supported, this can be dangerous if this function is called many times in an application because the memory leak will overwhelm the page quickly and crash the wasm.
Source§impl Closure<dyn FnOnce()>
impl Closure<dyn FnOnce()>
Sourcepub fn once<F, A, R>(
fn_once: F,
) -> Closure<<F as WasmClosureFnOnce<A, R>>::FnMut>where
F: 'static + WasmClosureFnOnce<A, R>,
Available on crate feature dep_js_sys
only.
pub fn once<F, A, R>(
fn_once: F,
) -> Closure<<F as WasmClosureFnOnce<A, R>>::FnMut>where
F: 'static + WasmClosureFnOnce<A, R>,
dep_js_sys
only.Create a Closure
from a function that can only be called once.
Since we have no way of enforcing that JS cannot attempt to call this
FnOne(A...) -> R
more than once, this produces a Closure<dyn FnMut(A...) -> R>
that will dynamically throw a JavaScript error if called more
than once.
§Example
use wasm_bindgen::prelude::*;
// Create an non-`Copy`, owned `String`.
let mut s = String::from("Hello");
// Close over `s`. Since `f` returns `s`, it is `FnOnce` and can only be
// called once. If it was called a second time, it wouldn't have any `s`
// to work with anymore!
let f = move || {
s += ", World!";
s
};
// Create a `Closure` from `f`. Note that the `Closure`'s type parameter
// is `FnMut`, even though `f` is `FnOnce`.
let closure: Closure<dyn FnMut() -> String> = Closure::once(f);
Sourcepub fn once_into_js<F, A, R>(fn_once: F) -> JsValuewhere
F: 'static + WasmClosureFnOnce<A, R>,
Available on crate feature dep_js_sys
only.
pub fn once_into_js<F, A, R>(fn_once: F) -> JsValuewhere
F: 'static + WasmClosureFnOnce<A, R>,
dep_js_sys
only.Convert a FnOnce(A...) -> R
into a JavaScript Function
object.
If the JavaScript function is invoked more than once, it will throw an exception.
Unlike Closure::once
, this does not return a Closure
that can be
dropped before the function is invoked to deallocate the closure. The
only way the FnOnce
is deallocated is by calling the JavaScript
function. If the JavaScript function is never called then the FnOnce
and everything it closes over will leak.
use wasm_bindgen::{prelude::*, JsCast};
let f = Closure::once_into_js(move || {
// ...
});
assert!(f.is_instance_of::<js_sys::Function>());
Trait Implementations§
Source§impl<T> IntoWasmAbi for &Closure<T>where
T: WasmClosure + ?Sized,
impl<T> IntoWasmAbi for &Closure<T>where
T: WasmClosure + ?Sized,
Auto Trait Implementations§
impl<T> Freeze for Closure<T>where
T: ?Sized,
impl<T> RefUnwindSafe for Closure<T>where
T: RefUnwindSafe + ?Sized,
impl<T> !Send for Closure<T>
impl<T> !Sync for Closure<T>
impl<T> Unpin for Closure<T>where
T: ?Sized,
impl<T> UnwindSafe for Closure<T>where
T: UnwindSafe + ?Sized,
Blanket Implementations§
§impl<T> ArchivePointee for T
impl<T> ArchivePointee for T
§type ArchivedMetadata = ()
type ArchivedMetadata = ()
§fn pointer_metadata(
_: &<T as ArchivePointee>::ArchivedMetadata,
) -> <T as Pointee>::Metadata
fn pointer_metadata( _: &<T as ArchivePointee>::ArchivedMetadata, ) -> <T as Pointee>::Metadata
Source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
Source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
Source§impl<T> ByteSized for T
impl<T> ByteSized for T
Source§const BYTE_ALIGN: usize = _
const BYTE_ALIGN: usize = _
Source§fn byte_align(&self) -> usize ⓘ
fn byte_align(&self) -> usize ⓘ
Source§fn ptr_size_ratio(&self) -> [usize; 2]
fn ptr_size_ratio(&self) -> [usize; 2]
Source§impl<T, R> Chain<R> for Twhere
T: ?Sized,
impl<T, R> Chain<R> for Twhere
T: ?Sized,
Source§impl<T> ExtAny for T
impl<T> ExtAny for T
Source§fn as_any_mut(&mut self) -> &mut dyn Anywhere
Self: Sized,
fn as_any_mut(&mut self) -> &mut dyn Anywhere
Self: Sized,
Source§impl<T> ExtMem for Twhere
T: ?Sized,
impl<T> ExtMem for Twhere
T: ?Sized,
Source§const NEEDS_DROP: bool = _
const NEEDS_DROP: bool = _
Source§fn mem_align_of_val(&self) -> usize ⓘ
fn mem_align_of_val(&self) -> usize ⓘ
Source§fn mem_size_of_val(&self) -> usize ⓘ
fn mem_size_of_val(&self) -> usize ⓘ
Source§fn mem_needs_drop(&self) -> bool
fn mem_needs_drop(&self) -> bool
true
if dropping values of this type matters. Read moreSource§fn mem_forget(self)where
Self: Sized,
fn mem_forget(self)where
Self: Sized,
self
without running its destructor. Read moreSource§fn mem_replace(&mut self, other: Self) -> Selfwhere
Self: Sized,
fn mem_replace(&mut self, other: Self) -> Selfwhere
Self: Sized,
Source§unsafe fn mem_zeroed<T>() -> T
unsafe fn mem_zeroed<T>() -> T
unsafe_layout
only.T
represented by the all-zero byte-pattern. Read moreSource§unsafe fn mem_transmute_copy<Src, Dst>(src: &Src) -> Dst
unsafe fn mem_transmute_copy<Src, Dst>(src: &Src) -> Dst
unsafe_layout
only.T
represented by the all-zero byte-pattern. Read moreSource§fn mem_as_bytes(&self) -> &[u8] ⓘ
fn mem_as_bytes(&self) -> &[u8] ⓘ
unsafe_slice
only.§impl<S> FromSample<S> for S
impl<S> FromSample<S> for S
fn from_sample_(s: S) -> S
Source§impl<T> Hook for T
impl<T> Hook for T
§impl<T> Instrument for T
impl<T> Instrument for T
§fn instrument(self, span: Span) -> Instrumented<Self> ⓘ
fn instrument(self, span: Span) -> Instrumented<Self> ⓘ
§fn in_current_span(self) -> Instrumented<Self> ⓘ
fn in_current_span(self) -> Instrumented<Self> ⓘ
Source§impl<T> IntoEither for T
impl<T> IntoEither for T
Source§fn into_either(self, into_left: bool) -> Either<Self, Self> ⓘ
fn into_either(self, into_left: bool) -> Either<Self, Self> ⓘ
self
into a Left
variant of Either<Self, Self>
if into_left
is true
.
Converts self
into a Right
variant of Either<Self, Self>
otherwise. Read moreSource§fn into_either_with<F>(self, into_left: F) -> Either<Self, Self> ⓘ
fn into_either_with<F>(self, into_left: F) -> Either<Self, Self> ⓘ
self
into a Left
variant of Either<Self, Self>
if into_left(&self)
returns true
.
Converts self
into a Right
variant of Either<Self, Self>
otherwise. Read more§impl<F, T> IntoSample<T> for Fwhere
T: FromSample<F>,
impl<F, T> IntoSample<T> for Fwhere
T: FromSample<F>,
fn into_sample(self) -> T
§impl<T> LayoutRaw for T
impl<T> LayoutRaw for T
§fn layout_raw(_: <T as Pointee>::Metadata) -> Result<Layout, LayoutError> ⓘ
fn layout_raw(_: <T as Pointee>::Metadata) -> Result<Layout, LayoutError> ⓘ
§impl<T, N1, N2> Niching<NichedOption<T, N1>> for N2
impl<T, N1, N2> Niching<NichedOption<T, N1>> for N2
§unsafe fn is_niched(niched: *const NichedOption<T, N1>) -> bool
unsafe fn is_niched(niched: *const NichedOption<T, N1>) -> bool
§fn resolve_niched(out: Place<NichedOption<T, N1>>)
fn resolve_niched(out: Place<NichedOption<T, N1>>)
out
indicating that a T
is niched.