@angular-libs/signal-storage 0.2.0-beta.2 → 0.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 CHANGED
@@ -1,14 +1,23 @@
1
1
  # @angular-libs/signal-storage
2
2
 
3
- A lightweight, type-safe Angular library for reactive state management using Signals. It provides seamless syncing with **localStorage**, **sessionStorage**, **IndexedDB** (via Angular Resources), and **In-Memory** storage.
3
+ Seamlessly sync Angular Signals with `localStorage`, `sessionStorage`, in-memory state, and `IndexedDB` with a strongly-typed reactive API.
4
4
 
5
- [StackBlitz demo](https://stackblitz.com/edit/signal-storage?file=src%2Fmain.ts) | `npm install @angular-libs/signal-storage`
5
+ `npm install @angular-libs/signal-storage`
6
6
 
7
7
  ---
8
8
 
9
- ## 🚀 1. Sync Storage (localStorage / sessionStorage)
9
+ ## Usage
10
10
 
11
- Perfect for lightweight, synchronous key-value storage. By default, it uses `localStorage` (falling back to memory for SSR).
11
+ ### 1. `SignalStorage` service
12
+
13
+ Perfect for lightweight, synchronous key-value storage.
14
+
15
+ Supports:
16
+
17
+ - `localStorage` (default)
18
+ - `sessionStorage`
19
+ - `MemoryStorage` (similar to NgRx SignalStore)
20
+ - Custom
12
21
 
13
22
  ```typescript
14
23
  import { Component, inject } from '@angular/core';
@@ -32,26 +41,32 @@ export class AppComponent {
32
41
  }
33
42
  ```
34
43
 
35
- ### Changing the Sync Storage Provider
44
+ #### Changing the Sync Storage Provider
36
45
 
37
- You can switch to `sessionStorage` or a purely `MemoryStorage` (similar to NgRx SignalStore without persistence) via providers:
46
+ You can switch to `sessionStorage`, `MemoryStorage` or custom via providers:
38
47
 
39
48
  ```typescript
40
- import { SIGNAL_STORAGE_TOKEN, MemoryStorage } from '@angular-libs/signal-storage';
49
+ import { SIGNAL_STORAGE_CONFIG, MemoryStorage } from '@angular-libs/signal-storage';
41
50
 
42
51
  // In app.config.ts
43
52
  providers: [
44
- // For Session Storage:
45
- { provide: SIGNAL_STORAGE_TOKEN, useFactory: () => window.sessionStorage },
46
-
47
- // OR for Pure Memory Storage:
48
- { provide: SIGNAL_STORAGE_TOKEN, useClass: MemoryStorage },
53
+ // MemoryStorage (similar to NgRx SignalStore):
54
+ {
55
+ provide: SIGNAL_STORAGE_CONFIG,
56
+ useValue: { storageFactory: () => new MemoryStorage() },
57
+ },
58
+
59
+ // sessionStorage:
60
+ {
61
+ provide: SIGNAL_STORAGE_CONFIG,
62
+ useValue: { storageFactory: () => window.sessionStorage },
63
+ },
49
64
  ];
50
65
  ```
51
66
 
52
67
  ---
53
68
 
54
- ## 2. Async Storage (IndexedDB + Angular Resources)
69
+ ### 2. Async `SignalIndexedDB` service (IndexedDB + Angular Resources)
55
70
 
56
71
  Ideal for larger amounts of data. It uses Angular's new `resource` API and supports automatic cross-tab synchronization.
57
72
 
@@ -100,25 +115,23 @@ export class ProfileComponent {
100
115
 
101
116
  ---
102
117
 
103
- ## 🛠 3. Advanced: Multiple Storage Instances
118
+ ### 🛠 Advanced: Multiple Storage Instances
104
119
 
105
120
  Need `localStorage` AND `MemoryStorage` in the same app? Extend the base class:
106
121
 
107
122
  ```typescript
108
- import { Injectable } from '@angular/core';
109
- import { SignalStorage, MemoryStorage } from '@angular-libs/signal-storage';
123
+ import { Injectable, inject, Injector } from '@angular/core';
124
+ import { MemoryStorage, SignalStorage, SignalIndexedDB } from '@angular-libs/signal-storage';
110
125
 
126
+ // Store 1
111
127
  interface LocalState {
112
128
  user: string;
113
129
  }
114
130
 
115
131
  @Injectable({ providedIn: 'root' })
116
- export class LocalStore extends SignalStorage<LocalState> {
117
- constructor() {
118
- super(typeof window !== 'undefined' ? window.localStorage : new MemoryStorage());
119
- }
120
- }
132
+ export class LocalStore extends SignalStorage<LocalState> {}
121
133
 
134
+ // Store 2
122
135
  interface TempState {
123
136
  tempId: string;
124
137
  }
@@ -126,17 +139,11 @@ interface TempState {
126
139
  @Injectable({ providedIn: 'root' })
127
140
  export class TempStore extends SignalStorage<TempState> {
128
141
  constructor() {
129
- super(new MemoryStorage());
142
+ super({ storageFactory: () => new MemoryStorage() });
130
143
  }
131
144
  }
132
- ```
133
-
134
- The same applies for `SignalIndexedDB` if you need multiple different databases:
135
-
136
- ```typescript
137
- import { Injectable, Injector, inject } from '@angular/core';
138
- import { SignalIndexedDB } from '@angular-libs/signal-storage';
139
145
 
146
+ // Store 3
140
147
  interface UserDbState {
141
148
  user: { name: string; email: string };
142
149
  }
@@ -147,17 +154,6 @@ export class UserDb extends SignalIndexedDB<UserDbState> {
147
154
  super(inject(Injector), { dbName: 'UserDb', storeName: 'keyValue', version: 1 });
148
155
  }
149
156
  }
150
-
151
- interface SettingsDbState {
152
- settings: { theme: 'light' | 'dark' };
153
- }
154
-
155
- @Injectable({ providedIn: 'root' })
156
- export class SettingsDb extends SignalIndexedDB<SettingsDbState> {
157
- constructor() {
158
- super(inject(Injector), { dbName: 'SettingsDb', storeName: 'keyValue', version: 1 });
159
- }
160
- }
161
157
  ```
162
158
 
163
159
  ---
@@ -86,22 +86,36 @@ class SignalStorage {
86
86
  switch (data.action) {
87
87
  case 'set':
88
88
  if (data.key) {
89
- this.set(data.key, data.value, false);
89
+ if (!this.isSharedStorage)
90
+ this.writeToStorage(data.key, data.value);
91
+ this.updateSignal(data.key, data.value);
90
92
  }
91
93
  break;
92
94
  case 'remove':
93
95
  if (data.key) {
94
- this.remove(data.key, false);
96
+ if (!this.isSharedStorage)
97
+ this.storage.removeItem(data.key);
98
+ this.updateSignalWithDefault(data.key);
95
99
  }
96
100
  break;
97
101
  case 'clear':
98
- this.clear(false);
102
+ if (!this.isSharedStorage)
103
+ this.storage.clear();
104
+ for (const key of this.signals.keys()) {
105
+ this.updateSignalWithDefault(key);
106
+ }
99
107
  break;
100
108
  }
101
109
  };
102
110
  }
103
111
  }
