@oneluiz/dual-datepicker 3.9.3 → 4.0.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.
Files changed (65) hide show
  1. package/README.md +26 -2
  2. package/fesm2022/oneluiz-dual-datepicker.mjs +696 -1040
  3. package/fesm2022/oneluiz-dual-datepicker.mjs.map +1 -1
  4. package/package.json +16 -8
  5. package/core/built-in-presets.d.ts +0 -98
  6. package/core/calendar-grid/cache.config.d.ts +0 -34
  7. package/core/calendar-grid/calendar-grid.cache.d.ts +0 -39
  8. package/core/calendar-grid/calendar-grid.factory.d.ts +0 -26
  9. package/core/calendar-grid/calendar-grid.types.d.ts +0 -57
  10. package/core/calendar-grid/index.d.ts +0 -56
  11. package/core/calendar-grid/range-highlighter.cache.d.ts +0 -106
  12. package/core/calendar-grid/range-highlighter.d.ts +0 -85
  13. package/core/calendar-grid/range-highlighter.types.d.ts +0 -182
  14. package/core/calendar-grid/virtual-weeks.logic.d.ts +0 -116
  15. package/core/calendar-grid/virtual-weeks.types.d.ts +0 -71
  16. package/core/date-adapter.d.ts +0 -298
  17. package/core/date-clock.d.ts +0 -82
  18. package/core/dual-date-range.store.d.ts +0 -113
  19. package/core/index.d.ts +0 -33
  20. package/core/native-date-adapter.d.ts +0 -152
  21. package/core/preset-providers.d.ts +0 -176
  22. package/core/preset-registry.d.ts +0 -181
  23. package/core/preset.engine.d.ts +0 -124
  24. package/core/range-preset.plugin.d.ts +0 -188
  25. package/core/range.validator.d.ts +0 -37
  26. package/core/system-clock.d.ts +0 -13
  27. package/date-adapter.d.ts +0 -116
  28. package/dual-datepicker.component.d.ts +0 -243
  29. package/esm2022/core/built-in-presets.mjs +0 -289
  30. package/esm2022/core/calendar-grid/cache.config.mjs +0 -35
  31. package/esm2022/core/calendar-grid/calendar-grid.cache.mjs +0 -98
  32. package/esm2022/core/calendar-grid/calendar-grid.factory.mjs +0 -97
  33. package/esm2022/core/calendar-grid/calendar-grid.types.mjs +0 -8
  34. package/esm2022/core/calendar-grid/index.mjs +0 -57
  35. package/esm2022/core/calendar-grid/range-highlighter.cache.mjs +0 -200
  36. package/esm2022/core/calendar-grid/range-highlighter.mjs +0 -185
  37. package/esm2022/core/calendar-grid/range-highlighter.types.mjs +0 -11
  38. package/esm2022/core/calendar-grid/virtual-weeks.logic.mjs +0 -149
  39. package/esm2022/core/calendar-grid/virtual-weeks.types.mjs +0 -11
  40. package/esm2022/core/date-adapter.mjs +0 -77
  41. package/esm2022/core/date-clock.mjs +0 -65
  42. package/esm2022/core/dual-date-range.store.mjs +0 -329
  43. package/esm2022/core/index.mjs +0 -37
  44. package/esm2022/core/native-date-adapter.mjs +0 -286
  45. package/esm2022/core/preset-providers.mjs +0 -243
  46. package/esm2022/core/preset-registry.mjs +0 -277
  47. package/esm2022/core/preset.engine.mjs +0 -179
  48. package/esm2022/core/range-preset.plugin.mjs +0 -70
  49. package/esm2022/core/range.validator.mjs +0 -105
  50. package/esm2022/core/system-clock.mjs +0 -34
  51. package/esm2022/date-adapter.mjs +0 -12
  52. package/esm2022/dual-datepicker.component.mjs +0 -1312
  53. package/esm2022/native-date-adapter.mjs +0 -117
  54. package/esm2022/oneluiz-dual-datepicker.mjs +0 -5
  55. package/esm2022/preset-utils.mjs +0 -276
  56. package/esm2022/public-api.mjs +0 -13
  57. package/index.d.ts +0 -5
  58. package/native-date-adapter.d.ts +0 -26
  59. package/preset-utils.d.ts +0 -91
  60. package/public-api.d.ts +0 -9
  61. package/src/themes/bootstrap.scss +0 -202
  62. package/src/themes/bulma.scss +0 -209
  63. package/src/themes/custom.scss +0 -236
  64. package/src/themes/foundation.scss +0 -201
  65. package/src/themes/tailwind.scss +0 -109
