@oneluiz/dual-datepicker 4.0.0 → 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.
- package/README.md +26 -2
- package/fesm2022/oneluiz-dual-datepicker.mjs +71 -69
- package/fesm2022/oneluiz-dual-datepicker.mjs.map +1 -1
- package/package.json +6 -8
- package/dual-datepicker.component.d.ts +0 -243
- package/esm2022/core/built-in-presets.mjs +0 -289
- package/esm2022/core/calendar-grid/cache.config.mjs +0 -35
- package/esm2022/core/calendar-grid/calendar-grid.cache.mjs +0 -98
- package/esm2022/core/calendar-grid/calendar-grid.factory.mjs +0 -97
- package/esm2022/core/calendar-grid/calendar-grid.types.mjs +0 -8
- package/esm2022/core/calendar-grid/range-highlighter.cache.mjs +0 -200
- package/esm2022/core/calendar-grid/range-highlighter.mjs +0 -185
- package/esm2022/core/calendar-grid/range-highlighter.types.mjs +0 -11
- package/esm2022/core/calendar-grid/virtual-weeks.logic.mjs +0 -149
- package/esm2022/core/calendar-grid/virtual-weeks.types.mjs +0 -11
- package/esm2022/core/date-adapter.mjs +0 -77
- package/esm2022/core/date-clock.mjs +0 -65
- package/esm2022/core/dual-date-range.store.mjs +0 -329
- package/esm2022/core/internal.mjs +0 -92
- package/esm2022/core/native-date-adapter.mjs +0 -286
- package/esm2022/core/preset-providers.mjs +0 -243
- package/esm2022/core/preset-registry.mjs +0 -277
- package/esm2022/core/preset.engine.mjs +0 -179
- package/esm2022/core/public.mjs +0 -94
- package/esm2022/core/range-preset.plugin.mjs +0 -70
- package/esm2022/core/range.validator.mjs +0 -105
- package/esm2022/core/system-clock.mjs +0 -34
- package/esm2022/core/testing/date.helpers.mjs +0 -49
- package/esm2022/core/testing/fixed-clock.mjs +0 -30
- package/esm2022/core/testing/index.mjs +0 -8
- package/esm2022/dual-datepicker.component.mjs +0 -1314
- package/esm2022/oneluiz-dual-datepicker.mjs +0 -5
- package/esm2022/public-api.mjs +0 -43
- package/index.d.ts +0 -5
- package/public-api.d.ts +0 -44
|
@@ -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"]}
|
package/esm2022/core/public.mjs
DELETED
|
@@ -1,94 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* PUBLIC API for @oneluiz/dual-datepicker
|
|
3
|
-
*
|
|
4
|
-
* This file defines the stable public API contract.
|
|
5
|
-
* Exports from this barrel are guaranteed to follow semantic versioning.
|
|
6
|
-
*
|
|
7
|
-
* @public
|
|
8
|
-
* @packageDocumentation
|
|
9
|
-
* @module @oneluiz/dual-datepicker/core
|
|
10
|
-
* @since v4.0.0
|
|
11
|
-
*/
|
|
12
|
-
// ============================================================================
|
|
13
|
-
// HEADLESS STORE
|
|
14
|
-
// ============================================================================
|
|
15
|
-
/**
|
|
16
|
-
* DualDateRangeStore - Signal-based headless date range store
|
|
17
|
-
*
|
|
18
|
-
* Core reactive store for managing date range state without UI dependencies.
|
|
19
|
-
* Perfect for SSR, global state, and custom UI implementations.
|
|
20
|
-
*
|
|
21
|
-
* @public
|
|
22
|
-
* @since v3.5.0
|
|
23
|
-
*/
|
|
24
|
-
export { DualDateRangeStore } from './dual-date-range.store';
|
|
25
|
-
export { DATE_ADAPTER } from './date-adapter';
|
|
26
|
-
/**
|
|
27
|
-
* NativeDateAdapter - Built-in implementation using native Date
|
|
28
|
-
*
|
|
29
|
-
* Default adapter that uses JavaScript's native Date object.
|
|
30
|
-
* Provides timezone-safe operations without external dependencies.
|
|
31
|
-
*
|
|
32
|
-
* @public
|
|
33
|
-
* @since v1.0.0
|
|
34
|
-
*/
|
|
35
|
-
export { NativeDateAdapter } from './native-date-adapter';
|
|
36
|
-
/**
|
|
37
|
-
* PresetRegistry - Plugin management and lifecycle
|
|
38
|
-
*
|
|
39
|
-
* Injectable service that manages preset plugins.
|
|
40
|
-
* Handles registration, resolution, and validation.
|
|
41
|
-
*
|
|
42
|
-
* @public
|
|
43
|
-
* @since v3.6.0
|
|
44
|
-
*/
|
|
45
|
-
export { PresetRegistry } from './preset-registry';
|
|
46
|
-
/**
|
|
47
|
-
* Built-in Presets - Standard date range presets as plugins
|
|
48
|
-
*
|
|
49
|
-
* Pre-configured plugins for common use cases:
|
|
50
|
-
* - TODAY, YESTERDAY
|
|
51
|
-
* - LAST_7_DAYS, LAST_30_DAYS
|
|
52
|
-
* - THIS_WEEK, LAST_WEEK
|
|
53
|
-
* - THIS_MONTH, LAST_MONTH
|
|
54
|
-
* - THIS_YEAR, LAST_YEAR
|
|
55
|
-
*
|
|
56
|
-
* @public
|
|
57
|
-
* @since v3.6.0
|
|
58
|
-
*/
|
|
59
|
-
export { BUILT_IN_PRESETS, TODAY_PRESET, YESTERDAY_PRESET, LAST_7_DAYS_PRESET, LAST_30_DAYS_PRESET, THIS_WEEK_PRESET, LAST_WEEK_PRESET, THIS_MONTH_PRESET, LAST_MONTH_PRESET, THIS_YEAR_PRESET, LAST_YEAR_PRESET } from './built-in-presets';
|
|
60
|
-
/**
|
|
61
|
-
* Provider Functions - Dependency injection helpers
|
|
62
|
-
*
|
|
63
|
-
* Functions for configuring the library via Angular DI:
|
|
64
|
-
* - provideCustomPresets() - Custom preset plugins
|
|
65
|
-
* - providePresetPackage() - Preset packages
|
|
66
|
-
*
|
|
67
|
-
* @public
|
|
68
|
-
* @since v3.6.0
|
|
69
|
-
*/
|
|
70
|
-
export { provideCustomPresets, providePresetPackage } from './preset-providers';
|
|
71
|
-
// ============================================================================
|
|
72
|
-
// VALIDATION & UTILITIES
|
|
73
|
-
// ============================================================================
|
|
74
|
-
/**
|
|
75
|
-
* Range Validation Utilities
|
|
76
|
-
*
|
|
77
|
-
* Pure functions for date range validation logic.
|
|
78
|
-
* Used internally but exposed for custom implementations.
|
|
79
|
-
*
|
|
80
|
-
* @public
|
|
81
|
-
* @since v3.5.0
|
|
82
|
-
*/
|
|
83
|
-
export { validateRangeOrder, applyBounds, isDateDisabled } from './range.validator';
|
|
84
|
-
/**
|
|
85
|
-
* Preset Engine - Preset resolution and management
|
|
86
|
-
*
|
|
87
|
-
* Core engine that handles preset plugin execution.
|
|
88
|
-
* Used internally but exposed for advanced use cases.
|
|
89
|
-
*
|
|
90
|
-
* @public
|
|
91
|
-
* @since v3.6.0
|
|
92
|
-
*/
|
|
93
|
-
export { PresetEngine } from './preset.engine';
|
|
94
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2NvcmUvcHVibGljLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7O0dBVUc7QUFFSCwrRUFBK0U7QUFDL0UsaUJBQWlCO0FBQ2pCLCtFQUErRTtBQUUvRTs7Ozs7Ozs7R0FRRztBQUNILE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBZ0I3RCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFFOUM7Ozs7Ozs7O0dBUUc7QUFDSCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQW1CMUQ7Ozs7Ozs7O0dBUUc7QUFDSCxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sbUJBQW1CLENBQUM7QUFFbkQ7Ozs7Ozs7Ozs7OztHQVlHO0FBQ0gsT0FBTyxFQUNMLGdCQUFnQixFQUNoQixZQUFZLEVBQ1osZ0JBQWdCLEVBQ2hCLGtCQUFrQixFQUNsQixtQkFBbUIsRUFDbkIsZ0JBQWdCLEVBQ2hCLGdCQUFnQixFQUNoQixpQkFBaUIsRUFDakIsaUJBQWlCLEVBQ2pCLGdCQUFnQixFQUNoQixnQkFBZ0IsRUFDakIsTUFBTSxvQkFBb0IsQ0FBQztBQUU1Qjs7Ozs7Ozs7O0dBU0c7QUFDSCxPQUFPLEVBQ0wsb0JBQW9CLEVBQ3BCLG9CQUFvQixFQUNyQixNQUFNLG9CQUFvQixDQUFDO0FBRTVCLCtFQUErRTtBQUMvRSx5QkFBeUI7QUFDekIsK0VBQStFO0FBRS9FOzs7Ozs7OztHQVFHO0FBQ0gsT0FBTyxFQUNMLGtCQUFrQixFQUNsQixXQUFXLEVBQ1gsY0FBYyxFQUVmLE1BQU0sbUJBQW1CLENBQUM7QUFFM0I7Ozs7Ozs7O0dBUUc7QUFDSCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFBVQkxJQyBBUEkgZm9yIEBvbmVsdWl6L2R1YWwtZGF0ZXBpY2tlclxuICogXG4gKiBUaGlzIGZpbGUgZGVmaW5lcyB0aGUgc3RhYmxlIHB1YmxpYyBBUEkgY29udHJhY3QuXG4gKiBFeHBvcnRzIGZyb20gdGhpcyBiYXJyZWwgYXJlIGd1YXJhbnRlZWQgdG8gZm9sbG93IHNlbWFudGljIHZlcnNpb25pbmcuXG4gKiBcbiAqIEBwdWJsaWNcbiAqIEBwYWNrYWdlRG9jdW1lbnRhdGlvblxuICogQG1vZHVsZSBAb25lbHVpei9kdWFsLWRhdGVwaWNrZXIvY29yZVxuICogQHNpbmNlIHY0LjAuMFxuICovXG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIEhFQURMRVNTIFNUT1JFXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbi8qKlxuICogRHVhbERhdGVSYW5nZVN0b3JlIC0gU2lnbmFsLWJhc2VkIGhlYWRsZXNzIGRhdGUgcmFuZ2Ugc3RvcmVcbiAqIFxuICogQ29yZSByZWFjdGl2ZSBzdG9yZSBmb3IgbWFuYWdpbmcgZGF0ZSByYW5nZSBzdGF0ZSB3aXRob3V0IFVJIGRlcGVuZGVuY2llcy5cbiAqIFBlcmZlY3QgZm9yIFNTUiwgZ2xvYmFsIHN0YXRlLCBhbmQgY3VzdG9tIFVJIGltcGxlbWVudGF0aW9ucy5cbiAqIFxuICogQHB1YmxpY1xuICogQHNpbmNlIHYzLjUuMFxuICovXG5leHBvcnQgeyBEdWFsRGF0ZVJhbmdlU3RvcmUgfSBmcm9tICcuL2R1YWwtZGF0ZS1yYW5nZS5zdG9yZSc7XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIERBVEUgQURBUFRFUiBTWVNURU1cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuLyoqXG4gKiBEYXRlQWRhcHRlciAtIEFic3RyYWN0IGRhdGUgbWFuaXB1bGF0aW9uIGludGVyZmFjZVxuICogXG4gKiBQcm92aWRlcyB0aW1lem9uZS1zYWZlIGRhdGUgb3BlcmF0aW9ucy5cbiAqIEltcGxlbWVudCB0aGlzIGludGVyZmFjZSBmb3IgY3VzdG9tIGRhdGUgbGlicmFyaWVzIChMdXhvbiwgZGF0ZS1mbnMsIGV0Yy4pXG4gKiBcbiAqIEBwdWJsaWNcbiAqIEBzaW5jZSB2MS4wLjBcbiAqL1xuZXhwb3J0IHR5cGUgeyBEYXRlQWRhcHRlciB9IGZyb20gJy4vZGF0ZS1hZGFwdGVyJztcbmV4cG9ydCB7IERBVEVfQURBUFRFUiB9IGZyb20gJy4vZGF0ZS1hZGFwdGVyJztcblxuLyoqXG4gKiBOYXRpdmVEYXRlQWRhcHRlciAtIEJ1aWx0LWluIGltcGxlbWVudGF0aW9uIHVzaW5nIG5hdGl2ZSBEYXRlXG4gKiBcbiAqIERlZmF1bHQgYWRhcHRlciB0aGF0IHVzZXMgSmF2YVNjcmlwdCdzIG5hdGl2ZSBEYXRlIG9iamVjdC5cbiAqIFByb3ZpZGVzIHRpbWV6b25lLXNhZmUgb3BlcmF0aW9ucyB3aXRob3V0IGV4dGVybmFsIGRlcGVuZGVuY2llcy5cbiAqIFxuICogQHB1YmxpY1xuICogQHNpbmNlIHYxLjAuMFxuICovXG5leHBvcnQgeyBOYXRpdmVEYXRlQWRhcHRlciB9IGZyb20gJy4vbmF0aXZlLWRhdGUtYWRhcHRlcic7XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIFBSRVNFVCBQTFVHSU4gU1lTVEVNICh2My42LjArKVxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4vKipcbiAqIFJhbmdlUHJlc2V0UGx1Z2luIC0gSW50ZXJmYWNlIGZvciBleHRlbnNpYmxlIGRhdGUgcmFuZ2UgcHJlc2V0c1xuICogXG4gKiBJbXBsZW1lbnQgdGhpcyBpbnRlcmZhY2UgdG8gY3JlYXRlIGN1c3RvbSBwcmVzZXQgcGx1Z2lucy5cbiAqIFN1cHBvcnRzIGR5bmFtaWMgcmFuZ2VzLCB2YWxpZGF0aW9uLCBhbmQgY3VzdG9tIGxvZ2ljLlxuICogXG4gKiBAcHVibGljXG4gKiBAc2luY2UgdjMuNi4wXG4gKi9cbmV4cG9ydCB0eXBlIHtcbiAgUmFuZ2VQcmVzZXRQbHVnaW5cbn0gZnJvbSAnLi9yYW5nZS1wcmVzZXQucGx1Z2luJztcblxuLyoqXG4gKiBQcmVzZXRSZWdpc3RyeSAtIFBsdWdpbiBtYW5hZ2VtZW50IGFuZCBsaWZlY3ljbGVcbiAqIFxuICogSW5qZWN0YWJsZSBzZXJ2aWNlIHRoYXQgbWFuYWdlcyBwcmVzZXQgcGx1Z2lucy5cbiAqIEhhbmRsZXMgcmVnaXN0cmF0aW9uLCByZXNvbHV0aW9uLCBhbmQgdmFsaWRhdGlvbi5cbiAqIFxuICogQHB1YmxpY1xuICogQHNpbmNlIHYzLjYuMFxuICovXG5leHBvcnQgeyBQcmVzZXRSZWdpc3RyeSB9IGZyb20gJy4vcHJlc2V0LXJlZ2lzdHJ5JztcblxuLyoqXG4gKiBCdWlsdC1pbiBQcmVzZXRzIC0gU3RhbmRhcmQgZGF0ZSByYW5nZSBwcmVzZXRzIGFzIHBsdWdpbnNcbiAqIFxuICogUHJlLWNvbmZpZ3VyZWQgcGx1Z2lucyBmb3IgY29tbW9uIHVzZSBjYXNlczpcbiAqIC0gVE9EQVksIFlFU1RFUkRBWVxuICogLSBMQVNUXzdfREFZUywgTEFTVF8zMF9EQVlTXG4gKiAtIFRISVNfV0VFSywgTEFTVF9XRUVLXG4gKiAtIFRISVNfTU9OVEgsIExBU1RfTU9OVEhcbiAqIC0gVEhJU19ZRUFSLCBMQVNUX1lFQVJcbiAqIFxuICogQHB1YmxpY1xuICogQHNpbmNlIHYzLjYuMFxuICovXG5leHBvcnQge1xuICBCVUlMVF9JTl9QUkVTRVRTLFxuICBUT0RBWV9QUkVTRVQsXG4gIFlFU1RFUkRBWV9QUkVTRVQsXG4gIExBU1RfN19EQVlTX1BSRVNFVCxcbiAgTEFTVF8zMF9EQVlTX1BSRVNFVCxcbiAgVEhJU19XRUVLX1BSRVNFVCxcbiAgTEFTVF9XRUVLX1BSRVNFVCxcbiAgVEhJU19NT05USF9QUkVTRVQsXG4gIExBU1RfTU9OVEhfUFJFU0VULFxuICBUSElTX1lFQVJfUFJFU0VULFxuICBMQVNUX1lFQVJfUFJFU0VUXG59IGZyb20gJy4vYnVpbHQtaW4tcHJlc2V0cyc7XG5cbi8qKlxuICogUHJvdmlkZXIgRnVuY3Rpb25zIC0gRGVwZW5kZW5jeSBpbmplY3Rpb24gaGVscGVyc1xuICogXG4gKiBGdW5jdGlvbnMgZm9yIGNvbmZpZ3VyaW5nIHRoZSBsaWJyYXJ5IHZpYSBBbmd1bGFyIERJOlxuICogLSBwcm92aWRlQ3VzdG9tUHJlc2V0cygpIC0gQ3VzdG9tIHByZXNldCBwbHVnaW5zXG4gKiAtIHByb3ZpZGVQcmVzZXRQYWNrYWdlKCkgLSBQcmVzZXQgcGFja2FnZXNcbiAqIFxuICogQHB1YmxpY1xuICogQHNpbmNlIHYzLjYuMFxuICovXG5leHBvcnQge1xuICBwcm92aWRlQ3VzdG9tUHJlc2V0cyxcbiAgcHJvdmlkZVByZXNldFBhY2thZ2Vcbn0gZnJvbSAnLi9wcmVzZXQtcHJvdmlkZXJzJztcblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gVkFMSURBVElPTiAmIFVUSUxJVElFU1xuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4vKipcbiAqIFJhbmdlIFZhbGlkYXRpb24gVXRpbGl0aWVzXG4gKiBcbiAqIFB1cmUgZnVuY3Rpb25zIGZvciBkYXRlIHJhbmdlIHZhbGlkYXRpb24gbG9naWMuXG4gKiBVc2VkIGludGVybmFsbHkgYnV0IGV4cG9zZWQgZm9yIGN1c3RvbSBpbXBsZW1lbnRhdGlvbnMuXG4gKiBcbiAqIEBwdWJsaWNcbiAqIEBzaW5jZSB2My41LjBcbiAqL1xuZXhwb3J0IHtcbiAgdmFsaWRhdGVSYW5nZU9yZGVyLFxuICBhcHBseUJvdW5kcyxcbiAgaXNEYXRlRGlzYWJsZWQsXG4gIHR5cGUgVmFsaWRhdGlvblJlc3VsdFxufSBmcm9tICcuL3JhbmdlLnZhbGlkYXRvcic7XG5cbi8qKlxuICogUHJlc2V0IEVuZ2luZSAtIFByZXNldCByZXNvbHV0aW9uIGFuZCBtYW5hZ2VtZW50XG4gKiBcbiAqIENvcmUgZW5naW5lIHRoYXQgaGFuZGxlcyBwcmVzZXQgcGx1Z2luIGV4ZWN1dGlvbi5cbiAqIFVzZWQgaW50ZXJuYWxseSBidXQgZXhwb3NlZCBmb3IgYWR2YW5jZWQgdXNlIGNhc2VzLlxuICogXG4gKiBAcHVibGljXG4gKiBAc2luY2UgdjMuNi4wXG4gKi9cbmV4cG9ydCB7IFByZXNldEVuZ2luZSB9IGZyb20gJy4vcHJlc2V0LmVuZ2luZSc7XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIFRZUEUgRVhQT1JUUyAoZm9yIFR5cGVTY3JpcHQgdXNlcnMpXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbi8qKlxuICogUmUtZXhwb3J0IHR5cGVzIGZvciBjb252ZW5pZW5jZVxuICogXG4gKiBUaGVzZSB0eXBlcyBhcmUgYWxyZWFkeSBleHBvcnRlZCBieSB0aGVpciByZXNwZWN0aXZlIG1vZHVsZXMsXG4gKiBidXQgd2UgcmUtZXhwb3J0IHRoZW0gaGVyZSBmb3IgZWFzaWVyIGltcG9ydHMuXG4gKi9cbmV4cG9ydCB0eXBlIHtcbiAgLy8gRnJvbSByYW5nZS52YWxpZGF0b3JcbiAgVmFsaWRhdGlvblJlc3VsdCBhcyBSYW5nZVZhbGlkYXRpb25SZXN1bHRcbn0gZnJvbSAnLi9yYW5nZS52YWxpZGF0b3InO1xuIl19
|