cross-state 0.49.3 → 0.50.0

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.
@@ -30,8 +30,10 @@ const split = (value, path) => {
30
30
  entries = new Map([...value].map((v, i) => [i, v]));
31
31
  } else if (Array.isArray(value)) {
32
32
  entries = new Map(value.map((v, i) => [i, v]));
33
- } else {
33
+ } else if (typeof value === "object" && value !== null) {
34
34
  entries = new Map(Object.entries(value));
35
+ } else {
36
+ return [{ path: [], value }];
35
37
  }
36
38
  if (first === "*") {
37
39
  return [...entries].flatMap(
@@ -219,7 +221,7 @@ class Persist {
219
221
  return;
220
222
  }
221
223
  if (!this.paths.find(
222
- (p) => p.path.length === key.path.length && isAncestor(p.path, key.path)
224
+ (p) => p.path.length === 1 && p.path[0] === "*" && key.path.length === 0 || p.path.length === key.path.length && isAncestor(p.path, key.path)
223
225
  )) {
224
226
  return null;
225
227
  }
@@ -1 +1 @@
1
- {"version":3,"file":"persist.cjs","sources":["../../src/lib/promiseChain.ts","../../src/persist/persistPathHelpers.ts","../../src/persist/persistStorage.ts","../../src/persist/persist.ts"],"sourcesContent":["import isPromise from '@lib/isPromise';\n\nexport interface Chain<T> {\n value: T;\n then<S>(fn: (value: Awaited<T>) => S): T extends Promise<any> ? Chain<Promise<S>> : Chain<S>;\n}\n\nexport default function promiseChain<T>(value: T | (() => T)): Chain<T> {\n if (value instanceof Function) {\n value = value();\n }\n\n return {\n value,\n then(fn) {\n const next = isPromise(value)\n ? value.then((value) => fn(value as Awaited<T>))\n : fn(value as Awaited<T>);\n\n return promiseChain(next) as any;\n },\n };\n}\n","import type { KeyType } from '@lib/path';\n\nexport const isAncestor = (ancestor: KeyType[], path: KeyType[]): boolean => {\n return (\n ancestor.length <= path.length &&\n ancestor.every((v, i) => v === '*' || path[i] === '*' || v === path[i])\n );\n};\n\nexport const split = (value: any, path: KeyType[]): { path: KeyType[]; value: unknown }[] => {\n const [first, ...rest] = path;\n if (first === undefined) return [{ path: [], value }];\n\n let entries: Map<KeyType, unknown>;\n if (value instanceof Map) {\n entries = value;\n } else if (value instanceof Set) {\n entries = new Map([...value].map((v, i) => [i, v]));\n } else if (Array.isArray(value)) {\n entries = new Map(value.map((v, i) => [i, v]));\n } else {\n entries = new Map(Object.entries(value));\n }\n\n if (first === '*') {\n return [...entries].flatMap(([k, v]) =>\n split(v, rest).map(({ path, value }) => ({ path: [k, ...path], value })),\n );\n }\n\n const subValue = entries.get(first);\n if (subValue === undefined) return [{ path: [], value }];\n\n return split(subValue, rest).map(({ path, value }) => ({ path: [first, ...path], value }));\n};\n","import isPromise from '@lib/isPromise';\nimport promiseChain from '@lib/promiseChain';\n\nexport interface PersistStorageBase {\n getItem: (key: string) => string | null | Promise<string | null>;\n setItem: (key: string, value: string) => void | Promise<unknown>;\n removeItem: (key: string) => void | Promise<unknown>;\n}\n\nexport interface PersistStorageWithKeys extends PersistStorageBase {\n keys: () => string[] | Promise<string[]>;\n}\n\nexport interface PersistStorageWithLength extends PersistStorageBase {\n length: number | (() => number | Promise<number>);\n key: (keyIndex: number) => string | null | Promise<string | null>;\n}\n\nexport interface PersistStorageWithListItems extends PersistStorageBase {\n listItems: () => Map<string, string> | Promise<Map<string, string>>;\n}\n\nexport type PersistStorage =\n | PersistStorageWithKeys\n | PersistStorageWithLength\n | PersistStorageWithListItems;\n\nexport function normalizeStorage(storage: PersistStorage): PersistStorageWithListItems {\n return {\n getItem: storage.getItem.bind(storage),\n setItem: storage.setItem.bind(storage),\n removeItem: storage.removeItem.bind(storage),\n\n listItems() {\n if ('listItems' in storage) {\n return storage.listItems();\n }\n\n return promiseChain(() => {\n if ('keys' in storage) {\n return storage.keys();\n } else {\n return promiseChain(\n storage.length instanceof Function ? storage.length() : storage.length,\n )\n .then((length) => {\n const keys = Array.from({ length }, (_, index) => storage.key(index));\n return keys.some(isPromise) ? Promise.all(keys) : (keys as (string | null)[]);\n })\n .then((keys) => {\n return keys.filter((key): key is string => typeof key === 'string');\n }).value;\n }\n })\n .then((keys) => {\n const results = keys.map(\n (key) =>\n promiseChain(storage.getItem(key)).then((value) => [key, value] as const).value,\n );\n\n return results.some(isPromise)\n ? Promise.all(results)\n : (results as [string, string | null][]);\n })\n .then((results) => {\n return new Map(results.filter(([, value]) => value !== null) as [string, string][]);\n }).value;\n },\n };\n}\n","import { type Cancel, type Duration, type Store } from '@core';\nimport { calcDuration } from '@lib/calcDuration';\nimport { shallowEqual } from '@lib/equals';\nimport { fromExtendedJsonString, toExtendedJsonString } from '@lib/extendedJson';\nimport isPromise from '@lib/isPromise';\nimport type { KeyType, WildcardPath } from '@lib/path';\nimport promiseChain from '@lib/promiseChain';\nimport { castArrayPath, get, remove, set } from '@lib/propAccess';\nimport { queue } from '@lib/queue';\nimport { subscribePatches } from '@patches/patchMethods';\nimport { isAncestor, split } from './persistPathHelpers';\nimport {\n normalizeStorage,\n type PersistStorage,\n type PersistStorageWithListItems,\n} from './persistStorage';\n\ntype PathOption<T> =\n | WildcardPath<T>\n | {\n path: WildcardPath<T>;\n // throttle?: Duration;\n };\n\ntype Key = { type: 'internal'; path: string } | { type: 'data'; path: KeyType[] };\n\nexport interface PersistOptions<T> {\n id: string;\n storage: PersistStorage;\n paths?: PathOption<T>[];\n throttle?: Duration;\n}\n\nexport class Persist<T> {\n readonly storage: PersistStorageWithListItems;\n readonly [Symbol.dispose]!: Cancel;\n\n readonly paths: {\n path: KeyType[];\n throttle?: number;\n }[];\n\n readonly initialized: Promise<void>;\n\n private resolveInitialized?: () => void;\n\n private channel: BroadcastChannel;\n\n private queue = queue();\n\n private handles = new Set<Cancel>();\n\n private stopped = false;\n\n private updateInProgress = new Map<string, unknown>();\n\n private prefix;\n\n constructor(\n public readonly store: Store<T>,\n public readonly options: PersistOptions<T>,\n ) {\n this.storage = normalizeStorage(options.storage);\n this.channel = new BroadcastChannel(`cross-state-persist_${options.id}`);\n this.prefix = `${options.id}:`;\n\n if (Symbol.dispose) {\n this[Symbol.dispose] = () => this.stop();\n }\n\n this.paths = (options.paths ?? [])\n .map<{\n path: KeyType[];\n throttle?: number;\n }>((p) => {\n if (isPlainPath(p)) {\n return {\n path: castArrayPath(p),\n throttle: options.throttle && calcDuration(options.throttle),\n };\n }\n\n const _p = p as { path: KeyType[]; throttle?: Duration };\n\n return {\n path: castArrayPath(_p.path),\n throttle:\n (_p.throttle && calcDuration(_p.throttle)) ??\n (options.throttle && calcDuration(options.throttle)),\n };\n })\n .sort((a, b) => b.path.length - a.path.length);\n\n if (this.paths.length === 0) {\n this.paths.push({\n path: ['*'],\n throttle: options.throttle && calcDuration(options.throttle),\n });\n }\n\n this.initialized = new Promise((resolve) => {\n this.resolveInitialized = resolve;\n });\n\n this.watchStore();\n this.watchStorage();\n }\n\n private watchStore() {\n const throttle = Math.min(...this.paths.map((p) => p.throttle ?? 0)) || undefined;\n\n const cancel = subscribePatches.apply(this.store as Store<unknown>, [\n (patches, reversePatches) => {\n let i = 0;\n for (const patch of patches) {\n const reversePatch = reversePatches[i++];\n\n const stringPath = JSON.stringify(patch.path);\n if (\n this.updateInProgress.has(stringPath) &&\n this.updateInProgress.get(stringPath) ===\n (patch.op === 'remove' ? undefined : patch.value)\n ) {\n continue;\n }\n\n const matchingPaths = this.paths.filter(\n (p) => isAncestor(p.path, patch.path) || isAncestor(patch.path, p.path),\n );\n\n for (const { path } of matchingPaths) {\n if (path.length <= patch.path.length) {\n const pathToSave = patch.path.slice(0, path.length);\n this.queue(() => this.save(pathToSave), pathToSave);\n } else if (patch.op === 'remove') {\n const subValues = split(\n reversePatch?.op === 'add' ? reversePatch.value : {},\n path.slice(patch.path.length),\n );\n\n for (const { path } of subValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n } else {\n const updatedValues = split(patch.value, path.slice(patch.path.length));\n const removedValues = split(\n reversePatch?.op !== 'remove' ? reversePatch?.value : {},\n path.slice(patch.path.length),\n ).filter((v) => !updatedValues.some((u) => shallowEqual(u.path, v.path)));\n\n for (const { path } of updatedValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n for (const { path } of removedValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n }\n }\n }\n },\n { runNow: false, passive: true, throttle },\n ]);\n\n this.handles.add(cancel);\n }\n\n private async watchStorage() {\n let items = this.storage.listItems();\n if (isPromise(items)) {\n items = await items;\n }\n\n if (this.stopped) {\n return;\n }\n\n const toLoad = new Map(\n [...items.entries()]\n .sort((a, b) => b[1].length - a[1].length)\n .map(([key, value]) => [this.parseKey(key), value])\n .filter(([key]) => key) as [Key, string][],\n );\n\n this.queue(() => this.load(toLoad));\n this.queue(() => this.resolveInitialized?.());\n\n const listener = (event: MessageEvent) => {\n this.queue(() => this.load([{ type: 'data', path: event.data }]));\n };\n\n this.channel.addEventListener('message', listener);\n this.handles.add(() => this.channel.removeEventListener('message', listener));\n }\n\n private buildKey({ type, path }: Key) {\n return `${this.prefix}${type === 'internal' ? path : JSON.stringify(path)}`;\n }\n\n private parseKey(key: string): Key | undefined {\n if (!key.startsWith(this.prefix)) {\n return;\n }\n\n key = key.slice(this.prefix.length);\n\n if (!key.startsWith('[')) {\n return { type: 'internal', path: key };\n }\n\n return { type: 'data', path: JSON.parse(key) as KeyType[] };\n }\n\n private load(items: Key[] | Map<Key, string>): void | Promise<void> {\n return promiseChain(() => {\n if (Array.isArray(items)) {\n return promiseChain(() => {\n const entries = items.map(\n (key) =>\n promiseChain(() => {\n return this.storage.getItem(this.buildKey(key));\n }).then((value) => [key, value] as const).value,\n );\n\n return entries.some(isPromise)\n ? Promise.all(entries)\n : (entries as [Key, string | null][]);\n }).then((entries) => {\n return entries.filter((entry) => entry !== null) as [Key, string][];\n }).value;\n } else {\n return [...items.entries()];\n }\n }).then((entries) => {\n if (this.stopped) {\n return;\n }\n\n const toWrite = entries\n .filter(([key, value]) => {\n if (key.type !== 'data' || !value) {\n return;\n }\n\n if (\n !this.paths.find(\n (p) => p.path.length === key.path.length && isAncestor(p.path, key.path),\n )\n ) {\n return null;\n }\n\n const inSaveQueue = this.queue\n .getRefs()\n .find((ref) => isAncestor(ref, key.path) || isAncestor(key.path, ref));\n return !inSaveQueue;\n })\n .map(([key, value]) => {\n try {\n return {\n path: key.path,\n value: !value || value === 'undefined' ? undefined : fromExtendedJsonString(value),\n };\n } catch {\n return undefined;\n }\n })\n .filter(Boolean) as { path: KeyType[]; value: unknown }[];\n\n if (toWrite.length > 0) {\n for (const { path, value } of toWrite) {\n this.updateInProgress.set(JSON.stringify(path), value);\n }\n\n this.store.set((state) => {\n for (const { path, value } of toWrite) {\n if (value === undefined) {\n state = remove(state, path as any);\n } else {\n state = set(state, path as any, value);\n }\n }\n\n return state;\n });\n\n this.updateInProgress.clear();\n }\n\n const versionEntry = entries.find(\n ([key]) => key.type === 'internal' && key.path === 'version',\n );\n if (versionEntry) {\n this.store.version = versionEntry[1];\n }\n }).value;\n }\n\n private save(path: KeyType[]): void | Promise<unknown> {\n const key = this.buildKey({ type: 'data', path });\n const value = get(this.store.get() as any, path);\n\n return promiseChain(value)\n .then((value) => {\n if (value === undefined) {\n return this.storage.removeItem(key);\n } else {\n return this.storage.setItem(key, toExtendedJsonString(value));\n }\n })\n .then(() => {\n this.channel.postMessage(path);\n\n if (this.store.version) {\n return this.storage.setItem(\n this.buildKey({ type: 'internal', path: 'version' }),\n this.store.version,\n );\n } else {\n return this.storage.removeItem(this.buildKey({ type: 'internal', path: 'version' }));\n }\n }).value;\n }\n\n async stop(): Promise<void> {\n this.stopped = true;\n\n for (const handle of this.handles) {\n handle();\n }\n\n await this.queue.whenDone();\n this.channel.close();\n }\n}\n\nexport function persist<T>(store: Store<T>, options: PersistOptions<T>): Persist<T> {\n return new Persist<T>(store, options);\n}\n\nfunction isPlainPath<T>(p: PathOption<T>): p is WildcardPath<T> & (KeyType[] | string) {\n return typeof p === 'string' || Array.isArray(p);\n}\n"],"names":["isPromise","value","path","store","queue","castArrayPath","calcDuration","subscribePatches","shallowEqual","fromExtendedJsonString","remove","set","get","toExtendedJsonString"],"mappings":";;;;;;AAOA,SAAwB,aAAgB,KAAgC,EAAA;AACtE,EAAA,IAAI,iBAAiB,QAAU,EAAA;AAC7B,IAAA,KAAA,GAAQ,KAAM,EAAA;AAAA;AAGhB,EAAO,OAAA;AAAA,IACL,KAAA;AAAA,IACA,KAAK,EAAI,EAAA;AACP,MAAA,MAAM,IAAO,GAAAA,eAAA,CAAU,KAAK,CAAA,GACxB,KAAM,CAAA,IAAA,CAAK,CAACC,MAAAA,KAAU,EAAGA,CAAAA,MAAmB,CAAC,CAAA,GAC7C,GAAG,KAAmB,CAAA;AAE1B,MAAA,OAAO,aAAa,IAAI,CAAA;AAAA;AAC1B,GACF;AACF;;ACpBa,MAAA,UAAA,GAAa,CAAC,QAAA,EAAqB,IAA6B,KAAA;AAC3E,EAAA,OACE,SAAS,MAAU,IAAA,IAAA,CAAK,UACxB,QAAS,CAAA,KAAA,CAAM,CAAC,CAAG,EAAA,CAAA,KAAM,CAAM,KAAA,GAAA,IAAO,KAAK,CAAC,CAAA,KAAM,OAAO,CAAM,KAAA,IAAA,CAAK,CAAC,CAAC,CAAA;AAE1E,CAAA;AAEa,MAAA,KAAA,GAAQ,CAAC,KAAA,EAAY,IAA2D,KAAA;AAC3F,EAAA,MAAM,CAAC,KAAA,EAAO,GAAG,IAAI,CAAI,GAAA,IAAA;AACzB,EAAI,IAAA,KAAA,KAAU,WAAkB,OAAA,CAAC,EAAE,IAAM,EAAA,EAAI,EAAA,KAAA,EAAO,CAAA;AAEpD,EAAI,IAAA,OAAA;AACJ,EAAA,IAAI,iBAAiB,GAAK,EAAA;AACxB,IAAU,OAAA,GAAA,KAAA;AAAA,GACZ,MAAA,IAAW,iBAAiB,GAAK,EAAA;AAC/B,IAAA,OAAA,GAAU,IAAI,GAAA,CAAI,CAAC,GAAG,KAAK,CAAE,CAAA,GAAA,CAAI,CAAC,CAAA,EAAG,CAAM,KAAA,CAAC,CAAG,EAAA,CAAC,CAAC,CAAC,CAAA;AAAA,GACzC,MAAA,IAAA,KAAA,CAAM,OAAQ,CAAA,KAAK,CAAG,EAAA;AAC/B,IAAU,OAAA,GAAA,IAAI,GAAI,CAAA,KAAA,CAAM,GAAI,CAAA,CAAC,CAAG,EAAA,CAAA,KAAM,CAAC,CAAA,EAAG,CAAC,CAAC,CAAC,CAAA;AAAA,GACxC,MAAA;AACL,IAAA,OAAA,GAAU,IAAI,GAAA,CAAI,MAAO,CAAA,OAAA,CAAQ,KAAK,CAAC,CAAA;AAAA;AAGzC,EAAA,IAAI,UAAU,GAAK,EAAA;AACjB,IAAO,OAAA,CAAC,GAAG,OAAO,CAAE,CAAA,OAAA;AAAA,MAAQ,CAAC,CAAC,CAAA,EAAG,CAAC,CAAA,KAChC,KAAM,CAAA,CAAA,EAAG,IAAI,CAAA,CAAE,GAAI,CAAA,CAAC,EAAE,IAAA,EAAAC,OAAM,KAAAD,EAAAA,MAAAA,EAAa,MAAA,EAAE,IAAM,EAAA,CAAC,CAAG,EAAA,GAAGC,KAAI,CAAA,EAAG,KAAAD,EAAAA,MAAAA,EAAQ,CAAA;AAAA,KACzE;AAAA;AAGF,EAAM,MAAA,QAAA,GAAW,OAAQ,CAAA,GAAA,CAAI,KAAK,CAAA;AAClC,EAAI,IAAA,QAAA,KAAa,WAAkB,OAAA,CAAC,EAAE,IAAM,EAAA,EAAI,EAAA,KAAA,EAAO,CAAA;AAEvD,EAAO,OAAA,KAAA,CAAM,UAAU,IAAI,CAAA,CAAE,IAAI,CAAC,EAAE,MAAAC,KAAM,EAAA,KAAA,EAAAD,QAAa,MAAA,EAAE,MAAM,CAAC,KAAA,EAAO,GAAGC,KAAI,CAAA,EAAG,KAAAD,EAAAA,MAAAA,EAAQ,CAAA,CAAA;AAC3F,CAAA;;ACPO,SAAS,iBAAiB,OAAsD,EAAA;AACrF,EAAO,OAAA;AAAA,IACL,OAAS,EAAA,OAAA,CAAQ,OAAQ,CAAA,IAAA,CAAK,OAAO,CAAA;AAAA,IACrC,OAAS,EAAA,OAAA,CAAQ,OAAQ,CAAA,IAAA,CAAK,OAAO,CAAA;AAAA,IACrC,UAAY,EAAA,OAAA,CAAQ,UAAW,CAAA,IAAA,CAAK,OAAO,CAAA;AAAA,IAE3C,SAAY,GAAA;AACV,MAAA,IAAI,eAAe,OAAS,EAAA;AAC1B,QAAA,OAAO,QAAQ,SAAU,EAAA;AAAA;AAG3B,MAAA,OAAO,aAAa,MAAM;AACxB,QAAA,IAAI,UAAU,OAAS,EAAA;AACrB,UAAA,OAAO,QAAQ,IAAK,EAAA;AAAA,SACf,MAAA;AACL,UAAO,OAAA,YAAA;AAAA,YACL,QAAQ,MAAkB,YAAA,QAAA,GAAW,OAAQ,CAAA,MAAA,KAAW,OAAQ,CAAA;AAAA,WAClE,CACG,IAAK,CAAA,CAAC,MAAW,KAAA;AAChB,YAAA,MAAM,IAAO,GAAA,KAAA,CAAM,IAAK,CAAA,EAAE,MAAO,EAAA,EAAG,CAAC,CAAA,EAAG,KAAU,KAAA,OAAA,CAAQ,GAAI,CAAA,KAAK,CAAC,CAAA;AACpE,YAAA,OAAO,KAAK,IAAK,CAAAD,eAAS,IAAI,OAAQ,CAAA,GAAA,CAAI,IAAI,CAAK,GAAA,IAAA;AAAA,WACpD,CAAA,CACA,IAAK,CAAA,CAAC,IAAS,KAAA;AACd,YAAA,OAAO,KAAK,MAAO,CAAA,CAAC,GAAuB,KAAA,OAAO,QAAQ,QAAQ,CAAA;AAAA,WACnE,CAAE,CAAA,KAAA;AAAA;AACP,OACD,CAAA,CACE,IAAK,CAAA,CAAC,IAAS,KAAA;AACd,QAAA,MAAM,UAAU,IAAK,CAAA,GAAA;AAAA,UACnB,CAAC,GAAA,KACC,YAAa,CAAA,OAAA,CAAQ,QAAQ,GAAG,CAAC,CAAE,CAAA,IAAA,CAAK,CAAC,KAAU,KAAA,CAAC,GAAK,EAAA,KAAK,CAAU,CAAE,CAAA;AAAA,SAC9E;AAEA,QAAA,OAAO,QAAQ,IAAK,CAAAA,eAAS,IACzB,OAAQ,CAAA,GAAA,CAAI,OAAO,CAClB,GAAA,OAAA;AAAA,OACN,CAAA,CACA,IAAK,CAAA,CAAC,OAAY,KAAA;AACjB,QAAO,OAAA,IAAI,GAAI,CAAA,OAAA,CAAQ,MAAO,CAAA,CAAC,GAAG,KAAK,CAAA,KAAM,KAAU,KAAA,IAAI,CAAuB,CAAA;AAAA,OACnF,CAAE,CAAA,KAAA;AAAA;AACP,GACF;AACF;;ACpCO,MAAM,OAAW,CAAA;AAAA,EAyBtB,WAAA,CACkBG,SACA,OAChB,EAAA;AAFgB,IAAA,IAAA,CAAA,KAAA,GAAAA,OAAA;AACA,IAAA,IAAA,CAAA,OAAA,GAAA,OAAA;AAZlB,IAAA,IAAA,CAAQ,QAAQC,WAAM,EAAA;AAEtB,IAAQ,IAAA,CAAA,OAAA,uBAAc,GAAY,EAAA;AAElC,IAAA,IAAA,CAAQ,OAAU,GAAA,KAAA;AAElB,IAAQ,IAAA,CAAA,gBAAA,uBAAuB,GAAqB,EAAA;AAQlD,IAAK,IAAA,CAAA,OAAA,GAAU,gBAAiB,CAAA,OAAA,CAAQ,OAAO,CAAA;AAC/C,IAAA,IAAA,CAAK,UAAU,IAAI,gBAAA,CAAiB,CAAuB,oBAAA,EAAA,OAAA,CAAQ,EAAE,CAAE,CAAA,CAAA;AACvE,IAAK,IAAA,CAAA,MAAA,GAAS,CAAG,EAAA,OAAA,CAAQ,EAAE,CAAA,CAAA,CAAA;AAE3B,IAAA,IAAI,OAAO,OAAS,EAAA;AAClB,MAAA,IAAA,CAAK,MAAO,CAAA,OAAO,CAAI,GAAA,MAAM,KAAK,IAAK,EAAA;AAAA;AAGzC,IAAA,IAAA,CAAK,SAAS,OAAQ,CAAA,KAAA,IAAS,EAC5B,EAAA,GAAA,CAGE,CAAC,CAAM,KAAA;AACR,MAAI,IAAA,WAAA,CAAY,CAAC,CAAG,EAAA;AAClB,QAAO,OAAA;AAAA,UACL,IAAA,EAAMC,yBAAc,CAAC,CAAA;AAAA,UACrB,QAAU,EAAA,OAAA,CAAQ,QAAY,IAAAC,kBAAA,CAAa,QAAQ,QAAQ;AAAA,SAC7D;AAAA;AAGF,MAAA,MAAM,EAAK,GAAA,CAAA;AAEX,MAAO,OAAA;AAAA,QACL,IAAA,EAAMD,wBAAc,CAAA,EAAA,CAAG,IAAI,CAAA;AAAA,QAC3B,QAAA,EAAA,CACG,EAAG,CAAA,QAAA,IAAYC,kBAAa,CAAA,EAAA,CAAG,QAAQ,CAAA,MACvC,OAAQ,CAAA,QAAA,IAAYA,kBAAa,CAAA,OAAA,CAAQ,QAAQ,CAAA;AAAA,OACtD;AAAA,KACD,CACA,CAAA,IAAA,CAAK,CAAC,CAAA,EAAG,CAAM,KAAA,CAAA,CAAE,IAAK,CAAA,MAAA,GAAS,CAAE,CAAA,IAAA,CAAK,MAAM,CAAA;AAE/C,IAAI,IAAA,IAAA,CAAK,KAAM,CAAA,MAAA,KAAW,CAAG,EAAA;AAC3B,MAAA,IAAA,CAAK,MAAM,IAAK,CAAA;AAAA,QACd,IAAA,EAAM,CAAC,GAAG,CAAA;AAAA,QACV,QAAU,EAAA,OAAA,CAAQ,QAAY,IAAAA,kBAAA,CAAa,QAAQ,QAAQ;AAAA,OAC5D,CAAA;AAAA;AAGH,IAAA,IAAA,CAAK,WAAc,GAAA,IAAI,OAAQ,CAAA,CAAC,OAAY,KAAA;AAC1C,MAAA,IAAA,CAAK,kBAAqB,GAAA,OAAA;AAAA,KAC3B,CAAA;AAED,IAAA,IAAA,CAAK,UAAW,EAAA;AAChB,IAAA,IAAA,CAAK,YAAa,EAAA;AAAA;AACpB,EAEQ,UAAa,GAAA;AACnB,IAAA,MAAM,QAAW,GAAA,IAAA,CAAK,GAAI,CAAA,GAAG,IAAK,CAAA,KAAA,CAAM,GAAI,CAAA,CAAC,CAAM,KAAA,CAAA,CAAE,QAAY,IAAA,CAAC,CAAC,CAAK,IAAA,SAAA;AAExE,IAAA,MAAM,MAAS,GAAAC,8BAAA,CAAiB,KAAM,CAAA,IAAA,CAAK,KAAyB,EAAA;AAAA,MAClE,CAAC,SAAS,cAAmB,KAAA;AAC3B,QAAA,IAAI,CAAI,GAAA,CAAA;AACR,QAAA,KAAA,MAAW,SAAS,OAAS,EAAA;AAC3B,UAAM,MAAA,YAAA,GAAe,eAAe,CAAG,EAAA,CAAA;AAEvC,UAAA,MAAM,UAAa,GAAA,IAAA,CAAK,SAAU,CAAA,KAAA,CAAM,IAAI,CAAA;AAC5C,UAAA,IACE,IAAK,CAAA,gBAAA,CAAiB,GAAI,CAAA,UAAU,KACpC,IAAK,CAAA,gBAAA,CAAiB,GAAI,CAAA,UAAU,OACjC,KAAM,CAAA,EAAA,KAAO,QAAW,GAAA,SAAA,GAAY,MAAM,KAC7C,CAAA,EAAA;AACA,YAAA;AAAA;AAGF,UAAM,MAAA,aAAA,GAAgB,KAAK,KAAM,CAAA,MAAA;AAAA,YAC/B,CAAC,CAAA,KAAM,UAAW,CAAA,CAAA,CAAE,IAAM,EAAA,KAAA,CAAM,IAAI,CAAA,IAAK,UAAW,CAAA,KAAA,CAAM,IAAM,EAAA,CAAA,CAAE,IAAI;AAAA,WACxE;AAEA,UAAW,KAAA,MAAA,EAAE,IAAK,EAAA,IAAK,aAAe,EAAA;AACpC,YAAA,IAAI,IAAK,CAAA,MAAA,IAAU,KAAM,CAAA,IAAA,CAAK,MAAQ,EAAA;AACpC,cAAA,MAAM,aAAa,KAAM,CAAA,IAAA,CAAK,KAAM,CAAA,CAAA,EAAG,KAAK,MAAM,CAAA;AAClD,cAAA,IAAA,CAAK,MAAM,MAAM,IAAA,CAAK,IAAK,CAAA,UAAU,GAAG,UAAU,CAAA;AAAA,aACpD,MAAA,IAAW,KAAM,CAAA,EAAA,KAAO,QAAU,EAAA;AAChC,cAAA,MAAM,SAAY,GAAA,KAAA;AAAA,gBAChB,YAAc,EAAA,EAAA,KAAO,KAAQ,GAAA,YAAA,CAAa,QAAQ,EAAC;AAAA,gBACnD,IAAK,CAAA,KAAA,CAAM,KAAM,CAAA,IAAA,CAAK,MAAM;AAAA,eAC9B;AAEA,cAAA,KAAA,MAAW,EAAE,IAAA,EAAAL,KAAK,EAAA,IAAK,SAAW,EAAA;AAChC,gBAAA,IAAA,CAAK,MAAM,MAAM,IAAA,CAAK,KAAK,CAAC,GAAG,MAAM,IAAM,EAAA,GAAGA,KAAI,CAAC,GAAG,CAAC,GAAG,MAAM,IAAM,EAAA,GAAGA,KAAI,CAAC,CAAA;AAAA;AAChF,aACK,MAAA;AACL,cAAM,MAAA,aAAA,GAAgB,MAAM,KAAM,CAAA,KAAA,EAAO,KAAK,KAAM,CAAA,KAAA,CAAM,IAAK,CAAA,MAAM,CAAC,CAAA;AACtE,cAAA,MAAM,aAAgB,GAAA,KAAA;AAAA,gBACpB,YAAc,EAAA,EAAA,KAAO,QAAW,GAAA,YAAA,EAAc,QAAQ,EAAC;AAAA,gBACvD,IAAK,CAAA,KAAA,CAAM,KAAM,CAAA,IAAA,CAAK,MAAM;AAAA,eAC5B,CAAA,MAAA,CAAO,CAAC,CAAA,KAAM,CAAC,aAAc,CAAA,IAAA,CAAK,CAAC,CAAA,KAAMM,wBAAa,CAAE,CAAA,IAAA,EAAM,CAAE,CAAA,IAAI,CAAC,CAAC,CAAA;AAExE,cAAA,KAAA,MAAW,EAAE,IAAA,EAAAN,KAAK,EAAA,IAAK,aAAe,EAAA;AACpC,gBAAA,IAAA,CAAK,MAAM,MAAM,IAAA,CAAK,KAAK,CAAC,GAAG,MAAM,IAAM,EAAA,GAAGA,KAAI,CAAC,GAAG,CAAC,GAAG,MAAM,IAAM,EAAA,GAAGA,KAAI,CAAC,CAAA;AAAA;AAEhF,cAAA,KAAA,MAAW,EAAE,IAAA,EAAAA,KAAK,EAAA,IAAK,aAAe,EAAA;AACpC,gBAAA,IAAA,CAAK,MAAM,MAAM,IAAA,CAAK,KAAK,CAAC,GAAG,MAAM,IAAM,EAAA,GAAGA,KAAI,CAAC,GAAG,CAAC,GAAG,MAAM,IAAM,EAAA,GAAGA,KAAI,CAAC,CAAA;AAAA;AAChF;AACF;AACF;AACF,OACF;AAAA,MACA,EAAE,MAAA,EAAQ,KAAO,EAAA,OAAA,EAAS,MAAM,QAAS;AAAA,KAC1C,CAAA;AAED,IAAK,IAAA,CAAA,OAAA,CAAQ,IAAI,MAAM,CAAA;AAAA;AACzB,EAEA,MAAc,YAAe,GAAA;AAC3B,IAAI,IAAA,KAAA,GAAQ,IAAK,CAAA,OAAA,CAAQ,SAAU,EAAA;AACnC,IAAI,IAAAF,eAAA,CAAU,KAAK,CAAG,EAAA;AACpB,MAAA,KAAA,GAAQ,MAAM,KAAA;AAAA;AAGhB,IAAA,IAAI,KAAK,OAAS,EAAA;AAChB,MAAA;AAAA;AAGF,IAAA,MAAM,SAAS,IAAI,GAAA;AAAA,MACjB,CAAC,GAAG,KAAA,CAAM,OAAQ,EAAC,EAChB,IAAK,CAAA,CAAC,CAAG,EAAA,CAAA,KAAM,EAAE,CAAC,CAAA,CAAE,SAAS,CAAE,CAAA,CAAC,EAAE,MAAM,CAAA,CACxC,GAAI,CAAA,CAAC,CAAC,GAAK,EAAA,KAAK,CAAM,KAAA,CAAC,KAAK,QAAS,CAAA,GAAG,CAAG,EAAA,KAAK,CAAC,CACjD,CAAA,MAAA,CAAO,CAAC,CAAC,GAAG,MAAM,GAAG;AAAA,KAC1B;AAEA,IAAA,IAAA,CAAK,KAAM,CAAA,MAAM,IAAK,CAAA,IAAA,CAAK,MAAM,CAAC,CAAA;AAClC,IAAA,IAAA,CAAK,KAAM,CAAA,MAAM,IAAK,CAAA,kBAAA,IAAsB,CAAA;AAE5C,IAAM,MAAA,QAAA,GAAW,CAAC,KAAwB,KAAA;AACxC,MAAA,IAAA,CAAK,KAAM,CAAA,MAAM,IAAK,CAAA,IAAA,CAAK,CAAC,EAAE,IAAM,EAAA,MAAA,EAAQ,IAAM,EAAA,KAAA,CAAM,IAAK,EAAC,CAAC,CAAC,CAAA;AAAA,KAClE;AAEA,IAAK,IAAA,CAAA,OAAA,CAAQ,gBAAiB,CAAA,SAAA,EAAW,QAAQ,CAAA;AACjD,IAAK,IAAA,CAAA,OAAA,CAAQ,IAAI,MAAM,IAAA,CAAK,QAAQ,mBAAoB,CAAA,SAAA,EAAW,QAAQ,CAAC,CAAA;AAAA;AAC9E,EAEQ,QAAS,CAAA,EAAE,IAAM,EAAA,IAAA,EAAa,EAAA;AACpC,IAAO,OAAA,CAAA,EAAG,IAAK,CAAA,MAAM,CAAG,EAAA,IAAA,KAAS,aAAa,IAAO,GAAA,IAAA,CAAK,SAAU,CAAA,IAAI,CAAC,CAAA,CAAA;AAAA;AAC3E,EAEQ,SAAS,GAA8B,EAAA;AAC7C,IAAA,IAAI,CAAC,GAAA,CAAI,UAAW,CAAA,IAAA,CAAK,MAAM,CAAG,EAAA;AAChC,MAAA;AAAA;AAGF,IAAA,GAAA,GAAM,GAAI,CAAA,KAAA,CAAM,IAAK,CAAA,MAAA,CAAO,MAAM,CAAA;AAElC,IAAA,IAAI,CAAC,GAAA,CAAI,UAAW,CAAA,GAAG,CAAG,EAAA;AACxB,MAAA,OAAO,EAAE,IAAA,EAAM,UAAY,EAAA,IAAA,EAAM,GAAI,EAAA;AAAA;AAGvC,IAAA,OAAO,EAAE,IAAM,EAAA,MAAA,EAAQ,MAAM,IAAK,CAAA,KAAA,CAAM,GAAG,CAAe,EAAA;AAAA;AAC5D,EAEQ,KAAK,KAAuD,EAAA;AAClE,IAAA,OAAO,aAAa,MAAM;AACxB,MAAI,IAAA,KAAA,CAAM,OAAQ,CAAA,KAAK,CAAG,EAAA;AACxB,QAAA,OAAO,aAAa,MAAM;AACxB,UAAA,MAAM,UAAU,KAAM,CAAA,GAAA;AAAA,YACpB,CAAC,GACC,KAAA,YAAA,CAAa,MAAM;AACjB,cAAA,OAAO,KAAK,OAAQ,CAAA,OAAA,CAAQ,IAAK,CAAA,QAAA,CAAS,GAAG,CAAC,CAAA;AAAA,aAC/C,EAAE,IAAK,CAAA,CAAC,UAAU,CAAC,GAAA,EAAK,KAAK,CAAU,CAAE,CAAA;AAAA,WAC9C;AAEA,UAAA,OAAO,QAAQ,IAAK,CAAAA,eAAS,IACzB,OAAQ,CAAA,GAAA,CAAI,OAAO,CAClB,GAAA,OAAA;AAAA,SACN,CAAA,CAAE,IAAK,CAAA,CAAC,OAAY,KAAA;AACnB,UAAA,OAAO,OAAQ,CAAA,MAAA,CAAO,CAAC,KAAA,KAAU,UAAU,IAAI,CAAA;AAAA,SAChD,CAAE,CAAA,KAAA;AAAA,OACE,MAAA;AACL,QAAA,OAAO,CAAC,GAAG,KAAM,CAAA,OAAA,EAAS,CAAA;AAAA;AAC5B,KACD,CAAA,CAAE,IAAK,CAAA,CAAC,OAAY,KAAA;AACnB,MAAA,IAAI,KAAK,OAAS,EAAA;AAChB,QAAA;AAAA;AAGF,MAAA,MAAM,UAAU,OACb,CAAA,MAAA,CAAO,CAAC,CAAC,GAAA,EAAK,KAAK,CAAM,KAAA;AACxB,QAAA,IAAI,GAAI,CAAA,IAAA,KAAS,MAAU,IAAA,CAAC,KAAO,EAAA;AACjC,UAAA;AAAA;AAGF,QACE,IAAA,CAAC,KAAK,KAAM,CAAA,IAAA;AAAA,UACV,CAAC,CAAA,KAAM,CAAE,CAAA,IAAA,CAAK,MAAW,KAAA,GAAA,CAAI,IAAK,CAAA,MAAA,IAAU,UAAW,CAAA,CAAA,CAAE,IAAM,EAAA,GAAA,CAAI,IAAI;AAAA,SAEzE,EAAA;AACA,UAAO,OAAA,IAAA;AAAA;AAGT,QAAA,MAAM,cAAc,IAAK,CAAA,KAAA,CACtB,OAAQ,EAAA,CACR,KAAK,CAAC,GAAA,KAAQ,UAAW,CAAA,GAAA,EAAK,IAAI,IAAI,CAAA,IAAK,WAAW,GAAI,CAAA,IAAA,EAAM,GAAG,CAAC,CAAA;AACvE,QAAA,OAAO,CAAC,WAAA;AAAA,OACT,CACA,CAAA,GAAA,CAAI,CAAC,CAAC,GAAA,EAAK,KAAK,CAAM,KAAA;AACrB,QAAI,IAAA;AACF,UAAO,OAAA;AAAA,YACL,MAAM,GAAI,CAAA,IAAA;AAAA,YACV,OAAO,CAAC,KAAA,IAAS,UAAU,WAAc,GAAA,KAAA,CAAA,GAAYS,qCAAuB,KAAK;AAAA,WACnF;AAAA,SACM,CAAA,MAAA;AACN,UAAO,OAAA,SAAA;AAAA;AACT,OACD,CACA,CAAA,MAAA,CAAO,OAAO,CAAA;AAEjB,MAAI,IAAA,OAAA,CAAQ,SAAS,CAAG,EAAA;AACtB,QAAA,KAAA,MAAW,EAAE,IAAA,EAAM,KAAM,EAAA,IAAK,OAAS,EAAA;AACrC,UAAA,IAAA,CAAK,iBAAiB,GAAI,CAAA,IAAA,CAAK,SAAU,CAAA,IAAI,GAAG,KAAK,CAAA;AAAA;AAGvD,QAAK,IAAA,CAAA,KAAA,CAAM,GAAI,CAAA,CAAC,KAAU,KAAA;AACxB,UAAA,KAAA,MAAW,EAAE,IAAA,EAAM,KAAM,EAAA,IAAK,OAAS,EAAA;AACrC,YAAA,IAAI,UAAU,SAAW,EAAA;AACvB,cAAQ,KAAA,GAAAC,iBAAA,CAAO,OAAO,IAAW,CAAA;AAAA,aAC5B,MAAA;AACL,cAAQ,KAAA,GAAAC,cAAA,CAAI,KAAO,EAAA,IAAA,EAAa,KAAK,CAAA;AAAA;AACvC;AAGF,UAAO,OAAA,KAAA;AAAA,SACR,CAAA;AAED,QAAA,IAAA,CAAK,iBAAiB,KAAM,EAAA;AAAA;AAG9B,MAAA,MAAM,eAAe,OAAQ,CAAA,IAAA;AAAA,QAC3B,CAAC,CAAC,GAAG,CAAA,KAAM,IAAI,IAAS,KAAA,UAAA,IAAc,IAAI,IAAS,KAAA;AAAA,OACrD;AACA,MAAA,IAAI,YAAc,EAAA;AAChB,QAAK,IAAA,CAAA,KAAA,CAAM,OAAU,GAAA,YAAA,CAAa,CAAC,CAAA;AAAA;AACrC,KACD,CAAE,CAAA,KAAA;AAAA;AACL,EAEQ,KAAK,IAA0C,EAAA;AACrD,IAAA,MAAM,MAAM,IAAK,CAAA,QAAA,CAAS,EAAE,IAAM,EAAA,MAAA,EAAQ,MAAM,CAAA;AAChD,IAAA,MAAM,QAAQC,cAAI,CAAA,IAAA,CAAK,KAAM,CAAA,GAAA,IAAc,IAAI,CAAA;AAE/C,IAAA,OAAO,YAAa,CAAA,KAAK,CACtB,CAAA,IAAA,CAAK,CAACX,MAAU,KAAA;AACf,MAAA,IAAIA,WAAU,SAAW,EAAA;AACvB,QAAO,OAAA,IAAA,CAAK,OAAQ,CAAA,UAAA,CAAW,GAAG,CAAA;AAAA,OAC7B,MAAA;AACL,QAAA,OAAO,KAAK,OAAQ,CAAA,OAAA,CAAQ,GAAK,EAAAY,kCAAA,CAAqBZ,MAAK,CAAC,CAAA;AAAA;AAC9D,KACD,CACA,CAAA,IAAA,CAAK,MAAM;AACV,MAAK,IAAA,CAAA,OAAA,CAAQ,YAAY,IAAI,CAAA;AAE7B,MAAI,IAAA,IAAA,CAAK,MAAM,OAAS,EAAA;AACtB,QAAA,OAAO,KAAK,OAAQ,CAAA,OAAA;AAAA,UAClB,KAAK,QAAS,CAAA,EAAE,MAAM,UAAY,EAAA,IAAA,EAAM,WAAW,CAAA;AAAA,UACnD,KAAK,KAAM,CAAA;AAAA,SACb;AAAA,OACK,MAAA;AACL,QAAO,OAAA,IAAA,CAAK,OAAQ,CAAA,UAAA,CAAW,IAAK,CAAA,QAAA,CAAS,EAAE,IAAA,EAAM,UAAY,EAAA,IAAA,EAAM,SAAU,EAAC,CAAC,CAAA;AAAA;AACrF,KACD,CAAE,CAAA,KAAA;AAAA;AACP,EAEA,MAAM,IAAsB,GAAA;AAC1B,IAAA,IAAA,CAAK,OAAU,GAAA,IAAA;AAEf,IAAW,KAAA,MAAA,MAAA,IAAU,KAAK,OAAS,EAAA;AACjC,MAAO,MAAA,EAAA;AAAA;AAGT,IAAM,MAAA,IAAA,CAAK,MAAM,QAAS,EAAA;AAC1B,IAAA,IAAA,CAAK,QAAQ,KAAM,EAAA;AAAA;AAEvB;AAEgB,SAAA,OAAA,CAAW,OAAiB,OAAwC,EAAA;AAClF,EAAO,OAAA,IAAI,OAAW,CAAA,KAAA,EAAO,OAAO,CAAA;AACtC;AAEA,SAAS,YAAe,CAA+D,EAAA;AACrF,EAAA,OAAO,OAAO,CAAA,KAAM,QAAY,IAAA,KAAA,CAAM,QAAQ,CAAC,CAAA;AACjD;;;;"}
1
+ {"version":3,"file":"persist.cjs","sources":["../../src/lib/promiseChain.ts","../../src/persist/persistPathHelpers.ts","../../src/persist/persistStorage.ts","../../src/persist/persist.ts"],"sourcesContent":["import isPromise from '@lib/isPromise';\n\nexport interface Chain<T> {\n value: T;\n then<S>(fn: (value: Awaited<T>) => S): T extends Promise<any> ? Chain<Promise<S>> : Chain<S>;\n}\n\nexport default function promiseChain<T>(value: T | (() => T)): Chain<T> {\n if (value instanceof Function) {\n value = value();\n }\n\n return {\n value,\n then(fn) {\n const next = isPromise(value)\n ? value.then((value) => fn(value as Awaited<T>))\n : fn(value as Awaited<T>);\n\n return promiseChain(next) as any;\n },\n };\n}\n","import type { KeyType } from '@lib/path';\n\nexport const isAncestor = (ancestor: KeyType[], path: KeyType[]): boolean => {\n return (\n ancestor.length <= path.length &&\n ancestor.every((v, i) => v === '*' || path[i] === '*' || v === path[i])\n );\n};\n\nexport const split = (value: any, path: KeyType[]): { path: KeyType[]; value: unknown }[] => {\n const [first, ...rest] = path;\n if (first === undefined) return [{ path: [], value }];\n\n let entries: Map<KeyType, unknown>;\n if (value instanceof Map) {\n entries = value;\n } else if (value instanceof Set) {\n entries = new Map([...value].map((v, i) => [i, v]));\n } else if (Array.isArray(value)) {\n entries = new Map(value.map((v, i) => [i, v]));\n } else if (typeof value === 'object' && value !== null) {\n entries = new Map(Object.entries(value));\n } else {\n return [{ path: [], value }];\n }\n\n if (first === '*') {\n return [...entries].flatMap(([k, v]) =>\n split(v, rest).map(({ path, value }) => ({ path: [k, ...path], value })),\n );\n }\n\n const subValue = entries.get(first);\n if (subValue === undefined) return [{ path: [], value }];\n\n return split(subValue, rest).map(({ path, value }) => ({ path: [first, ...path], value }));\n};\n","import isPromise from '@lib/isPromise';\nimport promiseChain from '@lib/promiseChain';\n\nexport interface PersistStorageBase {\n getItem: (key: string) => string | null | Promise<string | null>;\n setItem: (key: string, value: string) => void | Promise<unknown>;\n removeItem: (key: string) => void | Promise<unknown>;\n}\n\nexport interface PersistStorageWithKeys extends PersistStorageBase {\n keys: () => string[] | Promise<string[]>;\n}\n\nexport interface PersistStorageWithLength extends PersistStorageBase {\n length: number | (() => number | Promise<number>);\n key: (keyIndex: number) => string | null | Promise<string | null>;\n}\n\nexport interface PersistStorageWithListItems extends PersistStorageBase {\n listItems: () => Map<string, string> | Promise<Map<string, string>>;\n}\n\nexport type PersistStorage =\n | PersistStorageWithKeys\n | PersistStorageWithLength\n | PersistStorageWithListItems;\n\nexport function normalizeStorage(storage: PersistStorage): PersistStorageWithListItems {\n return {\n getItem: storage.getItem.bind(storage),\n setItem: storage.setItem.bind(storage),\n removeItem: storage.removeItem.bind(storage),\n\n listItems() {\n if ('listItems' in storage) {\n return storage.listItems();\n }\n\n return promiseChain(() => {\n if ('keys' in storage) {\n return storage.keys();\n } else {\n return promiseChain(\n storage.length instanceof Function ? storage.length() : storage.length,\n )\n .then((length) => {\n const keys = Array.from({ length }, (_, index) => storage.key(index));\n return keys.some(isPromise) ? Promise.all(keys) : (keys as (string | null)[]);\n })\n .then((keys) => {\n return keys.filter((key): key is string => typeof key === 'string');\n }).value;\n }\n })\n .then((keys) => {\n const results = keys.map(\n (key) =>\n promiseChain(storage.getItem(key)).then((value) => [key, value] as const).value,\n );\n\n return results.some(isPromise)\n ? Promise.all(results)\n : (results as [string, string | null][]);\n })\n .then((results) => {\n return new Map(results.filter(([, value]) => value !== null) as [string, string][]);\n }).value;\n },\n };\n}\n","import { type Cancel, type Duration, type Store } from '@core';\nimport { calcDuration } from '@lib/calcDuration';\nimport { shallowEqual } from '@lib/equals';\nimport { fromExtendedJsonString, toExtendedJsonString } from '@lib/extendedJson';\nimport isPromise from '@lib/isPromise';\nimport type { KeyType, WildcardPath } from '@lib/path';\nimport promiseChain from '@lib/promiseChain';\nimport { castArrayPath, get, remove, set } from '@lib/propAccess';\nimport { queue } from '@lib/queue';\nimport { subscribePatches } from '@patches/patchMethods';\nimport { isAncestor, split } from './persistPathHelpers';\nimport {\n normalizeStorage,\n type PersistStorage,\n type PersistStorageWithListItems,\n} from './persistStorage';\n\ntype PathOption<T> =\n | WildcardPath<T>\n | {\n path: WildcardPath<T>;\n // throttle?: Duration;\n };\n\ntype Key = { type: 'internal'; path: string } | { type: 'data'; path: KeyType[] };\n\nexport interface PersistOptions<T> {\n id: string;\n storage: PersistStorage;\n paths?: PathOption<T>[];\n throttle?: Duration;\n}\n\nexport class Persist<T> {\n readonly storage: PersistStorageWithListItems;\n readonly [Symbol.dispose]!: Cancel;\n\n readonly paths: {\n path: KeyType[];\n throttle?: number;\n }[];\n\n readonly initialized: Promise<void>;\n\n private resolveInitialized?: () => void;\n\n private channel: BroadcastChannel;\n\n private queue = queue();\n\n private handles = new Set<Cancel>();\n\n private stopped = false;\n\n private updateInProgress = new Map<string, unknown>();\n\n private prefix;\n\n constructor(\n public readonly store: Store<T>,\n public readonly options: PersistOptions<T>,\n ) {\n this.storage = normalizeStorage(options.storage);\n this.channel = new BroadcastChannel(`cross-state-persist_${options.id}`);\n this.prefix = `${options.id}:`;\n\n if (Symbol.dispose) {\n this[Symbol.dispose] = () => this.stop();\n }\n\n this.paths = (options.paths ?? [])\n .map<{\n path: KeyType[];\n throttle?: number;\n }>((p) => {\n if (isPlainPath(p)) {\n return {\n path: castArrayPath(p),\n throttle: options.throttle && calcDuration(options.throttle),\n };\n }\n\n const _p = p as { path: KeyType[]; throttle?: Duration };\n\n return {\n path: castArrayPath(_p.path),\n throttle:\n (_p.throttle && calcDuration(_p.throttle)) ??\n (options.throttle && calcDuration(options.throttle)),\n };\n })\n .sort((a, b) => b.path.length - a.path.length);\n\n if (this.paths.length === 0) {\n this.paths.push({\n path: ['*'],\n throttle: options.throttle && calcDuration(options.throttle),\n });\n }\n\n this.initialized = new Promise((resolve) => {\n this.resolveInitialized = resolve;\n });\n\n this.watchStore();\n this.watchStorage();\n }\n\n private watchStore() {\n const throttle = Math.min(...this.paths.map((p) => p.throttle ?? 0)) || undefined;\n\n const cancel = subscribePatches.apply(this.store as Store<unknown>, [\n (patches, reversePatches) => {\n let i = 0;\n for (const patch of patches) {\n const reversePatch = reversePatches[i++];\n\n const stringPath = JSON.stringify(patch.path);\n if (\n this.updateInProgress.has(stringPath) &&\n this.updateInProgress.get(stringPath) ===\n (patch.op === 'remove' ? undefined : patch.value)\n ) {\n continue;\n }\n\n const matchingPaths = this.paths.filter(\n (p) => isAncestor(p.path, patch.path) || isAncestor(patch.path, p.path),\n );\n\n for (const { path } of matchingPaths) {\n if (path.length <= patch.path.length) {\n const pathToSave = patch.path.slice(0, path.length);\n this.queue(() => this.save(pathToSave), pathToSave);\n } else if (patch.op === 'remove') {\n const subValues = split(\n reversePatch?.op === 'add' ? reversePatch.value : {},\n path.slice(patch.path.length),\n );\n\n for (const { path } of subValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n } else {\n const updatedValues = split(patch.value, path.slice(patch.path.length));\n const removedValues = split(\n reversePatch?.op !== 'remove' ? reversePatch?.value : {},\n path.slice(patch.path.length),\n ).filter((v) => !updatedValues.some((u) => shallowEqual(u.path, v.path)));\n\n for (const { path } of updatedValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n for (const { path } of removedValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n }\n }\n }\n },\n { runNow: false, passive: true, throttle },\n ]);\n\n this.handles.add(cancel);\n }\n\n private async watchStorage() {\n let items = this.storage.listItems();\n if (isPromise(items)) {\n items = await items;\n }\n\n if (this.stopped) {\n return;\n }\n\n const toLoad = new Map(\n [...items.entries()]\n .sort((a, b) => b[1].length - a[1].length)\n .map(([key, value]) => [this.parseKey(key), value])\n .filter(([key]) => key) as [Key, string][],\n );\n\n this.queue(() => this.load(toLoad));\n this.queue(() => this.resolveInitialized?.());\n\n const listener = (event: MessageEvent) => {\n this.queue(() => this.load([{ type: 'data', path: event.data }]));\n };\n\n this.channel.addEventListener('message', listener);\n this.handles.add(() => this.channel.removeEventListener('message', listener));\n }\n\n private buildKey({ type, path }: Key) {\n return `${this.prefix}${type === 'internal' ? path : JSON.stringify(path)}`;\n }\n\n private parseKey(key: string): Key | undefined {\n if (!key.startsWith(this.prefix)) {\n return;\n }\n\n key = key.slice(this.prefix.length);\n\n if (!key.startsWith('[')) {\n return { type: 'internal', path: key };\n }\n\n return { type: 'data', path: JSON.parse(key) as KeyType[] };\n }\n\n private load(items: Key[] | Map<Key, string>): void | Promise<void> {\n return promiseChain(() => {\n if (Array.isArray(items)) {\n return promiseChain(() => {\n const entries = items.map(\n (key) =>\n promiseChain(() => {\n return this.storage.getItem(this.buildKey(key));\n }).then((value) => [key, value] as const).value,\n );\n\n return entries.some(isPromise)\n ? Promise.all(entries)\n : (entries as [Key, string | null][]);\n }).then((entries) => {\n return entries.filter((entry) => entry !== null) as [Key, string][];\n }).value;\n } else {\n return [...items.entries()];\n }\n }).then((entries) => {\n if (this.stopped) {\n return;\n }\n\n const toWrite = entries\n .filter(([key, value]) => {\n if (key.type !== 'data' || !value) {\n return;\n }\n\n if (\n !this.paths.find(\n (p) =>\n (p.path.length === 1 && p.path[0] === '*' && key.path.length === 0) ||\n (p.path.length === key.path.length && isAncestor(p.path, key.path)),\n )\n ) {\n return null;\n }\n\n const inSaveQueue = this.queue\n .getRefs()\n .find((ref) => isAncestor(ref, key.path) || isAncestor(key.path, ref));\n return !inSaveQueue;\n })\n .map(([key, value]) => {\n try {\n return {\n path: key.path,\n value: !value || value === 'undefined' ? undefined : fromExtendedJsonString(value),\n };\n } catch {\n return undefined;\n }\n })\n .filter(Boolean) as { path: KeyType[]; value: unknown }[];\n\n if (toWrite.length > 0) {\n for (const { path, value } of toWrite) {\n this.updateInProgress.set(JSON.stringify(path), value);\n }\n\n this.store.set((state) => {\n for (const { path, value } of toWrite) {\n if (value === undefined) {\n state = remove(state, path as any);\n } else {\n state = set(state, path as any, value);\n }\n }\n\n return state;\n });\n\n this.updateInProgress.clear();\n }\n\n const versionEntry = entries.find(\n ([key]) => key.type === 'internal' && key.path === 'version',\n );\n if (versionEntry) {\n this.store.version = versionEntry[1];\n }\n }).value;\n }\n\n private save(path: KeyType[]): void | Promise<unknown> {\n const key = this.buildKey({ type: 'data', path });\n const value = get(this.store.get() as any, path);\n\n return promiseChain(value)\n .then((value) => {\n if (value === undefined) {\n return this.storage.removeItem(key);\n } else {\n return this.storage.setItem(key, toExtendedJsonString(value));\n }\n })\n .then(() => {\n this.channel.postMessage(path);\n\n if (this.store.version) {\n return this.storage.setItem(\n this.buildKey({ type: 'internal', path: 'version' }),\n this.store.version,\n );\n } else {\n return this.storage.removeItem(this.buildKey({ type: 'internal', path: 'version' }));\n }\n }).value;\n }\n\n async stop(): Promise<void> {\n this.stopped = true;\n\n for (const handle of this.handles) {\n handle();\n }\n\n await this.queue.whenDone();\n this.channel.close();\n }\n}\n\nexport function persist<T>(store: Store<T>, options: PersistOptions<T>): Persist<T> {\n return new Persist<T>(store, options);\n}\n\nfunction isPlainPath<T>(p: PathOption<T>): p is WildcardPath<T> & (KeyType[] | string) {\n return typeof p === 'string' || Array.isArray(p);\n}\n"],"names":["isPromise","value","path","store","queue","castArrayPath","calcDuration","subscribePatches","shallowEqual","fromExtendedJsonString","remove","set","get","toExtendedJsonString"],"mappings":"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"}
@@ -33,18 +33,19 @@ function connectUrl(store$1, {
33
33
  serialize = defaultSerializer,
34
34
  deserialize = defaultDeserializer,
35
35
  defaultValue = undefined,
36
+ writeDefaultValue,
36
37
  onCommit,
37
38
  debounce: debounceTime = 500
38
39
  }) {
39
40
  const serializedDefaultValue = defaultValue !== undefined ? serialize(defaultValue) : undefined;
40
- let isDirty = false;
41
+ let isDirty = writeDefaultValue ?? false;
41
42
  const commit = store.debounce(() => {
42
43
  if (isDirty) {
43
44
  const value = store$1.get();
44
45
  const url = new URL(window.location.href);
45
46
  const parameters = new URLSearchParams(url[type].slice(1));
46
47
  const serializedValue = value !== undefined ? serialize(value) : undefined;
47
- if (serializedValue === undefined || serializedValue === serializedDefaultValue) {
48
+ if (serializedValue === undefined || !writeDefaultValue && serializedValue === serializedDefaultValue) {
48
49
  parameters.delete(key);
49
50
  } else {
50
51
  parameters.set(key, serializedValue);
@@ -70,7 +71,7 @@ function connectUrl(store$1, {
70
71
  isDirty = true;
71
72
  commit();
72
73
  },
73
- { runNow: false }
74
+ { runNow: writeDefaultValue ?? false }
74
75
  );
75
76
  return store.disposable(() => {
76
77
  cancelUrlListener();
@@ -1 +1 @@
1
- {"version":3,"file":"urlStore.cjs","sources":["../../src/core/urlStore.ts"],"sourcesContent":["import disposable from '@lib/disposable';\nimport { type DisposableCancel, type Duration } from './commonTypes';\nimport { createStore, type Store, type StoreOptions } from './store';\nimport { debounce } from '@lib/debounce';\n\nexport interface UrlStoreOptions<T> extends StoreOptions<T | undefined> {\n key: string;\n type?: 'search' | 'hash';\n serialize?: (value: T) => string;\n deserialize?: (value: string) => T;\n defaultValue?: T;\n onCommit?: (value: T | undefined) => void;\n debounce?: Duration;\n}\n\nexport interface UrlStoreOptionsWithDefaults<T> extends UrlStoreOptions<T> {\n defaultValue: T;\n}\n\nexport type UrlStoreOptionsRequired<T> = UrlStoreOptions<T> &\n Required<Pick<UrlStoreOptions<T>, 'type' | 'serialize' | 'deserialize' | 'defaultValue'>>;\n\nconst urlStore = createStore(() => (typeof window !== 'undefined' ? window.location.href : ''));\n\nurlStore.addEffect(() => {\n const originalPushState = window.history.pushState;\n const originalReplaceState = window.history.replaceState;\n\n const update = () => {\n urlStore.set(window.location.href);\n };\n\n window.history.pushState = (...args) => {\n originalPushState.apply(window.history, args);\n update();\n };\n\n window.history.replaceState = (...args) => {\n originalReplaceState.apply(window.history, args);\n update();\n };\n\n window.addEventListener('popstate', update);\n\n return () => {\n window.history.pushState = originalPushState;\n window.history.replaceState = originalReplaceState;\n window.removeEventListener('popstate', update);\n };\n});\n\nexport function updateUrlStore(): void {\n urlStore.set(window.location.href);\n}\n\nexport function connectUrl<T>(\n store: Store<T>,\n options: UrlStoreOptionsWithDefaults<T>,\n): DisposableCancel;\n\nexport function connectUrl<T>(\n store: Store<T | undefined>,\n options: UrlStoreOptions<T>,\n): DisposableCancel;\n\nexport function connectUrl<T>(\n store: Store<T>,\n {\n key,\n type = 'search',\n serialize = defaultSerializer,\n deserialize = defaultDeserializer,\n defaultValue = undefined as T,\n onCommit,\n debounce: debounceTime = 500,\n }: UrlStoreOptions<T>,\n): DisposableCancel {\n const serializedDefaultValue = defaultValue !== undefined ? serialize(defaultValue) : undefined;\n let isDirty = false;\n\n const commit = debounce(() => {\n if (isDirty) {\n const value = store.get();\n const url = new URL(window.location.href);\n const parameters = new URLSearchParams(url[type].slice(1));\n const serializedValue = value !== undefined ? serialize(value) : undefined;\n\n if (serializedValue === undefined || serializedValue === serializedDefaultValue) {\n parameters.delete(key);\n } else {\n parameters.set(key, serializedValue);\n }\n\n url[type] = parameters.toString();\n\n window.history.replaceState(window.history.state, '', url.toString());\n window.dispatchEvent(new PopStateEvent('popstate'));\n\n onCommit?.(value);\n isDirty = false;\n }\n }, debounceTime);\n\n const cancelUrlListener = urlStore.subscribe((_url) => {\n if (isDirty) {\n return;\n }\n\n const url = new URL(_url);\n const parameters = new URLSearchParams(url[type].slice(1));\n const urlValue = parameters.get(key);\n\n store.set(urlValue !== null ? deserialize(urlValue) : defaultValue);\n });\n\n const cancelSubscription = store.subscribe(\n () => {\n isDirty = true;\n commit();\n },\n { runNow: false },\n );\n\n return disposable(() => {\n cancelUrlListener();\n cancelSubscription();\n commit.flush();\n });\n}\n\nfunction defaultDeserializer(value: string): any {\n if (value === undefined) {\n return undefined;\n }\n\n try {\n return JSON.parse(value, (_k, v) => {\n if (typeof v === 'object' && v !== null && '__set' in v) {\n return new Set(v.__set);\n }\n if (typeof v === 'object' && v !== null && '__map' in v) {\n return new Map(v.__map);\n }\n return v;\n });\n } catch {\n return undefined;\n }\n}\n\nfunction defaultSerializer(value: any): string {\n return JSON.stringify(value, (_k, v) => {\n if (v instanceof Set) {\n return { __set: Array.from(v) };\n }\n if (v instanceof Map) {\n return { __map: Array.from(v) };\n }\n return v;\n });\n}\n\nexport function createUrlStore<T>(options: UrlStoreOptionsWithDefaults<T>): Store<T>;\n\nexport function createUrlStore<T>(options: UrlStoreOptions<T>): Store<T | undefined>;\n\nexport function createUrlStore<T>(options: UrlStoreOptions<T>) {\n const store = createStore(options.defaultValue, options);\n connectUrl(store, options);\n return store;\n}\n"],"names":["createStore","store","debounce","disposable"],"mappings":"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"}
1
+ {"version":3,"file":"urlStore.cjs","sources":["../../src/core/urlStore.ts"],"sourcesContent":["import disposable from '@lib/disposable';\nimport { type DisposableCancel, type Duration } from './commonTypes';\nimport { createStore, type Store, type StoreOptions } from './store';\nimport { debounce } from '@lib/debounce';\n\nexport interface UrlStoreOptions<T> extends StoreOptions<T | undefined> {\n key: string;\n type?: 'search' | 'hash';\n serialize?: (value: T) => string;\n deserialize?: (value: string) => T;\n defaultValue?: T;\n writeDefaultValue?: boolean;\n onCommit?: (value: T | undefined) => void;\n debounce?: Duration;\n}\n\nexport interface UrlStoreOptionsWithDefaults<T> extends UrlStoreOptions<T> {\n defaultValue: T;\n}\n\nexport type UrlStoreOptionsRequired<T> = UrlStoreOptions<T> &\n Required<Pick<UrlStoreOptions<T>, 'type' | 'serialize' | 'deserialize' | 'defaultValue'>>;\n\nconst urlStore = createStore(() => (typeof window !== 'undefined' ? window.location.href : ''));\n\nurlStore.addEffect(() => {\n const originalPushState = window.history.pushState;\n const originalReplaceState = window.history.replaceState;\n\n const update = () => {\n urlStore.set(window.location.href);\n };\n\n window.history.pushState = (...args) => {\n originalPushState.apply(window.history, args);\n update();\n };\n\n window.history.replaceState = (...args) => {\n originalReplaceState.apply(window.history, args);\n update();\n };\n\n window.addEventListener('popstate', update);\n\n return () => {\n window.history.pushState = originalPushState;\n window.history.replaceState = originalReplaceState;\n window.removeEventListener('popstate', update);\n };\n});\n\nexport function updateUrlStore(): void {\n urlStore.set(window.location.href);\n}\n\nexport function connectUrl<T>(\n store: Store<T>,\n options: UrlStoreOptionsWithDefaults<T>,\n): DisposableCancel;\n\nexport function connectUrl<T>(\n store: Store<T | undefined>,\n options: UrlStoreOptions<T>,\n): DisposableCancel;\n\nexport function connectUrl<T>(\n store: Store<T>,\n {\n key,\n type = 'search',\n serialize = defaultSerializer,\n deserialize = defaultDeserializer,\n defaultValue = undefined as T,\n writeDefaultValue,\n onCommit,\n debounce: debounceTime = 500,\n }: UrlStoreOptions<T>,\n): DisposableCancel {\n const serializedDefaultValue = defaultValue !== undefined ? serialize(defaultValue) : undefined;\n let isDirty = writeDefaultValue ?? false;\n\n const commit = debounce(() => {\n if (isDirty) {\n const value = store.get();\n const url = new URL(window.location.href);\n const parameters = new URLSearchParams(url[type].slice(1));\n const serializedValue = value !== undefined ? serialize(value) : undefined;\n\n if (\n serializedValue === undefined ||\n (!writeDefaultValue && serializedValue === serializedDefaultValue)\n ) {\n parameters.delete(key);\n } else {\n parameters.set(key, serializedValue);\n }\n\n url[type] = parameters.toString();\n\n window.history.replaceState(window.history.state, '', url.toString());\n window.dispatchEvent(new PopStateEvent('popstate'));\n\n onCommit?.(value);\n isDirty = false;\n }\n }, debounceTime);\n\n const cancelUrlListener = urlStore.subscribe((_url) => {\n if (isDirty) {\n return;\n }\n\n const url = new URL(_url);\n const parameters = new URLSearchParams(url[type].slice(1));\n const urlValue = parameters.get(key);\n\n store.set(urlValue !== null ? deserialize(urlValue) : defaultValue);\n });\n\n const cancelSubscription = store.subscribe(\n () => {\n isDirty = true;\n commit();\n },\n { runNow: writeDefaultValue ?? false },\n );\n\n return disposable(() => {\n cancelUrlListener();\n cancelSubscription();\n commit.flush();\n });\n}\n\nfunction defaultDeserializer(value: string): any {\n if (value === undefined) {\n return undefined;\n }\n\n try {\n return JSON.parse(value, (_k, v) => {\n if (typeof v === 'object' && v !== null && '__set' in v) {\n return new Set(v.__set);\n }\n if (typeof v === 'object' && v !== null && '__map' in v) {\n return new Map(v.__map);\n }\n return v;\n });\n } catch {\n return undefined;\n }\n}\n\nfunction defaultSerializer(value: any): string {\n return JSON.stringify(value, (_k, v) => {\n if (v instanceof Set) {\n return { __set: Array.from(v) };\n }\n if (v instanceof Map) {\n return { __map: Array.from(v) };\n }\n return v;\n });\n}\n\nexport function createUrlStore<T>(options: UrlStoreOptionsWithDefaults<T>): Store<T>;\n\nexport function createUrlStore<T>(options: UrlStoreOptions<T>): Store<T | undefined>;\n\nexport function createUrlStore<T>(options: UrlStoreOptions<T>) {\n const store = createStore(options.defaultValue, options);\n connectUrl(store, options);\n return store;\n}\n"],"names":["createStore","store","debounce","disposable"],"mappings":"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"}
@@ -28,8 +28,10 @@ const split = (value, path) => {
28
28
  entries = new Map([...value].map((v, i) => [i, v]));
29
29
  } else if (Array.isArray(value)) {
30
30
  entries = new Map(value.map((v, i) => [i, v]));
31
- } else {
31
+ } else if (typeof value === "object" && value !== null) {
32
32
  entries = new Map(Object.entries(value));
33
+ } else {
34
+ return [{ path: [], value }];
33
35
  }
34
36
  if (first === "*") {
35
37
  return [...entries].flatMap(
@@ -217,7 +219,7 @@ class Persist {
217
219
  return;
218
220
  }
219
221
  if (!this.paths.find(
220
- (p) => p.path.length === key.path.length && isAncestor(p.path, key.path)
222
+ (p) => p.path.length === 1 && p.path[0] === "*" && key.path.length === 0 || p.path.length === key.path.length && isAncestor(p.path, key.path)
221
223
  )) {
222
224
  return null;
223
225
  }
@@ -1 +1 @@
1
- {"version":3,"file":"persist.mjs","sources":["../../src/lib/promiseChain.ts","../../src/persist/persistPathHelpers.ts","../../src/persist/persistStorage.ts","../../src/persist/persist.ts"],"sourcesContent":["import isPromise from '@lib/isPromise';\n\nexport interface Chain<T> {\n value: T;\n then<S>(fn: (value: Awaited<T>) => S): T extends Promise<any> ? Chain<Promise<S>> : Chain<S>;\n}\n\nexport default function promiseChain<T>(value: T | (() => T)): Chain<T> {\n if (value instanceof Function) {\n value = value();\n }\n\n return {\n value,\n then(fn) {\n const next = isPromise(value)\n ? value.then((value) => fn(value as Awaited<T>))\n : fn(value as Awaited<T>);\n\n return promiseChain(next) as any;\n },\n };\n}\n","import type { KeyType } from '@lib/path';\n\nexport const isAncestor = (ancestor: KeyType[], path: KeyType[]): boolean => {\n return (\n ancestor.length <= path.length &&\n ancestor.every((v, i) => v === '*' || path[i] === '*' || v === path[i])\n );\n};\n\nexport const split = (value: any, path: KeyType[]): { path: KeyType[]; value: unknown }[] => {\n const [first, ...rest] = path;\n if (first === undefined) return [{ path: [], value }];\n\n let entries: Map<KeyType, unknown>;\n if (value instanceof Map) {\n entries = value;\n } else if (value instanceof Set) {\n entries = new Map([...value].map((v, i) => [i, v]));\n } else if (Array.isArray(value)) {\n entries = new Map(value.map((v, i) => [i, v]));\n } else {\n entries = new Map(Object.entries(value));\n }\n\n if (first === '*') {\n return [...entries].flatMap(([k, v]) =>\n split(v, rest).map(({ path, value }) => ({ path: [k, ...path], value })),\n );\n }\n\n const subValue = entries.get(first);\n if (subValue === undefined) return [{ path: [], value }];\n\n return split(subValue, rest).map(({ path, value }) => ({ path: [first, ...path], value }));\n};\n","import isPromise from '@lib/isPromise';\nimport promiseChain from '@lib/promiseChain';\n\nexport interface PersistStorageBase {\n getItem: (key: string) => string | null | Promise<string | null>;\n setItem: (key: string, value: string) => void | Promise<unknown>;\n removeItem: (key: string) => void | Promise<unknown>;\n}\n\nexport interface PersistStorageWithKeys extends PersistStorageBase {\n keys: () => string[] | Promise<string[]>;\n}\n\nexport interface PersistStorageWithLength extends PersistStorageBase {\n length: number | (() => number | Promise<number>);\n key: (keyIndex: number) => string | null | Promise<string | null>;\n}\n\nexport interface PersistStorageWithListItems extends PersistStorageBase {\n listItems: () => Map<string, string> | Promise<Map<string, string>>;\n}\n\nexport type PersistStorage =\n | PersistStorageWithKeys\n | PersistStorageWithLength\n | PersistStorageWithListItems;\n\nexport function normalizeStorage(storage: PersistStorage): PersistStorageWithListItems {\n return {\n getItem: storage.getItem.bind(storage),\n setItem: storage.setItem.bind(storage),\n removeItem: storage.removeItem.bind(storage),\n\n listItems() {\n if ('listItems' in storage) {\n return storage.listItems();\n }\n\n return promiseChain(() => {\n if ('keys' in storage) {\n return storage.keys();\n } else {\n return promiseChain(\n storage.length instanceof Function ? storage.length() : storage.length,\n )\n .then((length) => {\n const keys = Array.from({ length }, (_, index) => storage.key(index));\n return keys.some(isPromise) ? Promise.all(keys) : (keys as (string | null)[]);\n })\n .then((keys) => {\n return keys.filter((key): key is string => typeof key === 'string');\n }).value;\n }\n })\n .then((keys) => {\n const results = keys.map(\n (key) =>\n promiseChain(storage.getItem(key)).then((value) => [key, value] as const).value,\n );\n\n return results.some(isPromise)\n ? Promise.all(results)\n : (results as [string, string | null][]);\n })\n .then((results) => {\n return new Map(results.filter(([, value]) => value !== null) as [string, string][]);\n }).value;\n },\n };\n}\n","import { type Cancel, type Duration, type Store } from '@core';\nimport { calcDuration } from '@lib/calcDuration';\nimport { shallowEqual } from '@lib/equals';\nimport { fromExtendedJsonString, toExtendedJsonString } from '@lib/extendedJson';\nimport isPromise from '@lib/isPromise';\nimport type { KeyType, WildcardPath } from '@lib/path';\nimport promiseChain from '@lib/promiseChain';\nimport { castArrayPath, get, remove, set } from '@lib/propAccess';\nimport { queue } from '@lib/queue';\nimport { subscribePatches } from '@patches/patchMethods';\nimport { isAncestor, split } from './persistPathHelpers';\nimport {\n normalizeStorage,\n type PersistStorage,\n type PersistStorageWithListItems,\n} from './persistStorage';\n\ntype PathOption<T> =\n | WildcardPath<T>\n | {\n path: WildcardPath<T>;\n // throttle?: Duration;\n };\n\ntype Key = { type: 'internal'; path: string } | { type: 'data'; path: KeyType[] };\n\nexport interface PersistOptions<T> {\n id: string;\n storage: PersistStorage;\n paths?: PathOption<T>[];\n throttle?: Duration;\n}\n\nexport class Persist<T> {\n readonly storage: PersistStorageWithListItems;\n readonly [Symbol.dispose]!: Cancel;\n\n readonly paths: {\n path: KeyType[];\n throttle?: number;\n }[];\n\n readonly initialized: Promise<void>;\n\n private resolveInitialized?: () => void;\n\n private channel: BroadcastChannel;\n\n private queue = queue();\n\n private handles = new Set<Cancel>();\n\n private stopped = false;\n\n private updateInProgress = new Map<string, unknown>();\n\n private prefix;\n\n constructor(\n public readonly store: Store<T>,\n public readonly options: PersistOptions<T>,\n ) {\n this.storage = normalizeStorage(options.storage);\n this.channel = new BroadcastChannel(`cross-state-persist_${options.id}`);\n this.prefix = `${options.id}:`;\n\n if (Symbol.dispose) {\n this[Symbol.dispose] = () => this.stop();\n }\n\n this.paths = (options.paths ?? [])\n .map<{\n path: KeyType[];\n throttle?: number;\n }>((p) => {\n if (isPlainPath(p)) {\n return {\n path: castArrayPath(p),\n throttle: options.throttle && calcDuration(options.throttle),\n };\n }\n\n const _p = p as { path: KeyType[]; throttle?: Duration };\n\n return {\n path: castArrayPath(_p.path),\n throttle:\n (_p.throttle && calcDuration(_p.throttle)) ??\n (options.throttle && calcDuration(options.throttle)),\n };\n })\n .sort((a, b) => b.path.length - a.path.length);\n\n if (this.paths.length === 0) {\n this.paths.push({\n path: ['*'],\n throttle: options.throttle && calcDuration(options.throttle),\n });\n }\n\n this.initialized = new Promise((resolve) => {\n this.resolveInitialized = resolve;\n });\n\n this.watchStore();\n this.watchStorage();\n }\n\n private watchStore() {\n const throttle = Math.min(...this.paths.map((p) => p.throttle ?? 0)) || undefined;\n\n const cancel = subscribePatches.apply(this.store as Store<unknown>, [\n (patches, reversePatches) => {\n let i = 0;\n for (const patch of patches) {\n const reversePatch = reversePatches[i++];\n\n const stringPath = JSON.stringify(patch.path);\n if (\n this.updateInProgress.has(stringPath) &&\n this.updateInProgress.get(stringPath) ===\n (patch.op === 'remove' ? undefined : patch.value)\n ) {\n continue;\n }\n\n const matchingPaths = this.paths.filter(\n (p) => isAncestor(p.path, patch.path) || isAncestor(patch.path, p.path),\n );\n\n for (const { path } of matchingPaths) {\n if (path.length <= patch.path.length) {\n const pathToSave = patch.path.slice(0, path.length);\n this.queue(() => this.save(pathToSave), pathToSave);\n } else if (patch.op === 'remove') {\n const subValues = split(\n reversePatch?.op === 'add' ? reversePatch.value : {},\n path.slice(patch.path.length),\n );\n\n for (const { path } of subValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n } else {\n const updatedValues = split(patch.value, path.slice(patch.path.length));\n const removedValues = split(\n reversePatch?.op !== 'remove' ? reversePatch?.value : {},\n path.slice(patch.path.length),\n ).filter((v) => !updatedValues.some((u) => shallowEqual(u.path, v.path)));\n\n for (const { path } of updatedValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n for (const { path } of removedValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n }\n }\n }\n },\n { runNow: false, passive: true, throttle },\n ]);\n\n this.handles.add(cancel);\n }\n\n private async watchStorage() {\n let items = this.storage.listItems();\n if (isPromise(items)) {\n items = await items;\n }\n\n if (this.stopped) {\n return;\n }\n\n const toLoad = new Map(\n [...items.entries()]\n .sort((a, b) => b[1].length - a[1].length)\n .map(([key, value]) => [this.parseKey(key), value])\n .filter(([key]) => key) as [Key, string][],\n );\n\n this.queue(() => this.load(toLoad));\n this.queue(() => this.resolveInitialized?.());\n\n const listener = (event: MessageEvent) => {\n this.queue(() => this.load([{ type: 'data', path: event.data }]));\n };\n\n this.channel.addEventListener('message', listener);\n this.handles.add(() => this.channel.removeEventListener('message', listener));\n }\n\n private buildKey({ type, path }: Key) {\n return `${this.prefix}${type === 'internal' ? path : JSON.stringify(path)}`;\n }\n\n private parseKey(key: string): Key | undefined {\n if (!key.startsWith(this.prefix)) {\n return;\n }\n\n key = key.slice(this.prefix.length);\n\n if (!key.startsWith('[')) {\n return { type: 'internal', path: key };\n }\n\n return { type: 'data', path: JSON.parse(key) as KeyType[] };\n }\n\n private load(items: Key[] | Map<Key, string>): void | Promise<void> {\n return promiseChain(() => {\n if (Array.isArray(items)) {\n return promiseChain(() => {\n const entries = items.map(\n (key) =>\n promiseChain(() => {\n return this.storage.getItem(this.buildKey(key));\n }).then((value) => [key, value] as const).value,\n );\n\n return entries.some(isPromise)\n ? Promise.all(entries)\n : (entries as [Key, string | null][]);\n }).then((entries) => {\n return entries.filter((entry) => entry !== null) as [Key, string][];\n }).value;\n } else {\n return [...items.entries()];\n }\n }).then((entries) => {\n if (this.stopped) {\n return;\n }\n\n const toWrite = entries\n .filter(([key, value]) => {\n if (key.type !== 'data' || !value) {\n return;\n }\n\n if (\n !this.paths.find(\n (p) => p.path.length === key.path.length && isAncestor(p.path, key.path),\n )\n ) {\n return null;\n }\n\n const inSaveQueue = this.queue\n .getRefs()\n .find((ref) => isAncestor(ref, key.path) || isAncestor(key.path, ref));\n return !inSaveQueue;\n })\n .map(([key, value]) => {\n try {\n return {\n path: key.path,\n value: !value || value === 'undefined' ? undefined : fromExtendedJsonString(value),\n };\n } catch {\n return undefined;\n }\n })\n .filter(Boolean) as { path: KeyType[]; value: unknown }[];\n\n if (toWrite.length > 0) {\n for (const { path, value } of toWrite) {\n this.updateInProgress.set(JSON.stringify(path), value);\n }\n\n this.store.set((state) => {\n for (const { path, value } of toWrite) {\n if (value === undefined) {\n state = remove(state, path as any);\n } else {\n state = set(state, path as any, value);\n }\n }\n\n return state;\n });\n\n this.updateInProgress.clear();\n }\n\n const versionEntry = entries.find(\n ([key]) => key.type === 'internal' && key.path === 'version',\n );\n if (versionEntry) {\n this.store.version = versionEntry[1];\n }\n }).value;\n }\n\n private save(path: KeyType[]): void | Promise<unknown> {\n const key = this.buildKey({ type: 'data', path });\n const value = get(this.store.get() as any, path);\n\n return promiseChain(value)\n .then((value) => {\n if (value === undefined) {\n return this.storage.removeItem(key);\n } else {\n return this.storage.setItem(key, toExtendedJsonString(value));\n }\n })\n .then(() => {\n this.channel.postMessage(path);\n\n if (this.store.version) {\n return this.storage.setItem(\n this.buildKey({ type: 'internal', path: 'version' }),\n this.store.version,\n );\n } else {\n return this.storage.removeItem(this.buildKey({ type: 'internal', path: 'version' }));\n }\n }).value;\n }\n\n async stop(): Promise<void> {\n this.stopped = true;\n\n for (const handle of this.handles) {\n handle();\n }\n\n await this.queue.whenDone();\n this.channel.close();\n }\n}\n\nexport function persist<T>(store: Store<T>, options: PersistOptions<T>): Persist<T> {\n return new Persist<T>(store, options);\n}\n\nfunction isPlainPath<T>(p: PathOption<T>): p is WildcardPath<T> & (KeyType[] | string) {\n return typeof p === 'string' || Array.isArray(p);\n}\n"],"names":["value","path"],"mappings":"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"}
1
+ {"version":3,"file":"persist.mjs","sources":["../../src/lib/promiseChain.ts","../../src/persist/persistPathHelpers.ts","../../src/persist/persistStorage.ts","../../src/persist/persist.ts"],"sourcesContent":["import isPromise from '@lib/isPromise';\n\nexport interface Chain<T> {\n value: T;\n then<S>(fn: (value: Awaited<T>) => S): T extends Promise<any> ? Chain<Promise<S>> : Chain<S>;\n}\n\nexport default function promiseChain<T>(value: T | (() => T)): Chain<T> {\n if (value instanceof Function) {\n value = value();\n }\n\n return {\n value,\n then(fn) {\n const next = isPromise(value)\n ? value.then((value) => fn(value as Awaited<T>))\n : fn(value as Awaited<T>);\n\n return promiseChain(next) as any;\n },\n };\n}\n","import type { KeyType } from '@lib/path';\n\nexport const isAncestor = (ancestor: KeyType[], path: KeyType[]): boolean => {\n return (\n ancestor.length <= path.length &&\n ancestor.every((v, i) => v === '*' || path[i] === '*' || v === path[i])\n );\n};\n\nexport const split = (value: any, path: KeyType[]): { path: KeyType[]; value: unknown }[] => {\n const [first, ...rest] = path;\n if (first === undefined) return [{ path: [], value }];\n\n let entries: Map<KeyType, unknown>;\n if (value instanceof Map) {\n entries = value;\n } else if (value instanceof Set) {\n entries = new Map([...value].map((v, i) => [i, v]));\n } else if (Array.isArray(value)) {\n entries = new Map(value.map((v, i) => [i, v]));\n } else if (typeof value === 'object' && value !== null) {\n entries = new Map(Object.entries(value));\n } else {\n return [{ path: [], value }];\n }\n\n if (first === '*') {\n return [...entries].flatMap(([k, v]) =>\n split(v, rest).map(({ path, value }) => ({ path: [k, ...path], value })),\n );\n }\n\n const subValue = entries.get(first);\n if (subValue === undefined) return [{ path: [], value }];\n\n return split(subValue, rest).map(({ path, value }) => ({ path: [first, ...path], value }));\n};\n","import isPromise from '@lib/isPromise';\nimport promiseChain from '@lib/promiseChain';\n\nexport interface PersistStorageBase {\n getItem: (key: string) => string | null | Promise<string | null>;\n setItem: (key: string, value: string) => void | Promise<unknown>;\n removeItem: (key: string) => void | Promise<unknown>;\n}\n\nexport interface PersistStorageWithKeys extends PersistStorageBase {\n keys: () => string[] | Promise<string[]>;\n}\n\nexport interface PersistStorageWithLength extends PersistStorageBase {\n length: number | (() => number | Promise<number>);\n key: (keyIndex: number) => string | null | Promise<string | null>;\n}\n\nexport interface PersistStorageWithListItems extends PersistStorageBase {\n listItems: () => Map<string, string> | Promise<Map<string, string>>;\n}\n\nexport type PersistStorage =\n | PersistStorageWithKeys\n | PersistStorageWithLength\n | PersistStorageWithListItems;\n\nexport function normalizeStorage(storage: PersistStorage): PersistStorageWithListItems {\n return {\n getItem: storage.getItem.bind(storage),\n setItem: storage.setItem.bind(storage),\n removeItem: storage.removeItem.bind(storage),\n\n listItems() {\n if ('listItems' in storage) {\n return storage.listItems();\n }\n\n return promiseChain(() => {\n if ('keys' in storage) {\n return storage.keys();\n } else {\n return promiseChain(\n storage.length instanceof Function ? storage.length() : storage.length,\n )\n .then((length) => {\n const keys = Array.from({ length }, (_, index) => storage.key(index));\n return keys.some(isPromise) ? Promise.all(keys) : (keys as (string | null)[]);\n })\n .then((keys) => {\n return keys.filter((key): key is string => typeof key === 'string');\n }).value;\n }\n })\n .then((keys) => {\n const results = keys.map(\n (key) =>\n promiseChain(storage.getItem(key)).then((value) => [key, value] as const).value,\n );\n\n return results.some(isPromise)\n ? Promise.all(results)\n : (results as [string, string | null][]);\n })\n .then((results) => {\n return new Map(results.filter(([, value]) => value !== null) as [string, string][]);\n }).value;\n },\n };\n}\n","import { type Cancel, type Duration, type Store } from '@core';\nimport { calcDuration } from '@lib/calcDuration';\nimport { shallowEqual } from '@lib/equals';\nimport { fromExtendedJsonString, toExtendedJsonString } from '@lib/extendedJson';\nimport isPromise from '@lib/isPromise';\nimport type { KeyType, WildcardPath } from '@lib/path';\nimport promiseChain from '@lib/promiseChain';\nimport { castArrayPath, get, remove, set } from '@lib/propAccess';\nimport { queue } from '@lib/queue';\nimport { subscribePatches } from '@patches/patchMethods';\nimport { isAncestor, split } from './persistPathHelpers';\nimport {\n normalizeStorage,\n type PersistStorage,\n type PersistStorageWithListItems,\n} from './persistStorage';\n\ntype PathOption<T> =\n | WildcardPath<T>\n | {\n path: WildcardPath<T>;\n // throttle?: Duration;\n };\n\ntype Key = { type: 'internal'; path: string } | { type: 'data'; path: KeyType[] };\n\nexport interface PersistOptions<T> {\n id: string;\n storage: PersistStorage;\n paths?: PathOption<T>[];\n throttle?: Duration;\n}\n\nexport class Persist<T> {\n readonly storage: PersistStorageWithListItems;\n readonly [Symbol.dispose]!: Cancel;\n\n readonly paths: {\n path: KeyType[];\n throttle?: number;\n }[];\n\n readonly initialized: Promise<void>;\n\n private resolveInitialized?: () => void;\n\n private channel: BroadcastChannel;\n\n private queue = queue();\n\n private handles = new Set<Cancel>();\n\n private stopped = false;\n\n private updateInProgress = new Map<string, unknown>();\n\n private prefix;\n\n constructor(\n public readonly store: Store<T>,\n public readonly options: PersistOptions<T>,\n ) {\n this.storage = normalizeStorage(options.storage);\n this.channel = new BroadcastChannel(`cross-state-persist_${options.id}`);\n this.prefix = `${options.id}:`;\n\n if (Symbol.dispose) {\n this[Symbol.dispose] = () => this.stop();\n }\n\n this.paths = (options.paths ?? [])\n .map<{\n path: KeyType[];\n throttle?: number;\n }>((p) => {\n if (isPlainPath(p)) {\n return {\n path: castArrayPath(p),\n throttle: options.throttle && calcDuration(options.throttle),\n };\n }\n\n const _p = p as { path: KeyType[]; throttle?: Duration };\n\n return {\n path: castArrayPath(_p.path),\n throttle:\n (_p.throttle && calcDuration(_p.throttle)) ??\n (options.throttle && calcDuration(options.throttle)),\n };\n })\n .sort((a, b) => b.path.length - a.path.length);\n\n if (this.paths.length === 0) {\n this.paths.push({\n path: ['*'],\n throttle: options.throttle && calcDuration(options.throttle),\n });\n }\n\n this.initialized = new Promise((resolve) => {\n this.resolveInitialized = resolve;\n });\n\n this.watchStore();\n this.watchStorage();\n }\n\n private watchStore() {\n const throttle = Math.min(...this.paths.map((p) => p.throttle ?? 0)) || undefined;\n\n const cancel = subscribePatches.apply(this.store as Store<unknown>, [\n (patches, reversePatches) => {\n let i = 0;\n for (const patch of patches) {\n const reversePatch = reversePatches[i++];\n\n const stringPath = JSON.stringify(patch.path);\n if (\n this.updateInProgress.has(stringPath) &&\n this.updateInProgress.get(stringPath) ===\n (patch.op === 'remove' ? undefined : patch.value)\n ) {\n continue;\n }\n\n const matchingPaths = this.paths.filter(\n (p) => isAncestor(p.path, patch.path) || isAncestor(patch.path, p.path),\n );\n\n for (const { path } of matchingPaths) {\n if (path.length <= patch.path.length) {\n const pathToSave = patch.path.slice(0, path.length);\n this.queue(() => this.save(pathToSave), pathToSave);\n } else if (patch.op === 'remove') {\n const subValues = split(\n reversePatch?.op === 'add' ? reversePatch.value : {},\n path.slice(patch.path.length),\n );\n\n for (const { path } of subValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n } else {\n const updatedValues = split(patch.value, path.slice(patch.path.length));\n const removedValues = split(\n reversePatch?.op !== 'remove' ? reversePatch?.value : {},\n path.slice(patch.path.length),\n ).filter((v) => !updatedValues.some((u) => shallowEqual(u.path, v.path)));\n\n for (const { path } of updatedValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n for (const { path } of removedValues) {\n this.queue(() => this.save([...patch.path, ...path]), [...patch.path, ...path]);\n }\n }\n }\n }\n },\n { runNow: false, passive: true, throttle },\n ]);\n\n this.handles.add(cancel);\n }\n\n private async watchStorage() {\n let items = this.storage.listItems();\n if (isPromise(items)) {\n items = await items;\n }\n\n if (this.stopped) {\n return;\n }\n\n const toLoad = new Map(\n [...items.entries()]\n .sort((a, b) => b[1].length - a[1].length)\n .map(([key, value]) => [this.parseKey(key), value])\n .filter(([key]) => key) as [Key, string][],\n );\n\n this.queue(() => this.load(toLoad));\n this.queue(() => this.resolveInitialized?.());\n\n const listener = (event: MessageEvent) => {\n this.queue(() => this.load([{ type: 'data', path: event.data }]));\n };\n\n this.channel.addEventListener('message', listener);\n this.handles.add(() => this.channel.removeEventListener('message', listener));\n }\n\n private buildKey({ type, path }: Key) {\n return `${this.prefix}${type === 'internal' ? path : JSON.stringify(path)}`;\n }\n\n private parseKey(key: string): Key | undefined {\n if (!key.startsWith(this.prefix)) {\n return;\n }\n\n key = key.slice(this.prefix.length);\n\n if (!key.startsWith('[')) {\n return { type: 'internal', path: key };\n }\n\n return { type: 'data', path: JSON.parse(key) as KeyType[] };\n }\n\n private load(items: Key[] | Map<Key, string>): void | Promise<void> {\n return promiseChain(() => {\n if (Array.isArray(items)) {\n return promiseChain(() => {\n const entries = items.map(\n (key) =>\n promiseChain(() => {\n return this.storage.getItem(this.buildKey(key));\n }).then((value) => [key, value] as const).value,\n );\n\n return entries.some(isPromise)\n ? Promise.all(entries)\n : (entries as [Key, string | null][]);\n }).then((entries) => {\n return entries.filter((entry) => entry !== null) as [Key, string][];\n }).value;\n } else {\n return [...items.entries()];\n }\n }).then((entries) => {\n if (this.stopped) {\n return;\n }\n\n const toWrite = entries\n .filter(([key, value]) => {\n if (key.type !== 'data' || !value) {\n return;\n }\n\n if (\n !this.paths.find(\n (p) =>\n (p.path.length === 1 && p.path[0] === '*' && key.path.length === 0) ||\n (p.path.length === key.path.length && isAncestor(p.path, key.path)),\n )\n ) {\n return null;\n }\n\n const inSaveQueue = this.queue\n .getRefs()\n .find((ref) => isAncestor(ref, key.path) || isAncestor(key.path, ref));\n return !inSaveQueue;\n })\n .map(([key, value]) => {\n try {\n return {\n path: key.path,\n value: !value || value === 'undefined' ? undefined : fromExtendedJsonString(value),\n };\n } catch {\n return undefined;\n }\n })\n .filter(Boolean) as { path: KeyType[]; value: unknown }[];\n\n if (toWrite.length > 0) {\n for (const { path, value } of toWrite) {\n this.updateInProgress.set(JSON.stringify(path), value);\n }\n\n this.store.set((state) => {\n for (const { path, value } of toWrite) {\n if (value === undefined) {\n state = remove(state, path as any);\n } else {\n state = set(state, path as any, value);\n }\n }\n\n return state;\n });\n\n this.updateInProgress.clear();\n }\n\n const versionEntry = entries.find(\n ([key]) => key.type === 'internal' && key.path === 'version',\n );\n if (versionEntry) {\n this.store.version = versionEntry[1];\n }\n }).value;\n }\n\n private save(path: KeyType[]): void | Promise<unknown> {\n const key = this.buildKey({ type: 'data', path });\n const value = get(this.store.get() as any, path);\n\n return promiseChain(value)\n .then((value) => {\n if (value === undefined) {\n return this.storage.removeItem(key);\n } else {\n return this.storage.setItem(key, toExtendedJsonString(value));\n }\n })\n .then(() => {\n this.channel.postMessage(path);\n\n if (this.store.version) {\n return this.storage.setItem(\n this.buildKey({ type: 'internal', path: 'version' }),\n this.store.version,\n );\n } else {\n return this.storage.removeItem(this.buildKey({ type: 'internal', path: 'version' }));\n }\n }).value;\n }\n\n async stop(): Promise<void> {\n this.stopped = true;\n\n for (const handle of this.handles) {\n handle();\n }\n\n await this.queue.whenDone();\n this.channel.close();\n }\n}\n\nexport function persist<T>(store: Store<T>, options: PersistOptions<T>): Persist<T> {\n return new Persist<T>(store, options);\n}\n\nfunction isPlainPath<T>(p: PathOption<T>): p is WildcardPath<T> & (KeyType[] | string) {\n return typeof p === 'string' || Array.isArray(p);\n}\n"],"names":["value","path"],"mappings":"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"}
@@ -31,18 +31,19 @@ function connectUrl(store, {
31
31
  serialize = defaultSerializer,
32
32
  deserialize = defaultDeserializer,
33
33
  defaultValue = undefined,
34
+ writeDefaultValue,
34
35
  onCommit,
35
36
  debounce: debounceTime = 500
36
37
  }) {
37
38
  const serializedDefaultValue = defaultValue !== undefined ? serialize(defaultValue) : undefined;
38
- let isDirty = false;
39
+ let isDirty = writeDefaultValue ?? false;
39
40
  const commit = debounce(() => {
40
41
  if (isDirty) {
41
42
  const value = store.get();
42
43
  const url = new URL(window.location.href);
43
44
  const parameters = new URLSearchParams(url[type].slice(1));
44
45
  const serializedValue = value !== undefined ? serialize(value) : undefined;
45
- if (serializedValue === undefined || serializedValue === serializedDefaultValue) {
46
+ if (serializedValue === undefined || !writeDefaultValue && serializedValue === serializedDefaultValue) {
46
47
  parameters.delete(key);
47
48
  } else {
48
49
  parameters.set(key, serializedValue);
@@ -68,7 +69,7 @@ function connectUrl(store, {
68
69
  isDirty = true;
69
70
  commit();
70
71
  },
71
- { runNow: false }
72
+ { runNow: writeDefaultValue ?? false }
72
73
  );
73
74
  return disposable(() => {
74
75
  cancelUrlListener();
@@ -1 +1 @@
1
- {"version":3,"file":"urlStore.mjs","sources":["../../src/core/urlStore.ts"],"sourcesContent":["import disposable from '@lib/disposable';\nimport { type DisposableCancel, type Duration } from './commonTypes';\nimport { createStore, type Store, type StoreOptions } from './store';\nimport { debounce } from '@lib/debounce';\n\nexport interface UrlStoreOptions<T> extends StoreOptions<T | undefined> {\n key: string;\n type?: 'search' | 'hash';\n serialize?: (value: T) => string;\n deserialize?: (value: string) => T;\n defaultValue?: T;\n onCommit?: (value: T | undefined) => void;\n debounce?: Duration;\n}\n\nexport interface UrlStoreOptionsWithDefaults<T> extends UrlStoreOptions<T> {\n defaultValue: T;\n}\n\nexport type UrlStoreOptionsRequired<T> = UrlStoreOptions<T> &\n Required<Pick<UrlStoreOptions<T>, 'type' | 'serialize' | 'deserialize' | 'defaultValue'>>;\n\nconst urlStore = createStore(() => (typeof window !== 'undefined' ? window.location.href : ''));\n\nurlStore.addEffect(() => {\n const originalPushState = window.history.pushState;\n const originalReplaceState = window.history.replaceState;\n\n const update = () => {\n urlStore.set(window.location.href);\n };\n\n window.history.pushState = (...args) => {\n originalPushState.apply(window.history, args);\n update();\n };\n\n window.history.replaceState = (...args) => {\n originalReplaceState.apply(window.history, args);\n update();\n };\n\n window.addEventListener('popstate', update);\n\n return () => {\n window.history.pushState = originalPushState;\n window.history.replaceState = originalReplaceState;\n window.removeEventListener('popstate', update);\n };\n});\n\nexport function updateUrlStore(): void {\n urlStore.set(window.location.href);\n}\n\nexport function connectUrl<T>(\n store: Store<T>,\n options: UrlStoreOptionsWithDefaults<T>,\n): DisposableCancel;\n\nexport function connectUrl<T>(\n store: Store<T | undefined>,\n options: UrlStoreOptions<T>,\n): DisposableCancel;\n\nexport function connectUrl<T>(\n store: Store<T>,\n {\n key,\n type = 'search',\n serialize = defaultSerializer,\n deserialize = defaultDeserializer,\n defaultValue = undefined as T,\n onCommit,\n debounce: debounceTime = 500,\n }: UrlStoreOptions<T>,\n): DisposableCancel {\n const serializedDefaultValue = defaultValue !== undefined ? serialize(defaultValue) : undefined;\n let isDirty = false;\n\n const commit = debounce(() => {\n if (isDirty) {\n const value = store.get();\n const url = new URL(window.location.href);\n const parameters = new URLSearchParams(url[type].slice(1));\n const serializedValue = value !== undefined ? serialize(value) : undefined;\n\n if (serializedValue === undefined || serializedValue === serializedDefaultValue) {\n parameters.delete(key);\n } else {\n parameters.set(key, serializedValue);\n }\n\n url[type] = parameters.toString();\n\n window.history.replaceState(window.history.state, '', url.toString());\n window.dispatchEvent(new PopStateEvent('popstate'));\n\n onCommit?.(value);\n isDirty = false;\n }\n }, debounceTime);\n\n const cancelUrlListener = urlStore.subscribe((_url) => {\n if (isDirty) {\n return;\n }\n\n const url = new URL(_url);\n const parameters = new URLSearchParams(url[type].slice(1));\n const urlValue = parameters.get(key);\n\n store.set(urlValue !== null ? deserialize(urlValue) : defaultValue);\n });\n\n const cancelSubscription = store.subscribe(\n () => {\n isDirty = true;\n commit();\n },\n { runNow: false },\n );\n\n return disposable(() => {\n cancelUrlListener();\n cancelSubscription();\n commit.flush();\n });\n}\n\nfunction defaultDeserializer(value: string): any {\n if (value === undefined) {\n return undefined;\n }\n\n try {\n return JSON.parse(value, (_k, v) => {\n if (typeof v === 'object' && v !== null && '__set' in v) {\n return new Set(v.__set);\n }\n if (typeof v === 'object' && v !== null && '__map' in v) {\n return new Map(v.__map);\n }\n return v;\n });\n } catch {\n return undefined;\n }\n}\n\nfunction defaultSerializer(value: any): string {\n return JSON.stringify(value, (_k, v) => {\n if (v instanceof Set) {\n return { __set: Array.from(v) };\n }\n if (v instanceof Map) {\n return { __map: Array.from(v) };\n }\n return v;\n });\n}\n\nexport function createUrlStore<T>(options: UrlStoreOptionsWithDefaults<T>): Store<T>;\n\nexport function createUrlStore<T>(options: UrlStoreOptions<T>): Store<T | undefined>;\n\nexport function createUrlStore<T>(options: UrlStoreOptions<T>) {\n const store = createStore(options.defaultValue, options);\n connectUrl(store, options);\n return store;\n}\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"urlStore.mjs","sources":["../../src/core/urlStore.ts"],"sourcesContent":["import disposable from '@lib/disposable';\nimport { type DisposableCancel, type Duration } from './commonTypes';\nimport { createStore, type Store, type StoreOptions } from './store';\nimport { debounce } from '@lib/debounce';\n\nexport interface UrlStoreOptions<T> extends StoreOptions<T | undefined> {\n key: string;\n type?: 'search' | 'hash';\n serialize?: (value: T) => string;\n deserialize?: (value: string) => T;\n defaultValue?: T;\n writeDefaultValue?: boolean;\n onCommit?: (value: T | undefined) => void;\n debounce?: Duration;\n}\n\nexport interface UrlStoreOptionsWithDefaults<T> extends UrlStoreOptions<T> {\n defaultValue: T;\n}\n\nexport type UrlStoreOptionsRequired<T> = UrlStoreOptions<T> &\n Required<Pick<UrlStoreOptions<T>, 'type' | 'serialize' | 'deserialize' | 'defaultValue'>>;\n\nconst urlStore = createStore(() => (typeof window !== 'undefined' ? window.location.href : ''));\n\nurlStore.addEffect(() => {\n const originalPushState = window.history.pushState;\n const originalReplaceState = window.history.replaceState;\n\n const update = () => {\n urlStore.set(window.location.href);\n };\n\n window.history.pushState = (...args) => {\n originalPushState.apply(window.history, args);\n update();\n };\n\n window.history.replaceState = (...args) => {\n originalReplaceState.apply(window.history, args);\n update();\n };\n\n window.addEventListener('popstate', update);\n\n return () => {\n window.history.pushState = originalPushState;\n window.history.replaceState = originalReplaceState;\n window.removeEventListener('popstate', update);\n };\n});\n\nexport function updateUrlStore(): void {\n urlStore.set(window.location.href);\n}\n\nexport function connectUrl<T>(\n store: Store<T>,\n options: UrlStoreOptionsWithDefaults<T>,\n): DisposableCancel;\n\nexport function connectUrl<T>(\n store: Store<T | undefined>,\n options: UrlStoreOptions<T>,\n): DisposableCancel;\n\nexport function connectUrl<T>(\n store: Store<T>,\n {\n key,\n type = 'search',\n serialize = defaultSerializer,\n deserialize = defaultDeserializer,\n defaultValue = undefined as T,\n writeDefaultValue,\n onCommit,\n debounce: debounceTime = 500,\n }: UrlStoreOptions<T>,\n): DisposableCancel {\n const serializedDefaultValue = defaultValue !== undefined ? serialize(defaultValue) : undefined;\n let isDirty = writeDefaultValue ?? false;\n\n const commit = debounce(() => {\n if (isDirty) {\n const value = store.get();\n const url = new URL(window.location.href);\n const parameters = new URLSearchParams(url[type].slice(1));\n const serializedValue = value !== undefined ? serialize(value) : undefined;\n\n if (\n serializedValue === undefined ||\n (!writeDefaultValue && serializedValue === serializedDefaultValue)\n ) {\n parameters.delete(key);\n } else {\n parameters.set(key, serializedValue);\n }\n\n url[type] = parameters.toString();\n\n window.history.replaceState(window.history.state, '', url.toString());\n window.dispatchEvent(new PopStateEvent('popstate'));\n\n onCommit?.(value);\n isDirty = false;\n }\n }, debounceTime);\n\n const cancelUrlListener = urlStore.subscribe((_url) => {\n if (isDirty) {\n return;\n }\n\n const url = new URL(_url);\n const parameters = new URLSearchParams(url[type].slice(1));\n const urlValue = parameters.get(key);\n\n store.set(urlValue !== null ? deserialize(urlValue) : defaultValue);\n });\n\n const cancelSubscription = store.subscribe(\n () => {\n isDirty = true;\n commit();\n },\n { runNow: writeDefaultValue ?? false },\n );\n\n return disposable(() => {\n cancelUrlListener();\n cancelSubscription();\n commit.flush();\n });\n}\n\nfunction defaultDeserializer(value: string): any {\n if (value === undefined) {\n return undefined;\n }\n\n try {\n return JSON.parse(value, (_k, v) => {\n if (typeof v === 'object' && v !== null && '__set' in v) {\n return new Set(v.__set);\n }\n if (typeof v === 'object' && v !== null && '__map' in v) {\n return new Map(v.__map);\n }\n return v;\n });\n } catch {\n return undefined;\n }\n}\n\nfunction defaultSerializer(value: any): string {\n return JSON.stringify(value, (_k, v) => {\n if (v instanceof Set) {\n return { __set: Array.from(v) };\n }\n if (v instanceof Map) {\n return { __map: Array.from(v) };\n }\n return v;\n });\n}\n\nexport function createUrlStore<T>(options: UrlStoreOptionsWithDefaults<T>): Store<T>;\n\nexport function createUrlStore<T>(options: UrlStoreOptions<T>): Store<T | undefined>;\n\nexport function createUrlStore<T>(options: UrlStoreOptions<T>) {\n const store = createStore(options.defaultValue, options);\n connectUrl(store, options);\n return store;\n}\n"],"names":[],"mappings":"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"}
@@ -6,6 +6,7 @@ export interface UrlStoreOptions<T> extends StoreOptions<T | undefined> {
6
6
  serialize?: (value: T) => string;
7
7
  deserialize?: (value: string) => T;
8
8
  defaultValue?: T;
9
+ writeDefaultValue?: boolean;
9
10
  onCommit?: (value: T | undefined) => void;
10
11
  debounce?: Duration;
11
12
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "cross-state",
3
- "version": "0.49.3",
3
+ "version": "0.50.0",
4
4
  "description": "(React) state library",
5
5
  "license": "ISC",
6
6
  "repository": "schummar/cross-state",