@@ -1,277 +0,0 @@
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"]}
@@ -1,179 +0,0 @@
1
- /**
2
- * Headless Preset Engine
3
- * Pure functions that resolve date ranges WITHOUT render dependency
4
- * Perfect for SSR, global state, dashboard filters
5
- *
6
- * v3.5.0: SSR-Safe via Clock Injection
7
- * All date calculations use DateClock instead of new Date()
8
- * This ensures server and client resolve identical presets
9
- *
10
- * v3.5.1: Timezone-Safe via DateAdapter
11
- * All date operations use DateAdapter for consistent behavior
12
- * Fixes timezone bugs common in ERP/BI/POS systems
13
- *
14
- * v3.6.0: Plugin-Driven Architecture
15
- * Preset Engine now uses PresetRegistry for plugin-based extensibility
16
- * Follows Open/Closed Principle - extend without modifying core
17
- * Supports external preset packages for industry-specific needs
18
- */
19
- import { Injectable, inject } from '@angular/core';
20
- import { DATE_CLOCK } from './date-clock';
21
- import { SystemClock } from './system-clock';
22
- import { DATE_ADAPTER } from './date-adapter';
23
- import { NativeDateAdapter } from './native-date-adapter';
24
- import { PresetRegistry } from './preset-registry';
25
- import * as i0 from "@angular/core";
26
- /**
27
- * Preset Engine - Plugin-Driven Architecture
28
- *
29
- * ARCHITECTURE (v3.6.0):
30
- * - NO longer contains presets internally
31
- * - Uses PresetRegistry for plugin management
32
- * - Injects DateClock for SSR-safe time
33
- * - Injects DateAdapter for timezone-safe operations
34
- * - Follows Open/Closed Principle
35
- *
36
- * BACKWARD COMPATIBILITY:
37
- * - Old API unchanged: resolve(), register(), getPresetKeys()
38
- * - Built-in presets auto-registered via provider
39
- * - Existing code continues to work
40
- *
41
- * EXTENSIBILITY:
42
- * ```typescript
43
- * // Register custom preset via registry
44
- * const registry = inject(PresetRegistry);
45
- * registry.register({
46
- * key: 'MY_PRESET',
47
- * resolve: (clock, adapter) => {
48
- * const now = clock.now();
49
- * return { start: now, end: now };
50
- * }
51
- * });
52
- *
53
- * // Use via engine
54
- * const engine = inject(PresetEngine);
55
- * const range = engine.resolve('MY_PRESET');
56
- * ```
57
- */
58
- export class PresetEngine {
59
- clock;
60
- adapter;
61
- registry;
62
- constructor() {
63
- // Inject dependencies with fallbacks
64
- try {
65
- this.clock = inject(DATE_CLOCK, { optional: true }) ?? new SystemClock();
66
- this.adapter = inject(DATE_ADAPTER, { optional: true }) ?? new NativeDateAdapter();
67
- this.registry = inject(PresetRegistry);
68
- }
69
- catch {
70
- // Fallback if inject() fails outside injection context
71
- this.clock = new SystemClock();
72
- this.adapter = new NativeDateAdapter();
73
- this.registry = new PresetRegistry();
74
- }
75
- }
76
- /**
77
- * Register a custom preset
78
- *
79
- * @deprecated Use PresetRegistry.register() directly for new code
80
- * Kept for backward compatibility
81
- *
82
- * @param key - Preset key (e.g., 'MY_CUSTOM_PRESET')
83
- * @param preset - Legacy RangePreset object
84
- */
85
- register(key, preset) {
86
- // Convert legacy RangePreset to RangePresetPlugin
87
- this.registry.register({
88
- key: key,
89
- resolve: (clock, adapter) => {
90
- const now = clock.now();
91
- return preset.resolve(now);
92
- }
93
- });
94
- }
95
- /**
96
- * Resolve a preset to date range
97
- *
98
- * Plugin Architecture:
99
- * 1. Looks up plugin in PresetRegistry
100
- * 2. Calls plugin.resolve(clock, adapter)
101
- * 3. Returns ISO date range
102
- *
103
- * SSR Note: Uses injected DateClock for deterministic resolution
104
- * Timezone Note: Uses injected DateAdapter for consistent operations
105
- *
106
- * @param key - Preset key (e.g., 'TODAY', 'LAST_7_DAYS')
107
- * @param now - Optional override for current date (defaults to clock.now())
108
- * @returns ISO date range or null if preset not found
109
- */
110
- resolve(key, now) {
111
- const plugin = this.registry.get(key);
112
- if (!plugin) {
113
- console.warn(`[PresetEngine] Preset "${key}" not found in registry`);
114
- return null;
115
- }
116
- // Create temporary clock if now is provided
117
- const effectiveClock = now
118
- ? { now: () => now }
119
- : this.clock;
120
- // Resolve via plugin
121
- const { start, end } = plugin.resolve(effectiveClock, this.adapter);
122
- // Convert to ISO format
123
- return {
124
- start: this.adapter.toISODate(start),
125
- end: this.adapter.toISODate(end)
126
- };
127
- }
128
- /**
129
- * Get all available preset keys
130
- *
131
- * Delegates to PresetRegistry
132
- *
133
- * @returns Array of registered preset keys
134
- */
135
- getPresetKeys() {
136
- return this.registry.getAllKeys();
137
- }
138
- /**
139
- * Check if a preset exists
140
- *
141
- * @param key - Preset key to check
142
- * @returns true if preset is registered
143
- */
144
- hasPreset(key) {
145
- return this.registry.has(key);
146
- }
147
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetEngine, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
148
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetEngine, providedIn: 'root' });
149
- }
150
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetEngine, decorators: [{
151
- type: Injectable,
152
- args: [{
153
- providedIn: 'root'
154
- }]
155
- }], ctorParameters: () => [] });
156
- /**
157
- * Create a custom preset from a function
158
- *
159
- * @deprecated Use RangePresetPlugin interface instead
160
- * Kept for backward compatibility
161
- */
162
- export function createPreset(resolver) {
163
- return { resolve: resolver };
164
- }
165
- /**
166
- * @deprecated Use dependency injection instead:
167
- * ```typescript
168
- * private engine = inject(PresetEngine);
169
- * ```
170
- *
171
- * Singleton preset engine instance for backward compatibility
172
- *
173
- * WARNING: This singleton uses SystemClock directly and is NOT SSR-safe.
174
- * For SSR applications, inject PresetEngine and override DATE_CLOCK token.
175
- *
176
- * This export will be removed in v4.0.0
177
- */
178
- export const presetEngine = new PresetEngine();
179
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"preset.engine.js","sourceRoot":"","sources":["../../../src/core/preset.engine.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Headless Preset Engine\n * Pure functions that resolve date ranges WITHOUT render dependency\n * Perfect for SSR, global state, dashboard filters\n * \n * v3.5.0: SSR-Safe via Clock Injection\n * All date calculations use DateClock instead of new Date()\n * This ensures server and client resolve identical presets\n * \n * v3.5.1: Timezone-Safe via DateAdapter\n * All date operations use DateAdapter for consistent behavior\n * Fixes timezone bugs common in ERP/BI/POS systems\n * \n * v3.6.0: Plugin-Driven Architecture\n * Preset Engine now uses PresetRegistry for plugin-based extensibility\n * Follows Open/Closed Principle - extend without modifying core\n * Supports external preset packages for industry-specific needs\n */\n\nimport { Injectable, inject } from '@angular/core';\nimport { DateClock, DATE_CLOCK } from './date-clock';\nimport { SystemClock } from './system-clock';\nimport { DateAdapter, DATE_ADAPTER } from './date-adapter';\nimport { NativeDateAdapter } from './native-date-adapter';\nimport { PresetRegistry } from './preset-registry';\n\n/**\n * @deprecated Use RangePresetPlugin from './range-preset.plugin' instead\n * Kept for backward compatibility\n */\nexport interface RangePreset {\n  /**\n   * Resolve preset to actual date range\n   * @param now - Current date for deterministic calculation\n   */\n  resolve(now: Date): { start: Date; end: Date };\n}\n\nexport interface PresetRange {\n  start: string; // ISO format\n  end: string; // ISO format\n}\n\n/**\n * Preset Engine - Plugin-Driven Architecture\n * \n * ARCHITECTURE (v3.6.0):\n * - NO longer contains presets internally\n * - Uses PresetRegistry for plugin management\n * - Injects DateClock for SSR-safe time\n * - Injects DateAdapter for timezone-safe operations\n * - Follows Open/Closed Principle\n * \n * BACKWARD COMPATIBILITY:\n * - Old API unchanged: resolve(), register(), getPresetKeys()\n * - Built-in presets auto-registered via provider\n * - Existing code continues to work\n * \n * EXTENSIBILITY:\n * ```typescript\n * // Register custom preset via registry\n * const registry = inject(PresetRegistry);\n * registry.register({\n *   key: 'MY_PRESET',\n *   resolve: (clock, adapter) => {\n *     const now = clock.now();\n *     return { start: now, end: now };\n *   }\n * });\n * \n * // Use via engine\n * const engine = inject(PresetEngine);\n * const range = engine.resolve('MY_PRESET');\n * ```\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class PresetEngine {\n  private clock: DateClock;\n  private adapter: DateAdapter;\n  private registry: PresetRegistry;\n\n  constructor() {\n    // Inject dependencies with fallbacks\n    try {\n      this.clock = inject(DATE_CLOCK, { optional: true }) ?? new SystemClock();\n      this.adapter = inject(DATE_ADAPTER, { optional: true }) ?? new NativeDateAdapter();\n      this.registry = inject(PresetRegistry);\n    } catch {\n      // Fallback if inject() fails outside injection context\n      this.clock = new SystemClock();\n      this.adapter = new NativeDateAdapter();\n      this.registry = new PresetRegistry();\n    }\n  }\n\n  /**\n   * Register a custom preset\n   * \n   * @deprecated Use PresetRegistry.register() directly for new code\n   * Kept for backward compatibility\n   * \n   * @param key - Preset key (e.g., 'MY_CUSTOM_PRESET')\n   * @param preset - Legacy RangePreset object\n   */\n  register(key: string, preset: RangePreset): void {\n    // Convert legacy RangePreset to RangePresetPlugin\n    this.registry.register({\n      key: key,\n      resolve: (clock, adapter) => {\n        const now = clock.now();\n        return preset.resolve(now);\n      }\n    });\n  }\n\n  /**\n   * Resolve a preset to date range\n   * \n   * Plugin Architecture:\n   * 1. Looks up plugin in PresetRegistry\n   * 2. Calls plugin.resolve(clock, adapter)\n   * 3. Returns ISO date range\n   * \n   * SSR Note: Uses injected DateClock for deterministic resolution\n   * Timezone Note: Uses injected DateAdapter for consistent operations\n   * \n   * @param key - Preset key (e.g., 'TODAY', 'LAST_7_DAYS')\n   * @param now - Optional override for current date (defaults to clock.now())\n   * @returns ISO date range or null if preset not found\n   */\n  resolve(key: string, now?: Date): PresetRange | null {\n    const plugin = this.registry.get(key);\n    \n    if (!plugin) {\n      console.warn(`[PresetEngine] Preset \"${key}\" not found in registry`);\n      return null;\n    }\n\n    // Create temporary clock if now is provided\n    const effectiveClock = now \n      ? { now: () => now } \n      : this.clock;\n\n    // Resolve via plugin\n    const { start, end } = plugin.resolve(effectiveClock, this.adapter);\n\n    // Convert to ISO format\n    return {\n      start: this.adapter.toISODate(start),\n      end: this.adapter.toISODate(end)\n    };\n  }\n\n  /**\n   * Get all available preset keys\n   * \n   * Delegates to PresetRegistry\n   * \n   * @returns Array of registered preset keys\n   */\n  getPresetKeys(): string[] {\n    return this.registry.getAllKeys();\n  }\n\n  /**\n   * Check if a preset exists\n   * \n   * @param key - Preset key to check\n   * @returns true if preset is registered\n   */\n  hasPreset(key: string): boolean {\n    return this.registry.has(key);\n  }\n}\n\n/**\n * Create a custom preset from a function\n * \n * @deprecated Use RangePresetPlugin interface instead\n * Kept for backward compatibility\n */\nexport function createPreset(\n  resolver: (now: Date) => { start: Date; end: Date }\n): RangePreset {\n  return { resolve: resolver };\n}\n\n/**\n * @deprecated Use dependency injection instead:\n * ```typescript\n * private engine = inject(PresetEngine);\n * ```\n * \n * Singleton preset engine instance for backward compatibility\n * \n * WARNING: This singleton uses SystemClock directly and is NOT SSR-safe.\n * For SSR applications, inject PresetEngine and override DATE_CLOCK token.\n * \n * This export will be removed in v4.0.0\n */\nexport const presetEngine = new PresetEngine();\n"]}
@@ -1,70 +0,0 @@
1
- /**
2
- * Range Preset Plugin System
3
- *
4
- * Version: 3.6.0
5
- *
6
- * Plugin-based architecture for date range presets following Open/Closed Principle.
7
- *
8
- * WHY THIS EXISTS:
9
- * - Enterprise apps need industry-specific presets (fiscal, hotel, logistics)
10
- * - Presets should be distributable as external packages
11
- * - Core should NOT know about all possible presets
12
- * - Users should extend presets without modifying library code
13
- *
14
- * ARCHITECTURE:
15
- * ```
16
- * RangePresetPlugin (interface) - Contract for all presets
17
- * ↓
18
- * PresetRegistry (service) - Manages plugin registration
19
- * ↓
20
- * PresetEngine (refactored) - Resolves presets via registry
21
- * ↓
22
- * DualDateRangeStore - No changes, backward compatible
23
- * ```
24
- *
25
- * USAGE:
26
- * ```typescript
27
- * // Built-in presets work automatically
28
- * store.applyPreset('LAST_7_DAYS'); // ✅ Works
29
- *
30
- * // Register custom preset
31
- * const registry = inject(PresetRegistry);
32
- * registry.register({
33
- * key: 'THIS_FISCAL_QUARTER',
34
- * resolve: (clock, adapter) => {
35
- * const now = clock.now();
36
- * const fiscalStart = adapter.startOfMonth(now);
37
- * const fiscalEnd = adapter.endOfMonth(now);
38
- * return { start: fiscalStart, end: fiscalEnd };
39
- * }
40
- * });
41
- *
42
- * // Use custom preset
43
- * store.applyPreset('THIS_FISCAL_QUARTER'); // ✅ Works
44
- * ```
45
- *
46
- * EXTERNAL PACKAGES:
47
- * ```typescript
48
- * // @acme/fiscal-presets package
49
- * export const FISCAL_PRESETS: RangePresetPlugin[] = [
50
- * { key: 'FISCAL_Q1', resolve: ... },
51
- * { key: 'FISCAL_Q2', resolve: ... }
52
- * ];
53
- *
54
- * // In app
55
- * FISCAL_PRESETS.forEach(p => registry.register(p));
56
- * ```
57
- */
58
- /**
59
- * Type guard to check if object is a valid RangePresetPlugin
60
- *
61
- * @param obj - Object to check
62
- * @returns true if object implements RangePresetPlugin interface
63
- */
64
- export function isRangePresetPlugin(obj) {
65
- return (obj &&
66
- typeof obj === 'object' &&
67
- typeof obj.key === 'string' &&
68
- typeof obj.resolve === 'function');
69
- }
70
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"range-preset.plugin.js","sourceRoot":"","sources":["../../../src/core/range-preset.plugin.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwDG;AAmIH;;;;;GAKG;AACH,MAAM,UAAU,mBAAmB,CAAC,GAAQ;IAC1C,OAAO,CACL,GAAG;QACH,OAAO,GAAG,KAAK,QAAQ;QACvB,OAAO,GAAG,CAAC,GAAG,KAAK,QAAQ;QAC3B,OAAO,GAAG,CAAC,OAAO,KAAK,UAAU,CAClC,CAAC;AACJ,CAAC","sourcesContent":["/**\n * Range Preset Plugin System\n * \n * Version: 3.6.0\n * \n * Plugin-based architecture for date range presets following Open/Closed Principle.\n * \n * WHY THIS EXISTS:\n * - Enterprise apps need industry-specific presets (fiscal, hotel, logistics)\n * - Presets should be distributable as external packages\n * - Core should NOT know about all possible presets\n * - Users should extend presets without modifying library code\n * \n * ARCHITECTURE:\n * ```\n * RangePresetPlugin (interface) - Contract for all presets\n *     ↓\n * PresetRegistry (service) - Manages plugin registration\n *     ↓\n * PresetEngine (refactored) - Resolves presets via registry\n *     ↓\n * DualDateRangeStore - No changes, backward compatible\n * ```\n * \n * USAGE:\n * ```typescript\n * // Built-in presets work automatically\n * store.applyPreset('LAST_7_DAYS'); // ✅ Works\n * \n * // Register custom preset\n * const registry = inject(PresetRegistry);\n * registry.register({\n *   key: 'THIS_FISCAL_QUARTER',\n *   resolve: (clock, adapter) => {\n *     const now = clock.now();\n *     const fiscalStart = adapter.startOfMonth(now);\n *     const fiscalEnd = adapter.endOfMonth(now);\n *     return { start: fiscalStart, end: fiscalEnd };\n *   }\n * });\n * \n * // Use custom preset\n * store.applyPreset('THIS_FISCAL_QUARTER'); // ✅ Works\n * ```\n * \n * EXTERNAL PACKAGES:\n * ```typescript\n * // @acme/fiscal-presets package\n * export const FISCAL_PRESETS: RangePresetPlugin[] = [\n *   { key: 'FISCAL_Q1', resolve: ... },\n *   { key: 'FISCAL_Q2', resolve: ... }\n * ];\n * \n * // In app\n * FISCAL_PRESETS.forEach(p => registry.register(p));\n * ```\n */\n\nimport { DateClock } from './date-clock';\nimport { DateAdapter } from './date-adapter';\n\n/**\n * Date range returned by preset plugins\n */\nexport interface DateRange {\n  /**\n   * Start date of the range (inclusive)\n   */\n  start: Date;\n\n  /**\n   * End date of the range (inclusive)\n   */\n  end: Date;\n}\n\n/**\n * Range Preset Plugin Interface\n * \n * All date range presets (built-in or external) implement this interface.\n * \n * DESIGN PRINCIPLES:\n * - **Deterministic**: Given the same clock.now(), always returns same range\n * - **Timezone-safe**: Uses DateAdapter for all date operations\n * - **SSR-compatible**: Uses DateClock injection, no global Date()\n * - **Testable**: Pure function, no side effects\n * \n * EXAMPLE - Built-in preset:\n * ```typescript\n * const todayPreset: RangePresetPlugin = {\n *   key: 'TODAY',\n *   resolve: (clock, adapter) => {\n *     const now = clock.now();\n *     const normalized = adapter.normalize(now);\n *     return { start: normalized, end: normalized };\n *   }\n * };\n * ```\n * \n * EXAMPLE - Custom fiscal preset:\n * ```typescript\n * const fiscalQuarterPreset: RangePresetPlugin = {\n *   key: 'THIS_FISCAL_QUARTER',\n *   resolve: (clock, adapter) => {\n *     const now = clock.now();\n *     const month = adapter.getMonth(now); // 0-11\n *     \n *     // Fiscal year starts in April (month 3)\n *     const fiscalMonth = (month + 9) % 12; // Offset to fiscal calendar\n *     const quarterStartMonth = Math.floor(fiscalMonth / 3) * 3;\n *     const adjustedMonth = (quarterStartMonth - 9 + 12) % 12;\n *     \n *     const yearOffset = month < 3 ? -1 : 0;\n *     const year = adapter.getYear(now) + yearOffset;\n *     \n *     const start = new Date(year, adjustedMonth, 1);\n *     const end = new Date(year, adjustedMonth + 3, 0);\n *     \n *     return {\n *       start: adapter.normalize(start),\n *       end: adapter.normalize(end)\n *     };\n *   }\n * };\n * ```\n * \n * EXAMPLE - Hotel industry preset:\n * ```typescript\n * const checkInWeekPreset: RangePresetPlugin = {\n *   key: 'CHECK_IN_WEEK',\n *   resolve: (clock, adapter) => {\n *     const now = clock.now();\n *     // Hotel check-ins are Friday to Friday\n *     const dayOfWeek = adapter.getDayOfWeek(now);\n *     const daysToFriday = dayOfWeek <= 5 ? 5 - dayOfWeek : 7 - dayOfWeek + 5;\n *     \n *     const nextFriday = adapter.addDays(now, daysToFriday);\n *     const followingFriday = adapter.addDays(nextFriday, 7);\n *     \n *     return { start: nextFriday, end: followingFriday };\n *   }\n * };\n * ```\n */\nexport interface RangePresetPlugin {\n  /**\n   * Unique identifier for the preset\n   * \n   * Convention: SCREAMING_SNAKE_CASE\n   * \n   * Examples:\n   * - Built-in: 'TODAY', 'LAST_7_DAYS', 'THIS_MONTH'\n   * - Fiscal: 'FISCAL_Q1', 'FISCAL_YEAR_TO_DATE'\n   * - Hotel: 'CHECK_IN_WEEK', 'NEXT_30_NIGHTS'\n   * - Logistics: 'SHIPPING_WEEK', 'DELIVERY_WINDOW'\n   */\n  key: string;\n\n  /**\n   * Resolve the date range for this preset\n   * \n   * MUST use:\n   * - `clock.now()` for current time (SSR-safe, deterministic)\n   * - `adapter.*` for all date operations (timezone-safe)\n   * \n   * MUST NOT use:\n   * - `new Date()` directly (breaks SSR determinism)\n   * - `date.toISOString()` (timezone bugs)\n   * - `date.setDate()` (mutates, use adapter.addDays() instead)\n   * \n   * @param clock - Injected DateClock for SSR-safe time access\n   * @param adapter - Injected DateAdapter for timezone-safe operations\n   * @returns Date range with start and end dates (both inclusive)\n   * \n   * @example\n   * ```typescript\n   * resolve: (clock, adapter) => {\n   *   const now = clock.now();\n   *   const start = adapter.addDays(now, -7);\n   *   const end = adapter.normalize(now);\n   *   return { start, end };\n   * }\n   * ```\n   */\n  resolve(clock: DateClock, adapter: DateAdapter): DateRange;\n}\n\n/**\n * Type guard to check if object is a valid RangePresetPlugin\n * \n * @param obj - Object to check\n * @returns true if object implements RangePresetPlugin interface\n */\nexport function isRangePresetPlugin(obj: any): obj is RangePresetPlugin {\n  return (\n    obj &&\n    typeof obj === 'object' &&\n    typeof obj.key === 'string' &&\n    typeof obj.resolve === 'function'\n  );\n}\n"]}