Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/xnought/svelte-store-anywidget

Use svelte writable stores synced with anywidget (store updates on notebook change or svelte change)
https://github.com/xnought/svelte-store-anywidget

anywidget jupyter-notebook svelte

Last synced: about 2 months ago
JSON representation

Use svelte writable stores synced with anywidget (store updates on notebook change or svelte change)

Awesome Lists containing this project

README

        

# Svelte Stores + AnyWidget

**Update: use the official implementation [`@anywidget/svelte`](https://www.npmjs.com/package/@anywidget/svelte) instead** (it has a better API too!)

If you want to create svelte components on jupyter notebooks use [anywidget](https://anywidget.dev/) + [svelte](https://svelte.dev/).

To make things easier I've created an interface to create svelte writable stores that sync with anywidget. You don't even have to think about anywidget!

https://github.com/xnought/svelte-store-anywidget/assets/65095341/b37a78d1-624f-4596-9661-dc123e8ddd91

## How?

Copy and paste the [`anywidgetStore.js`](anywidgetStore.js) file into your svelte project or just import it like I did (in [`Counter.svelte`](svelte/src/Counter.svelte)).

Then, in your svelte file you must register the anywidget model (returned from the render function (see [`main.js`](svelte/src/main.js) and [`Counter.svelte`](svelte/src/Counter.svelte)) in the svelte project).

```svelte

import { anywritable, register } from "./anywidgetStore"
export let model; // anywidget model
register(model); // register anywidget model once ever, then can use anywritable() across any component

// these are normal svelte stores synced with your notebook
// you change the value in the python notebook, this svelte store will automatically update
// you change the svelte store value, the python notebook value will auotmatically update
const count = anywritable("count"); // "count" is the name of the reactive value on the anywidget python class

{
$count++;
}}
>
{$count}

```

Notice the svelte store syntax `$`. It's that easy!

And if you want to access the store in other components, just call
`const count = anywritable("count");` again in that file. It will not create another store, but give you the one you initially created.

Note that you only need to call `register(model)` once. Then you can call `anywritable` in any component.

If you want to dispose the listeners, call `dispose(name)` where name is the name of the traitlet/value.

Or just call `disposeAll()` if you're lazy like me.

## Want to try it?

First install dependencies for the svelte components.

```bash
cd svelte
yarn
```

then run the dev server

```bash
yarn dev
```

Then out the example notebook called [`example.ipynb`](example.ipynb)

Run the notebook!

Check out the svelte files to see how I import things!

## Thank you internet

I borrowed half the code from [Trevor Manz](https://gist.github.com/manzt/af46972d7a0a8e870f5228da66c52891) to get svelte up and running with anywidget.

## Experimental Syntax

(WIP, still kind of buggy)

Instead of lugging around the `anywritable` function everywhere, this new syntax simply syncs an existing svelte store with the notebook value from anywidget.

Create the store in a file importable by any component. I'll put it in `store.js` for future reference.

```js
import { writable } from "svelte/store";

export const count = writable(0);
```

Then when I have access to the `model`, I can sync the `count` svelte store with the notebook by calling `syncAnywidget` on the `count`. Like this

```svelte

import { onDestroy } from "svelte";

import { count } from "./store";
import { syncAnywidget } from "../../../anywidgetStore";

export let model;

// call the sync once, then the stores react to changes and the notebook reacts to store changes
const disposeCount = syncAnywidget(count, model, "count");

// disposes all listeners created in the sync
onDestroy(() => {
disposeCount();
});

(Experimental sync syntax)

{
$count++;
}}
>
Increment: {$count}

```

Now, I can simply import the store in other files, and they will be usable like normal svelte stores! No need to sync again. Just once.

Check out the [`CounterExperiment.svelte`](./svelte/src/experiment/CounterExperiment.svelte) to see a usecase.