@oneluiz/dual-datepicker 3.5.1 → 3.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/core/built-in-presets.d.ts +98 -0
- package/core/index.d.ts +11 -0
- package/core/preset-providers.d.ts +176 -0
- package/core/preset-registry.d.ts +181 -0
- package/core/preset.engine.d.ts +57 -21
- package/core/range-preset.plugin.d.ts +188 -0
- package/esm2022/core/built-in-presets.mjs +289 -0
- package/esm2022/core/index.mjs +13 -1
- package/esm2022/core/preset-providers.mjs +243 -0
- package/esm2022/core/preset-registry.mjs +277 -0
- package/esm2022/core/preset.engine.mjs +86 -210
- package/esm2022/core/range-preset.plugin.mjs +70 -0
- package/esm2022/dual-datepicker.component.mjs +26 -2
- package/fesm2022/oneluiz-dual-datepicker.mjs +988 -211
- package/fesm2022/oneluiz-dual-datepicker.mjs.map +1 -1
- package/package.json +2 -2
|
@@ -10,77 +10,116 @@
|
|
|
10
10
|
* v3.5.1: Timezone-Safe via DateAdapter
|
|
11
11
|
* All date operations use DateAdapter for consistent behavior
|
|
12
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
|
|
13
18
|
*/
|
|
14
19
|
import { Injectable, inject } from '@angular/core';
|
|
15
20
|
import { DATE_CLOCK } from './date-clock';
|
|
16
21
|
import { SystemClock } from './system-clock';
|
|
17
22
|
import { DATE_ADAPTER } from './date-adapter';
|
|
18
23
|
import { NativeDateAdapter } from './native-date-adapter';
|
|
24
|
+
import { PresetRegistry } from './preset-registry';
|
|
19
25
|
import * as i0 from "@angular/core";
|
|
20
26
|
/**
|
|
21
|
-
*
|
|
22
|
-
*
|
|
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
|
|
23
35
|
*
|
|
24
|
-
*
|
|
25
|
-
* -
|
|
26
|
-
* -
|
|
27
|
-
* -
|
|
28
|
-
*
|
|
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
|
+
* });
|
|
29
52
|
*
|
|
30
|
-
*
|
|
31
|
-
*
|
|
32
|
-
*
|
|
33
|
-
*
|
|
34
|
-
* - Override DATE_ADAPTER for Luxon/DayJS/custom implementations
|
|
53
|
+
* // Use via engine
|
|
54
|
+
* const engine = inject(PresetEngine);
|
|
55
|
+
* const range = engine.resolve('MY_PRESET');
|
|
56
|
+
* ```
|
|
35
57
|
*/
|
|
36
58
|
export class PresetEngine {
|
|
37
|
-
presets = new Map();
|
|
38
59
|
clock;
|
|
39
60
|
adapter;
|
|
61
|
+
registry;
|
|
40
62
|
constructor() {
|
|
41
|
-
//
|
|
63
|
+
// Inject dependencies with fallbacks
|
|
42
64
|
try {
|
|
43
65
|
this.clock = inject(DATE_CLOCK, { optional: true }) ?? new SystemClock();
|
|
44
|
-
}
|
|
45
|
-
catch {
|
|
46
|
-
// In case inject() fails (e.g., called outside injection context)
|
|
47
|
-
this.clock = new SystemClock();
|
|
48
|
-
}
|
|
49
|
-
// Try to inject DATE_ADAPTER, fallback to NativeDateAdapter if not provided
|
|
50
|
-
try {
|
|
51
66
|
this.adapter = inject(DATE_ADAPTER, { optional: true }) ?? new NativeDateAdapter();
|
|
67
|
+
this.registry = inject(PresetRegistry);
|
|
52
68
|
}
|
|
53
69
|
catch {
|
|
54
|
-
//
|
|
70
|
+
// Fallback if inject() fails outside injection context
|
|
71
|
+
this.clock = new SystemClock();
|
|
55
72
|
this.adapter = new NativeDateAdapter();
|
|
73
|
+
this.registry = new PresetRegistry();
|
|
56
74
|
}
|
|
57
|
-
this.registerBuiltInPresets();
|
|
58
75
|
}
|
|
59
76
|
/**
|
|
60
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
|
|
61
84
|
*/
|
|
62
85
|
register(key, preset) {
|
|
63
|
-
|
|
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
|
+
});
|
|
64
94
|
}
|
|
65
95
|
/**
|
|
66
96
|
* Resolve a preset to date range
|
|
67
97
|
*
|
|
68
|
-
*
|
|
69
|
-
*
|
|
98
|
+
* Plugin Architecture:
|
|
99
|
+
* 1. Looks up plugin in PresetRegistry
|
|
100
|
+
* 2. Calls plugin.resolve(clock, adapter)
|
|
101
|
+
* 3. Returns ISO date range
|
|
70
102
|
*
|
|
71
|
-
*
|
|
72
|
-
*
|
|
73
|
-
* - DATE_ADAPTER: Control date operations (e.g., Luxon for timezone support)
|
|
103
|
+
* SSR Note: Uses injected DateClock for deterministic resolution
|
|
104
|
+
* Timezone Note: Uses injected DateAdapter for consistent operations
|
|
74
105
|
*
|
|
75
106
|
* @param key - Preset key (e.g., 'TODAY', 'LAST_7_DAYS')
|
|
76
107
|
* @param now - Optional override for current date (defaults to clock.now())
|
|
108
|
+
* @returns ISO date range or null if preset not found
|
|
77
109
|
*/
|
|
78
110
|
resolve(key, now) {
|
|
79
|
-
const
|
|
80
|
-
if (!
|
|
111
|
+
const plugin = this.registry.get(key);
|
|
112
|
+
if (!plugin) {
|
|
113
|
+
console.warn(`[PresetEngine] Preset "${key}" not found in registry`);
|
|
81
114
|
return null;
|
|
82
|
-
|
|
83
|
-
|
|
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
|
|
84
123
|
return {
|
|
85
124
|
start: this.adapter.toISODate(start),
|
|
86
125
|
end: this.adapter.toISODate(end)
|
|
@@ -88,188 +127,22 @@ export class PresetEngine {
|
|
|
88
127
|
}
|
|
89
128
|
/**
|
|
90
129
|
* Get all available preset keys
|
|
130
|
+
*
|
|
131
|
+
* Delegates to PresetRegistry
|
|
132
|
+
*
|
|
133
|
+
* @returns Array of registered preset keys
|
|
91
134
|
*/
|
|
92
135
|
getPresetKeys() {
|
|
93
|
-
return
|
|
136
|
+
return this.registry.getAllKeys();
|
|
94
137
|
}
|
|
95
138
|
/**
|
|
96
|
-
*
|
|
139
|
+
* Check if a preset exists
|
|
97
140
|
*
|
|
98
|
-
*
|
|
141
|
+
* @param key - Preset key to check
|
|
142
|
+
* @returns true if preset is registered
|
|
99
143
|
*/
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
// Today
|
|
103
|
-
this.register('TODAY', {
|
|
104
|
-
resolve: (now) => {
|
|
105
|
-
const normalized = adapter.normalize(now);
|
|
106
|
-
return { start: normalized, end: normalized };
|
|
107
|
-
}
|
|
108
|
-
});
|
|
109
|
-
// Yesterday
|
|
110
|
-
this.register('YESTERDAY', {
|
|
111
|
-
resolve: (now) => {
|
|
112
|
-
const date = adapter.addDays(now, -1);
|
|
113
|
-
return { start: date, end: date };
|
|
114
|
-
}
|
|
115
|
-
});
|
|
116
|
-
// Last N Days
|
|
117
|
-
this.register('LAST_7_DAYS', {
|
|
118
|
-
resolve: (now) => {
|
|
119
|
-
const end = adapter.normalize(now);
|
|
120
|
-
const start = adapter.addDays(now, -6);
|
|
121
|
-
return { start, end };
|
|
122
|
-
}
|
|
123
|
-
});
|
|
124
|
-
this.register('LAST_14_DAYS', {
|
|
125
|
-
resolve: (now) => {
|
|
126
|
-
const end = adapter.normalize(now);
|
|
127
|
-
const start = adapter.addDays(now, -13);
|
|
128
|
-
return { start, end };
|
|
129
|
-
}
|
|
130
|
-
});
|
|
131
|
-
this.register('LAST_30_DAYS', {
|
|
132
|
-
resolve: (now) => {
|
|
133
|
-
const end = adapter.normalize(now);
|
|
134
|
-
const start = adapter.addDays(now, -29);
|
|
135
|
-
return { start, end };
|
|
136
|
-
}
|
|
137
|
-
});
|
|
138
|
-
this.register('LAST_60_DAYS', {
|
|
139
|
-
resolve: (now) => {
|
|
140
|
-
const end = adapter.normalize(now);
|
|
141
|
-
const start = adapter.addDays(now, -59);
|
|
142
|
-
return { start, end };
|
|
143
|
-
}
|
|
144
|
-
});
|
|
145
|
-
this.register('LAST_90_DAYS', {
|
|
146
|
-
resolve: (now) => {
|
|
147
|
-
const end = adapter.normalize(now);
|
|
148
|
-
const start = adapter.addDays(now, -89);
|
|
149
|
-
return { start, end };
|
|
150
|
-
}
|
|
151
|
-
});
|
|
152
|
-
// This Week (Monday to Sunday)
|
|
153
|
-
this.register('THIS_WEEK', {
|
|
154
|
-
resolve: (now) => {
|
|
155
|
-
const dayOfWeek = adapter.getDay(now);
|
|
156
|
-
const daysToMonday = dayOfWeek === 0 ? 6 : dayOfWeek - 1;
|
|
157
|
-
const start = adapter.addDays(now, -daysToMonday);
|
|
158
|
-
const end = adapter.addDays(start, 6);
|
|
159
|
-
return { start, end };
|
|
160
|
-
}
|
|
161
|
-
});
|
|
162
|
-
// Last Week
|
|
163
|
-
this.register('LAST_WEEK', {
|
|
164
|
-
resolve: (now) => {
|
|
165
|
-
const dayOfWeek = adapter.getDay(now);
|
|
166
|
-
const daysToMonday = dayOfWeek === 0 ? 6 : dayOfWeek - 1;
|
|
167
|
-
const lastMonday = adapter.addDays(now, -daysToMonday - 7);
|
|
168
|
-
const lastSunday = adapter.addDays(lastMonday, 6);
|
|
169
|
-
return { start: lastMonday, end: lastSunday };
|
|
170
|
-
}
|
|
171
|
-
});
|
|
172
|
-
// This Month
|
|
173
|
-
this.register('THIS_MONTH', {
|
|
174
|
-
resolve: (now) => {
|
|
175
|
-
const start = adapter.startOfMonth(now);
|
|
176
|
-
const end = adapter.endOfMonth(now);
|
|
177
|
-
return { start, end };
|
|
178
|
-
}
|
|
179
|
-
});
|
|
180
|
-
// Last Month
|
|
181
|
-
this.register('LAST_MONTH', {
|
|
182
|
-
resolve: (now) => {
|
|
183
|
-
const lastMonth = adapter.addMonths(now, -1);
|
|
184
|
-
const start = adapter.startOfMonth(lastMonth);
|
|
185
|
-
const end = adapter.endOfMonth(lastMonth);
|
|
186
|
-
return { start, end };
|
|
187
|
-
}
|
|
188
|
-
});
|
|
189
|
-
// Month to Date
|
|
190
|
-
this.register('MONTH_TO_DATE', {
|
|
191
|
-
resolve: (now) => {
|
|
192
|
-
const start = adapter.startOfMonth(now);
|
|
193
|
-
const end = adapter.normalize(now);
|
|
194
|
-
return { start, end };
|
|
195
|
-
}
|
|
196
|
-
});
|
|
197
|
-
// This Quarter
|
|
198
|
-
this.register('THIS_QUARTER', {
|
|
199
|
-
resolve: (now) => {
|
|
200
|
-
const currentMonth = adapter.getMonth(now);
|
|
201
|
-
const quarterStartMonth = Math.floor(currentMonth / 3) * 3;
|
|
202
|
-
const year = adapter.getYear(now);
|
|
203
|
-
// Construct start of quarter
|
|
204
|
-
const start = new Date(year, quarterStartMonth, 1);
|
|
205
|
-
const normalizedStart = adapter.normalize(start);
|
|
206
|
-
// Construct end of quarter (last day of 3rd month)
|
|
207
|
-
const end = new Date(year, quarterStartMonth + 3, 0);
|
|
208
|
-
const normalizedEnd = adapter.normalize(end);
|
|
209
|
-
return { start: normalizedStart, end: normalizedEnd };
|
|
210
|
-
}
|
|
211
|
-
});
|
|
212
|
-
// Last Quarter
|
|
213
|
-
this.register('LAST_QUARTER', {
|
|
214
|
-
resolve: (now) => {
|
|
215
|
-
const currentMonth = adapter.getMonth(now);
|
|
216
|
-
const lastQuarterStartMonth = Math.floor(currentMonth / 3) * 3 - 3;
|
|
217
|
-
const year = adapter.getYear(now);
|
|
218
|
-
// Handle year rollover (Q1 - 1 = Q4 of previous year)
|
|
219
|
-
const adjustedYear = lastQuarterStartMonth < 0 ? year - 1 : year;
|
|
220
|
-
const adjustedMonth = lastQuarterStartMonth < 0 ? 9 : lastQuarterStartMonth;
|
|
221
|
-
const start = new Date(adjustedYear, adjustedMonth, 1);
|
|
222
|
-
const normalizedStart = adapter.normalize(start);
|
|
223
|
-
const end = new Date(adjustedYear, adjustedMonth + 3, 0);
|
|
224
|
-
const normalizedEnd = adapter.normalize(end);
|
|
225
|
-
return { start: normalizedStart, end: normalizedEnd };
|
|
226
|
-
}
|
|
227
|
-
});
|
|
228
|
-
// Quarter to Date
|
|
229
|
-
this.register('QUARTER_TO_DATE', {
|
|
230
|
-
resolve: (now) => {
|
|
231
|
-
const currentMonth = adapter.getMonth(now);
|
|
232
|
-
const quarterStartMonth = Math.floor(currentMonth / 3) * 3;
|
|
233
|
-
const year = adapter.getYear(now);
|
|
234
|
-
const start = new Date(year, quarterStartMonth, 1);
|
|
235
|
-
const normalizedStart = adapter.normalize(start);
|
|
236
|
-
const end = adapter.normalize(now);
|
|
237
|
-
return { start: normalizedStart, end };
|
|
238
|
-
}
|
|
239
|
-
});
|
|
240
|
-
// This Year
|
|
241
|
-
this.register('THIS_YEAR', {
|
|
242
|
-
resolve: (now) => {
|
|
243
|
-
const year = adapter.getYear(now);
|
|
244
|
-
const start = new Date(year, 0, 1);
|
|
245
|
-
const end = new Date(year, 11, 31);
|
|
246
|
-
return {
|
|
247
|
-
start: adapter.normalize(start),
|
|
248
|
-
end: adapter.normalize(end)
|
|
249
|
-
};
|
|
250
|
-
}
|
|
251
|
-
});
|
|
252
|
-
// Last Year
|
|
253
|
-
this.register('LAST_YEAR', {
|
|
254
|
-
resolve: (now) => {
|
|
255
|
-
const year = adapter.getYear(now);
|
|
256
|
-
const start = new Date(year - 1, 0, 1);
|
|
257
|
-
const end = new Date(year - 1, 11, 31);
|
|
258
|
-
return {
|
|
259
|
-
start: adapter.normalize(start),
|
|
260
|
-
end: adapter.normalize(end)
|
|
261
|
-
};
|
|
262
|
-
}
|
|
263
|
-
});
|
|
264
|
-
// Year to Date
|
|
265
|
-
this.register('YEAR_TO_DATE', {
|
|
266
|
-
resolve: (now) => {
|
|
267
|
-
const year = adapter.getYear(now);
|
|
268
|
-
const start = new Date(year, 0, 1);
|
|
269
|
-
const end = adapter.normalize(now);
|
|
270
|
-
return { start: adapter.normalize(start), end };
|
|
271
|
-
}
|
|
272
|
-
});
|
|
144
|
+
hasPreset(key) {
|
|
145
|
+
return this.registry.has(key);
|
|
273
146
|
}
|
|
274
147
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetEngine, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
275
148
|
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetEngine, providedIn: 'root' });
|
|
@@ -282,6 +155,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
282
155
|
}], ctorParameters: () => [] });
|
|
283
156
|
/**
|
|
284
157
|
* Create a custom preset from a function
|
|
158
|
+
*
|
|
159
|
+
* @deprecated Use RangePresetPlugin interface instead
|
|
160
|
+
* Kept for backward compatibility
|
|
285
161
|
*/
|
|
286
162
|
export function createPreset(resolver) {
|
|
287
163
|
return { resolve: resolver };
|
|
@@ -300,4 +176,4 @@ export function createPreset(resolver) {
|
|
|
300
176
|
* This export will be removed in v4.0.0
|
|
301
177
|
*/
|
|
302
178
|
export const presetEngine = new PresetEngine();
|
|
303
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"preset.engine.js","sourceRoot":"","sources":["../../../src/core/preset.engine.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AAEH,OAAO,EAAE,UAAU,EAAE,MAAM,EAAY,MAAM,eAAe,CAAC;AAC7D,OAAO,EAAa,UAAU,EAAE,MAAM,cAAc,CAAC;AACrD,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAe,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAC3D,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;;AAe1D;;;;;;;;;;;;;;;GAeG;AAIH,MAAM,OAAO,YAAY;IACf,OAAO,GAAG,IAAI,GAAG,EAAuB,CAAC;IACzC,KAAK,CAAY;IACjB,OAAO,CAAc;IAE7B;QACE,oEAAoE;QACpE,IAAI,CAAC;YACH,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,UAAU,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,IAAI,IAAI,WAAW,EAAE,CAAC;QAC3E,CAAC;QAAC,MAAM,CAAC;YACP,kEAAkE;YAClE,IAAI,CAAC,KAAK,GAAG,IAAI,WAAW,EAAE,CAAC;QACjC,CAAC;QAED,4EAA4E;QAC5E,IAAI,CAAC;YACH,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC,YAAY,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,IAAI,IAAI,iBAAiB,EAAE,CAAC;QACrF,CAAC;QAAC,MAAM,CAAC;YACP,kEAAkE;YAClE,IAAI,CAAC,OAAO,GAAG,IAAI,iBAAiB,EAAE,CAAC;QACzC,CAAC;QAED,IAAI,CAAC,sBAAsB,EAAE,CAAC;IAChC,CAAC;IAED;;OAEG;IACH,QAAQ,CAAC,GAAW,EAAE,MAAmB;QACvC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;IAChC,CAAC;IAED;;;;;;;;;;;;OAYG;IACH,OAAO,CAAC,GAAW,EAAE,GAAU;QAC7B,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACrC,IAAI,CAAC,MAAM;YAAE,OAAO,IAAI,CAAC;QAEzB,MAAM,WAAW,GAAG,GAAG,IAAI,IAAI,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC;QAC5C,MAAM,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;QACnD,OAAO;YACL,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC;YACpC,GAAG,EAAE,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC;SACjC,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,aAAa;QACX,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;IACzC,CAAC;IAED;;;;OAIG;IACK,sBAAsB;QAC5B,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC;QAE7B,QAAQ;QACR,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE;YACrB,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,UAAU,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBAC1C,OAAO,EAAE,KAAK,EAAE,UAAU,EAAE,GAAG,EAAE,UAAU,EAAE,CAAC;YAChD,CAAC;SACF,CAAC,CAAC;QAEH,YAAY;QACZ,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE;YACzB,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,IAAI,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;gBACtC,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,GAAG,EAAE,IAAI,EAAE,CAAC;YACpC,CAAC;SACF,CAAC,CAAC;QAEH,cAAc;QACd,IAAI,CAAC,QAAQ,CAAC,aAAa,EAAE;YAC3B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,GAAG,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBACnC,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;gBACvC,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE;YAC5B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,GAAG,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBACnC,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,CAAC;gBACxC,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE;YAC5B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,GAAG,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBACnC,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,CAAC;gBACxC,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE;YAC5B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,GAAG,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBACnC,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,CAAC;gBACxC,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE;YAC5B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,GAAG,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBACnC,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,CAAC;gBACxC,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEH,+BAA+B;QAC/B,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE;YACzB,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;gBACtC,MAAM,YAAY,GAAG,SAAS,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC;gBACzD,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,YAAY,CAAC,CAAC;gBAClD,MAAM,GAAG,GAAG,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;gBACtC,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEH,YAAY;QACZ,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE;YACzB,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;gBACtC,MAAM,YAAY,GAAG,SAAS,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC;gBACzD,MAAM,UAAU,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,YAAY,GAAG,CAAC,CAAC,CAAC;gBAC3D,MAAM,UAAU,GAAG,OAAO,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC;gBAClD,OAAO,EAAE,KAAK,EAAE,UAAU,EAAE,GAAG,EAAE,UAAU,EAAE,CAAC;YAChD,CAAC;SACF,CAAC,CAAC;QAEH,aAAa;QACb,IAAI,CAAC,QAAQ,CAAC,YAAY,EAAE;YAC1B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,KAAK,GAAG,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;gBACxC,MAAM,GAAG,GAAG,OAAO,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;gBACpC,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEH,aAAa;QACb,IAAI,CAAC,QAAQ,CAAC,YAAY,EAAE;YAC1B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,SAAS,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;gBAC7C,MAAM,KAAK,GAAG,OAAO,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;gBAC9C,MAAM,GAAG,GAAG,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;gBAC1C,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEH,gBAAgB;QAChB,IAAI,CAAC,QAAQ,CAAC,eAAe,EAAE;YAC7B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,KAAK,GAAG,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;gBACxC,MAAM,GAAG,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBACnC,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEH,eAAe;QACf,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE;YAC5B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,YAAY,GAAG,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;gBAC3C,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC;gBAC3D,MAAM,IAAI,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;gBAElC,6BAA6B;gBAC7B,MAAM,KAAK,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,iBAAiB,EAAE,CAAC,CAAC,CAAC;gBACnD,MAAM,eAAe,GAAG,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;gBAEjD,mDAAmD;gBACnD,MAAM,GAAG,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,iBAAiB,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;gBACrD,MAAM,aAAa,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBAE7C,OAAO,EAAE,KAAK,EAAE,eAAe,EAAE,GAAG,EAAE,aAAa,EAAE,CAAC;YACxD,CAAC;SACF,CAAC,CAAC;QAEH,eAAe;QACf,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE;YAC5B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,YAAY,GAAG,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;gBAC3C,MAAM,qBAAqB,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;gBACnE,MAAM,IAAI,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;gBAElC,sDAAsD;gBACtD,MAAM,YAAY,GAAG,qBAAqB,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;gBACjE,MAAM,aAAa,GAAG,qBAAqB,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,qBAAqB,CAAC;gBAE5E,MAAM,KAAK,GAAG,IAAI,IAAI,CAAC,YAAY,EAAE,aAAa,EAAE,CAAC,CAAC,CAAC;gBACvD,MAAM,eAAe,GAAG,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;gBAEjD,MAAM,GAAG,GAAG,IAAI,IAAI,CAAC,YAAY,EAAE,aAAa,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;gBACzD,MAAM,aAAa,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBAE7C,OAAO,EAAE,KAAK,EAAE,eAAe,EAAE,GAAG,EAAE,aAAa,EAAE,CAAC;YACxD,CAAC;SACF,CAAC,CAAC;QAEH,kBAAkB;QAClB,IAAI,CAAC,QAAQ,CAAC,iBAAiB,EAAE;YAC/B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,YAAY,GAAG,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;gBAC3C,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC;gBAC3D,MAAM,IAAI,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;gBAElC,MAAM,KAAK,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,iBAAiB,EAAE,CAAC,CAAC,CAAC;gBACnD,MAAM,eAAe,GAAG,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;gBACjD,MAAM,GAAG,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBAEnC,OAAO,EAAE,KAAK,EAAE,eAAe,EAAE,GAAG,EAAE,CAAC;YACzC,CAAC;SACF,CAAC,CAAC;QAEH,YAAY;QACZ,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE;YACzB,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,IAAI,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;gBAClC,MAAM,KAAK,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;gBACnC,MAAM,GAAG,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,EAAE,EAAE,EAAE,CAAC,CAAC;gBACnC,OAAO;oBACL,KAAK,EAAE,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC;oBAC/B,GAAG,EAAE,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC;iBAC5B,CAAC;YACJ,CAAC;SACF,CAAC,CAAC;QAEH,YAAY;QACZ,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE;YACzB,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,IAAI,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;gBAClC,MAAM,KAAK,GAAG,IAAI,IAAI,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;gBACvC,MAAM,GAAG,GAAG,IAAI,IAAI,CAAC,IAAI,GAAG,CAAC,EAAE,EAAE,EAAE,EAAE,CAAC,CAAC;gBACvC,OAAO;oBACL,KAAK,EAAE,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC;oBAC/B,GAAG,EAAE,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC;iBAC5B,CAAC;YACJ,CAAC;SACF,CAAC,CAAC;QAEH,eAAe;QACf,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE;YAC5B,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBACf,MAAM,IAAI,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;gBAClC,MAAM,KAAK,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;gBACnC,MAAM,GAAG,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBACnC,OAAO,EAAE,KAAK,EAAE,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE,GAAG,EAAE,CAAC;YAClD,CAAC;SACF,CAAC,CAAC;IACL,CAAC;wGA7QU,YAAY;4GAAZ,YAAY,cAFX,MAAM;;4FAEP,YAAY;kBAHxB,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB;;AAiRD;;GAEG;AACH,MAAM,UAAU,YAAY,CAC1B,QAAmD;IAEnD,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,CAAC;AAC/B,CAAC;AAED;;;;;;;;;;;;GAYG;AACH,MAAM,CAAC,MAAM,YAAY,GAAG,IAAI,YAAY,EAAE,CAAC","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\nimport { Injectable, inject, Optional } 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';\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 * Registry of built-in presets\n * Can be extended by consumers\n * \n * SSR-Safe Architecture:\n * - Injects DateClock via DI\n * - All presets use clock.now() instead of new Date()\n * - Deterministic: same clock.now() → same preset\n * - Override DATE_CLOCK token in SSR to ensure consistency\n * \n * Timezone-Safe Architecture:\n * - Injects DateAdapter via DI\n * - All date operations use adapter methods\n * - Prevents timezone bugs in cross-timezone scenarios\n * - Override DATE_ADAPTER for Luxon/DayJS/custom implementations\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class PresetEngine {\n  private presets = new Map<string, RangePreset>();\n  private clock: DateClock;\n  private adapter: DateAdapter;\n\n  constructor() {\n    // Try to inject DATE_CLOCK, fallback to SystemClock if not provided\n    try {\n      this.clock = inject(DATE_CLOCK, { optional: true }) ?? new SystemClock();\n    } catch {\n      // In case inject() fails (e.g., called outside injection context)\n      this.clock = new SystemClock();\n    }\n\n    // Try to inject DATE_ADAPTER, fallback to NativeDateAdapter if not provided\n    try {\n      this.adapter = inject(DATE_ADAPTER, { optional: true }) ?? new NativeDateAdapter();\n    } catch {\n      // In case inject() fails (e.g., called outside injection context)\n      this.adapter = new NativeDateAdapter();\n    }\n\n    this.registerBuiltInPresets();\n  }\n\n  /**\n   * Register a custom preset\n   */\n  register(key: string, preset: RangePreset): void {\n    this.presets.set(key, preset);\n  }\n\n  /**\n   * Resolve a preset to date range\n   * \n   * SSR Note: Uses injected DateClock for deterministic resolution\n   * Timezone Note: Uses injected DateAdapter for consistent date operations\n   * \n   * Override tokens in SSR scenarios:\n   * - DATE_CLOCK: Control current time\n   * - DATE_ADAPTER: Control date operations (e.g., Luxon for timezone support)\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   */\n  resolve(key: string, now?: Date): PresetRange | null {\n    const preset = this.presets.get(key);\n    if (!preset) return null;\n\n    const currentDate = now ?? this.clock.now();\n    const { start, end } = preset.resolve(currentDate);\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  getPresetKeys(): string[] {\n    return Array.from(this.presets.keys());\n  }\n\n  /**\n   * Register all built-in presets\n   * \n   * All presets now use DateAdapter for timezone-safe operations\n   */\n  private registerBuiltInPresets(): void {\n    const adapter = this.adapter;\n\n    // Today\n    this.register('TODAY', {\n      resolve: (now) => {\n        const normalized = adapter.normalize(now);\n        return { start: normalized, end: normalized };\n      }\n    });\n\n    // Yesterday\n    this.register('YESTERDAY', {\n      resolve: (now) => {\n        const date = adapter.addDays(now, -1);\n        return { start: date, end: date };\n      }\n    });\n\n    // Last N Days\n    this.register('LAST_7_DAYS', {\n      resolve: (now) => {\n        const end = adapter.normalize(now);\n        const start = adapter.addDays(now, -6);\n        return { start, end };\n      }\n    });\n\n    this.register('LAST_14_DAYS', {\n      resolve: (now) => {\n        const end = adapter.normalize(now);\n        const start = adapter.addDays(now, -13);\n        return { start, end };\n      }\n    });\n\n    this.register('LAST_30_DAYS', {\n      resolve: (now) => {\n        const end = adapter.normalize(now);\n        const start = adapter.addDays(now, -29);\n        return { start, end };\n      }\n    });\n\n    this.register('LAST_60_DAYS', {\n      resolve: (now) => {\n        const end = adapter.normalize(now);\n        const start = adapter.addDays(now, -59);\n        return { start, end };\n      }\n    });\n\n    this.register('LAST_90_DAYS', {\n      resolve: (now) => {\n        const end = adapter.normalize(now);\n        const start = adapter.addDays(now, -89);\n        return { start, end };\n      }\n    });\n\n    // This Week (Monday to Sunday)\n    this.register('THIS_WEEK', {\n      resolve: (now) => {\n        const dayOfWeek = adapter.getDay(now);\n        const daysToMonday = dayOfWeek === 0 ? 6 : dayOfWeek - 1;\n        const start = adapter.addDays(now, -daysToMonday);\n        const end = adapter.addDays(start, 6);\n        return { start, end };\n      }\n    });\n\n    // Last Week\n    this.register('LAST_WEEK', {\n      resolve: (now) => {\n        const dayOfWeek = adapter.getDay(now);\n        const daysToMonday = dayOfWeek === 0 ? 6 : dayOfWeek - 1;\n        const lastMonday = adapter.addDays(now, -daysToMonday - 7);\n        const lastSunday = adapter.addDays(lastMonday, 6);\n        return { start: lastMonday, end: lastSunday };\n      }\n    });\n\n    // This Month\n    this.register('THIS_MONTH', {\n      resolve: (now) => {\n        const start = adapter.startOfMonth(now);\n        const end = adapter.endOfMonth(now);\n        return { start, end };\n      }\n    });\n\n    // Last Month\n    this.register('LAST_MONTH', {\n      resolve: (now) => {\n        const lastMonth = adapter.addMonths(now, -1);\n        const start = adapter.startOfMonth(lastMonth);\n        const end = adapter.endOfMonth(lastMonth);\n        return { start, end };\n      }\n    });\n\n    // Month to Date\n    this.register('MONTH_TO_DATE', {\n      resolve: (now) => {\n        const start = adapter.startOfMonth(now);\n        const end = adapter.normalize(now);\n        return { start, end };\n      }\n    });\n\n    // This Quarter\n    this.register('THIS_QUARTER', {\n      resolve: (now) => {\n        const currentMonth = adapter.getMonth(now);\n        const quarterStartMonth = Math.floor(currentMonth / 3) * 3;\n        const year = adapter.getYear(now);\n        \n        // Construct start of quarter\n        const start = new Date(year, quarterStartMonth, 1);\n        const normalizedStart = adapter.normalize(start);\n        \n        // Construct end of quarter (last day of 3rd month)\n        const end = new Date(year, quarterStartMonth + 3, 0);\n        const normalizedEnd = adapter.normalize(end);\n        \n        return { start: normalizedStart, end: normalizedEnd };\n      }\n    });\n\n    // Last Quarter\n    this.register('LAST_QUARTER', {\n      resolve: (now) => {\n        const currentMonth = adapter.getMonth(now);\n        const lastQuarterStartMonth = Math.floor(currentMonth / 3) * 3 - 3;\n        const year = adapter.getYear(now);\n        \n        // Handle year rollover (Q1 - 1 = Q4 of previous year)\n        const adjustedYear = lastQuarterStartMonth < 0 ? year - 1 : year;\n        const adjustedMonth = lastQuarterStartMonth < 0 ? 9 : lastQuarterStartMonth;\n        \n        const start = new Date(adjustedYear, adjustedMonth, 1);\n        const normalizedStart = adapter.normalize(start);\n        \n        const end = new Date(adjustedYear, adjustedMonth + 3, 0);\n        const normalizedEnd = adapter.normalize(end);\n        \n        return { start: normalizedStart, end: normalizedEnd };\n      }\n    });\n\n    // Quarter to Date\n    this.register('QUARTER_TO_DATE', {\n      resolve: (now) => {\n        const currentMonth = adapter.getMonth(now);\n        const quarterStartMonth = Math.floor(currentMonth / 3) * 3;\n        const year = adapter.getYear(now);\n        \n        const start = new Date(year, quarterStartMonth, 1);\n        const normalizedStart = adapter.normalize(start);\n        const end = adapter.normalize(now);\n        \n        return { start: normalizedStart, end };\n      }\n    });\n\n    // This Year\n    this.register('THIS_YEAR', {\n      resolve: (now) => {\n        const year = adapter.getYear(now);\n        const start = new Date(year, 0, 1);\n        const end = new Date(year, 11, 31);\n        return { \n          start: adapter.normalize(start), \n          end: adapter.normalize(end) \n        };\n      }\n    });\n\n    // Last Year\n    this.register('LAST_YEAR', {\n      resolve: (now) => {\n        const year = adapter.getYear(now);\n        const start = new Date(year - 1, 0, 1);\n        const end = new Date(year - 1, 11, 31);\n        return { \n          start: adapter.normalize(start), \n          end: adapter.normalize(end) \n        };\n      }\n    });\n\n    // Year to Date\n    this.register('YEAR_TO_DATE', {\n      resolve: (now) => {\n        const year = adapter.getYear(now);\n        const start = new Date(year, 0, 1);\n        const end = adapter.normalize(now);\n        return { start: adapter.normalize(start), end };\n      }\n    });\n  }\n}\n\n/**\n * Create a custom preset from a function\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"]}
|
|
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"]}
|
|
@@ -0,0 +1,70 @@
|
|
|
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"]}
|