@oneluiz/dual-datepicker 3.5.0 → 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/README.md +51 -3
- package/core/built-in-presets.d.ts +98 -0
- package/core/date-adapter.d.ts +298 -0
- package/core/date-clock.d.ts +82 -0
- package/core/dual-date-range.store.d.ts +9 -0
- package/core/index.d.ts +15 -0
- package/core/native-date-adapter.d.ts +152 -0
- package/core/preset-providers.d.ts +176 -0
- package/core/preset-registry.d.ts +181 -0
- package/core/preset.engine.d.ts +79 -9
- package/core/range-preset.plugin.d.ts +188 -0
- package/core/system-clock.d.ts +13 -0
- package/dual-datepicker.component.d.ts +17 -11
- package/esm2022/core/built-in-presets.mjs +289 -0
- package/esm2022/core/date-adapter.mjs +77 -0
- package/esm2022/core/date-clock.mjs +65 -0
- package/esm2022/core/dual-date-range.store.mjs +51 -15
- package/esm2022/core/index.mjs +17 -1
- package/esm2022/core/native-date-adapter.mjs +286 -0
- package/esm2022/core/preset-providers.mjs +243 -0
- package/esm2022/core/preset-registry.mjs +277 -0
- package/esm2022/core/preset.engine.mjs +139 -169
- package/esm2022/core/range-preset.plugin.mjs +70 -0
- package/esm2022/core/system-clock.mjs +34 -0
- package/esm2022/dual-datepicker.component.mjs +264 -196
- package/fesm2022/oneluiz-dual-datepicker.mjs +3219 -1822
- package/fesm2022/oneluiz-dual-datepicker.mjs.map +1 -1
- package/package.json +5 -3
|
@@ -0,0 +1,286 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Native Date Adapter Implementation
|
|
3
|
+
*
|
|
4
|
+
* Default adapter using JavaScript Date with timezone-safe operations.
|
|
5
|
+
*
|
|
6
|
+
* Zero dependencies. Conservative implementation that:
|
|
7
|
+
* - Normalizes all dates to start of day (00:00:00.000)
|
|
8
|
+
* - Uses manual YYYY-MM-DD construction (avoids toISOString() timezone shift)
|
|
9
|
+
* - Parses ISO dates to local timezone (avoids UTC interpretation)
|
|
10
|
+
* - Handles month overflow correctly (Jan 31 + 1 month = Feb 28, not Mar 3)
|
|
11
|
+
* - All comparisons work on normalized dates
|
|
12
|
+
*
|
|
13
|
+
* Limitations:
|
|
14
|
+
* - No timezone awareness (all operations in local timezone)
|
|
15
|
+
* - No DST-safe calculations across timezone changes
|
|
16
|
+
* - For advanced timezone needs, use LuxonDateAdapter or DayJSDateAdapter
|
|
17
|
+
*
|
|
18
|
+
* Perfect for:
|
|
19
|
+
* ✅ Most enterprise apps (ERP, POS, BI) where local timezone is sufficient
|
|
20
|
+
* ✅ Applications without cross-timezone requirements
|
|
21
|
+
* ✅ Minimizing bundle size (zero deps)
|
|
22
|
+
* ✅ Simple, predictable date handling
|
|
23
|
+
*/
|
|
24
|
+
import { Injectable } from '@angular/core';
|
|
25
|
+
import * as i0 from "@angular/core";
|
|
26
|
+
export class NativeDateAdapter {
|
|
27
|
+
/**
|
|
28
|
+
* Normalize date to start of day (00:00:00.000) in local timezone
|
|
29
|
+
*
|
|
30
|
+
* This is the foundation of timezone-safe operations.
|
|
31
|
+
* All other methods use normalized dates for comparisons.
|
|
32
|
+
*/
|
|
33
|
+
normalize(date) {
|
|
34
|
+
const normalized = new Date(date);
|
|
35
|
+
normalized.setHours(0, 0, 0, 0);
|
|
36
|
+
return normalized;
|
|
37
|
+
}
|
|
38
|
+
/**
|
|
39
|
+
* Check if two dates are the same calendar day
|
|
40
|
+
*
|
|
41
|
+
* Implementation: Compare YYYY-MM-DD components directly
|
|
42
|
+
* Avoids timezone issues from valueOf() comparisons
|
|
43
|
+
*/
|
|
44
|
+
isSameDay(a, b) {
|
|
45
|
+
return (a.getFullYear() === b.getFullYear() &&
|
|
46
|
+
a.getMonth() === b.getMonth() &&
|
|
47
|
+
a.getDate() === b.getDate());
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Check if date A is before date B (calendar day level)
|
|
51
|
+
*
|
|
52
|
+
* Implementation: Compare normalized dates using valueOf()
|
|
53
|
+
*/
|
|
54
|
+
isBeforeDay(a, b) {
|
|
55
|
+
return this.normalize(a).valueOf() < this.normalize(b).valueOf();
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Check if date A is after date B (calendar day level)
|
|
59
|
+
*
|
|
60
|
+
* Implementation: Compare normalized dates using valueOf()
|
|
61
|
+
*/
|
|
62
|
+
isAfterDay(a, b) {
|
|
63
|
+
return this.normalize(a).valueOf() > this.normalize(b).valueOf();
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Add days to a date
|
|
67
|
+
*
|
|
68
|
+
* Implementation: Use setDate() which handles month rollover automatically
|
|
69
|
+
*
|
|
70
|
+
* Example:
|
|
71
|
+
* Jan 31 + 3 days → Feb 3 ✅
|
|
72
|
+
* Feb 28 + 1 day → Mar 1 ✅ (non-leap year)
|
|
73
|
+
*/
|
|
74
|
+
addDays(date, days) {
|
|
75
|
+
const result = new Date(date);
|
|
76
|
+
result.setDate(result.getDate() + days);
|
|
77
|
+
return this.normalize(result);
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Add months to a date
|
|
81
|
+
*
|
|
82
|
+
* CRITICAL: Handles month overflow correctly
|
|
83
|
+
*
|
|
84
|
+
* Algorithm:
|
|
85
|
+
* 1. Add months using setMonth()
|
|
86
|
+
* 2. If day-of-month changed (overflow), set to last day of target month
|
|
87
|
+
*
|
|
88
|
+
* Examples:
|
|
89
|
+
* - Jan 31 + 1 month → Feb 28 (or Feb 29 in leap year) ✅
|
|
90
|
+
* - Jan 31 + 2 months → Mar 31 ✅
|
|
91
|
+
* - Mar 31 + 1 month → Apr 30 ✅
|
|
92
|
+
* - Dec 31 + 1 month → Jan 31 (next year) ✅
|
|
93
|
+
*/
|
|
94
|
+
addMonths(date, months) {
|
|
95
|
+
const result = new Date(date);
|
|
96
|
+
const originalDay = result.getDate();
|
|
97
|
+
// Add months
|
|
98
|
+
result.setMonth(result.getMonth() + months);
|
|
99
|
+
// Check for day overflow (e.g., Jan 31 → Feb 31 becomes Mar 3)
|
|
100
|
+
if (result.getDate() !== originalDay) {
|
|
101
|
+
// Overflow detected: set to last day of target month
|
|
102
|
+
// Go to 1st of next month, then subtract 1 day
|
|
103
|
+
result.setDate(0); // Sets to last day of previous month
|
|
104
|
+
}
|
|
105
|
+
return this.normalize(result);
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Get start of day (00:00:00.000)
|
|
109
|
+
*
|
|
110
|
+
* Alias for normalize() with explicit intent
|
|
111
|
+
*/
|
|
112
|
+
startOfDay(date) {
|
|
113
|
+
return this.normalize(date);
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Get end of day (23:59:59.999)
|
|
117
|
+
*
|
|
118
|
+
* Useful for inclusive range queries
|
|
119
|
+
*/
|
|
120
|
+
endOfDay(date) {
|
|
121
|
+
const result = new Date(date);
|
|
122
|
+
result.setHours(23, 59, 59, 999);
|
|
123
|
+
return result;
|
|
124
|
+
}
|
|
125
|
+
/**
|
|
126
|
+
* Get first day of month (00:00:00.000)
|
|
127
|
+
*/
|
|
128
|
+
startOfMonth(date) {
|
|
129
|
+
const result = new Date(date);
|
|
130
|
+
result.setDate(1);
|
|
131
|
+
return this.normalize(result);
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* Get last day of month (23:59:59.999)
|
|
135
|
+
*
|
|
136
|
+
* Algorithm: Go to 1st of next month, subtract 1 day
|
|
137
|
+
*/
|
|
138
|
+
endOfMonth(date) {
|
|
139
|
+
const result = new Date(date);
|
|
140
|
+
result.setMonth(result.getMonth() + 1, 0); // Day 0 = last day of previous month
|
|
141
|
+
return this.endOfDay(result);
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Get year (4-digit)
|
|
145
|
+
*/
|
|
146
|
+
getYear(date) {
|
|
147
|
+
return date.getFullYear();
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* Get month (0-11)
|
|
151
|
+
*/
|
|
152
|
+
getMonth(date) {
|
|
153
|
+
return date.getMonth();
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Get day of month (1-31)
|
|
157
|
+
*/
|
|
158
|
+
getDate(date) {
|
|
159
|
+
return date.getDate();
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* Get day of week (0-6, Sunday=0)
|
|
163
|
+
*/
|
|
164
|
+
getDay(date) {
|
|
165
|
+
return date.getDay();
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* Convert Date to ISO date string (YYYY-MM-DD)
|
|
169
|
+
*
|
|
170
|
+
* CRITICAL: DO NOT use toISOString() - it converts to UTC!
|
|
171
|
+
*
|
|
172
|
+
* Manual construction ensures local timezone is preserved:
|
|
173
|
+
*
|
|
174
|
+
* Example problem with toISOString():
|
|
175
|
+
* ```
|
|
176
|
+
* // Local timezone: GMT-6 (CST)
|
|
177
|
+
* const date = new Date('2026-02-21T23:00:00'); // 11 PM Feb 21 local
|
|
178
|
+
*
|
|
179
|
+
* // WRONG ❌
|
|
180
|
+
* date.toISOString().split('T')[0]
|
|
181
|
+
* // Returns "2026-02-22" (converted to UTC = Feb 22 05:00 AM)
|
|
182
|
+
*
|
|
183
|
+
* // CORRECT ✅
|
|
184
|
+
* toISODate(date)
|
|
185
|
+
* // Returns "2026-02-21" (local date preserved)
|
|
186
|
+
* ```
|
|
187
|
+
*
|
|
188
|
+
* Implementation: Build YYYY-MM-DD manually from local date components
|
|
189
|
+
*/
|
|
190
|
+
toISODate(date) {
|
|
191
|
+
const year = date.getFullYear();
|
|
192
|
+
const month = String(date.getMonth() + 1).padStart(2, '0');
|
|
193
|
+
const day = String(date.getDate()).padStart(2, '0');
|
|
194
|
+
return `${year}-${month}-${day}`;
|
|
195
|
+
}
|
|
196
|
+
/**
|
|
197
|
+
* Parse ISO date string (YYYY-MM-DD) to Date
|
|
198
|
+
*
|
|
199
|
+
* CRITICAL: DO NOT use new Date(isoString) - may parse as UTC!
|
|
200
|
+
*
|
|
201
|
+
* Example problem with Date constructor:
|
|
202
|
+
* ```
|
|
203
|
+
* // Local timezone: GMT-6 (CST)
|
|
204
|
+
*
|
|
205
|
+
* // WRONG ❌
|
|
206
|
+
* new Date('2026-02-21')
|
|
207
|
+
* // Parsed as UTC: 2026-02-21T00:00:00Z
|
|
208
|
+
* // In local timezone: Feb 20, 2026 6:00 PM (previous day!)
|
|
209
|
+
*
|
|
210
|
+
* // CORRECT ✅
|
|
211
|
+
* parseISODate('2026-02-21')
|
|
212
|
+
* // Returns: 2026-02-21T00:00:00 local time
|
|
213
|
+
* ```
|
|
214
|
+
*
|
|
215
|
+
* Implementation: Parse components and construct Date in local timezone
|
|
216
|
+
*/
|
|
217
|
+
parseISODate(isoDate) {
|
|
218
|
+
if (!isoDate || typeof isoDate !== 'string') {
|
|
219
|
+
return null;
|
|
220
|
+
}
|
|
221
|
+
// Match YYYY-MM-DD format
|
|
222
|
+
const match = /^(\d{4})-(\d{2})-(\d{2})$/.exec(isoDate.trim());
|
|
223
|
+
if (!match) {
|
|
224
|
+
return null;
|
|
225
|
+
}
|
|
226
|
+
const year = parseInt(match[1], 10);
|
|
227
|
+
const month = parseInt(match[2], 10) - 1; // 0-indexed
|
|
228
|
+
const day = parseInt(match[3], 10);
|
|
229
|
+
// Validate ranges
|
|
230
|
+
if (month < 0 || month > 11) {
|
|
231
|
+
return null;
|
|
232
|
+
}
|
|
233
|
+
if (day < 1 || day > 31) {
|
|
234
|
+
return null;
|
|
235
|
+
}
|
|
236
|
+
// Construct date in local timezone
|
|
237
|
+
const date = new Date(year, month, day);
|
|
238
|
+
// Verify date is valid (e.g., Feb 31 would roll to Mar 3)
|
|
239
|
+
if (date.getFullYear() !== year ||
|
|
240
|
+
date.getMonth() !== month ||
|
|
241
|
+
date.getDate() !== day) {
|
|
242
|
+
return null;
|
|
243
|
+
}
|
|
244
|
+
return this.normalize(date);
|
|
245
|
+
}
|
|
246
|
+
/**
|
|
247
|
+
* Get week start day for locale
|
|
248
|
+
*
|
|
249
|
+
* Default: Sunday (0) for most locales
|
|
250
|
+
* Monday (1) for Europe, ISO 8601
|
|
251
|
+
*
|
|
252
|
+
* Implementation: Simple locale detection
|
|
253
|
+
* For advanced needs, use Intl.Locale or external library
|
|
254
|
+
*/
|
|
255
|
+
getWeekStart(locale) {
|
|
256
|
+
if (!locale) {
|
|
257
|
+
locale = typeof navigator !== 'undefined' ? navigator.language : 'en-US';
|
|
258
|
+
}
|
|
259
|
+
// ISO 8601: Monday start
|
|
260
|
+
const mondayStartLocales = [
|
|
261
|
+
'en-GB', 'en-IE', 'en-AU', 'en-NZ', 'en-CA',
|
|
262
|
+
'es', 'es-ES', 'es-MX',
|
|
263
|
+
'fr', 'fr-FR', 'fr-CA',
|
|
264
|
+
'de', 'de-DE', 'de-AT', 'de-CH',
|
|
265
|
+
'it', 'it-IT',
|
|
266
|
+
'pt', 'pt-PT', 'pt-BR',
|
|
267
|
+
'nl', 'nl-NL', 'nl-BE',
|
|
268
|
+
'ru', 'ru-RU',
|
|
269
|
+
'zh', 'zh-CN', 'zh-TW',
|
|
270
|
+
'ja', 'ja-JP',
|
|
271
|
+
'ko', 'ko-KR'
|
|
272
|
+
];
|
|
273
|
+
const normalizedLocale = locale.toLowerCase();
|
|
274
|
+
const startsWithMonday = mondayStartLocales.some(loc => normalizedLocale.startsWith(loc.toLowerCase()));
|
|
275
|
+
return startsWithMonday ? 1 : 0;
|
|
276
|
+
}
|
|
277
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: NativeDateAdapter, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
278
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: NativeDateAdapter, providedIn: 'root' });
|
|
279
|
+
}
|
|
280
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: NativeDateAdapter, decorators: [{
|
|
281
|
+
type: Injectable,
|
|
282
|
+
args: [{
|
|
283
|
+
providedIn: 'root'
|
|
284
|
+
}]
|
|
285
|
+
}] });
|
|
286
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"native-date-adapter.js","sourceRoot":"","sources":["../../../src/core/native-date-adapter.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Native Date Adapter Implementation\n * \n * Default adapter using JavaScript Date with timezone-safe operations.\n * \n * Zero dependencies. Conservative implementation that:\n * - Normalizes all dates to start of day (00:00:00.000)\n * - Uses manual YYYY-MM-DD construction (avoids toISOString() timezone shift)\n * - Parses ISO dates to local timezone (avoids UTC interpretation)\n * - Handles month overflow correctly (Jan 31 + 1 month = Feb 28, not Mar 3)\n * - All comparisons work on normalized dates\n * \n * Limitations:\n * - No timezone awareness (all operations in local timezone)\n * - No DST-safe calculations across timezone changes\n * - For advanced timezone needs, use LuxonDateAdapter or DayJSDateAdapter\n * \n * Perfect for:\n * ✅ Most enterprise apps (ERP, POS, BI) where local timezone is sufficient\n * ✅ Applications without cross-timezone requirements\n * ✅ Minimizing bundle size (zero deps)\n * ✅ Simple, predictable date handling\n */\n\nimport { Injectable } from '@angular/core';\nimport { DateAdapter } from './date-adapter';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class NativeDateAdapter implements DateAdapter {\n  /**\n   * Normalize date to start of day (00:00:00.000) in local timezone\n   * \n   * This is the foundation of timezone-safe operations.\n   * All other methods use normalized dates for comparisons.\n   */\n  normalize(date: Date): Date {\n    const normalized = new Date(date);\n    normalized.setHours(0, 0, 0, 0);\n    return normalized;\n  }\n\n  /**\n   * Check if two dates are the same calendar day\n   * \n   * Implementation: Compare YYYY-MM-DD components directly\n   * Avoids timezone issues from valueOf() comparisons\n   */\n  isSameDay(a: Date, b: Date): boolean {\n    return (\n      a.getFullYear() === b.getFullYear() &&\n      a.getMonth() === b.getMonth() &&\n      a.getDate() === b.getDate()\n    );\n  }\n\n  /**\n   * Check if date A is before date B (calendar day level)\n   * \n   * Implementation: Compare normalized dates using valueOf()\n   */\n  isBeforeDay(a: Date, b: Date): boolean {\n    return this.normalize(a).valueOf() < this.normalize(b).valueOf();\n  }\n\n  /**\n   * Check if date A is after date B (calendar day level)\n   * \n   * Implementation: Compare normalized dates using valueOf()\n   */\n  isAfterDay(a: Date, b: Date): boolean {\n    return this.normalize(a).valueOf() > this.normalize(b).valueOf();\n  }\n\n  /**\n   * Add days to a date\n   * \n   * Implementation: Use setDate() which handles month rollover automatically\n   * \n   * Example:\n   * Jan 31 + 3 days → Feb 3 ✅\n   * Feb 28 + 1 day → Mar 1 ✅ (non-leap year)\n   */\n  addDays(date: Date, days: number): Date {\n    const result = new Date(date);\n    result.setDate(result.getDate() + days);\n    return this.normalize(result);\n  }\n\n  /**\n   * Add months to a date\n   * \n   * CRITICAL: Handles month overflow correctly\n   * \n   * Algorithm:\n   * 1. Add months using setMonth()\n   * 2. If day-of-month changed (overflow), set to last day of target month\n   * \n   * Examples:\n   * - Jan 31 + 1 month → Feb 28 (or Feb 29 in leap year) ✅\n   * - Jan 31 + 2 months → Mar 31 ✅\n   * - Mar 31 + 1 month → Apr 30 ✅\n   * - Dec 31 + 1 month → Jan 31 (next year) ✅\n   */\n  addMonths(date: Date, months: number): Date {\n    const result = new Date(date);\n    const originalDay = result.getDate();\n    \n    // Add months\n    result.setMonth(result.getMonth() + months);\n    \n    // Check for day overflow (e.g., Jan 31 → Feb 31 becomes Mar 3)\n    if (result.getDate() !== originalDay) {\n      // Overflow detected: set to last day of target month\n      // Go to 1st of next month, then subtract 1 day\n      result.setDate(0); // Sets to last day of previous month\n    }\n    \n    return this.normalize(result);\n  }\n\n  /**\n   * Get start of day (00:00:00.000)\n   * \n   * Alias for normalize() with explicit intent\n   */\n  startOfDay(date: Date): Date {\n    return this.normalize(date);\n  }\n\n  /**\n   * Get end of day (23:59:59.999)\n   * \n   * Useful for inclusive range queries\n   */\n  endOfDay(date: Date): Date {\n    const result = new Date(date);\n    result.setHours(23, 59, 59, 999);\n    return result;\n  }\n\n  /**\n   * Get first day of month (00:00:00.000)\n   */\n  startOfMonth(date: Date): Date {\n    const result = new Date(date);\n    result.setDate(1);\n    return this.normalize(result);\n  }\n\n  /**\n   * Get last day of month (23:59:59.999)\n   * \n   * Algorithm: Go to 1st of next month, subtract 1 day\n   */\n  endOfMonth(date: Date): Date {\n    const result = new Date(date);\n    result.setMonth(result.getMonth() + 1, 0); // Day 0 = last day of previous month\n    return this.endOfDay(result);\n  }\n\n  /**\n   * Get year (4-digit)\n   */\n  getYear(date: Date): number {\n    return date.getFullYear();\n  }\n\n  /**\n   * Get month (0-11)\n   */\n  getMonth(date: Date): number {\n    return date.getMonth();\n  }\n\n  /**\n   * Get day of month (1-31)\n   */\n  getDate(date: Date): number {\n    return date.getDate();\n  }\n\n  /**\n   * Get day of week (0-6, Sunday=0)\n   */\n  getDay(date: Date): number {\n    return date.getDay();\n  }\n\n  /**\n   * Convert Date to ISO date string (YYYY-MM-DD)\n   * \n   * CRITICAL: DO NOT use toISOString() - it converts to UTC!\n   * \n   * Manual construction ensures local timezone is preserved:\n   * \n   * Example problem with toISOString():\n   * ```\n   * // Local timezone: GMT-6 (CST)\n   * const date = new Date('2026-02-21T23:00:00'); // 11 PM Feb 21 local\n   * \n   * // WRONG ❌\n   * date.toISOString().split('T')[0]\n   * // Returns \"2026-02-22\" (converted to UTC = Feb 22 05:00 AM)\n   * \n   * // CORRECT ✅\n   * toISODate(date)\n   * // Returns \"2026-02-21\" (local date preserved)\n   * ```\n   * \n   * Implementation: Build YYYY-MM-DD manually from local date components\n   */\n  toISODate(date: Date): string {\n    const year = date.getFullYear();\n    const month = String(date.getMonth() + 1).padStart(2, '0');\n    const day = String(date.getDate()).padStart(2, '0');\n    return `${year}-${month}-${day}`;\n  }\n\n  /**\n   * Parse ISO date string (YYYY-MM-DD) to Date\n   * \n   * CRITICAL: DO NOT use new Date(isoString) - may parse as UTC!\n   * \n   * Example problem with Date constructor:\n   * ```\n   * // Local timezone: GMT-6 (CST)\n   * \n   * // WRONG ❌\n   * new Date('2026-02-21')\n   * // Parsed as UTC: 2026-02-21T00:00:00Z\n   * // In local timezone: Feb 20, 2026 6:00 PM (previous day!)\n   * \n   * // CORRECT ✅\n   * parseISODate('2026-02-21')\n   * // Returns: 2026-02-21T00:00:00 local time\n   * ```\n   * \n   * Implementation: Parse components and construct Date in local timezone\n   */\n  parseISODate(isoDate: string): Date | null {\n    if (!isoDate || typeof isoDate !== 'string') {\n      return null;\n    }\n\n    // Match YYYY-MM-DD format\n    const match = /^(\\d{4})-(\\d{2})-(\\d{2})$/.exec(isoDate.trim());\n    \n    if (!match) {\n      return null;\n    }\n\n    const year = parseInt(match[1], 10);\n    const month = parseInt(match[2], 10) - 1; // 0-indexed\n    const day = parseInt(match[3], 10);\n\n    // Validate ranges\n    if (month < 0 || month > 11) {\n      return null;\n    }\n\n    if (day < 1 || day > 31) {\n      return null;\n    }\n\n    // Construct date in local timezone\n    const date = new Date(year, month, day);\n    \n    // Verify date is valid (e.g., Feb 31 would roll to Mar 3)\n    if (\n      date.getFullYear() !== year ||\n      date.getMonth() !== month ||\n      date.getDate() !== day\n    ) {\n      return null;\n    }\n\n    return this.normalize(date);\n  }\n\n  /**\n   * Get week start day for locale\n   * \n   * Default: Sunday (0) for most locales\n   * Monday (1) for Europe, ISO 8601\n   * \n   * Implementation: Simple locale detection\n   * For advanced needs, use Intl.Locale or external library\n   */\n  getWeekStart(locale?: string): 0 | 1 | 2 | 3 | 4 | 5 | 6 {\n    if (!locale) {\n      locale = typeof navigator !== 'undefined' ? navigator.language : 'en-US';\n    }\n\n    // ISO 8601: Monday start\n    const mondayStartLocales = [\n      'en-GB', 'en-IE', 'en-AU', 'en-NZ', 'en-CA',\n      'es', 'es-ES', 'es-MX',\n      'fr', 'fr-FR', 'fr-CA',\n      'de', 'de-DE', 'de-AT', 'de-CH',\n      'it', 'it-IT',\n      'pt', 'pt-PT', 'pt-BR',\n      'nl', 'nl-NL', 'nl-BE',\n      'ru', 'ru-RU',\n      'zh', 'zh-CN', 'zh-TW',\n      'ja', 'ja-JP',\n      'ko', 'ko-KR'\n    ];\n\n    const normalizedLocale = locale.toLowerCase();\n    const startsWithMonday = mondayStartLocales.some(loc => \n      normalizedLocale.startsWith(loc.toLowerCase())\n    );\n\n    return startsWithMonday ? 1 : 0;\n  }\n}\n"]}
|
|
@@ -0,0 +1,243 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Provider Functions for Built-in Presets
|
|
3
|
+
*
|
|
4
|
+
* Version: 3.6.0
|
|
5
|
+
*
|
|
6
|
+
* Automatic registration of built-in date range presets.
|
|
7
|
+
* These providers ensure backward compatibility by auto-registering
|
|
8
|
+
* all standard presets (TODAY, LAST_7_DAYS, THIS_MONTH, etc.)
|
|
9
|
+
*
|
|
10
|
+
* USAGE IN LIBRARY (Internal):
|
|
11
|
+
* Built-in presets are registered automatically via Angular providers.
|
|
12
|
+
* Library consumers don't need to do anything.
|
|
13
|
+
*
|
|
14
|
+
* USAGE IN APP (Custom Presets):
|
|
15
|
+
* ```typescript
|
|
16
|
+
* // app.config.ts
|
|
17
|
+
* export const appConfig: ApplicationConfig = {
|
|
18
|
+
* providers: [
|
|
19
|
+
* // ... other providers
|
|
20
|
+
* provideCustomPresets([
|
|
21
|
+
* {
|
|
22
|
+
* key: 'THIS_FISCAL_QUARTER',
|
|
23
|
+
* resolve: (clock, adapter) => {
|
|
24
|
+
* const now = clock.now();
|
|
25
|
+
* // ... fiscal logic
|
|
26
|
+
* return { start, end };
|
|
27
|
+
* }
|
|
28
|
+
* }
|
|
29
|
+
* ])
|
|
30
|
+
* ]
|
|
31
|
+
* };
|
|
32
|
+
* ```
|
|
33
|
+
*/
|
|
34
|
+
import { APP_INITIALIZER, makeEnvironmentProviders } from '@angular/core';
|
|
35
|
+
import { PresetRegistry } from './preset-registry';
|
|
36
|
+
import { BUILT_IN_PRESETS } from './built-in-presets';
|
|
37
|
+
/**
|
|
38
|
+
* Initializer function that registers built-in presets
|
|
39
|
+
*
|
|
40
|
+
* Runs at application startup (APP_INITIALIZER)
|
|
41
|
+
*
|
|
42
|
+
* @param registry - PresetRegistry instance
|
|
43
|
+
* @returns Initialization function
|
|
44
|
+
*/
|
|
45
|
+
function initializeBuiltInPresets(registry) {
|
|
46
|
+
return () => {
|
|
47
|
+
// Register all built-in presets
|
|
48
|
+
BUILT_IN_PRESETS.forEach(preset => {
|
|
49
|
+
registry.register(preset);
|
|
50
|
+
});
|
|
51
|
+
// Log registration for debugging (can be removed in production)
|
|
52
|
+
if (typeof console !== 'undefined' && console.debug) {
|
|
53
|
+
console.debug(`[ng-dual-datepicker] Registered ${BUILT_IN_PRESETS.length} built-in presets:`, BUILT_IN_PRESETS.map(p => p.key).join(', '));
|
|
54
|
+
}
|
|
55
|
+
};
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Provide built-in date range presets
|
|
59
|
+
*
|
|
60
|
+
* This provider is automatically included in the library's root providers.
|
|
61
|
+
* Library consumers don't need to add this manually.
|
|
62
|
+
*
|
|
63
|
+
* @returns EnvironmentProviders for built-in presets
|
|
64
|
+
*
|
|
65
|
+
* @internal
|
|
66
|
+
*/
|
|
67
|
+
export function provideBuiltInPresets() {
|
|
68
|
+
return makeEnvironmentProviders([
|
|
69
|
+
{
|
|
70
|
+
provide: APP_INITIALIZER,
|
|
71
|
+
multi: true,
|
|
72
|
+
useFactory: initializeBuiltInPresets,
|
|
73
|
+
deps: [PresetRegistry]
|
|
74
|
+
}
|
|
75
|
+
]);
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Provide custom date range presets
|
|
79
|
+
*
|
|
80
|
+
* Use this to register your own industry-specific presets:
|
|
81
|
+
* - Fiscal presets
|
|
82
|
+
* - Hotel/hospitality presets
|
|
83
|
+
* - Logistics presets
|
|
84
|
+
* - Custom business logic
|
|
85
|
+
*
|
|
86
|
+
* @param presets - Array of custom RangePresetPlugin implementations
|
|
87
|
+
* @returns EnvironmentProviders for custom presets
|
|
88
|
+
*
|
|
89
|
+
* @example
|
|
90
|
+
* ```typescript
|
|
91
|
+
* // Fiscal presets
|
|
92
|
+
* const FISCAL_PRESETS: RangePresetPlugin[] = [
|
|
93
|
+
* {
|
|
94
|
+
* key: 'THIS_FISCAL_QUARTER',
|
|
95
|
+
* resolve: (clock, adapter) => {
|
|
96
|
+
* const now = clock.now();
|
|
97
|
+
* const month = adapter.getMonth(now);
|
|
98
|
+
*
|
|
99
|
+
* // Fiscal year starts April (month 3)
|
|
100
|
+
* const fiscalMonth = (month + 9) % 12;
|
|
101
|
+
* const quarterStart = Math.floor(fiscalMonth / 3) * 3;
|
|
102
|
+
* const calendarMonth = (quarterStart - 9 + 12) % 12;
|
|
103
|
+
*
|
|
104
|
+
* const year = adapter.getYear(now);
|
|
105
|
+
* const fiscalYear = month < 3 ? year - 1 : year;
|
|
106
|
+
*
|
|
107
|
+
* const start = new Date(fiscalYear, calendarMonth, 1);
|
|
108
|
+
* const end = new Date(fiscalYear, calendarMonth + 3, 0);
|
|
109
|
+
*
|
|
110
|
+
* return {
|
|
111
|
+
* start: adapter.normalize(start),
|
|
112
|
+
* end: adapter.normalize(end)
|
|
113
|
+
* };
|
|
114
|
+
* }
|
|
115
|
+
* },
|
|
116
|
+
* {
|
|
117
|
+
* key: 'FISCAL_YEAR_TO_DATE',
|
|
118
|
+
* resolve: (clock, adapter) => {
|
|
119
|
+
* const now = clock.now();
|
|
120
|
+
* const month = adapter.getMonth(now);
|
|
121
|
+
* const year = adapter.getYear(now);
|
|
122
|
+
*
|
|
123
|
+
* // Fiscal year starts April 1
|
|
124
|
+
* const fiscalYearStart = month >= 3
|
|
125
|
+
* ? new Date(year, 3, 1)
|
|
126
|
+
* : new Date(year - 1, 3, 1);
|
|
127
|
+
*
|
|
128
|
+
* return {
|
|
129
|
+
* start: adapter.normalize(fiscalYearStart),
|
|
130
|
+
* end: adapter.normalize(now)
|
|
131
|
+
* };
|
|
132
|
+
* }
|
|
133
|
+
* }
|
|
134
|
+
* ];
|
|
135
|
+
*
|
|
136
|
+
* // In app.config.ts
|
|
137
|
+
* export const appConfig: ApplicationConfig = {
|
|
138
|
+
* providers: [
|
|
139
|
+
* provideCustomPresets(FISCAL_PRESETS)
|
|
140
|
+
* ]
|
|
141
|
+
* };
|
|
142
|
+
*
|
|
143
|
+
* // Use in components
|
|
144
|
+
* store.applyPreset('THIS_FISCAL_QUARTER');
|
|
145
|
+
* store.applyPreset('FISCAL_YEAR_TO_DATE');
|
|
146
|
+
* ```
|
|
147
|
+
*
|
|
148
|
+
* @example
|
|
149
|
+
* ```typescript
|
|
150
|
+
* // Hotel presets
|
|
151
|
+
* const HOTEL_PRESETS: RangePresetPlugin[] = [
|
|
152
|
+
* {
|
|
153
|
+
* key: 'CHECK_IN_WEEK',
|
|
154
|
+
* resolve: (clock, adapter) => {
|
|
155
|
+
* const now = clock.now();
|
|
156
|
+
* const dayOfWeek = adapter.getDayOfWeek(now);
|
|
157
|
+
*
|
|
158
|
+
* // Check-in week: Friday to Friday
|
|
159
|
+
* const daysToNextFriday = dayOfWeek <= 5
|
|
160
|
+
* ? 5 - dayOfWeek
|
|
161
|
+
* : 7 - dayOfWeek + 5;
|
|
162
|
+
*
|
|
163
|
+
* const nextFriday = adapter.addDays(now, daysToNextFriday);
|
|
164
|
+
* const followingFriday = adapter.addDays(nextFriday, 7);
|
|
165
|
+
*
|
|
166
|
+
* return { start: nextFriday, end: followingFriday };
|
|
167
|
+
* }
|
|
168
|
+
* },
|
|
169
|
+
* {
|
|
170
|
+
* key: 'NEXT_30_NIGHTS',
|
|
171
|
+
* resolve: (clock, adapter) => {
|
|
172
|
+
* const now = clock.now();
|
|
173
|
+
* const tomorrow = adapter.addDays(now, 1);
|
|
174
|
+
* const end = adapter.addDays(tomorrow, 30);
|
|
175
|
+
* return { start: tomorrow, end };
|
|
176
|
+
* }
|
|
177
|
+
* }
|
|
178
|
+
* ];
|
|
179
|
+
*
|
|
180
|
+
* providers: [provideCustomPresets(HOTEL_PRESETS)]
|
|
181
|
+
* ```
|
|
182
|
+
*/
|
|
183
|
+
export function provideCustomPresets(presets) {
|
|
184
|
+
return makeEnvironmentProviders([
|
|
185
|
+
{
|
|
186
|
+
provide: APP_INITIALIZER,
|
|
187
|
+
multi: true,
|
|
188
|
+
useFactory: (registry) => {
|
|
189
|
+
return () => {
|
|
190
|
+
presets.forEach(preset => {
|
|
191
|
+
registry.register(preset);
|
|
192
|
+
});
|
|
193
|
+
if (typeof console !== 'undefined' && console.debug) {
|
|
194
|
+
console.debug(`[ng-dual-datepicker] Registered ${presets.length} custom presets:`, presets.map(p => p.key).join(', '));
|
|
195
|
+
}
|
|
196
|
+
};
|
|
197
|
+
},
|
|
198
|
+
deps: [PresetRegistry]
|
|
199
|
+
}
|
|
200
|
+
]);
|
|
201
|
+
}
|
|
202
|
+
/**
|
|
203
|
+
* Provide preset package
|
|
204
|
+
*
|
|
205
|
+
* Convenience function for external preset packages.
|
|
206
|
+
*
|
|
207
|
+
* @param packageName - Name of the preset package (for logging)
|
|
208
|
+
* @param presets - Array of presets from the package
|
|
209
|
+
* @returns EnvironmentProviders
|
|
210
|
+
*
|
|
211
|
+
* @example
|
|
212
|
+
* ```typescript
|
|
213
|
+
* // @acme/fiscal-presets package
|
|
214
|
+
* export function provideFiscalPresets(): EnvironmentProviders {
|
|
215
|
+
* return providePresetPackage('@acme/fiscal-presets', FISCAL_PRESETS);
|
|
216
|
+
* }
|
|
217
|
+
*
|
|
218
|
+
* // In app
|
|
219
|
+
* import { provideFiscalPresets } from '@acme/fiscal-presets';
|
|
220
|
+
*
|
|
221
|
+
* providers: [provideFiscalPresets()]
|
|
222
|
+
* ```
|
|
223
|
+
*/
|
|
224
|
+
export function providePresetPackage(packageName, presets) {
|
|
225
|
+
return makeEnvironmentProviders([
|
|
226
|
+
{
|
|
227
|
+
provide: APP_INITIALIZER,
|
|
228
|
+
multi: true,
|
|
229
|
+
useFactory: (registry) => {
|
|
230
|
+
return () => {
|
|
231
|
+
presets.forEach(preset => {
|
|
232
|
+
registry.register(preset);
|
|
233
|
+
});
|
|
234
|
+
if (typeof console !== 'undefined' && console.debug) {
|
|
235
|
+
console.debug(`[${packageName}] Registered ${presets.length} presets:`, presets.map(p => p.key).join(', '));
|
|
236
|
+
}
|
|
237
|
+
};
|
|
238
|
+
},
|
|
239
|
+
deps: [PresetRegistry]
|
|
240
|
+
}
|
|
241
|
+
]);
|
|
242
|
+
}
|
|
243
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"preset-providers.js","sourceRoot":"","sources":["../../../src/core/preset-providers.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Provider Functions for Built-in Presets\n * \n * Version: 3.6.0\n * \n * Automatic registration of built-in date range presets.\n * These providers ensure backward compatibility by auto-registering\n * all standard presets (TODAY, LAST_7_DAYS, THIS_MONTH, etc.)\n * \n * USAGE IN LIBRARY (Internal):\n * Built-in presets are registered automatically via Angular providers.\n * Library consumers don't need to do anything.\n * \n * USAGE IN APP (Custom Presets):\n * ```typescript\n * // app.config.ts\n * export const appConfig: ApplicationConfig = {\n *   providers: [\n *     // ... other providers\n *     provideCustomPresets([\n *       {\n *         key: 'THIS_FISCAL_QUARTER',\n *         resolve: (clock, adapter) => {\n *           const now = clock.now();\n *           // ... fiscal logic\n *           return { start, end };\n *         }\n *       }\n *     ])\n *   ]\n * };\n * ```\n */\n\nimport { APP_INITIALIZER, EnvironmentProviders, makeEnvironmentProviders, inject } from '@angular/core';\nimport { PresetRegistry } from './preset-registry';\nimport { BUILT_IN_PRESETS } from './built-in-presets';\nimport { RangePresetPlugin } from './range-preset.plugin';\n\n/**\n * Initializer function that registers built-in presets\n * \n * Runs at application startup (APP_INITIALIZER)\n * \n * @param registry - PresetRegistry instance\n * @returns Initialization function\n */\nfunction initializeBuiltInPresets(registry: PresetRegistry): () => void {\n  return () => {\n    // Register all built-in presets\n    BUILT_IN_PRESETS.forEach(preset => {\n      registry.register(preset);\n    });\n\n    // Log registration for debugging (can be removed in production)\n    if (typeof console !== 'undefined' && console.debug) {\n      console.debug(\n        `[ng-dual-datepicker] Registered ${BUILT_IN_PRESETS.length} built-in presets:`,\n        BUILT_IN_PRESETS.map(p => p.key).join(', ')\n      );\n    }\n  };\n}\n\n/**\n * Provide built-in date range presets\n * \n * This provider is automatically included in the library's root providers.\n * Library consumers don't need to add this manually.\n * \n * @returns EnvironmentProviders for built-in presets\n * \n * @internal\n */\nexport function provideBuiltInPresets(): EnvironmentProviders {\n  return makeEnvironmentProviders([\n    {\n      provide: APP_INITIALIZER,\n      multi: true,\n      useFactory: initializeBuiltInPresets,\n      deps: [PresetRegistry]\n    }\n  ]);\n}\n\n/**\n * Provide custom date range presets\n * \n * Use this to register your own industry-specific presets:\n * - Fiscal presets\n * - Hotel/hospitality presets\n * - Logistics presets\n * - Custom business logic\n * \n * @param presets - Array of custom RangePresetPlugin implementations\n * @returns EnvironmentProviders for custom presets\n * \n * @example\n * ```typescript\n * // Fiscal presets\n * const FISCAL_PRESETS: RangePresetPlugin[] = [\n *   {\n *     key: 'THIS_FISCAL_QUARTER',\n *     resolve: (clock, adapter) => {\n *       const now = clock.now();\n *       const month = adapter.getMonth(now);\n *       \n *       // Fiscal year starts April (month 3)\n *       const fiscalMonth = (month + 9) % 12;\n *       const quarterStart = Math.floor(fiscalMonth / 3) * 3;\n *       const calendarMonth = (quarterStart - 9 + 12) % 12;\n *       \n *       const year = adapter.getYear(now);\n *       const fiscalYear = month < 3 ? year - 1 : year;\n *       \n *       const start = new Date(fiscalYear, calendarMonth, 1);\n *       const end = new Date(fiscalYear, calendarMonth + 3, 0);\n *       \n *       return {\n *         start: adapter.normalize(start),\n *         end: adapter.normalize(end)\n *       };\n *     }\n *   },\n *   {\n *     key: 'FISCAL_YEAR_TO_DATE',\n *     resolve: (clock, adapter) => {\n *       const now = clock.now();\n *       const month = adapter.getMonth(now);\n *       const year = adapter.getYear(now);\n *       \n *       // Fiscal year starts April 1\n *       const fiscalYearStart = month >= 3 \n *         ? new Date(year, 3, 1)\n *         : new Date(year - 1, 3, 1);\n *       \n *       return {\n *         start: adapter.normalize(fiscalYearStart),\n *         end: adapter.normalize(now)\n *       };\n *     }\n *   }\n * ];\n * \n * // In app.config.ts\n * export const appConfig: ApplicationConfig = {\n *   providers: [\n *     provideCustomPresets(FISCAL_PRESETS)\n *   ]\n * };\n * \n * // Use in components\n * store.applyPreset('THIS_FISCAL_QUARTER');\n * store.applyPreset('FISCAL_YEAR_TO_DATE');\n * ```\n * \n * @example\n * ```typescript\n * // Hotel presets\n * const HOTEL_PRESETS: RangePresetPlugin[] = [\n *   {\n *     key: 'CHECK_IN_WEEK',\n *     resolve: (clock, adapter) => {\n *       const now = clock.now();\n *       const dayOfWeek = adapter.getDayOfWeek(now);\n *       \n *       // Check-in week: Friday to Friday\n *       const daysToNextFriday = dayOfWeek <= 5 \n *         ? 5 - dayOfWeek \n *         : 7 - dayOfWeek + 5;\n *       \n *       const nextFriday = adapter.addDays(now, daysToNextFriday);\n *       const followingFriday = adapter.addDays(nextFriday, 7);\n *       \n *       return { start: nextFriday, end: followingFriday };\n *     }\n *   },\n *   {\n *     key: 'NEXT_30_NIGHTS',\n *     resolve: (clock, adapter) => {\n *       const now = clock.now();\n *       const tomorrow = adapter.addDays(now, 1);\n *       const end = adapter.addDays(tomorrow, 30);\n *       return { start: tomorrow, end };\n *     }\n *   }\n * ];\n * \n * providers: [provideCustomPresets(HOTEL_PRESETS)]\n * ```\n */\nexport function provideCustomPresets(presets: RangePresetPlugin[]): EnvironmentProviders {\n  return makeEnvironmentProviders([\n    {\n      provide: APP_INITIALIZER,\n      multi: true,\n      useFactory: (registry: PresetRegistry) => {\n        return () => {\n          presets.forEach(preset => {\n            registry.register(preset);\n          });\n\n          if (typeof console !== 'undefined' && console.debug) {\n            console.debug(\n              `[ng-dual-datepicker] Registered ${presets.length} custom presets:`,\n              presets.map(p => p.key).join(', ')\n            );\n          }\n        };\n      },\n      deps: [PresetRegistry]\n    }\n  ]);\n}\n\n/**\n * Provide preset package\n * \n * Convenience function for external preset packages.\n * \n * @param packageName - Name of the preset package (for logging)\n * @param presets - Array of presets from the package\n * @returns EnvironmentProviders\n * \n * @example\n * ```typescript\n * // @acme/fiscal-presets package\n * export function provideFiscalPresets(): EnvironmentProviders {\n *   return providePresetPackage('@acme/fiscal-presets', FISCAL_PRESETS);\n * }\n * \n * // In app\n * import { provideFiscalPresets } from '@acme/fiscal-presets';\n * \n * providers: [provideFiscalPresets()]\n * ```\n */\nexport function providePresetPackage(\n  packageName: string,\n  presets: RangePresetPlugin[]\n): EnvironmentProviders {\n  return makeEnvironmentProviders([\n    {\n      provide: APP_INITIALIZER,\n      multi: true,\n      useFactory: (registry: PresetRegistry) => {\n        return () => {\n          presets.forEach(preset => {\n            registry.register(preset);\n          });\n\n          if (typeof console !== 'undefined' && console.debug) {\n            console.debug(\n              `[${packageName}] Registered ${presets.length} presets:`,\n              presets.map(p => p.key).join(', ')\n            );\n          }\n        };\n      },\n      deps: [PresetRegistry]\n    }\n  ]);\n}\n"]}
|