@oneluiz/dual-datepicker 3.5.1 → 3.9.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.
- package/core/built-in-presets.d.ts +98 -0
- package/core/calendar-grid/calendar-grid.cache.d.ts +39 -0
- package/core/calendar-grid/calendar-grid.factory.d.ts +26 -0
- package/core/calendar-grid/calendar-grid.types.d.ts +57 -0
- package/core/calendar-grid/index.d.ts +50 -0
- package/core/calendar-grid/range-highlighter.cache.d.ts +105 -0
- package/core/calendar-grid/range-highlighter.d.ts +85 -0
- package/core/calendar-grid/range-highlighter.types.d.ts +182 -0
- package/core/calendar-grid/virtual-weeks.logic.d.ts +116 -0
- package/core/calendar-grid/virtual-weeks.types.d.ts +71 -0
- package/core/index.d.ts +22 -0
- package/core/preset-providers.d.ts +176 -0
- package/core/preset-registry.d.ts +181 -0
- package/core/preset.engine.d.ts +57 -21
- package/core/range-preset.plugin.d.ts +188 -0
- package/dual-datepicker.component.d.ts +78 -1
- package/esm2022/core/built-in-presets.mjs +289 -0
- package/esm2022/core/calendar-grid/calendar-grid.cache.mjs +92 -0
- package/esm2022/core/calendar-grid/calendar-grid.factory.mjs +97 -0
- package/esm2022/core/calendar-grid/calendar-grid.types.mjs +8 -0
- package/esm2022/core/calendar-grid/index.mjs +51 -0
- package/esm2022/core/calendar-grid/range-highlighter.cache.mjs +198 -0
- package/esm2022/core/calendar-grid/range-highlighter.mjs +185 -0
- package/esm2022/core/calendar-grid/range-highlighter.types.mjs +11 -0
- package/esm2022/core/calendar-grid/virtual-weeks.logic.mjs +149 -0
- package/esm2022/core/calendar-grid/virtual-weeks.types.mjs +11 -0
- package/esm2022/core/index.mjs +26 -1
- package/esm2022/core/preset-providers.mjs +243 -0
- package/esm2022/core/preset-registry.mjs +277 -0
- package/esm2022/core/preset.engine.mjs +86 -210
- package/esm2022/core/range-preset.plugin.mjs +70 -0
- package/esm2022/dual-datepicker.component.mjs +202 -39
- package/fesm2022/oneluiz-dual-datepicker.mjs +1985 -282
- package/fesm2022/oneluiz-dual-datepicker.mjs.map +1 -1
- package/package.json +2 -2
|
@@ -0,0 +1,243 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Provider Functions for Built-in Presets
|
|
3
|
+
*
|
|
4
|
+
* Version: 3.6.0
|
|
5
|
+
*
|
|
6
|
+
* Automatic registration of built-in date range presets.
|
|
7
|
+
* These providers ensure backward compatibility by auto-registering
|
|
8
|
+
* all standard presets (TODAY, LAST_7_DAYS, THIS_MONTH, etc.)
|
|
9
|
+
*
|
|
10
|
+
* USAGE IN LIBRARY (Internal):
|
|
11
|
+
* Built-in presets are registered automatically via Angular providers.
|
|
12
|
+
* Library consumers don't need to do anything.
|
|
13
|
+
*
|
|
14
|
+
* USAGE IN APP (Custom Presets):
|
|
15
|
+
* ```typescript
|
|
16
|
+
* // app.config.ts
|
|
17
|
+
* export const appConfig: ApplicationConfig = {
|
|
18
|
+
* providers: [
|
|
19
|
+
* // ... other providers
|
|
20
|
+
* provideCustomPresets([
|
|
21
|
+
* {
|
|
22
|
+
* key: 'THIS_FISCAL_QUARTER',
|
|
23
|
+
* resolve: (clock, adapter) => {
|
|
24
|
+
* const now = clock.now();
|
|
25
|
+
* // ... fiscal logic
|
|
26
|
+
* return { start, end };
|
|
27
|
+
* }
|
|
28
|
+
* }
|
|
29
|
+
* ])
|
|
30
|
+
* ]
|
|
31
|
+
* };
|
|
32
|
+
* ```
|
|
33
|
+
*/
|
|
34
|
+
import { APP_INITIALIZER, makeEnvironmentProviders } from '@angular/core';
|
|
35
|
+
import { PresetRegistry } from './preset-registry';
|
|
36
|
+
import { BUILT_IN_PRESETS } from './built-in-presets';
|
|
37
|
+
/**
|
|
38
|
+
* Initializer function that registers built-in presets
|
|
39
|
+
*
|
|
40
|
+
* Runs at application startup (APP_INITIALIZER)
|
|
41
|
+
*
|
|
42
|
+
* @param registry - PresetRegistry instance
|
|
43
|
+
* @returns Initialization function
|
|
44
|
+
*/
|
|
45
|
+
function initializeBuiltInPresets(registry) {
|
|
46
|
+
return () => {
|
|
47
|
+
// Register all built-in presets
|
|
48
|
+
BUILT_IN_PRESETS.forEach(preset => {
|
|
49
|
+
registry.register(preset);
|
|
50
|
+
});
|
|
51
|
+
// Log registration for debugging (can be removed in production)
|
|
52
|
+
if (typeof console !== 'undefined' && console.debug) {
|
|
53
|
+
console.debug(`[ng-dual-datepicker] Registered ${BUILT_IN_PRESETS.length} built-in presets:`, BUILT_IN_PRESETS.map(p => p.key).join(', '));
|
|
54
|
+
}
|
|
55
|
+
};
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Provide built-in date range presets
|
|
59
|
+
*
|
|
60
|
+
* This provider is automatically included in the library's root providers.
|
|
61
|
+
* Library consumers don't need to add this manually.
|
|
62
|
+
*
|
|
63
|
+
* @returns EnvironmentProviders for built-in presets
|
|
64
|
+
*
|
|
65
|
+
* @internal
|
|
66
|
+
*/
|
|
67
|
+
export function provideBuiltInPresets() {
|
|
68
|
+
return makeEnvironmentProviders([
|
|
69
|
+
{
|
|
70
|
+
provide: APP_INITIALIZER,
|
|
71
|
+
multi: true,
|
|
72
|
+
useFactory: initializeBuiltInPresets,
|
|
73
|
+
deps: [PresetRegistry]
|
|
74
|
+
}
|
|
75
|
+
]);
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Provide custom date range presets
|
|
79
|
+
*
|
|
80
|
+
* Use this to register your own industry-specific presets:
|
|
81
|
+
* - Fiscal presets
|
|
82
|
+
* - Hotel/hospitality presets
|
|
83
|
+
* - Logistics presets
|
|
84
|
+
* - Custom business logic
|
|
85
|
+
*
|
|
86
|
+
* @param presets - Array of custom RangePresetPlugin implementations
|
|
87
|
+
* @returns EnvironmentProviders for custom presets
|
|
88
|
+
*
|
|
89
|
+
* @example
|
|
90
|
+
* ```typescript
|
|
91
|
+
* // Fiscal presets
|
|
92
|
+
* const FISCAL_PRESETS: RangePresetPlugin[] = [
|
|
93
|
+
* {
|
|
94
|
+
* key: 'THIS_FISCAL_QUARTER',
|
|
95
|
+
* resolve: (clock, adapter) => {
|
|
96
|
+
* const now = clock.now();
|
|
97
|
+
* const month = adapter.getMonth(now);
|
|
98
|
+
*
|
|
99
|
+
* // Fiscal year starts April (month 3)
|
|
100
|
+
* const fiscalMonth = (month + 9) % 12;
|
|
101
|
+
* const quarterStart = Math.floor(fiscalMonth / 3) * 3;
|
|
102
|
+
* const calendarMonth = (quarterStart - 9 + 12) % 12;
|
|
103
|
+
*
|
|
104
|
+
* const year = adapter.getYear(now);
|
|
105
|
+
* const fiscalYear = month < 3 ? year - 1 : year;
|
|
106
|
+
*
|
|
107
|
+
* const start = new Date(fiscalYear, calendarMonth, 1);
|
|
108
|
+
* const end = new Date(fiscalYear, calendarMonth + 3, 0);
|
|
109
|
+
*
|
|
110
|
+
* return {
|
|
111
|
+
* start: adapter.normalize(start),
|
|
112
|
+
* end: adapter.normalize(end)
|
|
113
|
+
* };
|
|
114
|
+
* }
|
|
115
|
+
* },
|
|
116
|
+
* {
|
|
117
|
+
* key: 'FISCAL_YEAR_TO_DATE',
|
|
118
|
+
* resolve: (clock, adapter) => {
|
|
119
|
+
* const now = clock.now();
|
|
120
|
+
* const month = adapter.getMonth(now);
|
|
121
|
+
* const year = adapter.getYear(now);
|
|
122
|
+
*
|
|
123
|
+
* // Fiscal year starts April 1
|
|
124
|
+
* const fiscalYearStart = month >= 3
|
|
125
|
+
* ? new Date(year, 3, 1)
|
|
126
|
+
* : new Date(year - 1, 3, 1);
|
|
127
|
+
*
|
|
128
|
+
* return {
|
|
129
|
+
* start: adapter.normalize(fiscalYearStart),
|
|
130
|
+
* end: adapter.normalize(now)
|
|
131
|
+
* };
|
|
132
|
+
* }
|
|
133
|
+
* }
|
|
134
|
+
* ];
|
|
135
|
+
*
|
|
136
|
+
* // In app.config.ts
|
|
137
|
+
* export const appConfig: ApplicationConfig = {
|
|
138
|
+
* providers: [
|
|
139
|
+
* provideCustomPresets(FISCAL_PRESETS)
|
|
140
|
+
* ]
|
|
141
|
+
* };
|
|
142
|
+
*
|
|
143
|
+
* // Use in components
|
|
144
|
+
* store.applyPreset('THIS_FISCAL_QUARTER');
|
|
145
|
+
* store.applyPreset('FISCAL_YEAR_TO_DATE');
|
|
146
|
+
* ```
|
|
147
|
+
*
|
|
148
|
+
* @example
|
|
149
|
+
* ```typescript
|
|
150
|
+
* // Hotel presets
|
|
151
|
+
* const HOTEL_PRESETS: RangePresetPlugin[] = [
|
|
152
|
+
* {
|
|
153
|
+
* key: 'CHECK_IN_WEEK',
|
|
154
|
+
* resolve: (clock, adapter) => {
|
|
155
|
+
* const now = clock.now();
|
|
156
|
+
* const dayOfWeek = adapter.getDayOfWeek(now);
|
|
157
|
+
*
|
|
158
|
+
* // Check-in week: Friday to Friday
|
|
159
|
+
* const daysToNextFriday = dayOfWeek <= 5
|
|
160
|
+
* ? 5 - dayOfWeek
|
|
161
|
+
* : 7 - dayOfWeek + 5;
|
|
162
|
+
*
|
|
163
|
+
* const nextFriday = adapter.addDays(now, daysToNextFriday);
|
|
164
|
+
* const followingFriday = adapter.addDays(nextFriday, 7);
|
|
165
|
+
*
|
|
166
|
+
* return { start: nextFriday, end: followingFriday };
|
|
167
|
+
* }
|
|
168
|
+
* },
|
|
169
|
+
* {
|
|
170
|
+
* key: 'NEXT_30_NIGHTS',
|
|
171
|
+
* resolve: (clock, adapter) => {
|
|
172
|
+
* const now = clock.now();
|
|
173
|
+
* const tomorrow = adapter.addDays(now, 1);
|
|
174
|
+
* const end = adapter.addDays(tomorrow, 30);
|
|
175
|
+
* return { start: tomorrow, end };
|
|
176
|
+
* }
|
|
177
|
+
* }
|
|
178
|
+
* ];
|
|
179
|
+
*
|
|
180
|
+
* providers: [provideCustomPresets(HOTEL_PRESETS)]
|
|
181
|
+
* ```
|
|
182
|
+
*/
|
|
183
|
+
export function provideCustomPresets(presets) {
|
|
184
|
+
return makeEnvironmentProviders([
|
|
185
|
+
{
|
|
186
|
+
provide: APP_INITIALIZER,
|
|
187
|
+
multi: true,
|
|
188
|
+
useFactory: (registry) => {
|
|
189
|
+
return () => {
|
|
190
|
+
presets.forEach(preset => {
|
|
191
|
+
registry.register(preset);
|
|
192
|
+
});
|
|
193
|
+
if (typeof console !== 'undefined' && console.debug) {
|
|
194
|
+
console.debug(`[ng-dual-datepicker] Registered ${presets.length} custom presets:`, presets.map(p => p.key).join(', '));
|
|
195
|
+
}
|
|
196
|
+
};
|
|
197
|
+
},
|
|
198
|
+
deps: [PresetRegistry]
|
|
199
|
+
}
|
|
200
|
+
]);
|
|
201
|
+
}
|
|
202
|
+
/**
|
|
203
|
+
* Provide preset package
|
|
204
|
+
*
|
|
205
|
+
* Convenience function for external preset packages.
|
|
206
|
+
*
|
|
207
|
+
* @param packageName - Name of the preset package (for logging)
|
|
208
|
+
* @param presets - Array of presets from the package
|
|
209
|
+
* @returns EnvironmentProviders
|
|
210
|
+
*
|
|
211
|
+
* @example
|
|
212
|
+
* ```typescript
|
|
213
|
+
* // @acme/fiscal-presets package
|
|
214
|
+
* export function provideFiscalPresets(): EnvironmentProviders {
|
|
215
|
+
* return providePresetPackage('@acme/fiscal-presets', FISCAL_PRESETS);
|
|
216
|
+
* }
|
|
217
|
+
*
|
|
218
|
+
* // In app
|
|
219
|
+
* import { provideFiscalPresets } from '@acme/fiscal-presets';
|
|
220
|
+
*
|
|
221
|
+
* providers: [provideFiscalPresets()]
|
|
222
|
+
* ```
|
|
223
|
+
*/
|
|
224
|
+
export function providePresetPackage(packageName, presets) {
|
|
225
|
+
return makeEnvironmentProviders([
|
|
226
|
+
{
|
|
227
|
+
provide: APP_INITIALIZER,
|
|
228
|
+
multi: true,
|
|
229
|
+
useFactory: (registry) => {
|
|
230
|
+
return () => {
|
|
231
|
+
presets.forEach(preset => {
|
|
232
|
+
registry.register(preset);
|
|
233
|
+
});
|
|
234
|
+
if (typeof console !== 'undefined' && console.debug) {
|
|
235
|
+
console.debug(`[${packageName}] Registered ${presets.length} presets:`, presets.map(p => p.key).join(', '));
|
|
236
|
+
}
|
|
237
|
+
};
|
|
238
|
+
},
|
|
239
|
+
deps: [PresetRegistry]
|
|
240
|
+
}
|
|
241
|
+
]);
|
|
242
|
+
}
|
|
243
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"preset-providers.js","sourceRoot":"","sources":["../../../src/core/preset-providers.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Provider Functions for Built-in Presets\n * \n * Version: 3.6.0\n * \n * Automatic registration of built-in date range presets.\n * These providers ensure backward compatibility by auto-registering\n * all standard presets (TODAY, LAST_7_DAYS, THIS_MONTH, etc.)\n * \n * USAGE IN LIBRARY (Internal):\n * Built-in presets are registered automatically via Angular providers.\n * Library consumers don't need to do anything.\n * \n * USAGE IN APP (Custom Presets):\n * ```typescript\n * // app.config.ts\n * export const appConfig: ApplicationConfig = {\n *   providers: [\n *     // ... other providers\n *     provideCustomPresets([\n *       {\n *         key: 'THIS_FISCAL_QUARTER',\n *         resolve: (clock, adapter) => {\n *           const now = clock.now();\n *           // ... fiscal logic\n *           return { start, end };\n *         }\n *       }\n *     ])\n *   ]\n * };\n * ```\n */\n\nimport { APP_INITIALIZER, EnvironmentProviders, makeEnvironmentProviders, inject } from '@angular/core';\nimport { PresetRegistry } from './preset-registry';\nimport { BUILT_IN_PRESETS } from './built-in-presets';\nimport { RangePresetPlugin } from './range-preset.plugin';\n\n/**\n * Initializer function that registers built-in presets\n * \n * Runs at application startup (APP_INITIALIZER)\n * \n * @param registry - PresetRegistry instance\n * @returns Initialization function\n */\nfunction initializeBuiltInPresets(registry: PresetRegistry): () => void {\n  return () => {\n    // Register all built-in presets\n    BUILT_IN_PRESETS.forEach(preset => {\n      registry.register(preset);\n    });\n\n    // Log registration for debugging (can be removed in production)\n    if (typeof console !== 'undefined' && console.debug) {\n      console.debug(\n        `[ng-dual-datepicker] Registered ${BUILT_IN_PRESETS.length} built-in presets:`,\n        BUILT_IN_PRESETS.map(p => p.key).join(', ')\n      );\n    }\n  };\n}\n\n/**\n * Provide built-in date range presets\n * \n * This provider is automatically included in the library's root providers.\n * Library consumers don't need to add this manually.\n * \n * @returns EnvironmentProviders for built-in presets\n * \n * @internal\n */\nexport function provideBuiltInPresets(): EnvironmentProviders {\n  return makeEnvironmentProviders([\n    {\n      provide: APP_INITIALIZER,\n      multi: true,\n      useFactory: initializeBuiltInPresets,\n      deps: [PresetRegistry]\n    }\n  ]);\n}\n\n/**\n * Provide custom date range presets\n * \n * Use this to register your own industry-specific presets:\n * - Fiscal presets\n * - Hotel/hospitality presets\n * - Logistics presets\n * - Custom business logic\n * \n * @param presets - Array of custom RangePresetPlugin implementations\n * @returns EnvironmentProviders for custom presets\n * \n * @example\n * ```typescript\n * // Fiscal presets\n * const FISCAL_PRESETS: RangePresetPlugin[] = [\n *   {\n *     key: 'THIS_FISCAL_QUARTER',\n *     resolve: (clock, adapter) => {\n *       const now = clock.now();\n *       const month = adapter.getMonth(now);\n *       \n *       // Fiscal year starts April (month 3)\n *       const fiscalMonth = (month + 9) % 12;\n *       const quarterStart = Math.floor(fiscalMonth / 3) * 3;\n *       const calendarMonth = (quarterStart - 9 + 12) % 12;\n *       \n *       const year = adapter.getYear(now);\n *       const fiscalYear = month < 3 ? year - 1 : year;\n *       \n *       const start = new Date(fiscalYear, calendarMonth, 1);\n *       const end = new Date(fiscalYear, calendarMonth + 3, 0);\n *       \n *       return {\n *         start: adapter.normalize(start),\n *         end: adapter.normalize(end)\n *       };\n *     }\n *   },\n *   {\n *     key: 'FISCAL_YEAR_TO_DATE',\n *     resolve: (clock, adapter) => {\n *       const now = clock.now();\n *       const month = adapter.getMonth(now);\n *       const year = adapter.getYear(now);\n *       \n *       // Fiscal year starts April 1\n *       const fiscalYearStart = month >= 3 \n *         ? new Date(year, 3, 1)\n *         : new Date(year - 1, 3, 1);\n *       \n *       return {\n *         start: adapter.normalize(fiscalYearStart),\n *         end: adapter.normalize(now)\n *       };\n *     }\n *   }\n * ];\n * \n * // In app.config.ts\n * export const appConfig: ApplicationConfig = {\n *   providers: [\n *     provideCustomPresets(FISCAL_PRESETS)\n *   ]\n * };\n * \n * // Use in components\n * store.applyPreset('THIS_FISCAL_QUARTER');\n * store.applyPreset('FISCAL_YEAR_TO_DATE');\n * ```\n * \n * @example\n * ```typescript\n * // Hotel presets\n * const HOTEL_PRESETS: RangePresetPlugin[] = [\n *   {\n *     key: 'CHECK_IN_WEEK',\n *     resolve: (clock, adapter) => {\n *       const now = clock.now();\n *       const dayOfWeek = adapter.getDayOfWeek(now);\n *       \n *       // Check-in week: Friday to Friday\n *       const daysToNextFriday = dayOfWeek <= 5 \n *         ? 5 - dayOfWeek \n *         : 7 - dayOfWeek + 5;\n *       \n *       const nextFriday = adapter.addDays(now, daysToNextFriday);\n *       const followingFriday = adapter.addDays(nextFriday, 7);\n *       \n *       return { start: nextFriday, end: followingFriday };\n *     }\n *   },\n *   {\n *     key: 'NEXT_30_NIGHTS',\n *     resolve: (clock, adapter) => {\n *       const now = clock.now();\n *       const tomorrow = adapter.addDays(now, 1);\n *       const end = adapter.addDays(tomorrow, 30);\n *       return { start: tomorrow, end };\n *     }\n *   }\n * ];\n * \n * providers: [provideCustomPresets(HOTEL_PRESETS)]\n * ```\n */\nexport function provideCustomPresets(presets: RangePresetPlugin[]): EnvironmentProviders {\n  return makeEnvironmentProviders([\n    {\n      provide: APP_INITIALIZER,\n      multi: true,\n      useFactory: (registry: PresetRegistry) => {\n        return () => {\n          presets.forEach(preset => {\n            registry.register(preset);\n          });\n\n          if (typeof console !== 'undefined' && console.debug) {\n            console.debug(\n              `[ng-dual-datepicker] Registered ${presets.length} custom presets:`,\n              presets.map(p => p.key).join(', ')\n            );\n          }\n        };\n      },\n      deps: [PresetRegistry]\n    }\n  ]);\n}\n\n/**\n * Provide preset package\n * \n * Convenience function for external preset packages.\n * \n * @param packageName - Name of the preset package (for logging)\n * @param presets - Array of presets from the package\n * @returns EnvironmentProviders\n * \n * @example\n * ```typescript\n * // @acme/fiscal-presets package\n * export function provideFiscalPresets(): EnvironmentProviders {\n *   return providePresetPackage('@acme/fiscal-presets', FISCAL_PRESETS);\n * }\n * \n * // In app\n * import { provideFiscalPresets } from '@acme/fiscal-presets';\n * \n * providers: [provideFiscalPresets()]\n * ```\n */\nexport function providePresetPackage(\n  packageName: string,\n  presets: RangePresetPlugin[]\n): EnvironmentProviders {\n  return makeEnvironmentProviders([\n    {\n      provide: APP_INITIALIZER,\n      multi: true,\n      useFactory: (registry: PresetRegistry) => {\n        return () => {\n          presets.forEach(preset => {\n            registry.register(preset);\n          });\n\n          if (typeof console !== 'undefined' && console.debug) {\n            console.debug(\n              `[${packageName}] Registered ${presets.length} presets:`,\n              presets.map(p => p.key).join(', ')\n            );\n          }\n        };\n      },\n      deps: [PresetRegistry]\n    }\n  ]);\n}\n"]}
|
|
@@ -0,0 +1,277 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Preset Registry Service
|
|
3
|
+
*
|
|
4
|
+
* Version: 3.6.0
|
|
5
|
+
*
|
|
6
|
+
* Central registry for all date range preset plugins (built-in and custom).
|
|
7
|
+
*
|
|
8
|
+
* ARCHITECTURE:
|
|
9
|
+
* - Singleton service (providedIn: 'root')
|
|
10
|
+
* - Manages Map<string, RangePresetPlugin>
|
|
11
|
+
* - Thread-safe registration
|
|
12
|
+
* - Supports plugin override (useful for testing)
|
|
13
|
+
*
|
|
14
|
+
* USAGE:
|
|
15
|
+
* ```typescript
|
|
16
|
+
* // In app
|
|
17
|
+
* const registry = inject(PresetRegistry);
|
|
18
|
+
*
|
|
19
|
+
* // Register custom preset
|
|
20
|
+
* registry.register({
|
|
21
|
+
* key: 'MY_CUSTOM_PRESET',
|
|
22
|
+
* resolve: (clock, adapter) => {
|
|
23
|
+
* const now = clock.now();
|
|
24
|
+
* return { start: now, end: now };
|
|
25
|
+
* }
|
|
26
|
+
* });
|
|
27
|
+
*
|
|
28
|
+
* // Check if preset exists
|
|
29
|
+
* if (registry.has('MY_CUSTOM_PRESET')) {
|
|
30
|
+
* store.applyPreset('MY_CUSTOM_PRESET');
|
|
31
|
+
* }
|
|
32
|
+
*
|
|
33
|
+
* // Get all registered presets
|
|
34
|
+
* const allPresets = registry.getAll();
|
|
35
|
+
* console.log('Available presets:', allPresets.map(p => p.key));
|
|
36
|
+
* ```
|
|
37
|
+
*
|
|
38
|
+
* EXTERNAL PACKAGES:
|
|
39
|
+
* ```typescript
|
|
40
|
+
* // @acme/fiscal-presets
|
|
41
|
+
* export function provideFiscalPresets() {
|
|
42
|
+
* return {
|
|
43
|
+
* provide: APP_INITIALIZER,
|
|
44
|
+
* multi: true,
|
|
45
|
+
* useFactory: (registry: PresetRegistry) => {
|
|
46
|
+
* return () => {
|
|
47
|
+
* FISCAL_PRESETS.forEach(p => registry.register(p));
|
|
48
|
+
* };
|
|
49
|
+
* },
|
|
50
|
+
* deps: [PresetRegistry]
|
|
51
|
+
* };
|
|
52
|
+
* }
|
|
53
|
+
* ```
|
|
54
|
+
*/
|
|
55
|
+
import { Injectable } from '@angular/core';
|
|
56
|
+
import { isRangePresetPlugin } from './range-preset.plugin';
|
|
57
|
+
import * as i0 from "@angular/core";
|
|
58
|
+
/**
|
|
59
|
+
* Preset Registry Service
|
|
60
|
+
*
|
|
61
|
+
* Manages registration and retrieval of date range preset plugins.
|
|
62
|
+
*
|
|
63
|
+
* DESIGN:
|
|
64
|
+
* - Uses Map for O(1) lookups
|
|
65
|
+
* - Immutable getAll() returns copy
|
|
66
|
+
* - Supports plugin override (last registration wins)
|
|
67
|
+
* - Validates plugins before registration
|
|
68
|
+
*/
|
|
69
|
+
export class PresetRegistry {
|
|
70
|
+
/**
|
|
71
|
+
* Internal map of registered presets
|
|
72
|
+
* Key: preset key (e.g., 'TODAY', 'LAST_7_DAYS')
|
|
73
|
+
* Value: RangePresetPlugin instance
|
|
74
|
+
*/
|
|
75
|
+
presets = new Map();
|
|
76
|
+
/**
|
|
77
|
+
* Register a date range preset plugin
|
|
78
|
+
*
|
|
79
|
+
* If a preset with the same key already exists, it will be overridden.
|
|
80
|
+
* This is useful for:
|
|
81
|
+
* - Testing (override built-in presets with mocks)
|
|
82
|
+
* - Customization (replace default behavior)
|
|
83
|
+
* - Hot-reload scenarios
|
|
84
|
+
*
|
|
85
|
+
* @param plugin - The preset plugin to register
|
|
86
|
+
* @throws Error if plugin is invalid (missing key or resolve function)
|
|
87
|
+
*
|
|
88
|
+
* @example
|
|
89
|
+
* ```typescript
|
|
90
|
+
* registry.register({
|
|
91
|
+
* key: 'LAST_BUSINESS_WEEK',
|
|
92
|
+
* resolve: (clock, adapter) => {
|
|
93
|
+
* const now = clock.now();
|
|
94
|
+
* const dayOfWeek = adapter.getDayOfWeek(now);
|
|
95
|
+
*
|
|
96
|
+
* // Go back to last Friday
|
|
97
|
+
* const daysToLastFriday = dayOfWeek === 0 ? 2 : (dayOfWeek + 2) % 7;
|
|
98
|
+
* const lastFriday = adapter.addDays(now, -daysToLastFriday);
|
|
99
|
+
*
|
|
100
|
+
* // Business week: Monday to Friday
|
|
101
|
+
* const monday = adapter.addDays(lastFriday, -4);
|
|
102
|
+
*
|
|
103
|
+
* return { start: monday, end: lastFriday };
|
|
104
|
+
* }
|
|
105
|
+
* });
|
|
106
|
+
* ```
|
|
107
|
+
*/
|
|
108
|
+
register(plugin) {
|
|
109
|
+
// Validate plugin
|
|
110
|
+
if (!isRangePresetPlugin(plugin)) {
|
|
111
|
+
throw new Error(`Invalid preset plugin: must have 'key' (string) and 'resolve' (function). ` +
|
|
112
|
+
`Received: ${JSON.stringify(plugin)}`);
|
|
113
|
+
}
|
|
114
|
+
// Validate key is not empty
|
|
115
|
+
if (!plugin.key || plugin.key.trim().length === 0) {
|
|
116
|
+
throw new Error('Preset plugin key cannot be empty');
|
|
117
|
+
}
|
|
118
|
+
// Register (override if exists)
|
|
119
|
+
const key = plugin.key.toUpperCase(); // Normalize to uppercase
|
|
120
|
+
// Warn if overriding
|
|
121
|
+
if (this.presets.has(key)) {
|
|
122
|
+
console.warn(`[PresetRegistry] Overriding existing preset: "${key}". ` +
|
|
123
|
+
`This is normal if you're customizing built-in presets.`);
|
|
124
|
+
}
|
|
125
|
+
this.presets.set(key, plugin);
|
|
126
|
+
}
|
|
127
|
+
/**
|
|
128
|
+
* Register multiple preset plugins at once
|
|
129
|
+
*
|
|
130
|
+
* Convenience method for bulk registration.
|
|
131
|
+
* Useful when importing preset packages.
|
|
132
|
+
*
|
|
133
|
+
* @param plugins - Array of preset plugins to register
|
|
134
|
+
*
|
|
135
|
+
* @example
|
|
136
|
+
* ```typescript
|
|
137
|
+
* import { FISCAL_PRESETS } from '@acme/fiscal-presets';
|
|
138
|
+
*
|
|
139
|
+
* registry.registerAll(FISCAL_PRESETS);
|
|
140
|
+
* ```
|
|
141
|
+
*/
|
|
142
|
+
registerAll(plugins) {
|
|
143
|
+
plugins.forEach(plugin => this.register(plugin));
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Get a preset plugin by key
|
|
147
|
+
*
|
|
148
|
+
* @param key - Preset key (case-insensitive, e.g., 'today' or 'TODAY')
|
|
149
|
+
* @returns The preset plugin or undefined if not found
|
|
150
|
+
*
|
|
151
|
+
* @example
|
|
152
|
+
* ```typescript
|
|
153
|
+
* const preset = registry.get('LAST_7_DAYS');
|
|
154
|
+
* if (preset) {
|
|
155
|
+
* const range = preset.resolve(clock, adapter);
|
|
156
|
+
* console.log('Range:', range);
|
|
157
|
+
* }
|
|
158
|
+
* ```
|
|
159
|
+
*/
|
|
160
|
+
get(key) {
|
|
161
|
+
return this.presets.get(key.toUpperCase());
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Check if a preset exists in the registry
|
|
165
|
+
*
|
|
166
|
+
* @param key - Preset key (case-insensitive)
|
|
167
|
+
* @returns true if preset exists, false otherwise
|
|
168
|
+
*
|
|
169
|
+
* @example
|
|
170
|
+
* ```typescript
|
|
171
|
+
* if (registry.has('THIS_FISCAL_QUARTER')) {
|
|
172
|
+
* store.applyPreset('THIS_FISCAL_QUARTER');
|
|
173
|
+
* } else {
|
|
174
|
+
* console.error('Fiscal quarter preset not registered');
|
|
175
|
+
* }
|
|
176
|
+
* ```
|
|
177
|
+
*/
|
|
178
|
+
has(key) {
|
|
179
|
+
return this.presets.has(key.toUpperCase());
|
|
180
|
+
}
|
|
181
|
+
/**
|
|
182
|
+
* Get all registered preset plugins
|
|
183
|
+
*
|
|
184
|
+
* Returns a NEW array (immutable) to prevent external modification.
|
|
185
|
+
*
|
|
186
|
+
* @returns Array of all registered preset plugins
|
|
187
|
+
*
|
|
188
|
+
* @example
|
|
189
|
+
* ```typescript
|
|
190
|
+
* const allPresets = registry.getAll();
|
|
191
|
+
* console.log('Available presets:');
|
|
192
|
+
* allPresets.forEach(preset => {
|
|
193
|
+
* console.log(`- ${preset.key}`);
|
|
194
|
+
* });
|
|
195
|
+
* ```
|
|
196
|
+
*/
|
|
197
|
+
getAll() {
|
|
198
|
+
return Array.from(this.presets.values());
|
|
199
|
+
}
|
|
200
|
+
/**
|
|
201
|
+
* Get all preset keys
|
|
202
|
+
*
|
|
203
|
+
* Convenience method to list available preset identifiers.
|
|
204
|
+
*
|
|
205
|
+
* @returns Array of preset keys (uppercase)
|
|
206
|
+
*
|
|
207
|
+
* @example
|
|
208
|
+
* ```typescript
|
|
209
|
+
* const keys = registry.getAllKeys();
|
|
210
|
+
* // ['TODAY', 'YESTERDAY', 'LAST_7_DAYS', 'THIS_MONTH', ...]
|
|
211
|
+
* ```
|
|
212
|
+
*/
|
|
213
|
+
getAllKeys() {
|
|
214
|
+
return Array.from(this.presets.keys());
|
|
215
|
+
}
|
|
216
|
+
/**
|
|
217
|
+
* Get count of registered presets
|
|
218
|
+
*
|
|
219
|
+
* @returns Number of registered presets
|
|
220
|
+
*
|
|
221
|
+
* @example
|
|
222
|
+
* ```typescript
|
|
223
|
+
* console.log(`${registry.count()} presets available`);
|
|
224
|
+
* ```
|
|
225
|
+
*/
|
|
226
|
+
count() {
|
|
227
|
+
return this.presets.size;
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* Unregister a preset plugin
|
|
231
|
+
*
|
|
232
|
+
* Useful for:
|
|
233
|
+
* - Testing cleanup
|
|
234
|
+
* - Dynamic preset management
|
|
235
|
+
* - Removing deprecated presets
|
|
236
|
+
*
|
|
237
|
+
* @param key - Preset key to remove (case-insensitive)
|
|
238
|
+
* @returns true if preset was removed, false if it didn't exist
|
|
239
|
+
*
|
|
240
|
+
* @example
|
|
241
|
+
* ```typescript
|
|
242
|
+
* registry.unregister('MY_TEMPORARY_PRESET');
|
|
243
|
+
* ```
|
|
244
|
+
*/
|
|
245
|
+
unregister(key) {
|
|
246
|
+
return this.presets.delete(key.toUpperCase());
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* Clear all registered presets
|
|
250
|
+
*
|
|
251
|
+
* ⚠️ USE WITH CAUTION: This removes ALL presets including built-ins.
|
|
252
|
+
*
|
|
253
|
+
* Useful for:
|
|
254
|
+
* - Test teardown
|
|
255
|
+
* - Complete re-initialization scenarios
|
|
256
|
+
*
|
|
257
|
+
* @example
|
|
258
|
+
* ```typescript
|
|
259
|
+
* // In test cleanup
|
|
260
|
+
* afterEach(() => {
|
|
261
|
+
* registry.clear();
|
|
262
|
+
* });
|
|
263
|
+
* ```
|
|
264
|
+
*/
|
|
265
|
+
clear() {
|
|
266
|
+
this.presets.clear();
|
|
267
|
+
}
|
|
268
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetRegistry, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
269
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetRegistry, providedIn: 'root' });
|
|
270
|
+
}
|
|
271
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetRegistry, decorators: [{
|
|
272
|
+
type: Injectable,
|
|
273
|
+
args: [{
|
|
274
|
+
providedIn: 'root'
|
|
275
|
+
}]
|
|
276
|
+
}] });
|
|
277
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"preset-registry.js","sourceRoot":"","sources":["../../../src/core/preset-registry.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqDG;AAEH,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EAAqB,mBAAmB,EAAE,MAAM,uBAAuB,CAAC;;AAE/E;;;;;;;;;;GAUG;AAIH,MAAM,OAAO,cAAc;IACzB;;;;OAIG;IACc,OAAO,GAAG,IAAI,GAAG,EAA6B,CAAC;IAEhE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA+BG;IACH,QAAQ,CAAC,MAAyB;QAChC,kBAAkB;QAClB,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,EAAE,CAAC;YACjC,MAAM,IAAI,KAAK,CACb,4EAA4E;gBAC5E,aAAa,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CACtC,CAAC;QACJ,CAAC;QAED,4BAA4B;QAC5B,IAAI,CAAC,MAAM,CAAC,GAAG,IAAI,MAAM,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAClD,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC;QACvD,CAAC;QAED,gCAAgC;QAChC,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC,yBAAyB;QAE/D,qBAAqB;QACrB,IAAI,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC;YAC1B,OAAO,CAAC,IAAI,CACV,iDAAiD,GAAG,KAAK;gBACzD,wDAAwD,CACzD,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;IAChC,CAAC;IAED;;;;;;;;;;;;;;OAcG;IACH,WAAW,CAAC,OAA4B;QACtC,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;IACnD,CAAC;IAED;;;;;;;;;;;;;;OAcG;IACH,GAAG,CAAC,GAAW;QACb,OAAO,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC;IAC7C,CAAC;IAED;;;;;;;;;;;;;;OAcG;IACH,GAAG,CAAC,GAAW;QACb,OAAO,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC;IAC7C,CAAC;IAED;;;;;;;;;;;;;;;OAeG;IACH,MAAM;QACJ,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;IAC3C,CAAC;IAED;;;;;;;;;;;;OAYG;IACH,UAAU;QACR,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;IACzC,CAAC;IAED;;;;;;;;;OASG;IACH,KAAK;QACH,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC;IAC3B,CAAC;IAED;;;;;;;;;;;;;;;OAeG;IACH,UAAU,CAAC,GAAW;QACpB,OAAO,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC;IAChD,CAAC;IAED;;;;;;;;;;;;;;;;OAgBG;IACH,KAAK;QACH,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;IACvB,CAAC;wGAvNU,cAAc;4GAAd,cAAc,cAFb,MAAM;;4FAEP,cAAc;kBAH1B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["/**\n * Preset Registry Service\n * \n * Version: 3.6.0\n * \n * Central registry for all date range preset plugins (built-in and custom).\n * \n * ARCHITECTURE:\n * - Singleton service (providedIn: 'root')\n * - Manages Map<string, RangePresetPlugin>\n * - Thread-safe registration\n * - Supports plugin override (useful for testing)\n * \n * USAGE:\n * ```typescript\n * // In app\n * const registry = inject(PresetRegistry);\n * \n * // Register custom preset\n * registry.register({\n *   key: 'MY_CUSTOM_PRESET',\n *   resolve: (clock, adapter) => {\n *     const now = clock.now();\n *     return { start: now, end: now };\n *   }\n * });\n * \n * // Check if preset exists\n * if (registry.has('MY_CUSTOM_PRESET')) {\n *   store.applyPreset('MY_CUSTOM_PRESET');\n * }\n * \n * // Get all registered presets\n * const allPresets = registry.getAll();\n * console.log('Available presets:', allPresets.map(p => p.key));\n * ```\n * \n * EXTERNAL PACKAGES:\n * ```typescript\n * // @acme/fiscal-presets\n * export function provideFiscalPresets() {\n *   return {\n *     provide: APP_INITIALIZER,\n *     multi: true,\n *     useFactory: (registry: PresetRegistry) => {\n *       return () => {\n *         FISCAL_PRESETS.forEach(p => registry.register(p));\n *       };\n *     },\n *     deps: [PresetRegistry]\n *   };\n * }\n * ```\n */\n\nimport { Injectable } from '@angular/core';\nimport { RangePresetPlugin, isRangePresetPlugin } from './range-preset.plugin';\n\n/**\n * Preset Registry Service\n * \n * Manages registration and retrieval of date range preset plugins.\n * \n * DESIGN:\n * - Uses Map for O(1) lookups\n * - Immutable getAll() returns copy\n * - Supports plugin override (last registration wins)\n * - Validates plugins before registration\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class PresetRegistry {\n  /**\n   * Internal map of registered presets\n   * Key: preset key (e.g., 'TODAY', 'LAST_7_DAYS')\n   * Value: RangePresetPlugin instance\n   */\n  private readonly presets = new Map<string, RangePresetPlugin>();\n\n  /**\n   * Register a date range preset plugin\n   * \n   * If a preset with the same key already exists, it will be overridden.\n   * This is useful for:\n   * - Testing (override built-in presets with mocks)\n   * - Customization (replace default behavior)\n   * - Hot-reload scenarios\n   * \n   * @param plugin - The preset plugin to register\n   * @throws Error if plugin is invalid (missing key or resolve function)\n   * \n   * @example\n   * ```typescript\n   * registry.register({\n   *   key: 'LAST_BUSINESS_WEEK',\n   *   resolve: (clock, adapter) => {\n   *     const now = clock.now();\n   *     const dayOfWeek = adapter.getDayOfWeek(now);\n   *     \n   *     // Go back to last Friday\n   *     const daysToLastFriday = dayOfWeek === 0 ? 2 : (dayOfWeek + 2) % 7;\n   *     const lastFriday = adapter.addDays(now, -daysToLastFriday);\n   *     \n   *     // Business week: Monday to Friday\n   *     const monday = adapter.addDays(lastFriday, -4);\n   *     \n   *     return { start: monday, end: lastFriday };\n   *   }\n   * });\n   * ```\n   */\n  register(plugin: RangePresetPlugin): void {\n    // Validate plugin\n    if (!isRangePresetPlugin(plugin)) {\n      throw new Error(\n        `Invalid preset plugin: must have 'key' (string) and 'resolve' (function). ` +\n        `Received: ${JSON.stringify(plugin)}`\n      );\n    }\n\n    // Validate key is not empty\n    if (!plugin.key || plugin.key.trim().length === 0) {\n      throw new Error('Preset plugin key cannot be empty');\n    }\n\n    // Register (override if exists)\n    const key = plugin.key.toUpperCase(); // Normalize to uppercase\n    \n    // Warn if overriding\n    if (this.presets.has(key)) {\n      console.warn(\n        `[PresetRegistry] Overriding existing preset: \"${key}\". ` +\n        `This is normal if you're customizing built-in presets.`\n      );\n    }\n\n    this.presets.set(key, plugin);\n  }\n\n  /**\n   * Register multiple preset plugins at once\n   * \n   * Convenience method for bulk registration.\n   * Useful when importing preset packages.\n   * \n   * @param plugins - Array of preset plugins to register\n   * \n   * @example\n   * ```typescript\n   * import { FISCAL_PRESETS } from '@acme/fiscal-presets';\n   * \n   * registry.registerAll(FISCAL_PRESETS);\n   * ```\n   */\n  registerAll(plugins: RangePresetPlugin[]): void {\n    plugins.forEach(plugin => this.register(plugin));\n  }\n\n  /**\n   * Get a preset plugin by key\n   * \n   * @param key - Preset key (case-insensitive, e.g., 'today' or 'TODAY')\n   * @returns The preset plugin or undefined if not found\n   * \n   * @example\n   * ```typescript\n   * const preset = registry.get('LAST_7_DAYS');\n   * if (preset) {\n   *   const range = preset.resolve(clock, adapter);\n   *   console.log('Range:', range);\n   * }\n   * ```\n   */\n  get(key: string): RangePresetPlugin | undefined {\n    return this.presets.get(key.toUpperCase());\n  }\n\n  /**\n   * Check if a preset exists in the registry\n   * \n   * @param key - Preset key (case-insensitive)\n   * @returns true if preset exists, false otherwise\n   * \n   * @example\n   * ```typescript\n   * if (registry.has('THIS_FISCAL_QUARTER')) {\n   *   store.applyPreset('THIS_FISCAL_QUARTER');\n   * } else {\n   *   console.error('Fiscal quarter preset not registered');\n   * }\n   * ```\n   */\n  has(key: string): boolean {\n    return this.presets.has(key.toUpperCase());\n  }\n\n  /**\n   * Get all registered preset plugins\n   * \n   * Returns a NEW array (immutable) to prevent external modification.\n   * \n   * @returns Array of all registered preset plugins\n   * \n   * @example\n   * ```typescript\n   * const allPresets = registry.getAll();\n   * console.log('Available presets:');\n   * allPresets.forEach(preset => {\n   *   console.log(`- ${preset.key}`);\n   * });\n   * ```\n   */\n  getAll(): RangePresetPlugin[] {\n    return Array.from(this.presets.values());\n  }\n\n  /**\n   * Get all preset keys\n   * \n   * Convenience method to list available preset identifiers.\n   * \n   * @returns Array of preset keys (uppercase)\n   * \n   * @example\n   * ```typescript\n   * const keys = registry.getAllKeys();\n   * // ['TODAY', 'YESTERDAY', 'LAST_7_DAYS', 'THIS_MONTH', ...]\n   * ```\n   */\n  getAllKeys(): string[] {\n    return Array.from(this.presets.keys());\n  }\n\n  /**\n   * Get count of registered presets\n   * \n   * @returns Number of registered presets\n   * \n   * @example\n   * ```typescript\n   * console.log(`${registry.count()} presets available`);\n   * ```\n   */\n  count(): number {\n    return this.presets.size;\n  }\n\n  /**\n   * Unregister a preset plugin\n   * \n   * Useful for:\n   * - Testing cleanup\n   * - Dynamic preset management\n   * - Removing deprecated presets\n   * \n   * @param key - Preset key to remove (case-insensitive)\n   * @returns true if preset was removed, false if it didn't exist\n   * \n   * @example\n   * ```typescript\n   * registry.unregister('MY_TEMPORARY_PRESET');\n   * ```\n   */\n  unregister(key: string): boolean {\n    return this.presets.delete(key.toUpperCase());\n  }\n\n  /**\n   * Clear all registered presets\n   * \n   * ⚠️ USE WITH CAUTION: This removes ALL presets including built-ins.\n   * \n   * Useful for:\n   * - Test teardown\n   * - Complete re-initialization scenarios\n   * \n   * @example\n   * ```typescript\n   * // In test cleanup\n   * afterEach(() => {\n   *   registry.clear();\n   * });\n   * ```\n   */\n  clear(): void {\n    this.presets.clear();\n  }\n}\n"]}
|