@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.
@@ -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"]}