104
112
  }
113
+ get isSharedStorage() {
114
+ return typeof window !== 'undefined' && this.storage === window.localStorage;
115
+ }
116
+ ngOnDestroy() {
117
+ this.channel?.close();
118
+ }
105
119
  get(key, defaultValue) {
106
120
  const item = this.storage.getItem(key);
107
121
  if (item === null) {
@@ -131,15 +145,8 @@ class SignalStorage {
131
145
  * @param broadcast Whether to broadcast to other tabs
132
146
  */
133
147
  set(key, value, broadcast = true) {
134
- try {
135
- this.storage.setItem(key, JSON.stringify(value));
136
- }
137
- catch (e) {
138
- console.error(`Error saving to storage for key "${String(key)}". Storage quota may be exceeded.`, e);
139
- }
140
- if (this.signals.has(key)) {
141
- this.signals.get(key).set(value);
142
- }
148
+ this.writeToStorage(key, value);
149
+ this.updateSignal(key, value);
143
150
  if (broadcast) {
144
151
  this.channel?.postMessage({ action: 'set', key, value });
145
152
  }
@@ -162,10 +169,7 @@ class SignalStorage {
162
169
  */
163
170
  remove(key, broadcast = true) {
164
171
  this.storage.removeItem(key);
165
- if (this.signals.has(key)) {
166
- const def = this.defaultValues.get(key);
167
- this.signals.get(key).set(def !== undefined ? def : null);
168
- }
172
+ this.updateSignalWithDefault(key);
169
173
  if (broadcast) {
170
174
  this.channel?.postMessage({ action: 'remove', key });
171
175
  }
@@ -184,14 +188,32 @@ class SignalStorage {
184
188
  */
185
189
  clear(broadcast = true) {
186
190
  this.storage.clear();
187
- for (const [key, sig] of this.signals.entries()) {
188
- const def = this.defaultValues.get(key);
189
- sig.set(def !== undefined ? def : null);
191
+ for (const key of this.signals.keys()) {
192
+ this.updateSignalWithDefault(key);
190
193
  }
191
194
  if (broadcast) {
192
195
  this.channel?.postMessage({ action: 'clear' });
193
196
  }
194
197
  }
198
+ writeToStorage(key, value) {
199
+ try {
200
+ this.storage.setItem(key, JSON.stringify(value));
201
+ }
202
+ catch (e) {
203
+ console.error(`Error saving to storage for key "${String(key)}". Storage quota may be exceeded.`, e);
204
+ }
205
+ }
206
+ updateSignal(key, value) {
207
+ if (this.signals.has(key)) {
208
+ this.signals.get(key).set(value);
209
+ }
210
+ }
211
+ updateSignalWithDefault(key) {
212
+ if (this.signals.has(key)) {
213
+ const def = this.defaultValues.get(key);
214
+ this.signals.get(key).set(def !== undefined ? def : null);
215
+ }
216
+ }
195
217
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.1.0", ngImport: i0, type: SignalStorage, deps: [{ token: SIGNAL_STORAGE_CONFIG, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
196
218
  static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "21.1.0", ngImport: i0, type: SignalStorage });
197
219
  }
@@ -1 +1 @@
1
- {"version":3,"file":"angular-libs-signal-storage.mjs","sources":["../../../../projects/angular-libs/signal-storage/src/lib/signal-storage.ts","../../../../projects/angular-libs/signal-storage/src/lib/signal-indexeddb.ts","../../../../projects/angular-libs/signal-storage/src/public-api.ts","../../../../projects/angular-libs/signal-storage/src/angular-libs-signal-storage.ts"],"sourcesContent":["import {\n signal,\n WritableSignal,\n Signal,\n Injectable,\n InjectionToken,\n Optional,\n Inject,\n} from '@angular/core';\n\nexport interface SignalStorageConfig {\n /** The channel name for BroadcastChannel sync. Default: `signal-storage-sync` */\n syncChannel: string;\n /**\n * A factory returning the Storage to use (e.g., `() => sessionStorage`).\n * Defaults to `() => localStorage` (or MemoryStorage in SSR environments).\n */\n storageFactory: () => Storage;\n}\n\nconst DEFAULT_CONFIG: SignalStorageConfig = {\n syncChannel: 'signal-storage-sync',\n storageFactory: () => (typeof window !== 'undefined' ? window.localStorage : new MemoryStorage()),\n};\n\nexport const SIGNAL_STORAGE_CONFIG = new InjectionToken<SignalStorageConfig>(\n 'SIGNAL_STORAGE_CONFIG',\n {\n providedIn: 'root',\n factory: () => DEFAULT_CONFIG,\n },\n);\n\nexport function provideSignalStorage(config?: Partial<SignalStorageConfig>) {\n return [\n {\n provide: SIGNAL_STORAGE_CONFIG,\n useValue: { ...DEFAULT_CONFIG, ...config },\n },\n SignalStorage,\n ];\n}\n\ntype SyncMessage<T> =\n | { action: 'set'; key: keyof T; value: any }\n | { action: 'remove'; key: keyof T }\n | { action: 'clear' };\n\n/**\n * A strongly-typed, reactive storage solution powered by Angular Signals.\n * It natively supports `localStorage`, `sessionStorage`, `MemoryStorage`, or any custom storage mechanism.\n * Additionally, it automatically coordinates signal state changes across multiple browser tabs via BroadcastChannel.\n *\n * Define a strict type for the storage keys and values using the generic parameter `T`.\n * You can configure the storage mechanism and sync channel using the `provideSignalStorage` provider function,\n * or by passing a configuration object directly via `useFactory` when extending the class.\n *\n * @typeParam T - An interface defining the expected shape of the storage data.\n *\n * @example\n * ```typescript\n * // 1. Define your storage shape\n * interface AppState {\n * theme: 'light' | 'dark';\n * metrics: { visits: number };\n * }\n *\n * // 2. Create a typed service\n * @Injectable({ providedIn: 'root' })\n * export class AppStateStorage extends SignalStorage<AppState> {}\n *\n * @Component({ ... })\n * export class MyComponent {\n * // 3. Inject the typed service\n * private storage = inject(AppStateStorage);\n *\n * // Reactive: A readonly Signal that auto-updates\n * readonly theme = this.storage.getSignal('theme', 'light');\n *\n * constructor() {\n * // Static: Get the current value statically (no reactivity)\n * const rawTheme = this.storage.get('theme', 'light');\n *\n * // Mutate: Sets value in memory, updates the signal, and persists to storage\n * this.storage.set('theme', 'dark');\n * }\n *\n * toggleTheme() {\n * // Mutate safely via callback\n * this.storage.update('theme', current => current === 'light' ? 'dark' : 'light');\n * }\n * }\n * ```\n */\n@Injectable()\nexport class SignalStorage<T extends Record<string, any> = {}> {\n private storage: Storage;\n private signals = new Map<keyof T, WritableSignal<any>>();\n private defaultValues = new Map<keyof T, any>();\n private channel?: BroadcastChannel;\n\n /**\n * Create a new SignalStorage instance\n * @param config The configuration for the storage and synchronization.\n */\n constructor(@Optional() @Inject(SIGNAL_STORAGE_CONFIG) config?: Partial<SignalStorageConfig>) {\n const mergedConfig = { ...DEFAULT_CONFIG, ...config };\n this.storage = mergedConfig.storageFactory();\n const { syncChannel } = mergedConfig;\n\n if (typeof window !== 'undefined') {\n // Seamless BroadcastChannel cross-tab sync (for ANY storage type)\n if (typeof BroadcastChannel !== 'undefined') {\n this.channel = new BroadcastChannel(syncChannel);\n this.channel.onmessage = (event: MessageEvent<SyncMessage<T>>) => {\n const data = event.data;\n\n switch (data.action) {\n case 'set':\n if (data.key) {\n this.set(data.key, data.value, false);\n }\n break;\n case 'remove':\n if (data.key) {\n this.remove(data.key, false);\n }\n break;\n case 'clear':\n this.clear(false);\n break;\n }\n };\n }\n }\n }\n /**\n * Get typed data from SignalStorage\n * @param key The key to retrieve\n * @returns The typed data or null if not found\n */\n get<K extends keyof T>(key: K): T[K] | null;\n /**\n * Get typed data from SignalStorage with a default value\n * @param key The key to retrieve\n * @param defaultValue The default value to return if key not found\n * @returns The stored data or the default value\n */\n get<K extends keyof T>(key: K, defaultValue: T[K]): T[K];\n get<K extends keyof T>(key: K, defaultValue?: T[K]): T[K] | null {\n const item = this.storage.getItem(key as string);\n if (item === null) {\n return defaultValue ?? null;\n }\n try {\n return JSON.parse(item) as T[K];\n } catch {\n return defaultValue ?? null;\n }\n }\n\n /**\n * Get a reactive Angular signal for a key\n * @param key The key to retrieve\n * @param defaultValue The default value to return if key not found\n * @returns A read-only Signal containing the stored data\n */\n getSignal<K extends keyof T>(key: K): Signal<T[K] | null>;\n getSignal<K extends keyof T>(key: K, defaultValue: T[K]): Signal<T[K]>;\n getSignal<K extends keyof T>(key: K, defaultValue?: T[K]): Signal<any> {\n if (!this.signals.has(key)) {\n if (defaultValue !== undefined) {\n this.defaultValues.set(key, defaultValue);\n }\n const initialValue = this.get(key, defaultValue as T[K]);\n this.signals.set(key, signal(initialValue));\n }\n return this.signals.get(key)!.asReadonly();\n }\n\n /**\n * Set typed data in localStorage\n * @param key The key to set\n * @param value The value to store\n * @param broadcast Whether to broadcast to other tabs\n */\n set<K extends keyof T>(key: K, value: T[K], broadcast = true): void {\n try {\n this.storage.setItem(key as string, JSON.stringify(value));\n } catch (e) {\n console.error(\n `Error saving to storage for key \"${String(key)}\". Storage quota may be exceeded.`,\n e,\n );\n }\n if (this.signals.has(key)) {\n this.signals.get(key)!.set(value);\n }\n if (broadcast) {\n this.channel?.postMessage({ action: 'set', key, value });\n }\n }\n\n /**\n * Update typed data based on current value using a callback\n * @param key The key to update\n * @param updateFn Callback function that receives the current value and returns the new value\n */\n update<K extends keyof T>(key: K, updateFn: (currentValue: T[K] | null) => T[K]): void {\n const fallback = this.defaultValues.get(key);\n const currentValue = fallback !== undefined ? this.get(key, fallback) : this.get(key);\n const newValue = updateFn(currentValue);\n this.set(key, newValue);\n }\n\n /**\n * Remove an item from localStorage\n * @param key The key to remove\n * @param broadcast Whether to broadcast the removal to other tabs\n */\n remove<K extends keyof T>(key: K, broadcast = true): void {\n this.storage.removeItem(key as string);\n if (this.signals.has(key)) {\n const def = this.defaultValues.get(key);\n this.signals.get(key)!.set(def !== undefined ? def : null);\n }\n if (broadcast) {\n this.channel?.postMessage({ action: 'remove', key });\n }\n }\n\n /**\n * Check if a key exists in localStorage\n * @param key The key to check\n * @returns true if the key exists, false otherwise\n */\n has<K extends keyof T>(key: K): boolean {\n return this.storage.getItem(key as string) !== null;\n }\n\n /**\n * Clear all localStorage\n * @param broadcast Whether to broadcast the clear to other tabs\n */\n clear(broadcast = true): void {\n this.storage.clear();\n for (const [key, sig] of this.signals.entries()) {\n const def = this.defaultValues.get(key);\n sig.set(def !== undefined ? def : null);\n }\n if (broadcast) {\n this.channel?.postMessage({ action: 'clear' });\n }\n }\n}\n\n/**\n * An in-memory implementation of the Storage interface.\n * Can be provided to SIGNAL_STORAGE_TOKEN to use SignalStorage for purely in-memory app state.\n */\nexport class MemoryStorage implements Storage {\n private data = new Map<string, string>();\n\n get length(): number {\n return this.data.size;\n }\n\n clear(): void {\n this.data.clear();\n }\n\n getItem(key: string): string | null {\n const value = this.data.get(key);\n return value === undefined ? null : value;\n }\n\n removeItem(key: string): void {\n this.data.delete(key);\n }\n\n setItem(key: string, value: string): void {\n this.data.set(key, String(value));\n }\n\n key(index: number): string | null {\n const keys = Array.from(this.data.keys());\n return keys[index] === undefined ? null : keys[index];\n }\n}\n","import {\n Injectable,\n InjectionToken,\n Injector,\n Optional,\n Inject,\n resource,\n ResourceRef,\n OnDestroy,\n} from '@angular/core';\n\nexport interface SignalIndexedDBConfig {\n /** Default: `SignalStorageDb` */\n dbName: string;\n /** Default: `keyValue` */\n storeName: string;\n /** Default: `1` */\n version: number;\n}\n\nconst DEFAULT_CONFIG: SignalIndexedDBConfig = {\n dbName: 'SignalStorageDb',\n storeName: 'keyValue',\n version: 1,\n};\n\nexport const SIGNAL_INDEXEDDB_CONFIG = new InjectionToken<SignalIndexedDBConfig>(\n 'SIGNAL_INDEXEDDB_CONFIG',\n {\n providedIn: 'root',\n factory: () => DEFAULT_CONFIG,\n },\n);\n\nexport function provideSignalIndexedDB(config?: Partial<SignalIndexedDBConfig>) {\n return [\n {\n provide: SIGNAL_INDEXEDDB_CONFIG,\n useValue: { ...DEFAULT_CONFIG, ...config },\n },\n // Explicitly provide the storage instance here!\n SignalIndexedDB,\n ];\n}\n\ntype SyncMessage<T> =\n | { action: 'set'; key: keyof T; value: any }\n | { action: 'remove'; key: keyof T }\n | { action: 'clear' };\n\n/**\n * A strongly-typed, reactive, asynchronous storage solution powered by Angular's `resource` API and IndexedDB.\n * It gracefully falls back to an in-memory map if IndexedDB is not supported (e.g. server-side rendering).\n * Additionally, it automatically coordinates resource state changes across multiple browser tabs via BroadcastChannel.\n *\n * Define a strict type for the storage keys and values using the generic parameter `T`.\n * You can configure the DB name, store name, and version using the `provideSignalIndexedDB` provider function.\n *\n * @typeParam T - An interface defining the expected shape of the IndexedDB data.\n *\n * @example\n * ```typescript\n * // 1. Define your storage shape\n * interface AppState {\n * theme: 'light' | 'dark';\n * metrics: { visits: number };\n * }\n *\n * // 2. Create a typed service\n * @Injectable({ providedIn: 'root' })\n * export class AppStateDb extends SignalIndexedDB<AppState> {}\n *\n * @Component({ ... })\n * export class MyComponent {\n * // 3. Inject the typed service\n * private db = inject(AppStateDb);\n *\n * // Reactive: Angular Resource containing the async IndexedDB data\n * readonly themeResource = this.db.getResource('theme', 'light');\n *\n * async toggleTheme() {\n * // Static: Get the current value asynchronously (no reactivity)\n * const rawTheme = await this.db.get('theme');\n *\n * // Mutate: Sets value, updates the resource, and persists to IndexedDB\n * await this.db.set('theme', 'dark');\n *\n * // Mutate safely via callback\n * await this.db.update('theme', current => current === 'light' ? 'dark' : 'light');\n * }\n * }\n * ```\n */\n@Injectable()\nexport class SignalIndexedDB<T extends Record<string, any> = {}> implements OnDestroy {\n private config: SignalIndexedDBConfig;\n private dbPromise: Promise<IDBDatabase | null>;\n private fallbackStorage = new Map<string, any>();\n private defaultValues = new Map<keyof T, any>();\n private isFallback = false;\n\n private resources = new Map<keyof T, ResourceRef<any>>();\n private broadcastChannel?: BroadcastChannel;\n\n constructor(\n private injector: Injector,\n @Optional() @Inject(SIGNAL_INDEXEDDB_CONFIG) config?: SignalIndexedDBConfig,\n ) {\n this.config = { ...DEFAULT_CONFIG, ...(config || {}) };\n this.dbPromise = this.initDB().catch((err) => {\n console.warn('IndexedDB initialization failed. Falling back to in-memory storage.', err);\n this.isFallback = true;\n return null;\n });\n\n if (typeof window !== 'undefined' && typeof BroadcastChannel !== 'undefined') {\n try {\n this.broadcastChannel = new BroadcastChannel(\n `${this.config.dbName}-${this.config.storeName}-sync`,\n );\n this.broadcastChannel.onmessage = (event: MessageEvent<SyncMessage<T>>) => {\n const data = event.data;\n\n if (data.action === 'clear') {\n if (this.isFallback) this.fallbackStorage.clear();\n this.resources.forEach((res, k) => res.value.set(this.defaultValues.get(k)));\n } else if (data.action === 'remove' && data.key) {\n if (this.isFallback) this.fallbackStorage.delete(data.key as string);\n this.resources.get(data.key)?.value.set(this.defaultValues.get(data.key));\n } else if (data.action === 'set' && data.key) {\n if (this.isFallback) this.fallbackStorage.set(data.key as string, data.value);\n this.resources.get(data.key)?.value.set(data.value);\n }\n };\n } catch {\n // Ignore broadcast channel errors inside environments that do not fully support it\n }\n }\n }\n\n private initDB(): Promise<IDBDatabase> {\n return new Promise((resolve, reject) => {\n if (typeof window === 'undefined' || !window.indexedDB) {\n return reject('IndexedDB is not supported');\n }\n\n const request = indexedDB.open(this.config.dbName, this.config.version);\n\n request.onerror = () => reject(request.error);\n request.onsuccess = () => {\n const db = request.result;\n if (!db.objectStoreNames.contains(this.config.storeName)) {\n reject(new Error(`ObjectStore '${this.config.storeName}' not found in IndexedDB.`));\n } else {\n resolve(db);\n }\n };\n\n request.onupgradeneeded = (event: IDBVersionChangeEvent) => {\n const db = (event.target as IDBOpenDBRequest).result;\n if (!db.objectStoreNames.contains(this.config.storeName)) {\n db.createObjectStore(this.config.storeName);\n }\n };\n });\n }\n\n /**\n * Retrieves an Angular `ResourceRef` for a specific key.\n * The resource will asynchronously load the initial value from IndexedDB.\n * Calling `set()` or `update()` on this key later will automatically update this resource's signal.\n *\n * @param key The strongly-typed key to retrieve.\n * @returns A `ResourceRef` containing the value (or undefined if not found).\n */\n getResource<K extends keyof T>(key: K): ResourceRef<T[K] | undefined>;\n getResource<K extends keyof T>(key: K, defaultValue: T[K]): ResourceRef<T[K]>;\n getResource<K extends keyof T>(key: K, defaultValue?: T[K]): ResourceRef<any> {\n if (defaultValue !== undefined && !this.defaultValues.has(key)) {\n this.defaultValues.set(key, defaultValue);\n }\n if (!this.resources.has(key)) {\n const dbResource = resource({\n defaultValue,\n loader: async () => {\n const val = await this.get(key);\n return val !== undefined ? val : this.defaultValues.get(key);\n },\n injector: this.injector,\n });\n\n this.resources.set(key, dbResource);\n }\n\n return this.resources.get(key)!;\n }\n\n /**\n * Retrieves data asynchronously from IndexedDB without creating a reactive dependency.\n * Useful for one-off reads where a Signal/Resource is not needed.\n *\n * @param key The strongly-typed key to retrieve.\n * @returns A promise resolving to the stored value, or undefined if it does not exist.\n */\n async get<K extends keyof T>(key: K): Promise<T[K] | undefined> {\n if (this.isFallback) return this.fallbackStorage.get(key as string);\n\n const db = await this.dbPromise;\n if (!db) return this.fallbackStorage.get(key as string);\n\n return new Promise((resolve, reject) => {\n try {\n const transaction = db.transaction(this.config.storeName, 'readonly');\n const store = transaction.objectStore(this.config.storeName);\n const request = store.get(key as string);\n\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve(request.result);\n } catch (e) {\n reject(e);\n }\n });\n }\n\n /**\n * Sets data asynchronously in IndexedDB.\n * This is a reactive operation: it will automatically update any Angular `ResourceRef`\n * observing this key and broadcast the change to other open browser tabs.\n *\n * @param key The strongly-typed key to set.\n * @param value The value to store.\n */\n async set<K extends keyof T>(key: K, value: T[K]): Promise<void> {\n if (value === undefined) {\n return this.remove(key);\n }\n\n const db = await this.dbPromise;\n if (this.isFallback || !db) {\n this.fallbackStorage.set(key as string, value);\n } else {\n await new Promise<void>((resolve, reject) => {\n try {\n const request = db\n .transaction(this.config.storeName, 'readwrite')\n .objectStore(this.config.storeName)\n .put(value, key as string);\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve();\n } catch (e) {\n reject(e);\n }\n });\n }\n\n this.resources.get(key)?.value.set(value);\n this.broadcastChannel?.postMessage({ action: 'set', key, value });\n }\n\n /**\n * Safely updates a value based on its previous state.\n * Useful when deriving the next value directly from the old value.\n *\n * @param key The strongly-typed key to update.\n * @param updater A callback function that receives the current value and returns the new value.\n */\n async update<K extends keyof T>(\n key: K,\n updater: (current: T[K] | undefined) => T[K],\n ): Promise<void> {\n const current = await this.get(key);\n const newValue = updater(current);\n await this.set(key, newValue);\n }\n\n /**\n * Removes a key asynchronously from IndexedDB.\n * This operation will update the local state to undefined for the given key.\n *\n * @param key The strongly-typed key to remove.\n */\n async remove<K extends keyof T>(key: K): Promise<void> {\n const db = await this.dbPromise;\n if (this.isFallback || !db) {\n this.fallbackStorage.delete(key as string);\n } else {\n await new Promise<void>((resolve, reject) => {\n try {\n const request = db\n .transaction(this.config.storeName, 'readwrite')\n .objectStore(this.config.storeName)\n .delete(key as string);\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve();\n } catch (e) {\n reject(e);\n }\n });\n }\n\n this.resources.get(key)?.value.set(this.defaultValues.get(key));\n this.broadcastChannel?.postMessage({ action: 'remove', key });\n }\n\n /**\n * Clears the entire object store in IndexedDB.\n * This operation will reset all active resource signals to undefined.\n */\n async clear(): Promise<void> {\n const db = await this.dbPromise;\n if (this.isFallback || !db) {\n this.fallbackStorage.clear();\n } else {\n await new Promise<void>((resolve, reject) => {\n try {\n const request = db\n .transaction(this.config.storeName, 'readwrite')\n .objectStore(this.config.storeName)\n .clear();\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve();\n } catch (e) {\n reject(e);\n }\n });\n }\n\n this.resources.forEach((res, k) => res.value.set(this.defaultValues.get(k)));\n this.broadcastChannel?.postMessage({ action: 'clear' });\n }\n\n /**\n * Gets all keys currently stored in the object store.\n *\n * @returns A promise resolving to an array of stored keys.\n */\n async keys(): Promise<string[]> {\n if (this.isFallback) return Array.from(this.fallbackStorage.keys());\n\n const db = await this.dbPromise;\n if (!db) return Array.from(this.fallbackStorage.keys());\n\n return new Promise((resolve, reject) => {\n try {\n const transaction = db.transaction(this.config.storeName, 'readonly');\n const store = transaction.objectStore(this.config.storeName);\n const request = store.getAllKeys();\n\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve(request.result as string[]);\n } catch (e) {\n reject(e);\n }\n });\n }\n\n ngOnDestroy(): void {\n if (this.broadcastChannel) {\n this.broadcastChannel.close();\n }\n }\n}\n","/*\n * Public API Surface of signal-storage\n */\n\nexport * from './lib/signal-storage';\nexport * from './lib/signal-indexeddb';\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":["DEFAULT_CONFIG"],"mappings":"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"}
1
+ {"version":3,"file":"angular-libs-signal-storage.mjs","sources":["../../../../projects/angular-libs/signal-storage/src/lib/signal-storage.ts","../../../../projects/angular-libs/signal-storage/src/lib/signal-indexeddb.ts","../../../../projects/angular-libs/signal-storage/src/public-api.ts","../../../../projects/angular-libs/signal-storage/src/angular-libs-signal-storage.ts"],"sourcesContent":["import {\n signal,\n WritableSignal,\n Signal,\n Injectable,\n InjectionToken,\n Optional,\n Inject,\n OnDestroy,\n} from '@angular/core';\nimport { SyncMessage } from './sync-message';\n\nexport interface SignalStorageConfig {\n /** The channel name for BroadcastChannel sync. Default: `signal-storage-sync` */\n syncChannel: string;\n /**\n * A factory returning the Storage to use (e.g., `() => sessionStorage`).\n * Defaults to `() => localStorage` (or MemoryStorage in SSR environments).\n */\n storageFactory: () => Storage;\n}\n\nconst DEFAULT_CONFIG: SignalStorageConfig = {\n syncChannel: 'signal-storage-sync',\n storageFactory: () => (typeof window !== 'undefined' ? window.localStorage : new MemoryStorage()),\n};\n\nexport const SIGNAL_STORAGE_CONFIG = new InjectionToken<SignalStorageConfig>(\n 'SIGNAL_STORAGE_CONFIG',\n {\n providedIn: 'root',\n factory: () => DEFAULT_CONFIG,\n },\n);\n\nexport function provideSignalStorage(config?: Partial<SignalStorageConfig>) {\n return [\n {\n provide: SIGNAL_STORAGE_CONFIG,\n useValue: { ...DEFAULT_CONFIG, ...config },\n },\n SignalStorage,\n ];\n}\n\n/**\n * A strongly-typed, reactive storage solution powered by Angular Signals.\n * It natively supports `localStorage`, `sessionStorage`, `MemoryStorage`, or any custom storage mechanism.\n * Additionally, it automatically coordinates signal state changes across multiple browser tabs via BroadcastChannel.\n *\n * Define a strict type for the storage keys and values using the generic parameter `T`.\n * You can configure the storage mechanism and sync channel using the `provideSignalStorage` provider function,\n * or by passing a configuration object directly via `useFactory` when extending the class.\n *\n * @typeParam T - An interface defining the expected shape of the storage data.\n *\n * @example\n * ```typescript\n * // 1. Define your storage shape\n * interface AppState {\n * theme: 'light' | 'dark';\n * metrics: { visits: number };\n * }\n *\n * // 2. Create a typed service\n * @Injectable({ providedIn: 'root' })\n * export class AppStateStorage extends SignalStorage<AppState> {}\n *\n * @Component({ ... })\n * export class MyComponent {\n * // 3. Inject the typed service\n * private storage = inject(AppStateStorage);\n *\n * // Reactive: A readonly Signal that auto-updates\n * readonly theme = this.storage.getSignal('theme', 'light');\n *\n * constructor() {\n * // Static: Get the current value statically (no reactivity)\n * const rawTheme = this.storage.get('theme', 'light');\n *\n * // Mutate: Sets value in memory, updates the signal, and persists to storage\n * this.storage.set('theme', 'dark');\n * }\n *\n * toggleTheme() {\n * // Mutate safely via callback\n * this.storage.update('theme', current => current === 'light' ? 'dark' : 'light');\n * }\n * }\n * ```\n */\n@Injectable()\nexport class SignalStorage<T extends Record<string, any> = {}> implements OnDestroy {\n private storage: Storage;\n private signals = new Map<keyof T, WritableSignal<any>>();\n private defaultValues = new Map<keyof T, any>();\n private channel?: BroadcastChannel;\n\n /**\n * Create a new SignalStorage instance\n * @param config The configuration for the storage and synchronization.\n */\n constructor(@Optional() @Inject(SIGNAL_STORAGE_CONFIG) config?: Partial<SignalStorageConfig>) {\n const mergedConfig = { ...DEFAULT_CONFIG, ...config };\n this.storage = mergedConfig.storageFactory();\n const { syncChannel } = mergedConfig;\n\n if (typeof window !== 'undefined') {\n // Seamless BroadcastChannel cross-tab sync (for ANY storage type)\n if (typeof BroadcastChannel !== 'undefined') {\n this.channel = new BroadcastChannel(syncChannel);\n this.channel.onmessage = (event: MessageEvent<SyncMessage<T>>) => {\n const data = event.data;\n\n switch (data.action) {\n case 'set':\n if (data.key) {\n if (!this.isSharedStorage) this.writeToStorage(data.key, data.value);\n this.updateSignal(data.key, data.value);\n }\n break;\n case 'remove':\n if (data.key) {\n if (!this.isSharedStorage) this.storage.removeItem(data.key as string);\n this.updateSignalWithDefault(data.key);\n }\n break;\n case 'clear':\n if (!this.isSharedStorage) this.storage.clear();\n for (const key of this.signals.keys()) {\n this.updateSignalWithDefault(key);\n }\n break;\n }\n };\n }\n }\n }\n\n private get isSharedStorage(): boolean {\n return typeof window !== 'undefined' && this.storage === window.localStorage;\n }\n\n ngOnDestroy(): void {\n this.channel?.close();\n }\n\n /**\n * Get typed data from SignalStorage\n * @param key The key to retrieve\n * @returns The typed data or null if not found\n */\n get<K extends keyof T>(key: K): T[K] | null;\n /**\n * Get typed data from SignalStorage with a default value\n * @param key The key to retrieve\n * @param defaultValue The default value to return if key not found\n * @returns The stored data or the default value\n */\n get<K extends keyof T>(key: K, defaultValue: T[K]): T[K];\n get<K extends keyof T>(key: K, defaultValue?: T[K]): T[K] | null {\n const item = this.storage.getItem(key as string);\n if (item === null) {\n return defaultValue ?? null;\n }\n try {\n return JSON.parse(item) as T[K];\n } catch {\n return defaultValue ?? null;\n }\n }\n\n /**\n * Get a reactive Angular signal for a key\n * @param key The key to retrieve\n * @param defaultValue The default value to return if key not found\n * @returns A read-only Signal containing the stored data\n */\n getSignal<K extends keyof T>(key: K): Signal<T[K] | null>;\n getSignal<K extends keyof T>(key: K, defaultValue: T[K]): Signal<T[K]>;\n getSignal<K extends keyof T>(key: K, defaultValue?: T[K]): Signal<any> {\n if (!this.signals.has(key)) {\n if (defaultValue !== undefined) {\n this.defaultValues.set(key, defaultValue);\n }\n const initialValue = this.get(key, defaultValue as T[K]);\n this.signals.set(key, signal(initialValue));\n }\n return this.signals.get(key)!.asReadonly();\n }\n\n /**\n * Set typed data in localStorage\n * @param key The key to set\n * @param value The value to store\n * @param broadcast Whether to broadcast to other tabs\n */\n set<K extends keyof T>(key: K, value: T[K], broadcast = true): void {\n this.writeToStorage(key, value);\n this.updateSignal(key, value);\n if (broadcast) {\n this.channel?.postMessage({ action: 'set', key, value });\n }\n }\n\n /**\n * Update typed data based on current value using a callback\n * @param key The key to update\n * @param updateFn Callback function that receives the current value and returns the new value\n */\n update<K extends keyof T>(key: K, updateFn: (currentValue: T[K] | null) => T[K]): void {\n const fallback = this.defaultValues.get(key);\n const currentValue = fallback !== undefined ? this.get(key, fallback) : this.get(key);\n const newValue = updateFn(currentValue);\n this.set(key, newValue);\n }\n\n /**\n * Remove an item from localStorage\n * @param key The key to remove\n * @param broadcast Whether to broadcast the removal to other tabs\n */\n remove<K extends keyof T>(key: K, broadcast = true): void {\n this.storage.removeItem(key as string);\n this.updateSignalWithDefault(key);\n if (broadcast) {\n this.channel?.postMessage({ action: 'remove', key });\n }\n }\n\n /**\n * Check if a key exists in localStorage\n * @param key The key to check\n * @returns true if the key exists, false otherwise\n */\n has<K extends keyof T>(key: K): boolean {\n return this.storage.getItem(key as string) !== null;\n }\n\n /**\n * Clear all localStorage\n * @param broadcast Whether to broadcast the clear to other tabs\n */\n clear(broadcast = true): void {\n this.storage.clear();\n for (const key of this.signals.keys()) {\n this.updateSignalWithDefault(key);\n }\n if (broadcast) {\n this.channel?.postMessage({ action: 'clear' });\n }\n }\n\n private writeToStorage(key: keyof T, value: any): void {\n try {\n this.storage.setItem(key as string, JSON.stringify(value));\n } catch (e) {\n console.error(\n `Error saving to storage for key \"${String(key)}\". Storage quota may be exceeded.`,\n e,\n );\n }\n }\n\n private updateSignal(key: keyof T, value: any): void {\n if (this.signals.has(key)) {\n this.signals.get(key)!.set(value);\n }\n }\n\n private updateSignalWithDefault(key: keyof T): void {\n if (this.signals.has(key)) {\n const def = this.defaultValues.get(key);\n this.signals.get(key)!.set(def !== undefined ? def : null);\n }\n }\n}\n\n/**\n * An in-memory implementation of the Storage interface.\n * Can be provided to SIGNAL_STORAGE_TOKEN to use SignalStorage for purely in-memory app state.\n */\nexport class MemoryStorage implements Storage {\n private data = new Map<string, string>();\n\n get length(): number {\n return this.data.size;\n }\n\n clear(): void {\n this.data.clear();\n }\n\n getItem(key: string): string | null {\n const value = this.data.get(key);\n return value === undefined ? null : value;\n }\n\n removeItem(key: string): void {\n this.data.delete(key);\n }\n\n setItem(key: string, value: string): void {\n this.data.set(key, String(value));\n }\n\n key(index: number): string | null {\n const keys = Array.from(this.data.keys());\n return keys[index] === undefined ? null : keys[index];\n }\n}\n","import {\n Injectable,\n InjectionToken,\n Injector,\n Optional,\n Inject,\n resource,\n ResourceRef,\n OnDestroy,\n} from '@angular/core';\nimport { SyncMessage } from './sync-message';\n\nexport interface SignalIndexedDBConfig {\n /** Default: `SignalStorageDb` */\n dbName: string;\n /** Default: `keyValue` */\n storeName: string;\n /** Default: `1` */\n version: number;\n}\n\nconst DEFAULT_CONFIG: SignalIndexedDBConfig = {\n dbName: 'SignalStorageDb',\n storeName: 'keyValue',\n version: 1,\n};\n\nexport const SIGNAL_INDEXEDDB_CONFIG = new InjectionToken<SignalIndexedDBConfig>(\n 'SIGNAL_INDEXEDDB_CONFIG',\n {\n providedIn: 'root',\n factory: () => DEFAULT_CONFIG,\n },\n);\n\nexport function provideSignalIndexedDB(config?: Partial<SignalIndexedDBConfig>) {\n return [\n {\n provide: SIGNAL_INDEXEDDB_CONFIG,\n useValue: { ...DEFAULT_CONFIG, ...config },\n },\n // Explicitly provide the storage instance here!\n SignalIndexedDB,\n ];\n}\n\n/**\n * A strongly-typed, reactive, asynchronous storage solution powered by Angular's `resource` API and IndexedDB.\n * It gracefully falls back to an in-memory map if IndexedDB is not supported (e.g. server-side rendering).\n * Additionally, it automatically coordinates resource state changes across multiple browser tabs via BroadcastChannel.\n *\n * Define a strict type for the storage keys and values using the generic parameter `T`.\n * You can configure the DB name, store name, and version using the `provideSignalIndexedDB` provider function.\n *\n * @typeParam T - An interface defining the expected shape of the IndexedDB data.\n *\n * @example\n * ```typescript\n * // 1. Define your storage shape\n * interface AppState {\n * theme: 'light' | 'dark';\n * metrics: { visits: number };\n * }\n *\n * // 2. Create a typed service\n * @Injectable({ providedIn: 'root' })\n * export class AppStateDb extends SignalIndexedDB<AppState> {}\n *\n * @Component({ ... })\n * export class MyComponent {\n * // 3. Inject the typed service\n * private db = inject(AppStateDb);\n *\n * // Reactive: Angular Resource containing the async IndexedDB data\n * readonly themeResource = this.db.getResource('theme', 'light');\n *\n * async toggleTheme() {\n * // Static: Get the current value asynchronously (no reactivity)\n * const rawTheme = await this.db.get('theme');\n *\n * // Mutate: Sets value, updates the resource, and persists to IndexedDB\n * await this.db.set('theme', 'dark');\n *\n * // Mutate safely via callback\n * await this.db.update('theme', current => current === 'light' ? 'dark' : 'light');\n * }\n * }\n * ```\n */\n@Injectable()\nexport class SignalIndexedDB<T extends Record<string, any> = {}> implements OnDestroy {\n private config: SignalIndexedDBConfig;\n private dbPromise: Promise<IDBDatabase | null>;\n private fallbackStorage = new Map<string, any>();\n private defaultValues = new Map<keyof T, any>();\n private isFallback = false;\n\n private resources = new Map<keyof T, ResourceRef<any>>();\n private broadcastChannel?: BroadcastChannel;\n\n constructor(\n private injector: Injector,\n @Optional() @Inject(SIGNAL_INDEXEDDB_CONFIG) config?: SignalIndexedDBConfig,\n ) {\n this.config = { ...DEFAULT_CONFIG, ...(config || {}) };\n this.dbPromise = this.initDB().catch((err) => {\n console.warn('IndexedDB initialization failed. Falling back to in-memory storage.', err);\n this.isFallback = true;\n return null;\n });\n\n if (typeof window !== 'undefined' && typeof BroadcastChannel !== 'undefined') {\n try {\n this.broadcastChannel = new BroadcastChannel(\n `${this.config.dbName}-${this.config.storeName}-sync`,\n );\n this.broadcastChannel.onmessage = (event: MessageEvent<SyncMessage<T>>) => {\n const data = event.data;\n\n if (data.action === 'clear') {\n if (this.isFallback) this.fallbackStorage.clear();\n this.resources.forEach((res, k) => res.value.set(this.defaultValues.get(k)));\n } else if (data.action === 'remove' && data.key) {\n if (this.isFallback) this.fallbackStorage.delete(data.key as string);\n this.resources.get(data.key)?.value.set(this.defaultValues.get(data.key));\n } else if (data.action === 'set' && data.key) {\n if (this.isFallback) this.fallbackStorage.set(data.key as string, data.value);\n this.resources.get(data.key)?.value.set(data.value);\n }\n };\n } catch {\n // Ignore broadcast channel errors inside environments that do not fully support it\n }\n }\n }\n\n private initDB(): Promise<IDBDatabase> {\n return new Promise((resolve, reject) => {\n if (typeof window === 'undefined' || !window.indexedDB) {\n return reject('IndexedDB is not supported');\n }\n\n const request = indexedDB.open(this.config.dbName, this.config.version);\n\n request.onerror = () => reject(request.error);\n request.onsuccess = () => {\n const db = request.result;\n if (!db.objectStoreNames.contains(this.config.storeName)) {\n reject(new Error(`ObjectStore '${this.config.storeName}' not found in IndexedDB.`));\n } else {\n resolve(db);\n }\n };\n\n request.onupgradeneeded = (event: IDBVersionChangeEvent) => {\n const db = (event.target as IDBOpenDBRequest).result;\n if (!db.objectStoreNames.contains(this.config.storeName)) {\n db.createObjectStore(this.config.storeName);\n }\n };\n });\n }\n\n /**\n * Retrieves an Angular `ResourceRef` for a specific key.\n * The resource will asynchronously load the initial value from IndexedDB.\n * Calling `set()` or `update()` on this key later will automatically update this resource's signal.\n *\n * @param key The strongly-typed key to retrieve.\n * @returns A `ResourceRef` containing the value (or undefined if not found).\n */\n getResource<K extends keyof T>(key: K): ResourceRef<T[K] | undefined>;\n getResource<K extends keyof T>(key: K, defaultValue: T[K]): ResourceRef<T[K]>;\n getResource<K extends keyof T>(key: K, defaultValue?: T[K]): ResourceRef<any> {\n if (defaultValue !== undefined && !this.defaultValues.has(key)) {\n this.defaultValues.set(key, defaultValue);\n }\n if (!this.resources.has(key)) {\n const dbResource = resource({\n defaultValue,\n loader: async () => {\n const val = await this.get(key);\n return val !== undefined ? val : this.defaultValues.get(key);\n },\n injector: this.injector,\n });\n\n this.resources.set(key, dbResource);\n }\n\n return this.resources.get(key)!;\n }\n\n /**\n * Retrieves data asynchronously from IndexedDB without creating a reactive dependency.\n * Useful for one-off reads where a Signal/Resource is not needed.\n *\n * @param key The strongly-typed key to retrieve.\n * @returns A promise resolving to the stored value, or undefined if it does not exist.\n */\n async get<K extends keyof T>(key: K): Promise<T[K] | undefined> {\n if (this.isFallback) return this.fallbackStorage.get(key as string);\n\n const db = await this.dbPromise;\n if (!db) return this.fallbackStorage.get(key as string);\n\n return new Promise((resolve, reject) => {\n try {\n const transaction = db.transaction(this.config.storeName, 'readonly');\n const store = transaction.objectStore(this.config.storeName);\n const request = store.get(key as string);\n\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve(request.result);\n } catch (e) {\n reject(e);\n }\n });\n }\n\n /**\n * Sets data asynchronously in IndexedDB.\n * This is a reactive operation: it will automatically update any Angular `ResourceRef`\n * observing this key and broadcast the change to other open browser tabs.\n *\n * @param key The strongly-typed key to set.\n * @param value The value to store.\n */\n async set<K extends keyof T>(key: K, value: T[K]): Promise<void> {\n if (value === undefined) {\n return this.remove(key);\n }\n\n const db = await this.dbPromise;\n if (this.isFallback || !db) {\n this.fallbackStorage.set(key as string, value);\n } else {\n await new Promise<void>((resolve, reject) => {\n try {\n const request = db\n .transaction(this.config.storeName, 'readwrite')\n .objectStore(this.config.storeName)\n .put(value, key as string);\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve();\n } catch (e) {\n reject(e);\n }\n });\n }\n\n this.resources.get(key)?.value.set(value);\n this.broadcastChannel?.postMessage({ action: 'set', key, value });\n }\n\n /**\n * Safely updates a value based on its previous state.\n * Useful when deriving the next value directly from the old value.\n *\n * @param key The strongly-typed key to update.\n * @param updater A callback function that receives the current value and returns the new value.\n */\n async update<K extends keyof T>(\n key: K,\n updater: (current: T[K] | undefined) => T[K],\n ): Promise<void> {\n const current = await this.get(key);\n const newValue = updater(current);\n await this.set(key, newValue);\n }\n\n /**\n * Removes a key asynchronously from IndexedDB.\n * This operation will update the local state to undefined for the given key.\n *\n * @param key The strongly-typed key to remove.\n */\n async remove<K extends keyof T>(key: K): Promise<void> {\n const db = await this.dbPromise;\n if (this.isFallback || !db) {\n this.fallbackStorage.delete(key as string);\n } else {\n await new Promise<void>((resolve, reject) => {\n try {\n const request = db\n .transaction(this.config.storeName, 'readwrite')\n .objectStore(this.config.storeName)\n .delete(key as string);\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve();\n } catch (e) {\n reject(e);\n }\n });\n }\n\n this.resources.get(key)?.value.set(this.defaultValues.get(key));\n this.broadcastChannel?.postMessage({ action: 'remove', key });\n }\n\n /**\n * Clears the entire object store in IndexedDB.\n * This operation will reset all active resource signals to undefined.\n */\n async clear(): Promise<void> {\n const db = await this.dbPromise;\n if (this.isFallback || !db) {\n this.fallbackStorage.clear();\n } else {\n await new Promise<void>((resolve, reject) => {\n try {\n const request = db\n .transaction(this.config.storeName, 'readwrite')\n .objectStore(this.config.storeName)\n .clear();\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve();\n } catch (e) {\n reject(e);\n }\n });\n }\n\n this.resources.forEach((res, k) => res.value.set(this.defaultValues.get(k)));\n this.broadcastChannel?.postMessage({ action: 'clear' });\n }\n\n /**\n * Gets all keys currently stored in the object store.\n *\n * @returns A promise resolving to an array of stored keys.\n */\n async keys(): Promise<string[]> {\n if (this.isFallback) return Array.from(this.fallbackStorage.keys());\n\n const db = await this.dbPromise;\n if (!db) return Array.from(this.fallbackStorage.keys());\n\n return new Promise((resolve, reject) => {\n try {\n const transaction = db.transaction(this.config.storeName, 'readonly');\n const store = transaction.objectStore(this.config.storeName);\n const request = store.getAllKeys();\n\n request.onerror = () => reject(request.error);\n request.onsuccess = () => resolve(request.result as string[]);\n } catch (e) {\n reject(e);\n }\n });\n }\n\n ngOnDestroy(): void {\n if (this.broadcastChannel) {\n this.broadcastChannel.close();\n }\n }\n}\n","/*\n * Public API Surface of signal-storage\n */\n\nexport * from './lib/signal-storage';\nexport * from './lib/signal-indexeddb';\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":["DEFAULT_CONFIG"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@angular-libs/signal-storage",
3
- "version": "0.2.0-beta.2",
3
+ "version": "0.2.1",
4
4
  "peerDependencies": {
5
5
  "@angular/common": ">=19.0.0",
6
6
  "@angular/core": ">=19.0.0"
@@ -11,6 +11,7 @@
11
11
  "dependencies": {
12
12
  "tslib": "^2.3.0"
13
13
  },
14
+ "homepage": "https://github.com/angular-lib/angular-libs/tree/main/projects/angular-libs/signal-storage",
14
15
  "license": "MIT",
15
16
  "sideEffects": false,
16
17
  "module": "fesm2022/angular-libs-signal-storage.mjs",
@@ -1,5 +1,5 @@
1
1
  import * as i0 from '@angular/core';
2
- import { InjectionToken, Signal, OnDestroy, Injector, ResourceRef } from '@angular/core';
2
+ import { InjectionToken, OnDestroy, Signal, Injector, ResourceRef } from '@angular/core';
3
3
 
4
4
  interface SignalStorageConfig {
5
5
  /** The channel name for BroadcastChannel sync. Default: `signal-storage-sync` */
@@ -64,7 +64,7 @@ declare function provideSignalStorage(config?: Partial<SignalStorageConfig>): (t
64
64
  * }
65
65
  * ```
66
66
  */
67
- declare class SignalStorage<T extends Record<string, any> = {}> {
67
+ declare class SignalStorage<T extends Record<string, any> = {}> implements OnDestroy {
68
68
  private storage;
69
69
  private signals;
70
70
  private defaultValues;
@@ -74,6 +74,8 @@ declare class SignalStorage<T extends Record<string, any> = {}> {
74
74
  * @param config The configuration for the storage and synchronization.
75
75
  */
76
76
  constructor(config?: Partial<SignalStorageConfig>);
77
+ private get isSharedStorage();
78
+ ngOnDestroy(): void;
77
79
  /**
78
80
  * Get typed data from SignalStorage
79
81
  * @param key The key to retrieve
@@ -125,6 +127,9 @@ declare class SignalStorage<T extends Record<string, any> = {}> {
125
127
  * @param broadcast Whether to broadcast the clear to other tabs
126
128
  */
127
129
  clear(broadcast?: boolean): void;
130
+ private writeToStorage;
131
+ private updateSignal;
132
+ private updateSignalWithDefault;
128
133
  static ɵfac: i0.ɵɵFactoryDeclaration<SignalStorage<any>, [{ optional: true; }]>;
129
134
  static ɵprov: i0.ɵɵInjectableDeclaration<SignalStorage<any>>;
130
135
  }