Functional & Reactive programming (FRP) simplifies handling complex, dynamic data flows: It is particularly useful in scenarios with asynchronous data sources, such as user interfaces or real-time data feeds.
cells is a simplified Functional & Reactive (but not strictly FRP) library
inspired by spreadsheets.
cells focuses on automation as it transparently manages:
asynccalls- undefined values
- errors
- pointers
Although it is a fully independent library, cells can be a powerful drop-in
replacement for Svelte stores as it implements the Svelte store interface. There
is also a React hook
and a demo with Vue 3.
import { Sheet } from "@okcontract/cells";
// create a new Sheet, typically one per application
const sheet = new Sheet();
// that's a cell
const cellA = sheet.new(1);
// no need to wait for anything, values and computations will
// be called automatically when they're ready
const cellB = sheet.new(getValueAfterDelay(2, 100));
// that's a _mapped_ cell, that is recomputed automatically when
// either dependent cells are updated
const sum = sheet.map([cellA, cellB], (a, b) => a + b);
// you can map a single cell directly
// and feel free to define mapped cells before their values are available
const prod = cellB.map((v) => 2 * v);
// we await _only_ when we need results
expect(await prod.get()).toBe(4);
// you can update _value_ cells directly (no await)
cellA.set(3);
expect(await sum.get()).toBe(5);
// map computations can be async too
const ok = someCell.map(async (v) => {
const errors = await validate(v);
return errors.length === 0;
});Note that:
.get()never returnsundefinedandcellssemantics waits until cell values are defined. If you need to immediately return a value, usenullinstead.- At any time, the actual cell value is accessible through
cell.value(possibly undefined) but it's advisable to avoid relying on a value that could be updated at any moment.
You can define subscribers functions to be called whenever a cell value change.
const cellA = sheet.new(1);
// create subscriptions on any cell
const unsubscribe = cellA.subscribe((v) => {
console.log({ v });
});
// later call `unsubscribe` to cancel the subscription
unsubscribe();When using cells in Svelte files, you can use the $cell sugar to subscribe
to a cell value for display. Note that there is a initial undefined value
when using that sugar as the reactive subscription are initially output by the
Svelte compiler as let $cell;.
Unlike Svelte, subscriptions are called in transactional batches, i.e. cells
wait until all updates are propagated in the Sheet to dispatch notifications
once to subscribers. This prevents stuttering.
Cells are not garbage collected automatically, since the best practice is to
keep a long-running Sheet for each application.
To delete a cell, use:
sheet.delete(cell);To simplify these operations, you can define a sub-graph of cells in a Proxy
that can be deleted at once.
import { SheetProxy } from "@okcontract/cells";
// create a proxy (for instance in a Svelte component)
const proxy = new SheetProxy(sheet);
// use the proxy to create new cells...
const cell = proxy.new(1);
// ...and mapped cells
const mappedCell = proxy.map([...cells], (...args)=>{...})
// delete the proxy when you're done
proxy.destroy();There are added benefits of proxies, including a single call to wait for all
cells in a Proxy to be computed.
await proxy.working.wait();Sometimes, you will need to return cells inside .map compute functions.
cells manages this automatically with pointers.
const cellA = proxy.new(...);
const cellB = proxy.new(...);
// pointerCell will either be cellA or cellB
const pointerCell = proxy.map([cellA, cellB], (a, b) =>
condition ? cellA : cellB
);
// breathe normally
await pointerCell.get();
pointerCell.map(...);It's easy to update cells depending on their previous value:
const counter = proxy.new(0);
// ...
counter.update((prev) => prev + 1);Note that update functions should return new arrays and objects, for
instance using spread operators [...prev, new] and {...prev, field: new}.
If you need more complex imperative updates, we suggest you to use immer:
import { produce } from "immer";
const patch = (prev) => {
// complex updates...
};
cell.update((prev) => produce(prev, patch));The design and philosophy of cells comes from frameworks such as RxJS,
MobX, and Recoil. Our main objective is to enhance developer experience by
simplifying the complexity typically associated with state management and
reactive programming.
We aim for ease of use and correction, so chasing down any bug is our top priority.
A non-goal is high-performance (we're not a tensor library!): cells is
slower than any direct implementation and should not be used for
computationally intensive tasks. However, cells optimizes the global
computation within a program, potentially leading to more optimal execution
than can be achieved through imperative computations from manually organized
functions or modules.
cells is built at OKcontract and is released under
the MIT license.
Contributors are welcome, feel free to submit PRs directly for small changes. You can also reach out in our Discord or contact us on Twitter in advance for larger contributions.
This work is supported in part by a RFG grant from Optimism.