@plasius/react-state 1.2.0 → 1.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +3 -1
- package/dist/index.cjs +10 -1
- package/dist/index.cjs.map +1 -1
- package/dist/index.js +10 -1
- package/dist/index.js.map +1 -1
- package/package.json +4 -1
package/README.md
CHANGED
|
@@ -1,13 +1,15 @@
|
|
|
1
1
|
# @plasius/react-state
|
|
2
2
|
|
|
3
3
|
[](https://www.npmjs.com/package/@plasius/react-state)
|
|
4
|
-
[](https://github.com/
|
|
4
|
+
[](https://github.com/Plasius-LTD/react-state/actions/workflows/ci.yml)
|
|
5
5
|
[](https://codecov.io/gh/Plasius-LTD/react-state)
|
|
6
6
|
[](./LICENSE)
|
|
7
7
|
[](./CODE_OF_CONDUCT.md)
|
|
8
8
|
[](./SECURITY.md)
|
|
9
9
|
[](./CHANGELOG.md)
|
|
10
10
|
|
|
11
|
+
Apache-2.0. ESM + CJS builds. TypeScript types included.
|
|
12
|
+
|
|
11
13
|
---
|
|
12
14
|
|
|
13
15
|
## Overview
|
package/dist/index.cjs
CHANGED
|
@@ -55,7 +55,16 @@ function deepFreeze(obj, seen = /* @__PURE__ */ new WeakSet()) {
|
|
|
55
55
|
|
|
56
56
|
// src/store.ts
|
|
57
57
|
var import_meta = {};
|
|
58
|
-
var DEV =
|
|
58
|
+
var DEV = (() => {
|
|
59
|
+
const metaEnv = import_meta.env;
|
|
60
|
+
if (metaEnv && Object.prototype.hasOwnProperty.call(metaEnv, "DEV")) {
|
|
61
|
+
return Boolean(metaEnv.DEV);
|
|
62
|
+
}
|
|
63
|
+
if (typeof process !== "undefined" && process.env) {
|
|
64
|
+
return process.env.NODE_ENV !== "production";
|
|
65
|
+
}
|
|
66
|
+
return true;
|
|
67
|
+
})();
|
|
59
68
|
function devTrack(name, props) {
|
|
60
69
|
if (!DEV) return;
|
|
61
70
|
try {
|
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts","../src/types.ts","../src/create-scoped-store.tsx","../src/freeze.ts","../src/store.ts","../src/provider.tsx","../src/metadata-store.ts"],"sourcesContent":["export * from \"./types.js\";\nexport * from \"./create-scoped-store.js\";\nexport * from \"./store.js\";\nexport * from \"./provider.js\";\nexport * from \"./metadata-store.js\";\n","export type Reducer<S, A> = (state: S, action: A) => S;\nexport type Listener = () => void;\nexport type Unsubscribe = () => void;\nexport const __noop = null;","import { createContext, useContext, useRef, useSyncExternalStore, useCallback, useEffect, useMemo } from \"react\";\nimport type { IState, IAction, Store } from \"./store.js\";\nimport { createStore } from \"./store.js\";\n\n// DEV-only tracking (no-op in production)\nconst __DEV__ = typeof process !== \"undefined\" ? process.env.NODE_ENV !== \"production\" : true;\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!__DEV__) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\n// Local microtask-based batching (no react-dom dependency)\nconst _queue = new Set<() => void>();\nlet _queued = false;\nconst _schedule = typeof queueMicrotask === \"function\"\n ? queueMicrotask\n : (fn: () => void) => Promise.resolve().then(fn);\n\nfunction enqueue(fn: () => void) {\n _queue.add(fn);\n if (_queued) return;\n _queued = true;\n _schedule(() => {\n _queued = false;\n const fns = Array.from(_queue);\n _queue.clear();\n for (const f of fns) f();\n devTrack(\"scoped:batch:flush\", { size: fns.length });\n });\n}\n\nfunction makeBatchedSubscribe(subscribe: (l: () => void) => () => void) {\n return (onChange: () => void) => {\n devTrack(\"scoped:sub:add\");\n const wrapped = () => {\n devTrack(\"scoped:notify:enqueue\");\n enqueue(onChange);\n };\n const unsubscribe = subscribe(wrapped);\n return () => {\n devTrack(\"scoped:sub:remove\");\n unsubscribe();\n };\n };\n}\n\nfunction shallowEqual(a: any, b: any) {\n if (Object.is(a, b)) return true;\n if (\n typeof a !== \"object\" ||\n a === null ||\n typeof b !== \"object\" ||\n b === null\n )\n return false;\n const ak = Object.keys(a),\n bk = Object.keys(b);\n if (ak.length !== bk.length) return false;\n for (let i = 0; i < ak.length; i++) {\n const k = ak[i] as string;\n if (\n !Object.prototype.hasOwnProperty.call(b, k) ||\n !Object.is((a as any)[k], (b as any)[k])\n )\n return false;\n }\n return true;\n}\n\nexport function createScopedStoreContext<S extends IState, A extends IAction>(\n reducer: (state: S, action: A) => S,\n initialState: S\n) {\n const Context = createContext<Store<S, A> | null>(null);\n\n // Each Provider instance gets its own store.\n const Provider = ({\n children,\n initialState: override,\n }: {\n children: React.ReactNode;\n initialState?: S;\n }) => {\n const store = useMemo(() => {\n const next = createStore(reducer, override ?? initialState);\n devTrack(\"scoped:store:create\");\n return next;\n }, [reducer, override, initialState]);\n\n useEffect(() => {\n devTrack(\"scoped:provider:mount\");\n return () => devTrack(\"scoped:provider:unmount\");\n }, []);\n\n return <Context.Provider value={store}>{children}</Context.Provider>;\n };\n\n const useStore = (): S => {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Store not found in context\");\n devTrack(\"scoped:useStore\");\n return useSyncExternalStore(\n makeBatchedSubscribe(ctx.subscribe),\n ctx.getState,\n ctx.getState\n );\n };\n\n const useDispatch = (): ((action: A) => void) => {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Dispatch not found in context\");\n return useCallback((action: A) => ctx.dispatch(action), [ctx]);\n };\n\n function useSelector<T>(\n selector: (state: S) => T,\n isEqual: (a: T, b: T) => boolean = shallowEqual\n ): T {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Store not found in context\");\n\n const lastRef = useRef<{ selected: T } | null>(null);\n\n const getSnapshot = () => {\n const nextSelected = selector(ctx.getState());\n const last = lastRef.current;\n if (last && isEqual(last.selected, nextSelected)) {\n devTrack(\"scoped:selector:cache-hit\");\n return last.selected;\n }\n devTrack(\"scoped:selector:cache-miss\");\n lastRef.current = { selected: nextSelected };\n return nextSelected;\n };\n\n const subscribe = (onChange: () => void) => {\n devTrack(\"scoped:selector:sub:add\");\n const unsubscribe = ctx.subscribeWithSelector(selector, (nextSelected) => {\n const last = lastRef.current;\n if (last && isEqual(last.selected, nextSelected)) {\n devTrack(\"scoped:selector:skip\");\n return;\n }\n lastRef.current = { selected: nextSelected };\n devTrack(\"scoped:selector:notify\");\n enqueue(onChange);\n }, isEqual);\n return () => {\n devTrack(\"scoped:selector:sub:remove\");\n unsubscribe();\n };\n };\n\n return useSyncExternalStore(subscribe, getSnapshot, getSnapshot);\n }\n\n return {\n Context,\n Provider,\n useStore,\n useDispatch,\n useSelector,\n };\n}\n","// freeze.ts\nexport function deepFreeze<T>(obj: T, seen = new WeakSet<object>()): T {\n if (obj === null || typeof obj !== \"object\") return obj;\n const o = obj as unknown as object;\n if (seen.has(o)) return obj;\n seen.add(o);\n\n // Freeze children first\n for (const key of Object.getOwnPropertyNames(o)) {\n // @ts-expect-error index access\n const val = (o as any)[key];\n if (val && typeof val === \"object\") deepFreeze(val, seen);\n }\n // Also handle symbols (rare but safe)\n for (const sym of Object.getOwnPropertySymbols(o)) {\n // @ts-expect-error index access\n const val = (o as any)[sym];\n if (val && typeof val === \"object\") deepFreeze(val, seen);\n }\n\n return Object.freeze(obj);\n}\n","import type { Reducer, Listener } from \"./types.js\";\nimport { deepFreeze } from \"./freeze.js\";\n\nconst DEV =\n typeof import.meta !== \"undefined\"\n ? (import.meta as unknown as { env?: { DEV?: boolean } }).env?.DEV\n : process.env.NODE_ENV !== \"production\";\n\n// Lightweight DEV-only tracker\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!DEV) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\n// Allow narrower parameter types for callbacks without fighting variance\ntype BivariantListener<T> = {\n bivarianceHack(value: T): void;\n}[\"bivarianceHack\"];\n\n// eslint-disable-next-line @typescript-eslint/no-empty-object-type\nexport interface IState {}\nexport interface IAction {\n type: string;\n}\n\nexport interface Store<S extends IState, A extends IAction> {\n getState(): S;\n dispatch(action: A): void;\n /**\n * Subscribe to all state changes.\n */\n subscribe(listener: Listener): () => void;\n /**\n * Subscribe to changes of a specific key in the state.\n */\n subscribeToKey<K extends keyof S>(\n key: K,\n listener: (value: S[K]) => void\n ): () => void;\n /**\n * Subscribe to changes in a selected value from the state.\n */\n subscribeWithSelector<T>(\n selector: (state: S) => T,\n listener: (selected: T) => void,\n isEqual?: (a: T, b: T) => boolean\n ): () => void;\n}\n\nexport function createStore<S extends IState, A extends IAction>(\n reducer: Reducer<S, A>,\n initialState: S\n): Store<S, A> {\n let state: S = DEV ? deepFreeze(initialState) : initialState;\n const listeners = new Set<Listener>();\n const keyListeners = new Map<keyof S, Set<BivariantListener<S[keyof S]>>>();\n\n interface SelectorEntry<T> {\n selector: (state: S) => T;\n listener: BivariantListener<T>;\n lastValue: T;\n isEqual?: (a: T, b: T) => boolean;\n }\n const selectorListeners = new Set<SelectorEntry<unknown>>();\n\n const getState = () => state;\n\n const dispatch = (action: A) => {\n const prevState = state;\n const nextState = reducer(state, action);\n\n if (DEV) deepFreeze(nextState);\n\n // Track the inbound action\n devTrack(\"store:dispatch\", { type: action?.type });\n\n // Distinct-until-changed: if the reducer returns the same reference,\n // skip all notifications (prevents unnecessary re-renders).\n if (Object.is(prevState, nextState)) {\n state = nextState; // keep any identity guarantees from reducer\n devTrack(\"store:no-op\", { type: action?.type });\n return;\n }\n\n state = nextState;\n\n // Compute changed keys (shallow) for diagnostics\n let changedKeys: (keyof S)[] | undefined;\n if (DEV) {\n changedKeys = Object.keys(nextState as Record<string, unknown>)\n .filter((k) => !Object.is((prevState as any)[k], (nextState as any)[k])) as (keyof S)[];\n devTrack(\"store:state-changed\", { type: action?.type, changedKeys });\n }\n\n // Notify global listeners (iterate over a snapshot so unsubscribe during\n // notify does not skip the next listener)\n const globalSnapshot = [...listeners];\n devTrack(\"store:notify:all\", { listeners: globalSnapshot.length });\n let firstError: unknown;\n for (const listener of globalSnapshot) {\n try {\n listener();\n } catch (err) {\n if (!firstError) firstError = err;\n }\n }\n\n // Notify key listeners only when that key actually changed (Object.is)\n for (const [key, set] of keyListeners.entries()) {\n if (!Object.is(prevState[key], state[key])) {\n devTrack(\"store:notify:key\", { key: String(key), listeners: set.size });\n for (const listener of [...set]) {\n try {\n listener(state[key]);\n } catch (err) {\n if (!firstError) firstError = err;\n }\n }\n }\n }\n\n // Notify selector listeners only when selected value changed (Object.is)\n let selNotifies = 0;\n for (const entry of [...selectorListeners]) {\n const nextValue = (entry.selector as (s: S) => unknown)(state);\n const equal = (entry.isEqual as ((a: unknown, b: unknown) => boolean) | undefined) ?? Object.is;\n let isSame = false;\n try {\n isSame = equal(entry.lastValue, nextValue);\n } catch (err) {\n if (!firstError) firstError = err;\n continue;\n }\n if (!isSame) {\n entry.lastValue = nextValue as unknown;\n try {\n (entry.listener as (v: unknown) => void)(nextValue);\n } catch (err) {\n if (!firstError) firstError = err;\n }\n selNotifies++;\n }\n }\n devTrack(\"store:notify:selector\", { listeners: selNotifies });\n if (firstError) throw firstError;\n };\n\n const subscribe = (listener: Listener) => {\n listeners.add(listener);\n devTrack(\"store:sub:all:add\", { size: listeners.size });\n return () => {\n listeners.delete(listener);\n devTrack(\"store:sub:all:remove\", { size: listeners.size });\n };\n };\n\n const subscribeToKey = <K extends keyof S>(\n key: K,\n listener: (value: S[K]) => void\n ) => {\n const set =\n keyListeners.get(key) ?? new Set<BivariantListener<S[keyof S]>>();\n set.add(listener as unknown as BivariantListener<S[keyof S]>);\n keyListeners.set(key, set);\n devTrack(\"store:sub:key:add\", { key: String(key), size: set.size });\n return () => {\n set.delete(listener as unknown as BivariantListener<S[keyof S]>);\n if (set.size === 0) keyListeners.delete(key);\n devTrack(\"store:sub:key:remove\", { key: String(key), size: set.size });\n };\n };\n\n const subscribeWithSelector = <T>(\n selector: (state: S) => T,\n listener: (selected: T) => void,\n isEqual?: (a: T, b: T) => boolean\n ) => {\n const entry: SelectorEntry<T> = {\n selector,\n listener: listener as BivariantListener<T>,\n lastValue: selector(state),\n isEqual,\n };\n selectorListeners.add(entry as unknown as SelectorEntry<unknown>);\n devTrack(\"store:sub:selector:add\", { size: selectorListeners.size });\n return () => {\n selectorListeners.delete(entry as unknown as SelectorEntry<unknown>);\n devTrack(\"store:sub:selector:remove\", { size: selectorListeners.size });\n };\n };\n\n return {\n getState,\n dispatch,\n subscribe,\n subscribeToKey,\n subscribeWithSelector,\n };\n}\n","import React, { createContext, useContext, useEffect, useSyncExternalStore } from \"react\";\nimport type { ReactNode } from \"react\";\nimport type { Store, IState, IAction } from \"./store.js\";\n\n// DEV-only tracking (no-op in production)\nconst __DEV__ = typeof process !== \"undefined\" ? process.env.NODE_ENV !== \"production\" : true;\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!__DEV__) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\nconst StoreContext = createContext<Store<IState, IAction> | undefined>(undefined);\n\nfunction useStoreInstance<S extends IState, A extends IAction>(): Store<S, A> {\n const store = useContext(StoreContext) as Store<S, A> | undefined;\n if (!store) {\n devTrack(\"store:provider:missing\");\n throw new Error(\n \"StoreProvider is missing in the React tree. Wrap your app with <StoreProvider store={...}>.\"\n );\n }\n return store;\n}\n\ninterface StoreProviderProps<S extends IState, A extends IAction> {\n store: Store<S, A>;\n children: ReactNode;\n}\n\nexport function StoreProvider<S extends IState, A extends IAction>({\n store,\n children,\n}: StoreProviderProps<S, A>) {\n useEffect(() => {\n devTrack(\"store:provider:mount\", { hasStore: !!store });\n return () => devTrack(\"store:provider:unmount\");\n }, [store]);\n return (\n <StoreContext.Provider value={store as unknown as Store<IState, IAction>}>\n {children}\n </StoreContext.Provider>\n );\n}\n\nexport function useStore<S extends IState>(): S {\n const store = useStoreInstance<S, IAction>();\n return useSyncExternalStore(\n (onStoreChange) => {\n devTrack(\"store:react:subscribe\");\n const unsubscribe = store.subscribe(() => {\n devTrack(\"store:react:notify\");\n onStoreChange();\n });\n return () => {\n devTrack(\"store:react:unsubscribe\");\n unsubscribe();\n };\n },\n store.getState,\n store.getState\n );\n}\n\nexport function useDispatch<A extends IAction>(): Store<IState, A>[\"dispatch\"] {\n const store = useStoreInstance<IState, A>();\n // Return the store's dispatch directly; consumers can call dispatch(action).\n return store.dispatch as Store<IState, A>[\"dispatch\"];\n}\n","// metadata-store.ts\nexport class MetadataStore<T extends object, Meta extends object> {\n private readonly symbol: symbol;\n\n constructor(description: string) {\n this.symbol = Symbol(description);\n }\n\n set(target: T, meta: Meta) {\n Object.defineProperty(target, this.symbol as PropertyKey, {\n value: meta,\n writable: false,\n enumerable: false,\n });\n }\n\n get(target: T): Meta | undefined {\n return (target as Record<PropertyKey, Meta>)[this.symbol as PropertyKey];\n }\n\n has(target: T): boolean {\n return (this.symbol as PropertyKey) in target;\n }\n}\n"],"mappings":"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","names":["devTrack","useStore","useDispatch","import_react","import_jsx_runtime","__DEV__","devTrack"]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/types.ts","../src/create-scoped-store.tsx","../src/freeze.ts","../src/store.ts","../src/provider.tsx","../src/metadata-store.ts"],"sourcesContent":["export * from \"./types.js\";\nexport * from \"./create-scoped-store.js\";\nexport * from \"./store.js\";\nexport * from \"./provider.js\";\nexport * from \"./metadata-store.js\";\n","export type Reducer<S, A> = (state: S, action: A) => S;\nexport type Listener = () => void;\nexport type Unsubscribe = () => void;\nexport const __noop = null;","import { createContext, useContext, useRef, useSyncExternalStore, useCallback, useEffect, useMemo } from \"react\";\nimport type { IState, IAction, Store } from \"./store.js\";\nimport { createStore } from \"./store.js\";\n\n// DEV-only tracking (no-op in production)\nconst __DEV__ = typeof process !== \"undefined\" ? process.env.NODE_ENV !== \"production\" : true;\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!__DEV__) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\n// Local microtask-based batching (no react-dom dependency)\nconst _queue = new Set<() => void>();\nlet _queued = false;\nconst _schedule = typeof queueMicrotask === \"function\"\n ? queueMicrotask\n : (fn: () => void) => Promise.resolve().then(fn);\n\nfunction enqueue(fn: () => void) {\n _queue.add(fn);\n if (_queued) return;\n _queued = true;\n _schedule(() => {\n _queued = false;\n const fns = Array.from(_queue);\n _queue.clear();\n for (const f of fns) f();\n devTrack(\"scoped:batch:flush\", { size: fns.length });\n });\n}\n\nfunction makeBatchedSubscribe(subscribe: (l: () => void) => () => void) {\n return (onChange: () => void) => {\n devTrack(\"scoped:sub:add\");\n const wrapped = () => {\n devTrack(\"scoped:notify:enqueue\");\n enqueue(onChange);\n };\n const unsubscribe = subscribe(wrapped);\n return () => {\n devTrack(\"scoped:sub:remove\");\n unsubscribe();\n };\n };\n}\n\nfunction shallowEqual(a: any, b: any) {\n if (Object.is(a, b)) return true;\n if (\n typeof a !== \"object\" ||\n a === null ||\n typeof b !== \"object\" ||\n b === null\n )\n return false;\n const ak = Object.keys(a),\n bk = Object.keys(b);\n if (ak.length !== bk.length) return false;\n for (let i = 0; i < ak.length; i++) {\n const k = ak[i] as string;\n if (\n !Object.prototype.hasOwnProperty.call(b, k) ||\n !Object.is((a as any)[k], (b as any)[k])\n )\n return false;\n }\n return true;\n}\n\nexport function createScopedStoreContext<S extends IState, A extends IAction>(\n reducer: (state: S, action: A) => S,\n initialState: S\n) {\n const Context = createContext<Store<S, A> | null>(null);\n\n // Each Provider instance gets its own store.\n const Provider = ({\n children,\n initialState: override,\n }: {\n children: React.ReactNode;\n initialState?: S;\n }) => {\n const store = useMemo(() => {\n const next = createStore(reducer, override ?? initialState);\n devTrack(\"scoped:store:create\");\n return next;\n }, [reducer, override, initialState]);\n\n useEffect(() => {\n devTrack(\"scoped:provider:mount\");\n return () => devTrack(\"scoped:provider:unmount\");\n }, []);\n\n return <Context.Provider value={store}>{children}</Context.Provider>;\n };\n\n const useStore = (): S => {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Store not found in context\");\n devTrack(\"scoped:useStore\");\n return useSyncExternalStore(\n makeBatchedSubscribe(ctx.subscribe),\n ctx.getState,\n ctx.getState\n );\n };\n\n const useDispatch = (): ((action: A) => void) => {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Dispatch not found in context\");\n return useCallback((action: A) => ctx.dispatch(action), [ctx]);\n };\n\n function useSelector<T>(\n selector: (state: S) => T,\n isEqual: (a: T, b: T) => boolean = shallowEqual\n ): T {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Store not found in context\");\n\n const lastRef = useRef<{ selected: T } | null>(null);\n\n const getSnapshot = () => {\n const nextSelected = selector(ctx.getState());\n const last = lastRef.current;\n if (last && isEqual(last.selected, nextSelected)) {\n devTrack(\"scoped:selector:cache-hit\");\n return last.selected;\n }\n devTrack(\"scoped:selector:cache-miss\");\n lastRef.current = { selected: nextSelected };\n return nextSelected;\n };\n\n const subscribe = (onChange: () => void) => {\n devTrack(\"scoped:selector:sub:add\");\n const unsubscribe = ctx.subscribeWithSelector(selector, (nextSelected) => {\n const last = lastRef.current;\n if (last && isEqual(last.selected, nextSelected)) {\n devTrack(\"scoped:selector:skip\");\n return;\n }\n lastRef.current = { selected: nextSelected };\n devTrack(\"scoped:selector:notify\");\n enqueue(onChange);\n }, isEqual);\n return () => {\n devTrack(\"scoped:selector:sub:remove\");\n unsubscribe();\n };\n };\n\n return useSyncExternalStore(subscribe, getSnapshot, getSnapshot);\n }\n\n return {\n Context,\n Provider,\n useStore,\n useDispatch,\n useSelector,\n };\n}\n","// freeze.ts\nexport function deepFreeze<T>(obj: T, seen = new WeakSet<object>()): T {\n if (obj === null || typeof obj !== \"object\") return obj;\n const o = obj as unknown as object;\n if (seen.has(o)) return obj;\n seen.add(o);\n\n // Freeze children first\n for (const key of Object.getOwnPropertyNames(o)) {\n // @ts-expect-error index access\n const val = (o as any)[key];\n if (val && typeof val === \"object\") deepFreeze(val, seen);\n }\n // Also handle symbols (rare but safe)\n for (const sym of Object.getOwnPropertySymbols(o)) {\n // @ts-expect-error index access\n const val = (o as any)[sym];\n if (val && typeof val === \"object\") deepFreeze(val, seen);\n }\n\n return Object.freeze(obj);\n}\n","import type { Reducer, Listener } from \"./types.js\";\nimport { deepFreeze } from \"./freeze.js\";\n\nconst DEV = (() => {\n const metaEnv = (import.meta as unknown as { env?: { DEV?: boolean } }).env;\n if (metaEnv && Object.prototype.hasOwnProperty.call(metaEnv, \"DEV\")) {\n return Boolean(metaEnv.DEV);\n }\n if (typeof process !== \"undefined\" && process.env) {\n return process.env.NODE_ENV !== \"production\";\n }\n return true;\n})();\n\n// Lightweight DEV-only tracker\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!DEV) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\n// Allow narrower parameter types for callbacks without fighting variance\ntype BivariantListener<T> = {\n bivarianceHack(value: T): void;\n}[\"bivarianceHack\"];\n\n// eslint-disable-next-line @typescript-eslint/no-empty-object-type\nexport interface IState {}\nexport interface IAction {\n type: string;\n}\n\nexport interface Store<S extends IState, A extends IAction> {\n getState(): S;\n dispatch(action: A): void;\n /**\n * Subscribe to all state changes.\n */\n subscribe(listener: Listener): () => void;\n /**\n * Subscribe to changes of a specific key in the state.\n */\n subscribeToKey<K extends keyof S>(\n key: K,\n listener: (value: S[K]) => void\n ): () => void;\n /**\n * Subscribe to changes in a selected value from the state.\n */\n subscribeWithSelector<T>(\n selector: (state: S) => T,\n listener: (selected: T) => void,\n isEqual?: (a: T, b: T) => boolean\n ): () => void;\n}\n\nexport function createStore<S extends IState, A extends IAction>(\n reducer: Reducer<S, A>,\n initialState: S\n): Store<S, A> {\n let state: S = DEV ? deepFreeze(initialState) : initialState;\n const listeners = new Set<Listener>();\n const keyListeners = new Map<keyof S, Set<BivariantListener<S[keyof S]>>>();\n\n interface SelectorEntry<T> {\n selector: (state: S) => T;\n listener: BivariantListener<T>;\n lastValue: T;\n isEqual?: (a: T, b: T) => boolean;\n }\n const selectorListeners = new Set<SelectorEntry<unknown>>();\n\n const getState = () => state;\n\n const dispatch = (action: A) => {\n const prevState = state;\n const nextState = reducer(state, action);\n\n if (DEV) deepFreeze(nextState);\n\n // Track the inbound action\n devTrack(\"store:dispatch\", { type: action?.type });\n\n // Distinct-until-changed: if the reducer returns the same reference,\n // skip all notifications (prevents unnecessary re-renders).\n if (Object.is(prevState, nextState)) {\n state = nextState; // keep any identity guarantees from reducer\n devTrack(\"store:no-op\", { type: action?.type });\n return;\n }\n\n state = nextState;\n\n // Compute changed keys (shallow) for diagnostics\n let changedKeys: (keyof S)[] | undefined;\n if (DEV) {\n changedKeys = Object.keys(nextState as Record<string, unknown>)\n .filter((k) => !Object.is((prevState as any)[k], (nextState as any)[k])) as (keyof S)[];\n devTrack(\"store:state-changed\", { type: action?.type, changedKeys });\n }\n\n // Notify global listeners (iterate over a snapshot so unsubscribe during\n // notify does not skip the next listener)\n const globalSnapshot = [...listeners];\n devTrack(\"store:notify:all\", { listeners: globalSnapshot.length });\n let firstError: unknown;\n for (const listener of globalSnapshot) {\n try {\n listener();\n } catch (err) {\n if (!firstError) firstError = err;\n }\n }\n\n // Notify key listeners only when that key actually changed (Object.is)\n for (const [key, set] of keyListeners.entries()) {\n if (!Object.is(prevState[key], state[key])) {\n devTrack(\"store:notify:key\", { key: String(key), listeners: set.size });\n for (const listener of [...set]) {\n try {\n listener(state[key]);\n } catch (err) {\n if (!firstError) firstError = err;\n }\n }\n }\n }\n\n // Notify selector listeners only when selected value changed (Object.is)\n let selNotifies = 0;\n for (const entry of [...selectorListeners]) {\n const nextValue = (entry.selector as (s: S) => unknown)(state);\n const equal = (entry.isEqual as ((a: unknown, b: unknown) => boolean) | undefined) ?? Object.is;\n let isSame = false;\n try {\n isSame = equal(entry.lastValue, nextValue);\n } catch (err) {\n if (!firstError) firstError = err;\n continue;\n }\n if (!isSame) {\n entry.lastValue = nextValue as unknown;\n try {\n (entry.listener as (v: unknown) => void)(nextValue);\n } catch (err) {\n if (!firstError) firstError = err;\n }\n selNotifies++;\n }\n }\n devTrack(\"store:notify:selector\", { listeners: selNotifies });\n if (firstError) throw firstError;\n };\n\n const subscribe = (listener: Listener) => {\n listeners.add(listener);\n devTrack(\"store:sub:all:add\", { size: listeners.size });\n return () => {\n listeners.delete(listener);\n devTrack(\"store:sub:all:remove\", { size: listeners.size });\n };\n };\n\n const subscribeToKey = <K extends keyof S>(\n key: K,\n listener: (value: S[K]) => void\n ) => {\n const set =\n keyListeners.get(key) ?? new Set<BivariantListener<S[keyof S]>>();\n set.add(listener as unknown as BivariantListener<S[keyof S]>);\n keyListeners.set(key, set);\n devTrack(\"store:sub:key:add\", { key: String(key), size: set.size });\n return () => {\n set.delete(listener as unknown as BivariantListener<S[keyof S]>);\n if (set.size === 0) keyListeners.delete(key);\n devTrack(\"store:sub:key:remove\", { key: String(key), size: set.size });\n };\n };\n\n const subscribeWithSelector = <T>(\n selector: (state: S) => T,\n listener: (selected: T) => void,\n isEqual?: (a: T, b: T) => boolean\n ) => {\n const entry: SelectorEntry<T> = {\n selector,\n listener: listener as BivariantListener<T>,\n lastValue: selector(state),\n isEqual,\n };\n selectorListeners.add(entry as unknown as SelectorEntry<unknown>);\n devTrack(\"store:sub:selector:add\", { size: selectorListeners.size });\n return () => {\n selectorListeners.delete(entry as unknown as SelectorEntry<unknown>);\n devTrack(\"store:sub:selector:remove\", { size: selectorListeners.size });\n };\n };\n\n return {\n getState,\n dispatch,\n subscribe,\n subscribeToKey,\n subscribeWithSelector,\n };\n}\n","import React, { createContext, useContext, useEffect, useSyncExternalStore } from \"react\";\nimport type { ReactNode } from \"react\";\nimport type { Store, IState, IAction } from \"./store.js\";\n\n// DEV-only tracking (no-op in production)\nconst __DEV__ = typeof process !== \"undefined\" ? process.env.NODE_ENV !== \"production\" : true;\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!__DEV__) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\nconst StoreContext = createContext<Store<IState, IAction> | undefined>(undefined);\n\nfunction useStoreInstance<S extends IState, A extends IAction>(): Store<S, A> {\n const store = useContext(StoreContext) as Store<S, A> | undefined;\n if (!store) {\n devTrack(\"store:provider:missing\");\n throw new Error(\n \"StoreProvider is missing in the React tree. Wrap your app with <StoreProvider store={...}>.\"\n );\n }\n return store;\n}\n\ninterface StoreProviderProps<S extends IState, A extends IAction> {\n store: Store<S, A>;\n children: ReactNode;\n}\n\nexport function StoreProvider<S extends IState, A extends IAction>({\n store,\n children,\n}: StoreProviderProps<S, A>) {\n useEffect(() => {\n devTrack(\"store:provider:mount\", { hasStore: !!store });\n return () => devTrack(\"store:provider:unmount\");\n }, [store]);\n return (\n <StoreContext.Provider value={store as unknown as Store<IState, IAction>}>\n {children}\n </StoreContext.Provider>\n );\n}\n\nexport function useStore<S extends IState>(): S {\n const store = useStoreInstance<S, IAction>();\n return useSyncExternalStore(\n (onStoreChange) => {\n devTrack(\"store:react:subscribe\");\n const unsubscribe = store.subscribe(() => {\n devTrack(\"store:react:notify\");\n onStoreChange();\n });\n return () => {\n devTrack(\"store:react:unsubscribe\");\n unsubscribe();\n };\n },\n store.getState,\n store.getState\n );\n}\n\nexport function useDispatch<A extends IAction>(): Store<IState, A>[\"dispatch\"] {\n const store = useStoreInstance<IState, A>();\n // Return the store's dispatch directly; consumers can call dispatch(action).\n return store.dispatch as Store<IState, A>[\"dispatch\"];\n}\n","// metadata-store.ts\nexport class MetadataStore<T extends object, Meta extends object> {\n private readonly symbol: symbol;\n\n constructor(description: string) {\n this.symbol = Symbol(description);\n }\n\n set(target: T, meta: Meta) {\n Object.defineProperty(target, this.symbol as PropertyKey, {\n value: meta,\n writable: false,\n enumerable: false,\n });\n }\n\n get(target: T): Meta | undefined {\n return (target as Record<PropertyKey, Meta>)[this.symbol as PropertyKey];\n }\n\n has(target: T): boolean {\n return (this.symbol as PropertyKey) in target;\n }\n}\n"],"mappings":"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","names":["devTrack","useStore","useDispatch","import_react","import_jsx_runtime","__DEV__","devTrack"]}
|
package/dist/index.js
CHANGED
|
@@ -22,7 +22,16 @@ function deepFreeze(obj, seen = /* @__PURE__ */ new WeakSet()) {
|
|
|
22
22
|
}
|
|
23
23
|
|
|
24
24
|
// src/store.ts
|
|
25
|
-
var DEV =
|
|
25
|
+
var DEV = (() => {
|
|
26
|
+
const metaEnv = import.meta.env;
|
|
27
|
+
if (metaEnv && Object.prototype.hasOwnProperty.call(metaEnv, "DEV")) {
|
|
28
|
+
return Boolean(metaEnv.DEV);
|
|
29
|
+
}
|
|
30
|
+
if (typeof process !== "undefined" && process.env) {
|
|
31
|
+
return process.env.NODE_ENV !== "production";
|
|
32
|
+
}
|
|
33
|
+
return true;
|
|
34
|
+
})();
|
|
26
35
|
function devTrack(name, props) {
|
|
27
36
|
if (!DEV) return;
|
|
28
37
|
try {
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/types.ts","../src/create-scoped-store.tsx","../src/freeze.ts","../src/store.ts","../src/provider.tsx","../src/metadata-store.ts"],"sourcesContent":["export type Reducer<S, A> = (state: S, action: A) => S;\nexport type Listener = () => void;\nexport type Unsubscribe = () => void;\nexport const __noop = null;","import { createContext, useContext, useRef, useSyncExternalStore, useCallback, useEffect, useMemo } from \"react\";\nimport type { IState, IAction, Store } from \"./store.js\";\nimport { createStore } from \"./store.js\";\n\n// DEV-only tracking (no-op in production)\nconst __DEV__ = typeof process !== \"undefined\" ? process.env.NODE_ENV !== \"production\" : true;\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!__DEV__) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\n// Local microtask-based batching (no react-dom dependency)\nconst _queue = new Set<() => void>();\nlet _queued = false;\nconst _schedule = typeof queueMicrotask === \"function\"\n ? queueMicrotask\n : (fn: () => void) => Promise.resolve().then(fn);\n\nfunction enqueue(fn: () => void) {\n _queue.add(fn);\n if (_queued) return;\n _queued = true;\n _schedule(() => {\n _queued = false;\n const fns = Array.from(_queue);\n _queue.clear();\n for (const f of fns) f();\n devTrack(\"scoped:batch:flush\", { size: fns.length });\n });\n}\n\nfunction makeBatchedSubscribe(subscribe: (l: () => void) => () => void) {\n return (onChange: () => void) => {\n devTrack(\"scoped:sub:add\");\n const wrapped = () => {\n devTrack(\"scoped:notify:enqueue\");\n enqueue(onChange);\n };\n const unsubscribe = subscribe(wrapped);\n return () => {\n devTrack(\"scoped:sub:remove\");\n unsubscribe();\n };\n };\n}\n\nfunction shallowEqual(a: any, b: any) {\n if (Object.is(a, b)) return true;\n if (\n typeof a !== \"object\" ||\n a === null ||\n typeof b !== \"object\" ||\n b === null\n )\n return false;\n const ak = Object.keys(a),\n bk = Object.keys(b);\n if (ak.length !== bk.length) return false;\n for (let i = 0; i < ak.length; i++) {\n const k = ak[i] as string;\n if (\n !Object.prototype.hasOwnProperty.call(b, k) ||\n !Object.is((a as any)[k], (b as any)[k])\n )\n return false;\n }\n return true;\n}\n\nexport function createScopedStoreContext<S extends IState, A extends IAction>(\n reducer: (state: S, action: A) => S,\n initialState: S\n) {\n const Context = createContext<Store<S, A> | null>(null);\n\n // Each Provider instance gets its own store.\n const Provider = ({\n children,\n initialState: override,\n }: {\n children: React.ReactNode;\n initialState?: S;\n }) => {\n const store = useMemo(() => {\n const next = createStore(reducer, override ?? initialState);\n devTrack(\"scoped:store:create\");\n return next;\n }, [reducer, override, initialState]);\n\n useEffect(() => {\n devTrack(\"scoped:provider:mount\");\n return () => devTrack(\"scoped:provider:unmount\");\n }, []);\n\n return <Context.Provider value={store}>{children}</Context.Provider>;\n };\n\n const useStore = (): S => {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Store not found in context\");\n devTrack(\"scoped:useStore\");\n return useSyncExternalStore(\n makeBatchedSubscribe(ctx.subscribe),\n ctx.getState,\n ctx.getState\n );\n };\n\n const useDispatch = (): ((action: A) => void) => {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Dispatch not found in context\");\n return useCallback((action: A) => ctx.dispatch(action), [ctx]);\n };\n\n function useSelector<T>(\n selector: (state: S) => T,\n isEqual: (a: T, b: T) => boolean = shallowEqual\n ): T {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Store not found in context\");\n\n const lastRef = useRef<{ selected: T } | null>(null);\n\n const getSnapshot = () => {\n const nextSelected = selector(ctx.getState());\n const last = lastRef.current;\n if (last && isEqual(last.selected, nextSelected)) {\n devTrack(\"scoped:selector:cache-hit\");\n return last.selected;\n }\n devTrack(\"scoped:selector:cache-miss\");\n lastRef.current = { selected: nextSelected };\n return nextSelected;\n };\n\n const subscribe = (onChange: () => void) => {\n devTrack(\"scoped:selector:sub:add\");\n const unsubscribe = ctx.subscribeWithSelector(selector, (nextSelected) => {\n const last = lastRef.current;\n if (last && isEqual(last.selected, nextSelected)) {\n devTrack(\"scoped:selector:skip\");\n return;\n }\n lastRef.current = { selected: nextSelected };\n devTrack(\"scoped:selector:notify\");\n enqueue(onChange);\n }, isEqual);\n return () => {\n devTrack(\"scoped:selector:sub:remove\");\n unsubscribe();\n };\n };\n\n return useSyncExternalStore(subscribe, getSnapshot, getSnapshot);\n }\n\n return {\n Context,\n Provider,\n useStore,\n useDispatch,\n useSelector,\n };\n}\n","// freeze.ts\nexport function deepFreeze<T>(obj: T, seen = new WeakSet<object>()): T {\n if (obj === null || typeof obj !== \"object\") return obj;\n const o = obj as unknown as object;\n if (seen.has(o)) return obj;\n seen.add(o);\n\n // Freeze children first\n for (const key of Object.getOwnPropertyNames(o)) {\n // @ts-expect-error index access\n const val = (o as any)[key];\n if (val && typeof val === \"object\") deepFreeze(val, seen);\n }\n // Also handle symbols (rare but safe)\n for (const sym of Object.getOwnPropertySymbols(o)) {\n // @ts-expect-error index access\n const val = (o as any)[sym];\n if (val && typeof val === \"object\") deepFreeze(val, seen);\n }\n\n return Object.freeze(obj);\n}\n","import type { Reducer, Listener } from \"./types.js\";\nimport { deepFreeze } from \"./freeze.js\";\n\nconst DEV =\n typeof import.meta !== \"undefined\"\n ? (import.meta as unknown as { env?: { DEV?: boolean } }).env?.DEV\n : process.env.NODE_ENV !== \"production\";\n\n// Lightweight DEV-only tracker\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!DEV) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\n// Allow narrower parameter types for callbacks without fighting variance\ntype BivariantListener<T> = {\n bivarianceHack(value: T): void;\n}[\"bivarianceHack\"];\n\n// eslint-disable-next-line @typescript-eslint/no-empty-object-type\nexport interface IState {}\nexport interface IAction {\n type: string;\n}\n\nexport interface Store<S extends IState, A extends IAction> {\n getState(): S;\n dispatch(action: A): void;\n /**\n * Subscribe to all state changes.\n */\n subscribe(listener: Listener): () => void;\n /**\n * Subscribe to changes of a specific key in the state.\n */\n subscribeToKey<K extends keyof S>(\n key: K,\n listener: (value: S[K]) => void\n ): () => void;\n /**\n * Subscribe to changes in a selected value from the state.\n */\n subscribeWithSelector<T>(\n selector: (state: S) => T,\n listener: (selected: T) => void,\n isEqual?: (a: T, b: T) => boolean\n ): () => void;\n}\n\nexport function createStore<S extends IState, A extends IAction>(\n reducer: Reducer<S, A>,\n initialState: S\n): Store<S, A> {\n let state: S = DEV ? deepFreeze(initialState) : initialState;\n const listeners = new Set<Listener>();\n const keyListeners = new Map<keyof S, Set<BivariantListener<S[keyof S]>>>();\n\n interface SelectorEntry<T> {\n selector: (state: S) => T;\n listener: BivariantListener<T>;\n lastValue: T;\n isEqual?: (a: T, b: T) => boolean;\n }\n const selectorListeners = new Set<SelectorEntry<unknown>>();\n\n const getState = () => state;\n\n const dispatch = (action: A) => {\n const prevState = state;\n const nextState = reducer(state, action);\n\n if (DEV) deepFreeze(nextState);\n\n // Track the inbound action\n devTrack(\"store:dispatch\", { type: action?.type });\n\n // Distinct-until-changed: if the reducer returns the same reference,\n // skip all notifications (prevents unnecessary re-renders).\n if (Object.is(prevState, nextState)) {\n state = nextState; // keep any identity guarantees from reducer\n devTrack(\"store:no-op\", { type: action?.type });\n return;\n }\n\n state = nextState;\n\n // Compute changed keys (shallow) for diagnostics\n let changedKeys: (keyof S)[] | undefined;\n if (DEV) {\n changedKeys = Object.keys(nextState as Record<string, unknown>)\n .filter((k) => !Object.is((prevState as any)[k], (nextState as any)[k])) as (keyof S)[];\n devTrack(\"store:state-changed\", { type: action?.type, changedKeys });\n }\n\n // Notify global listeners (iterate over a snapshot so unsubscribe during\n // notify does not skip the next listener)\n const globalSnapshot = [...listeners];\n devTrack(\"store:notify:all\", { listeners: globalSnapshot.length });\n let firstError: unknown;\n for (const listener of globalSnapshot) {\n try {\n listener();\n } catch (err) {\n if (!firstError) firstError = err;\n }\n }\n\n // Notify key listeners only when that key actually changed (Object.is)\n for (const [key, set] of keyListeners.entries()) {\n if (!Object.is(prevState[key], state[key])) {\n devTrack(\"store:notify:key\", { key: String(key), listeners: set.size });\n for (const listener of [...set]) {\n try {\n listener(state[key]);\n } catch (err) {\n if (!firstError) firstError = err;\n }\n }\n }\n }\n\n // Notify selector listeners only when selected value changed (Object.is)\n let selNotifies = 0;\n for (const entry of [...selectorListeners]) {\n const nextValue = (entry.selector as (s: S) => unknown)(state);\n const equal = (entry.isEqual as ((a: unknown, b: unknown) => boolean) | undefined) ?? Object.is;\n let isSame = false;\n try {\n isSame = equal(entry.lastValue, nextValue);\n } catch (err) {\n if (!firstError) firstError = err;\n continue;\n }\n if (!isSame) {\n entry.lastValue = nextValue as unknown;\n try {\n (entry.listener as (v: unknown) => void)(nextValue);\n } catch (err) {\n if (!firstError) firstError = err;\n }\n selNotifies++;\n }\n }\n devTrack(\"store:notify:selector\", { listeners: selNotifies });\n if (firstError) throw firstError;\n };\n\n const subscribe = (listener: Listener) => {\n listeners.add(listener);\n devTrack(\"store:sub:all:add\", { size: listeners.size });\n return () => {\n listeners.delete(listener);\n devTrack(\"store:sub:all:remove\", { size: listeners.size });\n };\n };\n\n const subscribeToKey = <K extends keyof S>(\n key: K,\n listener: (value: S[K]) => void\n ) => {\n const set =\n keyListeners.get(key) ?? new Set<BivariantListener<S[keyof S]>>();\n set.add(listener as unknown as BivariantListener<S[keyof S]>);\n keyListeners.set(key, set);\n devTrack(\"store:sub:key:add\", { key: String(key), size: set.size });\n return () => {\n set.delete(listener as unknown as BivariantListener<S[keyof S]>);\n if (set.size === 0) keyListeners.delete(key);\n devTrack(\"store:sub:key:remove\", { key: String(key), size: set.size });\n };\n };\n\n const subscribeWithSelector = <T>(\n selector: (state: S) => T,\n listener: (selected: T) => void,\n isEqual?: (a: T, b: T) => boolean\n ) => {\n const entry: SelectorEntry<T> = {\n selector,\n listener: listener as BivariantListener<T>,\n lastValue: selector(state),\n isEqual,\n };\n selectorListeners.add(entry as unknown as SelectorEntry<unknown>);\n devTrack(\"store:sub:selector:add\", { size: selectorListeners.size });\n return () => {\n selectorListeners.delete(entry as unknown as SelectorEntry<unknown>);\n devTrack(\"store:sub:selector:remove\", { size: selectorListeners.size });\n };\n };\n\n return {\n getState,\n dispatch,\n subscribe,\n subscribeToKey,\n subscribeWithSelector,\n };\n}\n","import React, { createContext, useContext, useEffect, useSyncExternalStore } from \"react\";\nimport type { ReactNode } from \"react\";\nimport type { Store, IState, IAction } from \"./store.js\";\n\n// DEV-only tracking (no-op in production)\nconst __DEV__ = typeof process !== \"undefined\" ? process.env.NODE_ENV !== \"production\" : true;\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!__DEV__) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\nconst StoreContext = createContext<Store<IState, IAction> | undefined>(undefined);\n\nfunction useStoreInstance<S extends IState, A extends IAction>(): Store<S, A> {\n const store = useContext(StoreContext) as Store<S, A> | undefined;\n if (!store) {\n devTrack(\"store:provider:missing\");\n throw new Error(\n \"StoreProvider is missing in the React tree. Wrap your app with <StoreProvider store={...}>.\"\n );\n }\n return store;\n}\n\ninterface StoreProviderProps<S extends IState, A extends IAction> {\n store: Store<S, A>;\n children: ReactNode;\n}\n\nexport function StoreProvider<S extends IState, A extends IAction>({\n store,\n children,\n}: StoreProviderProps<S, A>) {\n useEffect(() => {\n devTrack(\"store:provider:mount\", { hasStore: !!store });\n return () => devTrack(\"store:provider:unmount\");\n }, [store]);\n return (\n <StoreContext.Provider value={store as unknown as Store<IState, IAction>}>\n {children}\n </StoreContext.Provider>\n );\n}\n\nexport function useStore<S extends IState>(): S {\n const store = useStoreInstance<S, IAction>();\n return useSyncExternalStore(\n (onStoreChange) => {\n devTrack(\"store:react:subscribe\");\n const unsubscribe = store.subscribe(() => {\n devTrack(\"store:react:notify\");\n onStoreChange();\n });\n return () => {\n devTrack(\"store:react:unsubscribe\");\n unsubscribe();\n };\n },\n store.getState,\n store.getState\n );\n}\n\nexport function useDispatch<A extends IAction>(): Store<IState, A>[\"dispatch\"] {\n const store = useStoreInstance<IState, A>();\n // Return the store's dispatch directly; consumers can call dispatch(action).\n return store.dispatch as Store<IState, A>[\"dispatch\"];\n}\n","// metadata-store.ts\nexport class MetadataStore<T extends object, Meta extends object> {\n private readonly symbol: symbol;\n\n constructor(description: string) {\n this.symbol = Symbol(description);\n }\n\n set(target: T, meta: Meta) {\n Object.defineProperty(target, this.symbol as PropertyKey, {\n value: meta,\n writable: false,\n enumerable: false,\n });\n }\n\n get(target: T): Meta | undefined {\n return (target as Record<PropertyKey, Meta>)[this.symbol as PropertyKey];\n }\n\n has(target: T): boolean {\n return (this.symbol as PropertyKey) in target;\n }\n}\n"],"mappings":";AAGO,IAAM,SAAS;;;ACHtB,SAAS,eAAe,YAAY,QAAQ,sBAAsB,aAAa,WAAW,eAAe;;;ACClG,SAAS,WAAc,KAAQ,OAAO,oBAAI,QAAgB,GAAM;AACrE,MAAI,QAAQ,QAAQ,OAAO,QAAQ,SAAU,QAAO;AACpD,QAAM,IAAI;AACV,MAAI,KAAK,IAAI,CAAC,EAAG,QAAO;AACxB,OAAK,IAAI,CAAC;AAGV,aAAW,OAAO,OAAO,oBAAoB,CAAC,GAAG;AAE/C,UAAM,MAAO,EAAU,GAAG;AAC1B,QAAI,OAAO,OAAO,QAAQ,SAAU,YAAW,KAAK,IAAI;AAAA,EAC1D;AAEA,aAAW,OAAO,OAAO,sBAAsB,CAAC,GAAG;AAEjD,UAAM,MAAO,EAAU,GAAG;AAC1B,QAAI,OAAO,OAAO,QAAQ,SAAU,YAAW,KAAK,IAAI;AAAA,EAC1D;AAEA,SAAO,OAAO,OAAO,GAAG;AAC1B;;;AClBA,IAAM,MACJ,OAAO,gBAAgB,cAClB,YAAuD,KAAK,MAC7D,QAAQ,IAAI,aAAa;AAG/B,SAAS,SAAS,MAAc,OAAiC;AAC/D,MAAI,CAAC,IAAK;AACV,MAAI;AACF,UAAM,IAAK,YAAoB;AAC/B,QAAI,OAAO,MAAM,WAAY,GAAE,MAAM,KAAK;AAAA,EAC5C,QAAQ;AAAA,EAAC;AACX;AAqCO,SAAS,YACd,SACA,cACa;AACb,MAAI,QAAW,MAAM,WAAW,YAAY,IAAI;AAChD,QAAM,YAAY,oBAAI,IAAc;AACpC,QAAM,eAAe,oBAAI,IAAiD;AAQ1E,QAAM,oBAAoB,oBAAI,IAA4B;AAE1D,QAAM,WAAW,MAAM;AAEvB,QAAM,WAAW,CAAC,WAAc;AAC9B,UAAM,YAAY;AAClB,UAAM,YAAY,QAAQ,OAAO,MAAM;AAEvC,QAAI,IAAK,YAAW,SAAS;AAG7B,aAAS,kBAAkB,EAAE,MAAM,QAAQ,KAAK,CAAC;AAIjD,QAAI,OAAO,GAAG,WAAW,SAAS,GAAG;AACnC,cAAQ;AACR,eAAS,eAAe,EAAE,MAAM,QAAQ,KAAK,CAAC;AAC9C;AAAA,IACF;AAEA,YAAQ;AAGR,QAAI;AACJ,QAAI,KAAK;AACP,oBAAc,OAAO,KAAK,SAAoC,EAC3D,OAAO,CAAC,MAAM,CAAC,OAAO,GAAI,UAAkB,CAAC,GAAI,UAAkB,CAAC,CAAC,CAAC;AACzE,eAAS,uBAAuB,EAAE,MAAM,QAAQ,MAAM,YAAY,CAAC;AAAA,IACrE;AAIA,UAAM,iBAAiB,CAAC,GAAG,SAAS;AACpC,aAAS,oBAAoB,EAAE,WAAW,eAAe,OAAO,CAAC;AACjE,QAAI;AACJ,eAAW,YAAY,gBAAgB;AACrC,UAAI;AACF,iBAAS;AAAA,MACX,SAAS,KAAK;AACZ,YAAI,CAAC,WAAY,cAAa;AAAA,MAChC;AAAA,IACF;AAGA,eAAW,CAAC,KAAK,GAAG,KAAK,aAAa,QAAQ,GAAG;AAC/C,UAAI,CAAC,OAAO,GAAG,UAAU,GAAG,GAAG,MAAM,GAAG,CAAC,GAAG;AAC1C,iBAAS,oBAAoB,EAAE,KAAK,OAAO,GAAG,GAAG,WAAW,IAAI,KAAK,CAAC;AACtE,mBAAW,YAAY,CAAC,GAAG,GAAG,GAAG;AAC/B,cAAI;AACF,qBAAS,MAAM,GAAG,CAAC;AAAA,UACrB,SAAS,KAAK;AACZ,gBAAI,CAAC,WAAY,cAAa;AAAA,UAChC;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAGA,QAAI,cAAc;AAClB,eAAW,SAAS,CAAC,GAAG,iBAAiB,GAAG;AAC1C,YAAM,YAAa,MAAM,SAA+B,KAAK;AAC7D,YAAM,QAAS,MAAM,WAAiE,OAAO;AAC7F,UAAI,SAAS;AACb,UAAI;AACF,iBAAS,MAAM,MAAM,WAAW,SAAS;AAAA,MAC3C,SAAS,KAAK;AACZ,YAAI,CAAC,WAAY,cAAa;AAC9B;AAAA,MACF;AACA,UAAI,CAAC,QAAQ;AACX,cAAM,YAAY;AAClB,YAAI;AACF,UAAC,MAAM,SAAkC,SAAS;AAAA,QACpD,SAAS,KAAK;AACZ,cAAI,CAAC,WAAY,cAAa;AAAA,QAChC;AACA;AAAA,MACF;AAAA,IACF;AACA,aAAS,yBAAyB,EAAE,WAAW,YAAY,CAAC;AAC5D,QAAI,WAAY,OAAM;AAAA,EACxB;AAEA,QAAM,YAAY,CAAC,aAAuB;AACxC,cAAU,IAAI,QAAQ;AACtB,aAAS,qBAAqB,EAAE,MAAM,UAAU,KAAK,CAAC;AACtD,WAAO,MAAM;AACX,gBAAU,OAAO,QAAQ;AACzB,eAAS,wBAAwB,EAAE,MAAM,UAAU,KAAK,CAAC;AAAA,IAC3D;AAAA,EACF;AAEA,QAAM,iBAAiB,CACrB,KACA,aACG;AACH,UAAM,MACJ,aAAa,IAAI,GAAG,KAAK,oBAAI,IAAmC;AAClE,QAAI,IAAI,QAAoD;AAC5D,iBAAa,IAAI,KAAK,GAAG;AACzB,aAAS,qBAAqB,EAAE,KAAK,OAAO,GAAG,GAAG,MAAM,IAAI,KAAK,CAAC;AAClE,WAAO,MAAM;AACX,UAAI,OAAO,QAAoD;AAC/D,UAAI,IAAI,SAAS,EAAG,cAAa,OAAO,GAAG;AAC3C,eAAS,wBAAwB,EAAE,KAAK,OAAO,GAAG,GAAG,MAAM,IAAI,KAAK,CAAC;AAAA,IACvE;AAAA,EACF;AAEA,QAAM,wBAAwB,CAC5B,UACA,UACA,YACG;AACH,UAAM,QAA0B;AAAA,MAC9B;AAAA,MACA;AAAA,MACA,WAAW,SAAS,KAAK;AAAA,MACzB;AAAA,IACF;AACA,sBAAkB,IAAI,KAA0C;AAChE,aAAS,0BAA0B,EAAE,MAAM,kBAAkB,KAAK,CAAC;AACnE,WAAO,MAAM;AACX,wBAAkB,OAAO,KAA0C;AACnE,eAAS,6BAA6B,EAAE,MAAM,kBAAkB,KAAK,CAAC;AAAA,IACxE;AAAA,EACF;AAEA,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF;AACF;;;AFxGW;AA5FX,IAAM,UAAU,OAAO,YAAY,cAAc,QAAQ,IAAI,aAAa,eAAe;AACzF,SAASA,UAAS,MAAc,OAAiC;AAC/D,MAAI,CAAC,QAAS;AACd,MAAI;AACF,UAAM,IAAK,YAAoB;AAC/B,QAAI,OAAO,MAAM,WAAY,GAAE,MAAM,KAAK;AAAA,EAC5C,QAAQ;AAAA,EAAC;AACX;AAGA,IAAM,SAAS,oBAAI,IAAgB;AACnC,IAAI,UAAU;AACd,IAAM,YAAY,OAAO,mBAAmB,aACxC,iBACA,CAAC,OAAmB,QAAQ,QAAQ,EAAE,KAAK,EAAE;AAEjD,SAAS,QAAQ,IAAgB;AAC/B,SAAO,IAAI,EAAE;AACb,MAAI,QAAS;AACb,YAAU;AACV,YAAU,MAAM;AACd,cAAU;AACV,UAAM,MAAM,MAAM,KAAK,MAAM;AAC7B,WAAO,MAAM;AACb,eAAW,KAAK,IAAK,GAAE;AACvB,IAAAA,UAAS,sBAAsB,EAAE,MAAM,IAAI,OAAO,CAAC;AAAA,EACrD,CAAC;AACH;AAEA,SAAS,qBAAqB,WAA0C;AACtE,SAAO,CAAC,aAAyB;AAC/B,IAAAA,UAAS,gBAAgB;AACzB,UAAM,UAAU,MAAM;AACpB,MAAAA,UAAS,uBAAuB;AAChC,cAAQ,QAAQ;AAAA,IAClB;AACA,UAAM,cAAc,UAAU,OAAO;AACrC,WAAO,MAAM;AACX,MAAAA,UAAS,mBAAmB;AAC5B,kBAAY;AAAA,IACd;AAAA,EACF;AACF;AAEA,SAAS,aAAa,GAAQ,GAAQ;AACpC,MAAI,OAAO,GAAG,GAAG,CAAC,EAAG,QAAO;AAC5B,MACE,OAAO,MAAM,YACb,MAAM,QACN,OAAO,MAAM,YACb,MAAM;AAEN,WAAO;AACT,QAAM,KAAK,OAAO,KAAK,CAAC,GACtB,KAAK,OAAO,KAAK,CAAC;AACpB,MAAI,GAAG,WAAW,GAAG,OAAQ,QAAO;AACpC,WAAS,IAAI,GAAG,IAAI,GAAG,QAAQ,KAAK;AAClC,UAAM,IAAI,GAAG,CAAC;AACd,QACE,CAAC,OAAO,UAAU,eAAe,KAAK,GAAG,CAAC,KAC1C,CAAC,OAAO,GAAI,EAAU,CAAC,GAAI,EAAU,CAAC,CAAC;AAEvC,aAAO;AAAA,EACX;AACA,SAAO;AACT;AAEO,SAAS,yBACd,SACA,cACA;AACA,QAAM,UAAU,cAAkC,IAAI;AAGtD,QAAM,WAAW,CAAC;AAAA,IAChB;AAAA,IACA,cAAc;AAAA,EAChB,MAGM;AACJ,UAAM,QAAQ,QAAQ,MAAM;AAC1B,YAAM,OAAO,YAAY,SAAS,YAAY,YAAY;AAC1D,MAAAA,UAAS,qBAAqB;AAC9B,aAAO;AAAA,IACT,GAAG,CAAC,SAAS,UAAU,YAAY,CAAC;AAEpC,cAAU,MAAM;AACd,MAAAA,UAAS,uBAAuB;AAChC,aAAO,MAAMA,UAAS,yBAAyB;AAAA,IACjD,GAAG,CAAC,CAAC;AAEL,WAAO,oBAAC,QAAQ,UAAR,EAAiB,OAAO,OAAQ,UAAS;AAAA,EACnD;AAEA,QAAMC,YAAW,MAAS;AACxB,UAAM,MAAM,WAAW,OAAO;AAC9B,QAAI,CAAC,IAAK,OAAM,IAAI,MAAM,4BAA4B;AACtD,IAAAD,UAAS,iBAAiB;AAC1B,WAAO;AAAA,MACL,qBAAqB,IAAI,SAAS;AAAA,MAClC,IAAI;AAAA,MACJ,IAAI;AAAA,IACN;AAAA,EACF;AAEA,QAAME,eAAc,MAA6B;AAC/C,UAAM,MAAM,WAAW,OAAO;AAC9B,QAAI,CAAC,IAAK,OAAM,IAAI,MAAM,+BAA+B;AACzD,WAAO,YAAY,CAAC,WAAc,IAAI,SAAS,MAAM,GAAG,CAAC,GAAG,CAAC;AAAA,EAC/D;AAEA,WAAS,YACP,UACA,UAAmC,cAChC;AACH,UAAM,MAAM,WAAW,OAAO;AAC9B,QAAI,CAAC,IAAK,OAAM,IAAI,MAAM,4BAA4B;AAEtD,UAAM,UAAU,OAA+B,IAAI;AAEnD,UAAM,cAAc,MAAM;AACxB,YAAM,eAAe,SAAS,IAAI,SAAS,CAAC;AAC5C,YAAM,OAAO,QAAQ;AACrB,UAAI,QAAQ,QAAQ,KAAK,UAAU,YAAY,GAAG;AAChD,QAAAF,UAAS,2BAA2B;AACpC,eAAO,KAAK;AAAA,MACd;AACA,MAAAA,UAAS,4BAA4B;AACrC,cAAQ,UAAU,EAAE,UAAU,aAAa;AAC3C,aAAO;AAAA,IACT;AAEA,UAAM,YAAY,CAAC,aAAyB;AAC1C,MAAAA,UAAS,yBAAyB;AAClC,YAAM,cAAc,IAAI,sBAAsB,UAAU,CAAC,iBAAiB;AACxE,cAAM,OAAO,QAAQ;AACrB,YAAI,QAAQ,QAAQ,KAAK,UAAU,YAAY,GAAG;AAChD,UAAAA,UAAS,sBAAsB;AAC/B;AAAA,QACF;AACA,gBAAQ,UAAU,EAAE,UAAU,aAAa;AAC3C,QAAAA,UAAS,wBAAwB;AACjC,gBAAQ,QAAQ;AAAA,MAClB,GAAG,OAAO;AACV,aAAO,MAAM;AACX,QAAAA,UAAS,4BAA4B;AACrC,oBAAY;AAAA,MACd;AAAA,IACF;AAEA,WAAO,qBAAqB,WAAW,aAAa,WAAW;AAAA,EACjE;AAEA,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA,UAAAC;AAAA,IACA,aAAAC;AAAA,IACA;AAAA,EACF;AACF;;;AGtKA,SAAgB,iBAAAC,gBAAe,cAAAC,aAAY,aAAAC,YAAW,wBAAAC,6BAA4B;AAyC9E,gBAAAC,YAAA;AApCJ,IAAMC,WAAU,OAAO,YAAY,cAAc,QAAQ,IAAI,aAAa,eAAe;AACzF,SAASC,UAAS,MAAc,OAAiC;AAC/D,MAAI,CAACD,SAAS;AACd,MAAI;AACF,UAAM,IAAK,YAAoB;AAC/B,QAAI,OAAO,MAAM,WAAY,GAAE,MAAM,KAAK;AAAA,EAC5C,QAAQ;AAAA,EAAC;AACX;AAEA,IAAM,eAAeL,eAAkD,MAAS;AAEhF,SAAS,mBAAqE;AAC5E,QAAM,QAAQC,YAAW,YAAY;AACrC,MAAI,CAAC,OAAO;AACV,IAAAK,UAAS,wBAAwB;AACjC,UAAM,IAAI;AAAA,MACR;AAAA,IACF;AAAA,EACF;AACA,SAAO;AACT;AAOO,SAAS,cAAmD;AAAA,EACjE;AAAA,EACA;AACF,GAA6B;AAC3B,EAAAJ,WAAU,MAAM;AACd,IAAAI,UAAS,wBAAwB,EAAE,UAAU,CAAC,CAAC,MAAM,CAAC;AACtD,WAAO,MAAMA,UAAS,wBAAwB;AAAA,EAChD,GAAG,CAAC,KAAK,CAAC;AACV,SACE,gBAAAF,KAAC,aAAa,UAAb,EAAsB,OAAO,OAC3B,UACH;AAEJ;AAEO,SAAS,WAAgC;AAC9C,QAAM,QAAQ,iBAA6B;AAC3C,SAAOD;AAAA,IACL,CAAC,kBAAkB;AACjB,MAAAG,UAAS,uBAAuB;AAChC,YAAM,cAAc,MAAM,UAAU,MAAM;AACxC,QAAAA,UAAS,oBAAoB;AAC7B,sBAAc;AAAA,MAChB,CAAC;AACD,aAAO,MAAM;AACX,QAAAA,UAAS,yBAAyB;AAClC,oBAAY;AAAA,MACd;AAAA,IACF;AAAA,IACA,MAAM;AAAA,IACN,MAAM;AAAA,EACR;AACF;AAEO,SAAS,cAA+D;AAC7E,QAAM,QAAQ,iBAA4B;AAE1C,SAAO,MAAM;AACf;;;ACrEO,IAAM,gBAAN,MAA2D;AAAA,EAC/C;AAAA,EAEjB,YAAY,aAAqB;AAC/B,SAAK,SAAS,OAAO,WAAW;AAAA,EAClC;AAAA,EAEA,IAAI,QAAW,MAAY;AACzB,WAAO,eAAe,QAAQ,KAAK,QAAuB;AAAA,MACxD,OAAO;AAAA,MACP,UAAU;AAAA,MACV,YAAY;AAAA,IACd,CAAC;AAAA,EACH;AAAA,EAEA,IAAI,QAA6B;AAC/B,WAAQ,OAAqC,KAAK,MAAqB;AAAA,EACzE;AAAA,EAEA,IAAI,QAAoB;AACtB,WAAQ,KAAK,UAA0B;AAAA,EACzC;AACF;","names":["devTrack","useStore","useDispatch","createContext","useContext","useEffect","useSyncExternalStore","jsx","__DEV__","devTrack"]}
|
|
1
|
+
{"version":3,"sources":["../src/types.ts","../src/create-scoped-store.tsx","../src/freeze.ts","../src/store.ts","../src/provider.tsx","../src/metadata-store.ts"],"sourcesContent":["export type Reducer<S, A> = (state: S, action: A) => S;\nexport type Listener = () => void;\nexport type Unsubscribe = () => void;\nexport const __noop = null;","import { createContext, useContext, useRef, useSyncExternalStore, useCallback, useEffect, useMemo } from \"react\";\nimport type { IState, IAction, Store } from \"./store.js\";\nimport { createStore } from \"./store.js\";\n\n// DEV-only tracking (no-op in production)\nconst __DEV__ = typeof process !== \"undefined\" ? process.env.NODE_ENV !== \"production\" : true;\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!__DEV__) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\n// Local microtask-based batching (no react-dom dependency)\nconst _queue = new Set<() => void>();\nlet _queued = false;\nconst _schedule = typeof queueMicrotask === \"function\"\n ? queueMicrotask\n : (fn: () => void) => Promise.resolve().then(fn);\n\nfunction enqueue(fn: () => void) {\n _queue.add(fn);\n if (_queued) return;\n _queued = true;\n _schedule(() => {\n _queued = false;\n const fns = Array.from(_queue);\n _queue.clear();\n for (const f of fns) f();\n devTrack(\"scoped:batch:flush\", { size: fns.length });\n });\n}\n\nfunction makeBatchedSubscribe(subscribe: (l: () => void) => () => void) {\n return (onChange: () => void) => {\n devTrack(\"scoped:sub:add\");\n const wrapped = () => {\n devTrack(\"scoped:notify:enqueue\");\n enqueue(onChange);\n };\n const unsubscribe = subscribe(wrapped);\n return () => {\n devTrack(\"scoped:sub:remove\");\n unsubscribe();\n };\n };\n}\n\nfunction shallowEqual(a: any, b: any) {\n if (Object.is(a, b)) return true;\n if (\n typeof a !== \"object\" ||\n a === null ||\n typeof b !== \"object\" ||\n b === null\n )\n return false;\n const ak = Object.keys(a),\n bk = Object.keys(b);\n if (ak.length !== bk.length) return false;\n for (let i = 0; i < ak.length; i++) {\n const k = ak[i] as string;\n if (\n !Object.prototype.hasOwnProperty.call(b, k) ||\n !Object.is((a as any)[k], (b as any)[k])\n )\n return false;\n }\n return true;\n}\n\nexport function createScopedStoreContext<S extends IState, A extends IAction>(\n reducer: (state: S, action: A) => S,\n initialState: S\n) {\n const Context = createContext<Store<S, A> | null>(null);\n\n // Each Provider instance gets its own store.\n const Provider = ({\n children,\n initialState: override,\n }: {\n children: React.ReactNode;\n initialState?: S;\n }) => {\n const store = useMemo(() => {\n const next = createStore(reducer, override ?? initialState);\n devTrack(\"scoped:store:create\");\n return next;\n }, [reducer, override, initialState]);\n\n useEffect(() => {\n devTrack(\"scoped:provider:mount\");\n return () => devTrack(\"scoped:provider:unmount\");\n }, []);\n\n return <Context.Provider value={store}>{children}</Context.Provider>;\n };\n\n const useStore = (): S => {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Store not found in context\");\n devTrack(\"scoped:useStore\");\n return useSyncExternalStore(\n makeBatchedSubscribe(ctx.subscribe),\n ctx.getState,\n ctx.getState\n );\n };\n\n const useDispatch = (): ((action: A) => void) => {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Dispatch not found in context\");\n return useCallback((action: A) => ctx.dispatch(action), [ctx]);\n };\n\n function useSelector<T>(\n selector: (state: S) => T,\n isEqual: (a: T, b: T) => boolean = shallowEqual\n ): T {\n const ctx = useContext(Context);\n if (!ctx) throw new Error(\"Store not found in context\");\n\n const lastRef = useRef<{ selected: T } | null>(null);\n\n const getSnapshot = () => {\n const nextSelected = selector(ctx.getState());\n const last = lastRef.current;\n if (last && isEqual(last.selected, nextSelected)) {\n devTrack(\"scoped:selector:cache-hit\");\n return last.selected;\n }\n devTrack(\"scoped:selector:cache-miss\");\n lastRef.current = { selected: nextSelected };\n return nextSelected;\n };\n\n const subscribe = (onChange: () => void) => {\n devTrack(\"scoped:selector:sub:add\");\n const unsubscribe = ctx.subscribeWithSelector(selector, (nextSelected) => {\n const last = lastRef.current;\n if (last && isEqual(last.selected, nextSelected)) {\n devTrack(\"scoped:selector:skip\");\n return;\n }\n lastRef.current = { selected: nextSelected };\n devTrack(\"scoped:selector:notify\");\n enqueue(onChange);\n }, isEqual);\n return () => {\n devTrack(\"scoped:selector:sub:remove\");\n unsubscribe();\n };\n };\n\n return useSyncExternalStore(subscribe, getSnapshot, getSnapshot);\n }\n\n return {\n Context,\n Provider,\n useStore,\n useDispatch,\n useSelector,\n };\n}\n","// freeze.ts\nexport function deepFreeze<T>(obj: T, seen = new WeakSet<object>()): T {\n if (obj === null || typeof obj !== \"object\") return obj;\n const o = obj as unknown as object;\n if (seen.has(o)) return obj;\n seen.add(o);\n\n // Freeze children first\n for (const key of Object.getOwnPropertyNames(o)) {\n // @ts-expect-error index access\n const val = (o as any)[key];\n if (val && typeof val === \"object\") deepFreeze(val, seen);\n }\n // Also handle symbols (rare but safe)\n for (const sym of Object.getOwnPropertySymbols(o)) {\n // @ts-expect-error index access\n const val = (o as any)[sym];\n if (val && typeof val === \"object\") deepFreeze(val, seen);\n }\n\n return Object.freeze(obj);\n}\n","import type { Reducer, Listener } from \"./types.js\";\nimport { deepFreeze } from \"./freeze.js\";\n\nconst DEV = (() => {\n const metaEnv = (import.meta as unknown as { env?: { DEV?: boolean } }).env;\n if (metaEnv && Object.prototype.hasOwnProperty.call(metaEnv, \"DEV\")) {\n return Boolean(metaEnv.DEV);\n }\n if (typeof process !== \"undefined\" && process.env) {\n return process.env.NODE_ENV !== \"production\";\n }\n return true;\n})();\n\n// Lightweight DEV-only tracker\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!DEV) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\n// Allow narrower parameter types for callbacks without fighting variance\ntype BivariantListener<T> = {\n bivarianceHack(value: T): void;\n}[\"bivarianceHack\"];\n\n// eslint-disable-next-line @typescript-eslint/no-empty-object-type\nexport interface IState {}\nexport interface IAction {\n type: string;\n}\n\nexport interface Store<S extends IState, A extends IAction> {\n getState(): S;\n dispatch(action: A): void;\n /**\n * Subscribe to all state changes.\n */\n subscribe(listener: Listener): () => void;\n /**\n * Subscribe to changes of a specific key in the state.\n */\n subscribeToKey<K extends keyof S>(\n key: K,\n listener: (value: S[K]) => void\n ): () => void;\n /**\n * Subscribe to changes in a selected value from the state.\n */\n subscribeWithSelector<T>(\n selector: (state: S) => T,\n listener: (selected: T) => void,\n isEqual?: (a: T, b: T) => boolean\n ): () => void;\n}\n\nexport function createStore<S extends IState, A extends IAction>(\n reducer: Reducer<S, A>,\n initialState: S\n): Store<S, A> {\n let state: S = DEV ? deepFreeze(initialState) : initialState;\n const listeners = new Set<Listener>();\n const keyListeners = new Map<keyof S, Set<BivariantListener<S[keyof S]>>>();\n\n interface SelectorEntry<T> {\n selector: (state: S) => T;\n listener: BivariantListener<T>;\n lastValue: T;\n isEqual?: (a: T, b: T) => boolean;\n }\n const selectorListeners = new Set<SelectorEntry<unknown>>();\n\n const getState = () => state;\n\n const dispatch = (action: A) => {\n const prevState = state;\n const nextState = reducer(state, action);\n\n if (DEV) deepFreeze(nextState);\n\n // Track the inbound action\n devTrack(\"store:dispatch\", { type: action?.type });\n\n // Distinct-until-changed: if the reducer returns the same reference,\n // skip all notifications (prevents unnecessary re-renders).\n if (Object.is(prevState, nextState)) {\n state = nextState; // keep any identity guarantees from reducer\n devTrack(\"store:no-op\", { type: action?.type });\n return;\n }\n\n state = nextState;\n\n // Compute changed keys (shallow) for diagnostics\n let changedKeys: (keyof S)[] | undefined;\n if (DEV) {\n changedKeys = Object.keys(nextState as Record<string, unknown>)\n .filter((k) => !Object.is((prevState as any)[k], (nextState as any)[k])) as (keyof S)[];\n devTrack(\"store:state-changed\", { type: action?.type, changedKeys });\n }\n\n // Notify global listeners (iterate over a snapshot so unsubscribe during\n // notify does not skip the next listener)\n const globalSnapshot = [...listeners];\n devTrack(\"store:notify:all\", { listeners: globalSnapshot.length });\n let firstError: unknown;\n for (const listener of globalSnapshot) {\n try {\n listener();\n } catch (err) {\n if (!firstError) firstError = err;\n }\n }\n\n // Notify key listeners only when that key actually changed (Object.is)\n for (const [key, set] of keyListeners.entries()) {\n if (!Object.is(prevState[key], state[key])) {\n devTrack(\"store:notify:key\", { key: String(key), listeners: set.size });\n for (const listener of [...set]) {\n try {\n listener(state[key]);\n } catch (err) {\n if (!firstError) firstError = err;\n }\n }\n }\n }\n\n // Notify selector listeners only when selected value changed (Object.is)\n let selNotifies = 0;\n for (const entry of [...selectorListeners]) {\n const nextValue = (entry.selector as (s: S) => unknown)(state);\n const equal = (entry.isEqual as ((a: unknown, b: unknown) => boolean) | undefined) ?? Object.is;\n let isSame = false;\n try {\n isSame = equal(entry.lastValue, nextValue);\n } catch (err) {\n if (!firstError) firstError = err;\n continue;\n }\n if (!isSame) {\n entry.lastValue = nextValue as unknown;\n try {\n (entry.listener as (v: unknown) => void)(nextValue);\n } catch (err) {\n if (!firstError) firstError = err;\n }\n selNotifies++;\n }\n }\n devTrack(\"store:notify:selector\", { listeners: selNotifies });\n if (firstError) throw firstError;\n };\n\n const subscribe = (listener: Listener) => {\n listeners.add(listener);\n devTrack(\"store:sub:all:add\", { size: listeners.size });\n return () => {\n listeners.delete(listener);\n devTrack(\"store:sub:all:remove\", { size: listeners.size });\n };\n };\n\n const subscribeToKey = <K extends keyof S>(\n key: K,\n listener: (value: S[K]) => void\n ) => {\n const set =\n keyListeners.get(key) ?? new Set<BivariantListener<S[keyof S]>>();\n set.add(listener as unknown as BivariantListener<S[keyof S]>);\n keyListeners.set(key, set);\n devTrack(\"store:sub:key:add\", { key: String(key), size: set.size });\n return () => {\n set.delete(listener as unknown as BivariantListener<S[keyof S]>);\n if (set.size === 0) keyListeners.delete(key);\n devTrack(\"store:sub:key:remove\", { key: String(key), size: set.size });\n };\n };\n\n const subscribeWithSelector = <T>(\n selector: (state: S) => T,\n listener: (selected: T) => void,\n isEqual?: (a: T, b: T) => boolean\n ) => {\n const entry: SelectorEntry<T> = {\n selector,\n listener: listener as BivariantListener<T>,\n lastValue: selector(state),\n isEqual,\n };\n selectorListeners.add(entry as unknown as SelectorEntry<unknown>);\n devTrack(\"store:sub:selector:add\", { size: selectorListeners.size });\n return () => {\n selectorListeners.delete(entry as unknown as SelectorEntry<unknown>);\n devTrack(\"store:sub:selector:remove\", { size: selectorListeners.size });\n };\n };\n\n return {\n getState,\n dispatch,\n subscribe,\n subscribeToKey,\n subscribeWithSelector,\n };\n}\n","import React, { createContext, useContext, useEffect, useSyncExternalStore } from \"react\";\nimport type { ReactNode } from \"react\";\nimport type { Store, IState, IAction } from \"./store.js\";\n\n// DEV-only tracking (no-op in production)\nconst __DEV__ = typeof process !== \"undefined\" ? process.env.NODE_ENV !== \"production\" : true;\nfunction devTrack(name: string, props?: Record<string, unknown>) {\n if (!__DEV__) return;\n try {\n const t = (globalThis as any)?.track;\n if (typeof t === \"function\") t(name, props);\n } catch {}\n}\n\nconst StoreContext = createContext<Store<IState, IAction> | undefined>(undefined);\n\nfunction useStoreInstance<S extends IState, A extends IAction>(): Store<S, A> {\n const store = useContext(StoreContext) as Store<S, A> | undefined;\n if (!store) {\n devTrack(\"store:provider:missing\");\n throw new Error(\n \"StoreProvider is missing in the React tree. Wrap your app with <StoreProvider store={...}>.\"\n );\n }\n return store;\n}\n\ninterface StoreProviderProps<S extends IState, A extends IAction> {\n store: Store<S, A>;\n children: ReactNode;\n}\n\nexport function StoreProvider<S extends IState, A extends IAction>({\n store,\n children,\n}: StoreProviderProps<S, A>) {\n useEffect(() => {\n devTrack(\"store:provider:mount\", { hasStore: !!store });\n return () => devTrack(\"store:provider:unmount\");\n }, [store]);\n return (\n <StoreContext.Provider value={store as unknown as Store<IState, IAction>}>\n {children}\n </StoreContext.Provider>\n );\n}\n\nexport function useStore<S extends IState>(): S {\n const store = useStoreInstance<S, IAction>();\n return useSyncExternalStore(\n (onStoreChange) => {\n devTrack(\"store:react:subscribe\");\n const unsubscribe = store.subscribe(() => {\n devTrack(\"store:react:notify\");\n onStoreChange();\n });\n return () => {\n devTrack(\"store:react:unsubscribe\");\n unsubscribe();\n };\n },\n store.getState,\n store.getState\n );\n}\n\nexport function useDispatch<A extends IAction>(): Store<IState, A>[\"dispatch\"] {\n const store = useStoreInstance<IState, A>();\n // Return the store's dispatch directly; consumers can call dispatch(action).\n return store.dispatch as Store<IState, A>[\"dispatch\"];\n}\n","// metadata-store.ts\nexport class MetadataStore<T extends object, Meta extends object> {\n private readonly symbol: symbol;\n\n constructor(description: string) {\n this.symbol = Symbol(description);\n }\n\n set(target: T, meta: Meta) {\n Object.defineProperty(target, this.symbol as PropertyKey, {\n value: meta,\n writable: false,\n enumerable: false,\n });\n }\n\n get(target: T): Meta | undefined {\n return (target as Record<PropertyKey, Meta>)[this.symbol as PropertyKey];\n }\n\n has(target: T): boolean {\n return (this.symbol as PropertyKey) in target;\n }\n}\n"],"mappings":"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","names":["devTrack","useStore","useDispatch","createContext","useContext","useEffect","useSyncExternalStore","jsx","__DEV__","devTrack"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@plasius/react-state",
|
|
3
|
-
"version": "1.2.
|
|
3
|
+
"version": "1.2.1",
|
|
4
4
|
"description": "Tiny, testable, typesafe React Scoped Store helper.",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"react",
|
|
@@ -31,6 +31,9 @@
|
|
|
31
31
|
"author": "Plasius LTD",
|
|
32
32
|
"type": "module",
|
|
33
33
|
"sideEffects": false,
|
|
34
|
+
"main": "./dist/index.cjs",
|
|
35
|
+
"module": "./dist/index.js",
|
|
36
|
+
"types": "./dist/index.d.ts",
|
|
34
37
|
"files": [
|
|
35
38
|
"dist"
|
|
36
39
|
],
|