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