@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.
Files changed (35) hide show
  1. package/core/built-in-presets.d.ts +98 -0
  2. package/core/calendar-grid/calendar-grid.cache.d.ts +39 -0
  3. package/core/calendar-grid/calendar-grid.factory.d.ts +26 -0
  4. package/core/calendar-grid/calendar-grid.types.d.ts +57 -0
  5. package/core/calendar-grid/index.d.ts +50 -0
  6. package/core/calendar-grid/range-highlighter.cache.d.ts +105 -0
  7. package/core/calendar-grid/range-highlighter.d.ts +85 -0
  8. package/core/calendar-grid/range-highlighter.types.d.ts +182 -0
  9. package/core/calendar-grid/virtual-weeks.logic.d.ts +116 -0
  10. package/core/calendar-grid/virtual-weeks.types.d.ts +71 -0
  11. package/core/index.d.ts +22 -0
  12. package/core/preset-providers.d.ts +176 -0
  13. package/core/preset-registry.d.ts +181 -0
  14. package/core/preset.engine.d.ts +57 -21
  15. package/core/range-preset.plugin.d.ts +188 -0
  16. package/dual-datepicker.component.d.ts +78 -1
  17. package/esm2022/core/built-in-presets.mjs +289 -0
  18. package/esm2022/core/calendar-grid/calendar-grid.cache.mjs +92 -0
  19. package/esm2022/core/calendar-grid/calendar-grid.factory.mjs +97 -0
  20. package/esm2022/core/calendar-grid/calendar-grid.types.mjs +8 -0
  21. package/esm2022/core/calendar-grid/index.mjs +51 -0
  22. package/esm2022/core/calendar-grid/range-highlighter.cache.mjs +198 -0
  23. package/esm2022/core/calendar-grid/range-highlighter.mjs +185 -0
  24. package/esm2022/core/calendar-grid/range-highlighter.types.mjs +11 -0
  25. package/esm2022/core/calendar-grid/virtual-weeks.logic.mjs +149 -0
  26. package/esm2022/core/calendar-grid/virtual-weeks.types.mjs +11 -0
  27. package/esm2022/core/index.mjs +26 -1
  28. package/esm2022/core/preset-providers.mjs +243 -0
  29. package/esm2022/core/preset-registry.mjs +277 -0
  30. package/esm2022/core/preset.engine.mjs +86 -210
  31. package/esm2022/core/range-preset.plugin.mjs +70 -0
  32. package/esm2022/dual-datepicker.component.mjs +202 -39
  33. package/fesm2022/oneluiz-dual-datepicker.mjs +1985 -282
  34. package/fesm2022/oneluiz-dual-datepicker.mjs.map +1 -1
  35. 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"]}