@oneluiz/dual-datepicker 3.5.0 → 3.5.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 +51 -3
- 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 +4 -0
- package/core/native-date-adapter.d.ts +152 -0
- package/core/preset.engine.d.ts +40 -6
- package/core/system-clock.d.ts +13 -0
- package/dual-datepicker.component.d.ts +17 -11
- 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 +5 -1
- package/esm2022/core/native-date-adapter.mjs +286 -0
- package/esm2022/core/preset.engine.mjs +154 -60
- package/esm2022/core/system-clock.mjs +34 -0
- package/esm2022/dual-datepicker.component.mjs +239 -195
- package/fesm2022/oneluiz-dual-datepicker.mjs +2470 -1850
- 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"]}
|