neon/types_impl/extract/with.rs
1use crate::{
2 context::Cx,
3 result::JsResult,
4 types::{extract::TryIntoJs, Value},
5};
6
7struct With<F>(pub F);
8
9/// Wraps a closure that will be lazily evaluated when [`TryIntoJs::try_into_js`] is
10/// called.
11///
12/// Useful for executing arbitrary code on the main thread before returning from a
13/// function exported with [`neon::export`](crate::export).
14///
15/// **Note:** The return type is [`JsResult`]. If you need to return a non-JavaScript type,
16/// call [`TryIntoJs::try_into_js`].
17///
18/// _See [`With`](With#Example) for example usage._
19///
20/// ## Example
21///
22/// ```
23/// # use neon::{prelude::*, types::extract::{self, TryIntoJs}};
24/// use std::time::Instant;
25///
26/// #[neon::export(task)]
27/// fn sum(nums: Vec<f64>) -> impl for<'cx> TryIntoJs<'cx> {
28/// let start = Instant::now();
29/// let sum = nums.into_iter().sum::<f64>();
30/// let log = format!("sum took {} ms", start.elapsed().as_millis());
31///
32/// extract::with(move |cx| -> NeonResult<_> {
33/// cx.global::<JsObject>("console")?
34/// .method(cx, "log")?
35/// .arg(&log)?
36/// .exec()?;
37///
38/// sum.try_into_js(cx)
39/// })
40/// }
41/// ```
42pub fn with<V, F>(f: F) -> impl for<'cx> TryIntoJs<'cx, Value = V>
43where
44 V: Value,
45 for<'cx> F: FnOnce(&mut Cx<'cx>) -> JsResult<'cx, V>,
46{
47 With(f)
48}
49
50impl<'cx, O, F> TryIntoJs<'cx> for With<F>
51where
52 O: TryIntoJs<'cx>,
53 F: FnOnce(&mut Cx<'cx>) -> O,
54{
55 type Value = O::Value;
56
57 fn try_into_js(self, cx: &mut Cx<'cx>) -> JsResult<'cx, Self::Value> {
58 (self.0)(cx).try_into_js(cx)
59 }
60}
61
62impl<F> super::private::Sealed for With<F> {}