Skip to main content

perspective_js/
view.rs

1// ┏━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
2// ┃ ██████ ██████ ██████       █      █      █      █      █ █▄  ▀███ █       ┃
3// ┃ ▄▄▄▄▄█ █▄▄▄▄▄ ▄▄▄▄▄█  ▀▀▀▀▀█▀▀▀▀▀ █ ▀▀▀▀▀█ ████████▌▐███ ███▄  ▀█ █ ▀▀▀▀▀ ┃
4// ┃ █▀▀▀▀▀ █▀▀▀▀▀ █▀██▀▀ ▄▄▄▄▄ █ ▄▄▄▄▄█ ▄▄▄▄▄█ ████████▌▐███ █████▄   █ ▄▄▄▄▄ ┃
5// ┃ █      ██████ █  ▀█▄       █ ██████      █      ███▌▐███ ███████▄ █       ┃
6// ┣━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┫
7// ┃ Copyright (c) 2017, the Perspective Authors.                              ┃
8// ┃ ╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌ ┃
9// ┃ This file is part of the Perspective library, distributed under the terms ┃
10// ┃ of the [Apache License 2.0](https://www.apache.org/licenses/LICENSE-2.0). ┃
11// ┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
12
13use js_sys::{Array, ArrayBuffer, Function, Object};
14use perspective_client::{
15    ColumnWindow, OnUpdateData, OnUpdateOptions, ViewWindow, assert_view_api,
16};
17use wasm_bindgen::prelude::*;
18use wasm_bindgen_futures::spawn_local;
19
20#[cfg(doc)]
21use crate::table::Table;
22use crate::utils::{ApiFuture, ApiResult, JsValueSerdeExt, LocalPollLoop};
23
24#[wasm_bindgen]
25unsafe extern "C" {
26    #[wasm_bindgen(typescript_type = "ViewWindow")]
27    #[derive(Clone)]
28    pub type JsViewWindow;
29
30    #[wasm_bindgen(typescript_type = "ColumnWindow")]
31    #[derive(Clone)]
32    pub type JsColumnWindow;
33
34    #[wasm_bindgen(method, setter, js_name = "formatted")]
35    pub fn set_formatted(this: &JsViewWindow, x: bool);
36
37    #[wasm_bindgen(typescript_type = "OnUpdateOptions")]
38    pub type JsOnUpdateOptions;
39
40}
41
42impl From<ViewWindow> for JsViewWindow {
43    fn from(value: ViewWindow) -> Self {
44        JsViewWindow::from_serde_ext(&value)
45            .unwrap()
46            .unchecked_into()
47    }
48}
49
50fn scalar_to_jsvalue(scalar: &perspective_client::config::Scalar) -> JsValue {
51    match scalar {
52        perspective_client::config::Scalar::Float(x) => JsValue::from_f64(*x),
53        perspective_client::config::Scalar::String(x) => JsValue::from_str(x),
54        perspective_client::config::Scalar::Bool(x) => JsValue::from_bool(*x),
55        perspective_client::config::Scalar::Null => JsValue::NULL,
56    }
57}
58
59/// The [`View`] struct is Perspective's query and serialization interface. It
60/// represents a query on the `Table`'s dataset and is always created from an
61/// existing `Table` instance via the [`Table::view`] method.
62///
63/// [`View`]s are immutable with respect to the arguments provided to the
64/// [`Table::view`] method; to change these parameters, you must create a new
65/// [`View`] on the same [`Table`]. However, each [`View`] is _live_ with
66/// respect to the [`Table`]'s data, and will (within a conflation window)
67/// update with the latest state as its parent [`Table`] updates, including
68/// incrementally recalculating all aggregates, pivots, filters, etc. [`View`]
69/// query parameters are composable, in that each parameter works independently
70/// _and_ in conjunction with each other, and there is no limit to the number of
71/// pivots, filters, etc. which can be applied.
72#[wasm_bindgen]
73#[derive(Clone)]
74pub struct View(pub(crate) perspective_client::View);
75
76assert_view_api!(View);
77
78impl From<perspective_client::View> for View {
79    fn from(value: perspective_client::View) -> Self {
80        View(value)
81    }
82}
83
84#[wasm_bindgen]
85impl View {
86    #[doc(hidden)]
87    pub fn __get_model(&self) -> View {
88        self.clone()
89    }
90
91    #[wasm_bindgen]
92    #[doc(hidden)]
93    pub fn __unsafe_get_name(&self) -> String {
94        self.0.name.clone()
95    }
96
97    /// Returns an array of strings containing the column paths of the [`View`]
98    /// without any of the source columns.
99    ///
100    /// A column path shows the columns that a given cell belongs to after
101    /// pivots are applied.
102    #[wasm_bindgen]
103    pub async fn column_paths(&self, window: Option<JsColumnWindow>) -> ApiResult<JsValue> {
104        let window = window.into_serde_ext::<Option<ColumnWindow>>()?;
105        let columns = self.0.column_paths(window.unwrap_or_default()).await?;
106        Ok(JsValue::from_serde_ext(&columns)?)
107    }
108
109    /// Delete this [`View`] and clean up all resources associated with it.
110    /// [`View`] objects do not stop consuming resources or processing
111    /// updates when they are garbage collected - you must call this method
112    /// to reclaim these.
113    #[wasm_bindgen]
114    pub async fn delete(self) -> ApiResult<()> {
115        self.0.delete().await?;
116        Ok(())
117    }
118
119    /// Returns this [`View`]'s _dimensions_, row and column count, as well as
120    /// those of the [`crate::Table`] from which it was derived.
121    ///
122    /// - `num_table_rows` - The number of rows in the underlying
123    ///   [`crate::Table`].
124    /// - `num_table_columns` - The number of columns in the underlying
125    ///   [`crate::Table`] (including the `index` column if this
126    ///   [`crate::Table`] was constructed with one).
127    /// - `num_view_rows` - The number of rows in this [`View`]. If this
128    ///   [`View`] has a `group_by` clause, `num_view_rows` will also include
129    ///   aggregated rows.
130    /// - `num_view_columns` - The number of columns in this [`View`]. If this
131    ///   [`View`] has a `split_by` clause, `num_view_columns` will include all
132    ///   _column paths_, e.g. the number of `columns` clause times the number
133    ///   of `split_by` groups.
134    #[wasm_bindgen]
135    pub async fn dimensions(&self) -> ApiResult<JsValue> {
136        let dimensions = self.0.dimensions().await?;
137        Ok(JsValue::from_serde_ext(&dimensions)?)
138    }
139
140    /// The expression schema of this [`View`], which contains only the
141    /// expressions created on this [`View`]. See [`View::schema`] for
142    /// details.
143    #[wasm_bindgen]
144    pub async fn expression_schema(&self) -> ApiResult<JsValue> {
145        let schema = self.0.expression_schema().await?;
146        Ok(JsValue::from_serde_ext(&schema)?)
147    }
148
149    /// A copy of the config object passed to the [`Table::view`] method which
150    /// created this [`View`].
151    #[wasm_bindgen]
152    pub async fn get_config(&self) -> ApiResult<JsValue> {
153        let config = self.0.get_config().await?;
154        Ok(JsValue::from_serde_ext(&config)?)
155    }
156
157    /// Calculates the [min, max] of the leaf nodes of a column `column_name`.
158    ///
159    /// # Returns
160    ///
161    /// A tuple of [min, max], whose types are column and aggregate dependent.
162    #[wasm_bindgen]
163    pub async fn get_min_max(&self, name: String) -> ApiResult<Array> {
164        let result = self.0.get_min_max(name).await?;
165        let arr = Array::new();
166        arr.push(&scalar_to_jsvalue(&result.0));
167        arr.push(&scalar_to_jsvalue(&result.1));
168        Ok(arr)
169    }
170
171    /// The number of aggregated rows in this [`View`]. This is affected by the
172    /// "group_by" configuration parameter supplied to this view's contructor.
173    ///
174    /// # Returns
175    ///
176    /// The number of aggregated rows.
177    #[wasm_bindgen]
178    pub async fn num_rows(&self) -> ApiResult<i32> {
179        let size = self.0.num_rows().await?;
180        Ok(size as i32)
181    }
182
183    /// The schema of this [`View`].
184    ///
185    /// The [`View`] schema differs from the `schema` returned by
186    /// [`Table::schema`]; it may have different column names due to
187    /// `expressions` or `columns` configs, or it maye have _different
188    /// column types_ due to the application og `group_by` and `aggregates`
189    /// config. You can think of [`Table::schema`] as the _input_ schema and
190    /// [`View::schema`] as the _output_ schema of a Perspective pipeline.
191    #[wasm_bindgen]
192    pub async fn schema(&self) -> ApiResult<JsValue> {
193        let schema = self.0.schema().await?;
194        Ok(JsValue::from_serde_ext(&schema)?)
195    }
196
197    /// Serializes a [`View`] to the Apache Arrow data format.
198    #[wasm_bindgen]
199    pub async fn to_arrow(&self, window: Option<JsViewWindow>) -> ApiResult<ArrayBuffer> {
200        let window = window.into_serde_ext::<Option<ViewWindow>>()?;
201        let arrow = self.0.to_arrow(window.unwrap_or_default()).await?;
202        Ok(js_sys::Uint8Array::from(&arrow[..])
203            .buffer()
204            .unchecked_into())
205    }
206
207    /// Serializes this [`View`] to a string of JSON data. Useful if you want to
208    /// save additional round trip serialize/deserialize cycles.
209    #[wasm_bindgen]
210    pub async fn to_columns_string(&self, window: Option<JsViewWindow>) -> ApiResult<String> {
211        let window = window.into_serde_ext::<Option<ViewWindow>>()?;
212        let json = self.0.to_columns_string(window.unwrap_or_default()).await?;
213        Ok(json)
214    }
215
216    /// Serializes this [`View`] to JavaScript objects in a column-oriented
217    /// format.
218    #[wasm_bindgen]
219    pub async fn to_columns(&self, window: Option<JsViewWindow>) -> ApiResult<Object> {
220        let json = self.to_columns_string(window).await?;
221        Ok(js_sys::JSON::parse(&json)?.unchecked_into())
222    }
223
224    /// Render this `View` as a JSON string.
225    #[wasm_bindgen]
226    pub async fn to_json_string(&self, window: Option<JsViewWindow>) -> ApiResult<String> {
227        let window = window.into_serde_ext::<Option<ViewWindow>>()?;
228        let json = self.0.to_json_string(window.unwrap_or_default()).await?;
229        Ok(json)
230    }
231
232    /// Serializes this [`View`] to JavaScript objects in a row-oriented
233    /// format.
234    #[wasm_bindgen]
235    pub async fn to_json(&self, window: Option<JsViewWindow>) -> ApiResult<Array> {
236        let json = self.to_json_string(window).await?;
237        Ok(js_sys::JSON::parse(&json)?.unchecked_into())
238    }
239
240    /// Renders this [`View`] as an [NDJSON](https://github.com/ndjson/ndjson-spec)
241    /// formatted [`String`].
242    #[wasm_bindgen]
243    pub async fn to_ndjson(&self, window: Option<JsViewWindow>) -> ApiResult<String> {
244        let window = window.into_serde_ext::<Option<ViewWindow>>()?;
245        let ndjson = self.0.to_ndjson(window.unwrap_or_default()).await?;
246        Ok(ndjson)
247    }
248
249    /// Serializes this [`View`] to CSV data in a standard format.
250    #[wasm_bindgen]
251    pub async fn to_csv(&self, window: Option<JsViewWindow>) -> ApiResult<String> {
252        let window = window.into_serde_ext::<Option<ViewWindow>>()?;
253        Ok(self.0.to_csv(window.unwrap_or_default()).await?)
254    }
255
256    /// Fetches columns from the [`View`] in Arrow format, decodes them, and
257    /// passes typed array views to `callback`. All arrays are only valid for
258    /// the duration of the callback — if `callback` returns a `Promise`, it
259    /// is awaited before the backing Arrow buffer is released, so async
260    /// callbacks may use the views for the full duration of the awaited
261    /// work (e.g. across an `await requestAnimationFrame`-backed promise).
262    ///
263    /// # Arguments
264    ///
265    /// - `window` - Optional [`TypedArrayWindow`] controlling row/column
266    ///   windowing and output options (e.g., `float32` mode).
267    /// - `callback` - A JS function called with `(names: string[], values:
268    ///   TypedArray[], validities: (Uint8Array|null)[], dictionaries:
269    ///   (string[]|null)[]) => void | Promise<void>`.
270    #[wasm_bindgen]
271    pub async fn with_typed_arrays(
272        &self,
273        window: Option<crate::typed_array::JsTypedArrayWindow>,
274        callback: Function,
275    ) -> ApiResult<()> {
276        let opts: crate::typed_array::TypedArrayWindow = window
277            .into_serde_ext::<Option<crate::typed_array::TypedArrayWindow>>()?
278            .unwrap_or_default();
279
280        let float32 = opts.float32;
281        let mut view_window: ViewWindow = opts.into();
282        view_window.emit_legacy_row_path_names = Some(false);
283        let arrow = self.0.to_arrow(view_window).await?;
284        crate::typed_array::decode_and_call(&arrow, float32, &callback).await?;
285        Ok(())
286    }
287
288    /// Register a callback with this [`View`]. Whenever the view's underlying
289    /// table emits an update, this callback will be invoked with an object
290    /// containing `port_id`, indicating which port the update fired on, and
291    /// optionally `delta`, which is the new data that was updated for each
292    /// cell or each row.
293    ///
294    /// # Arguments
295    ///
296    /// - `on_update` - A callback function invoked on update, which receives an
297    ///   object with two keys: `port_id`, indicating which port the update was
298    ///   triggered on, and `delta`, whose value is dependent on the mode
299    ///   parameter.
300    /// - `options` - If this is provided as `OnUpdateOptions { mode:
301    ///   Some(OnUpdateMode::Row) }`, then `delta` is an Arrow of the updated
302    ///   rows. Otherwise `delta` will be [`Option::None`].
303    ///
304    /// # JavaScript Examples
305    ///
306    /// ```javascript
307    /// // Attach an `on_update` callback
308    /// view.on_update((updated) => console.log(updated.port_id));
309    /// ```
310    ///
311    /// ```javascript
312    /// // `on_update` with row deltas
313    /// view.on_update((updated) => console.log(updated.delta), { mode: "row" });
314    /// ```
315    #[wasm_bindgen]
316    pub fn on_update(
317        &self,
318        on_update_js: Function,
319        options: Option<JsOnUpdateOptions>,
320    ) -> ApiFuture<u32> {
321        let poll_loop = LocalPollLoop::new(move |args: OnUpdateData| {
322            let js_obj = JsValue::from_serde_ext(&*args)?;
323            on_update_js.call1(&JsValue::UNDEFINED, &js_obj)
324        });
325
326        let on_update = Box::new(move |msg| poll_loop.poll(msg));
327        let view = self.0.clone();
328        ApiFuture::new(async move {
329            let on_update_opts = options
330                .into_serde_ext::<Option<OnUpdateOptions>>()?
331                .unwrap_or_default();
332
333            let id = view.on_update(on_update, on_update_opts).await?;
334            Ok(id)
335        })
336    }
337
338    /// Unregister a previously registered update callback with this [`View`].
339    ///
340    /// # Arguments
341    ///
342    /// - `id` - A callback `id` as returned by a recipricol call to
343    ///   [`View::on_update`].
344    #[wasm_bindgen]
345    pub async fn remove_update(&self, callback_id: u32) -> ApiResult<()> {
346        Ok(self.0.remove_update(callback_id).await?)
347    }
348
349    /// Register a callback with this [`View`]. Whenever the [`View`] is
350    /// deleted, this callback will be invoked.
351    #[wasm_bindgen]
352    pub fn on_delete(&self, on_delete: Function) -> ApiFuture<u32> {
353        let view = self.clone();
354        ApiFuture::new(async move {
355            let emit = LocalPollLoop::new(move |()| on_delete.call0(&JsValue::UNDEFINED));
356            let on_delete = Box::new(move || spawn_local(emit.poll(())));
357            Ok(view.0.on_delete(on_delete).await?)
358        })
359    }
360
361    /// The number of aggregated columns in this [`View`]. This is affected by
362    /// the "split_by" configuration parameter supplied to this view's
363    /// contructor.
364    ///
365    /// # Returns
366    ///
367    /// The number of aggregated columns.
368    #[wasm_bindgen]
369    pub async fn num_columns(&self) -> ApiResult<u32> {
370        // TODO: This is broken because of how split by creates a
371        // cartesian product of columns * unique values.
372        Ok(self.0.dimensions().await?.num_view_columns)
373    }
374
375    /// Unregister a previously registered [`View::on_delete`] callback.
376    #[wasm_bindgen]
377    pub fn remove_delete(&self, callback_id: u32) -> ApiFuture<()> {
378        let client = self.0.clone();
379        ApiFuture::new(async move {
380            client.remove_delete(callback_id).await?;
381            Ok(())
382        })
383    }
384
385    /// Collapses the `group_by` row at `row_index`.
386    #[wasm_bindgen]
387    pub async fn collapse(&self, row_index: u32) -> ApiResult<u32> {
388        Ok(self.0.collapse(row_index).await?)
389    }
390
391    /// Expand the `group_by` row at `row_index`.
392    #[wasm_bindgen]
393    pub async fn expand(&self, row_index: u32) -> ApiResult<u32> {
394        Ok(self.0.expand(row_index).await?)
395    }
396
397    /// Set expansion `depth` of the `group_by` tree.
398    #[wasm_bindgen]
399    pub async fn set_depth(&self, depth: u32) -> ApiResult<()> {
400        Ok(self.0.set_depth(depth).await?)
401    }
402}