Zustand adapter to share state between pages (content script, injected script, popup, devtools, etc..) and background in web extensions.
- Runtime contexts: window (injected script), popup, devtools, content script, background, options, sidepanel (planned)
- Browsers: Chrome, Firefox, Safari, Opera, Edge + others supported by webextension-polyfill
npm install -S @webext-pegasus/transport @webext-pegasus/store-zustand
- Create a store based on https://github.com/pmndrs/zustand.
- You can create a store either reactive way or vanilla.
- Wrap the store with
initPegasusZustandStoreBackend
. Import the store from the background or any other extension context. - Wait for the store to be ready (connected to background) via
pegasusZustandStoreReady
.
That's it! Now your store is available from everywhere.
Tip
Refer to ./packages/example-extension for more examples.
store.ts
import { create } from 'zustand'
// or import { createStore } from 'zustand/vanilla'
import { initPegasusZustandStoreBackend, pegasusZustandStoreReady } from 'webext-zustand'
interface BearState {
bears: number
increase: (by: number) => void
}
export const useBearStore = create<BearState>()((set) => ({
bears: 0,
increase: (by) => set((state) => ({ bears: state.bears + by })),
}))
export const STORE_NAME = 'GlobalBearStore';
export const bearStoreBackendReady = () =>
initPegasusZustandStoreBackend(STORE_NAME, useBearStore);
export const bearStoreReady = () =>
pegasusZustandStoreReady(STORE_NAME, useBearStore);
background.ts
import { initPegasusTransport } from '@webext-pegasus/transport/background';
import { initPegasusZustandStoreBackend } from '@webext-pegasus/store-zustand';
import {bearStoreBackendReady} from './store';
initPegasusTransport();
bearStoreBackendReady().then(store => {
// listen state changes
store.subscribe((state) => {
// console.log(state);
});
// dispatch
// store.getState().increase(2);
});
popup.tsx
import React from 'react';
import { createRoot } from 'react-dom/client';
import {initPegasusTransport} from '@webext-pegasus/transport/popup';
import { useBearStore, bearStoreReady } from './store';
const Popup = () => {
const bears = useBearStore((state) => state.bears);
const increase = useBearStore((state) => state.increase);
return (
<div>
Popup
<div>
<span>Bears: {bears}</span>
<br />
<button onClick={() => increase(1)}>Increment +</button>
</div>
</div>
);
};
// Init surface specific APIs
initPegasusTransport();
bearStoreReady().then(() => {
createRoot(document.getElementById("root") as HTMLElement).render(
<React.StrictMode>
<Popup />
</React.StrictMode>
);
});
content-script.tsx
import React from 'react';
import { createRoot } from 'react-dom/client';
import {initPegasusTransport} from '@webext-pegasus/transport/content-script';
import { useBearStore, bearStoreReady } from './store';
const Content = () => {
const bears = useBearStore((state) => state.bears);
const increase = useBearStore((state) => state.increase);
return (
<div>
Content
<div>
<span>Bears: {bears}</span>
<br />
<button onClick={() => increase(1)}>Increment +</button>
</div>
</div>
);
};
// Init surface specific APIs
initPegasusTransport();
bearStoreReady().then(() => {
const root = document.createElement("div");
document.body.prepend(root);
createRoot(root).render(
<React.StrictMode>
<Content />
</React.StrictMode>
);
});
In MV3 extensions A service worker replaces the extension's background or event page to ensure that background code stays off the main thread. This enables extensions to run only when needed, saving resources.
However this also means that on the contrary to the constantly running background script MV3 SW terminates when not in use, you'll need to persist application states rather than rely on global variables.
Fortunately this library provides a simple solution for this problem! You only need to do couple of changes to your extension:
-
Enable
storage
permission for your extension. State for your Pegasus Stores will be persisted viabrowser.storage
API -
Pass and additional
storageStrategy
parameter to witin your background script:initPegasusZustandStoreBackend(STORE_NAME, store, { storageStrategy: 'session' // use "local" to persist store across browser sessions or "sync" for store used for extension settings });
This library is based on the Sinan Bekar's implementation of the webext-zustand. However it adds support of the injected scripts & utilizes newer, simplified transport layer.