@oneluiz/dual-datepicker 3.4.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 +134 -1
- package/core/date-adapter.d.ts +298 -0
- package/core/date-clock.d.ts +82 -0
- package/core/dual-date-range.store.d.ts +113 -0
- package/core/index.d.ts +11 -0
- package/core/native-date-adapter.d.ts +152 -0
- package/core/preset.engine.d.ts +88 -0
- package/core/range.validator.d.ts +37 -0
- 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 +329 -0
- package/esm2022/core/index.mjs +12 -0
- package/esm2022/core/native-date-adapter.mjs +286 -0
- package/esm2022/core/preset.engine.mjs +303 -0
- package/esm2022/core/range.validator.mjs +105 -0
- package/esm2022/core/system-clock.mjs +34 -0
- package/esm2022/dual-datepicker.component.mjs +239 -195
- package/esm2022/public-api.mjs +5 -2
- package/fesm2022/oneluiz-dual-datepicker.mjs +1431 -204
- package/fesm2022/oneluiz-dual-datepicker.mjs.map +1 -1
- package/package.json +5 -3
- package/public-api.d.ts +1 -0
|
@@ -0,0 +1,152 @@
|
|
|
1
|
+
import { DateAdapter } from './date-adapter';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export declare class NativeDateAdapter implements DateAdapter {
|
|
4
|
+
/**
|
|
5
|
+
* Normalize date to start of day (00:00:00.000) in local timezone
|
|
6
|
+
*
|
|
7
|
+
* This is the foundation of timezone-safe operations.
|
|
8
|
+
* All other methods use normalized dates for comparisons.
|
|
9
|
+
*/
|
|
10
|
+
normalize(date: Date): Date;
|
|
11
|
+
/**
|
|
12
|
+
* Check if two dates are the same calendar day
|
|
13
|
+
*
|
|
14
|
+
* Implementation: Compare YYYY-MM-DD components directly
|
|
15
|
+
* Avoids timezone issues from valueOf() comparisons
|
|
16
|
+
*/
|
|
17
|
+
isSameDay(a: Date, b: Date): boolean;
|
|
18
|
+
/**
|
|
19
|
+
* Check if date A is before date B (calendar day level)
|
|
20
|
+
*
|
|
21
|
+
* Implementation: Compare normalized dates using valueOf()
|
|
22
|
+
*/
|
|
23
|
+
isBeforeDay(a: Date, b: Date): boolean;
|
|
24
|
+
/**
|
|
25
|
+
* Check if date A is after date B (calendar day level)
|
|
26
|
+
*
|
|
27
|
+
* Implementation: Compare normalized dates using valueOf()
|
|
28
|
+
*/
|
|
29
|
+
isAfterDay(a: Date, b: Date): boolean;
|
|
30
|
+
/**
|
|
31
|
+
* Add days to a date
|
|
32
|
+
*
|
|
33
|
+
* Implementation: Use setDate() which handles month rollover automatically
|
|
34
|
+
*
|
|
35
|
+
* Example:
|
|
36
|
+
* Jan 31 + 3 days → Feb 3 ✅
|
|
37
|
+
* Feb 28 + 1 day → Mar 1 ✅ (non-leap year)
|
|
38
|
+
*/
|
|
39
|
+
addDays(date: Date, days: number): Date;
|
|
40
|
+
/**
|
|
41
|
+
* Add months to a date
|
|
42
|
+
*
|
|
43
|
+
* CRITICAL: Handles month overflow correctly
|
|
44
|
+
*
|
|
45
|
+
* Algorithm:
|
|
46
|
+
* 1. Add months using setMonth()
|
|
47
|
+
* 2. If day-of-month changed (overflow), set to last day of target month
|
|
48
|
+
*
|
|
49
|
+
* Examples:
|
|
50
|
+
* - Jan 31 + 1 month → Feb 28 (or Feb 29 in leap year) ✅
|
|
51
|
+
* - Jan 31 + 2 months → Mar 31 ✅
|
|
52
|
+
* - Mar 31 + 1 month → Apr 30 ✅
|
|
53
|
+
* - Dec 31 + 1 month → Jan 31 (next year) ✅
|
|
54
|
+
*/
|
|
55
|
+
addMonths(date: Date, months: number): Date;
|
|
56
|
+
/**
|
|
57
|
+
* Get start of day (00:00:00.000)
|
|
58
|
+
*
|
|
59
|
+
* Alias for normalize() with explicit intent
|
|
60
|
+
*/
|
|
61
|
+
startOfDay(date: Date): Date;
|
|
62
|
+
/**
|
|
63
|
+
* Get end of day (23:59:59.999)
|
|
64
|
+
*
|
|
65
|
+
* Useful for inclusive range queries
|
|
66
|
+
*/
|
|
67
|
+
endOfDay(date: Date): Date;
|
|
68
|
+
/**
|
|
69
|
+
* Get first day of month (00:00:00.000)
|
|
70
|
+
*/
|
|
71
|
+
startOfMonth(date: Date): Date;
|
|
72
|
+
/**
|
|
73
|
+
* Get last day of month (23:59:59.999)
|
|
74
|
+
*
|
|
75
|
+
* Algorithm: Go to 1st of next month, subtract 1 day
|
|
76
|
+
*/
|
|
77
|
+
endOfMonth(date: Date): Date;
|
|
78
|
+
/**
|
|
79
|
+
* Get year (4-digit)
|
|
80
|
+
*/
|
|
81
|
+
getYear(date: Date): number;
|
|
82
|
+
/**
|
|
83
|
+
* Get month (0-11)
|
|
84
|
+
*/
|
|
85
|
+
getMonth(date: Date): number;
|
|
86
|
+
/**
|
|
87
|
+
* Get day of month (1-31)
|
|
88
|
+
*/
|
|
89
|
+
getDate(date: Date): number;
|
|
90
|
+
/**
|
|
91
|
+
* Get day of week (0-6, Sunday=0)
|
|
92
|
+
*/
|
|
93
|
+
getDay(date: Date): number;
|
|
94
|
+
/**
|
|
95
|
+
* Convert Date to ISO date string (YYYY-MM-DD)
|
|
96
|
+
*
|
|
97
|
+
* CRITICAL: DO NOT use toISOString() - it converts to UTC!
|
|
98
|
+
*
|
|
99
|
+
* Manual construction ensures local timezone is preserved:
|
|
100
|
+
*
|
|
101
|
+
* Example problem with toISOString():
|
|
102
|
+
* ```
|
|
103
|
+
* // Local timezone: GMT-6 (CST)
|
|
104
|
+
* const date = new Date('2026-02-21T23:00:00'); // 11 PM Feb 21 local
|
|
105
|
+
*
|
|
106
|
+
* // WRONG ❌
|
|
107
|
+
* date.toISOString().split('T')[0]
|
|
108
|
+
* // Returns "2026-02-22" (converted to UTC = Feb 22 05:00 AM)
|
|
109
|
+
*
|
|
110
|
+
* // CORRECT ✅
|
|
111
|
+
* toISODate(date)
|
|
112
|
+
* // Returns "2026-02-21" (local date preserved)
|
|
113
|
+
* ```
|
|
114
|
+
*
|
|
115
|
+
* Implementation: Build YYYY-MM-DD manually from local date components
|
|
116
|
+
*/
|
|
117
|
+
toISODate(date: Date): string;
|
|
118
|
+
/**
|
|
119
|
+
* Parse ISO date string (YYYY-MM-DD) to Date
|
|
120
|
+
*
|
|
121
|
+
* CRITICAL: DO NOT use new Date(isoString) - may parse as UTC!
|
|
122
|
+
*
|
|
123
|
+
* Example problem with Date constructor:
|
|
124
|
+
* ```
|
|
125
|
+
* // Local timezone: GMT-6 (CST)
|
|
126
|
+
*
|
|
127
|
+
* // WRONG ❌
|
|
128
|
+
* new Date('2026-02-21')
|
|
129
|
+
* // Parsed as UTC: 2026-02-21T00:00:00Z
|
|
130
|
+
* // In local timezone: Feb 20, 2026 6:00 PM (previous day!)
|
|
131
|
+
*
|
|
132
|
+
* // CORRECT ✅
|
|
133
|
+
* parseISODate('2026-02-21')
|
|
134
|
+
* // Returns: 2026-02-21T00:00:00 local time
|
|
135
|
+
* ```
|
|
136
|
+
*
|
|
137
|
+
* Implementation: Parse components and construct Date in local timezone
|
|
138
|
+
*/
|
|
139
|
+
parseISODate(isoDate: string): Date | null;
|
|
140
|
+
/**
|
|
141
|
+
* Get week start day for locale
|
|
142
|
+
*
|
|
143
|
+
* Default: Sunday (0) for most locales
|
|
144
|
+
* Monday (1) for Europe, ISO 8601
|
|
145
|
+
*
|
|
146
|
+
* Implementation: Simple locale detection
|
|
147
|
+
* For advanced needs, use Intl.Locale or external library
|
|
148
|
+
*/
|
|
149
|
+
getWeekStart(locale?: string): 0 | 1 | 2 | 3 | 4 | 5 | 6;
|
|
150
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<NativeDateAdapter, never>;
|
|
151
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<NativeDateAdapter>;
|
|
152
|
+
}
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
import * as i0 from "@angular/core";
|
|
2
|
+
export interface RangePreset {
|
|
3
|
+
/**
|
|
4
|
+
* Resolve preset to actual date range
|
|
5
|
+
* @param now - Current date for deterministic calculation
|
|
6
|
+
*/
|
|
7
|
+
resolve(now: Date): {
|
|
8
|
+
start: Date;
|
|
9
|
+
end: Date;
|
|
10
|
+
};
|
|
11
|
+
}
|
|
12
|
+
export interface PresetRange {
|
|
13
|
+
start: string;
|
|
14
|
+
end: string;
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Registry of built-in presets
|
|
18
|
+
* Can be extended by consumers
|
|
19
|
+
*
|
|
20
|
+
* SSR-Safe Architecture:
|
|
21
|
+
* - Injects DateClock via DI
|
|
22
|
+
* - All presets use clock.now() instead of new Date()
|
|
23
|
+
* - Deterministic: same clock.now() → same preset
|
|
24
|
+
* - Override DATE_CLOCK token in SSR to ensure consistency
|
|
25
|
+
*
|
|
26
|
+
* Timezone-Safe Architecture:
|
|
27
|
+
* - Injects DateAdapter via DI
|
|
28
|
+
* - All date operations use adapter methods
|
|
29
|
+
* - Prevents timezone bugs in cross-timezone scenarios
|
|
30
|
+
* - Override DATE_ADAPTER for Luxon/DayJS/custom implementations
|
|
31
|
+
*/
|
|
32
|
+
export declare class PresetEngine {
|
|
33
|
+
private presets;
|
|
34
|
+
private clock;
|
|
35
|
+
private adapter;
|
|
36
|
+
constructor();
|
|
37
|
+
/**
|
|
38
|
+
* Register a custom preset
|
|
39
|
+
*/
|
|
40
|
+
register(key: string, preset: RangePreset): void;
|
|
41
|
+
/**
|
|
42
|
+
* Resolve a preset to date range
|
|
43
|
+
*
|
|
44
|
+
* SSR Note: Uses injected DateClock for deterministic resolution
|
|
45
|
+
* Timezone Note: Uses injected DateAdapter for consistent date operations
|
|
46
|
+
*
|
|
47
|
+
* Override tokens in SSR scenarios:
|
|
48
|
+
* - DATE_CLOCK: Control current time
|
|
49
|
+
* - DATE_ADAPTER: Control date operations (e.g., Luxon for timezone support)
|
|
50
|
+
*
|
|
51
|
+
* @param key - Preset key (e.g., 'TODAY', 'LAST_7_DAYS')
|
|
52
|
+
* @param now - Optional override for current date (defaults to clock.now())
|
|
53
|
+
*/
|
|
54
|
+
resolve(key: string, now?: Date): PresetRange | null;
|
|
55
|
+
/**
|
|
56
|
+
* Get all available preset keys
|
|
57
|
+
*/
|
|
58
|
+
getPresetKeys(): string[];
|
|
59
|
+
/**
|
|
60
|
+
* Register all built-in presets
|
|
61
|
+
*
|
|
62
|
+
* All presets now use DateAdapter for timezone-safe operations
|
|
63
|
+
*/
|
|
64
|
+
private registerBuiltInPresets;
|
|
65
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<PresetEngine, never>;
|
|
66
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<PresetEngine>;
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Create a custom preset from a function
|
|
70
|
+
*/
|
|
71
|
+
export declare function createPreset(resolver: (now: Date) => {
|
|
72
|
+
start: Date;
|
|
73
|
+
end: Date;
|
|
74
|
+
}): RangePreset;
|
|
75
|
+
/**
|
|
76
|
+
* @deprecated Use dependency injection instead:
|
|
77
|
+
* ```typescript
|
|
78
|
+
* private engine = inject(PresetEngine);
|
|
79
|
+
* ```
|
|
80
|
+
*
|
|
81
|
+
* Singleton preset engine instance for backward compatibility
|
|
82
|
+
*
|
|
83
|
+
* WARNING: This singleton uses SystemClock directly and is NOT SSR-safe.
|
|
84
|
+
* For SSR applications, inject PresetEngine and override DATE_CLOCK token.
|
|
85
|
+
*
|
|
86
|
+
* This export will be removed in v4.0.0
|
|
87
|
+
*/
|
|
88
|
+
export declare const presetEngine: PresetEngine;
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pure validation functions for date ranges
|
|
3
|
+
* No dependencies, no side effects - just logic
|
|
4
|
+
* Perfect for SSR, testing, and reusability
|
|
5
|
+
*/
|
|
6
|
+
export interface ValidationResult {
|
|
7
|
+
valid: boolean;
|
|
8
|
+
error?: string;
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* Validate that end date is not before start date
|
|
12
|
+
*/
|
|
13
|
+
export declare function validateRangeOrder(start: Date | null, end: Date | null): ValidationResult;
|
|
14
|
+
/**
|
|
15
|
+
* Validate that date is within min/max bounds
|
|
16
|
+
*/
|
|
17
|
+
export declare function validateDateBounds(date: Date | null, minDate?: Date, maxDate?: Date): ValidationResult;
|
|
18
|
+
/**
|
|
19
|
+
* Validate that a range is within bounds
|
|
20
|
+
*/
|
|
21
|
+
export declare function validateRangeBounds(start: Date | null, end: Date | null, minDate?: Date, maxDate?: Date): ValidationResult;
|
|
22
|
+
/**
|
|
23
|
+
* Check if a date is disabled
|
|
24
|
+
*/
|
|
25
|
+
export declare function isDateDisabled(date: Date, disabledDates?: Date[] | ((date: Date) => boolean)): boolean;
|
|
26
|
+
/**
|
|
27
|
+
* Apply bounds to a date (clamp it)
|
|
28
|
+
*/
|
|
29
|
+
export declare function applyBounds(date: Date, minDate?: Date, maxDate?: Date): Date;
|
|
30
|
+
/**
|
|
31
|
+
* Parse ISO date string to Date object (deterministic)
|
|
32
|
+
*/
|
|
33
|
+
export declare function parseISODate(dateStr: string): Date | null;
|
|
34
|
+
/**
|
|
35
|
+
* Format Date to ISO string (YYYY-MM-DD)
|
|
36
|
+
*/
|
|
37
|
+
export declare function formatISODate(date: Date | null): string;
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { DateClock } from './date-clock';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export declare class SystemClock implements DateClock {
|
|
4
|
+
/**
|
|
5
|
+
* Returns current system time
|
|
6
|
+
*
|
|
7
|
+
* This is the standard behavior for client-side applications.
|
|
8
|
+
* For SSR, override DATE_CLOCK token with a fixed Date.
|
|
9
|
+
*/
|
|
10
|
+
now(): Date;
|
|
11
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<SystemClock, never>;
|
|
12
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<SystemClock>;
|
|
13
|
+
}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { EventEmitter, OnInit, OnChanges, SimpleChanges, ElementRef } from '@angular/core';
|
|
2
2
|
import { ControlValueAccessor } from '@angular/forms';
|
|
3
|
+
import { DualDateRangeStore } from './core/dual-date-range.store';
|
|
3
4
|
import * as i0 from "@angular/core";
|
|
4
5
|
export interface DateRange {
|
|
5
6
|
startDate: string;
|
|
@@ -30,8 +31,10 @@ export type ThemeType = 'default' | 'bootstrap' | 'bulma' | 'foundation' | 'tail
|
|
|
30
31
|
export declare class DualDatepickerComponent implements OnInit, OnChanges, ControlValueAccessor {
|
|
31
32
|
private elementRef;
|
|
32
33
|
placeholder: string;
|
|
33
|
-
startDate: string;
|
|
34
|
-
|
|
34
|
+
set startDate(value: string);
|
|
35
|
+
get startDate(): string;
|
|
36
|
+
set endDate(value: string);
|
|
37
|
+
get endDate(): string;
|
|
35
38
|
showPresets: boolean;
|
|
36
39
|
showClearButton: boolean;
|
|
37
40
|
multiRange: boolean;
|
|
@@ -61,24 +64,22 @@ export declare class DualDatepickerComponent implements OnInit, OnChanges, Contr
|
|
|
61
64
|
multiDateRangeChange: EventEmitter<MultiDateRange>;
|
|
62
65
|
multiDateRangeSelected: EventEmitter<MultiDateRange>;
|
|
63
66
|
private dateAdapter;
|
|
67
|
+
protected readonly rangeStore: DualDateRangeStore;
|
|
64
68
|
showDatePicker: import("@angular/core").WritableSignal<boolean>;
|
|
65
|
-
dateRangeText: import("@angular/core").WritableSignal<string>;
|
|
66
|
-
selectingStartDate: import("@angular/core").WritableSignal<boolean>;
|
|
67
69
|
currentMonth: import("@angular/core").WritableSignal<any>;
|
|
68
70
|
previousMonth: import("@angular/core").WritableSignal<any>;
|
|
69
71
|
currentMonthDays: import("@angular/core").WritableSignal<any[]>;
|
|
70
72
|
previousMonthDays: import("@angular/core").WritableSignal<any[]>;
|
|
71
73
|
isDisabled: import("@angular/core").WritableSignal<boolean>;
|
|
72
|
-
pendingStartDate: string;
|
|
73
|
-
pendingEndDate: string;
|
|
74
|
-
hasPendingChanges: import("@angular/core").WritableSignal<boolean>;
|
|
75
|
-
startHour: number;
|
|
76
|
-
startMinute: number;
|
|
77
|
-
endHour: number;
|
|
78
|
-
endMinute: number;
|
|
79
74
|
showStartTimePicker: import("@angular/core").WritableSignal<boolean>;
|
|
80
75
|
showEndTimePicker: import("@angular/core").WritableSignal<boolean>;
|
|
81
76
|
hoverDate: import("@angular/core").WritableSignal<string>;
|
|
77
|
+
get startHour(): number;
|
|
78
|
+
get startMinute(): number;
|
|
79
|
+
get endHour(): number;
|
|
80
|
+
get endMinute(): number;
|
|
81
|
+
private setStartHourMinute;
|
|
82
|
+
private setEndHourMinute;
|
|
82
83
|
selectedRanges: import("@angular/core").WritableSignal<DateRange[]>;
|
|
83
84
|
currentRangeIndex: import("@angular/core").WritableSignal<number>;
|
|
84
85
|
focusedDay: import("@angular/core").WritableSignal<{
|
|
@@ -88,6 +89,11 @@ export declare class DualDatepickerComponent implements OnInit, OnChanges, Contr
|
|
|
88
89
|
currentMonthName: import("@angular/core").Signal<string>;
|
|
89
90
|
previousMonthName: import("@angular/core").Signal<string>;
|
|
90
91
|
weekDayNames: import("@angular/core").Signal<string[]>;
|
|
92
|
+
dateRangeText: import("@angular/core").Signal<string>;
|
|
93
|
+
selectingStartDate: import("@angular/core").Signal<boolean>;
|
|
94
|
+
hasPendingChanges: import("@angular/core").Signal<boolean>;
|
|
95
|
+
get pendingStartDate(): string;
|
|
96
|
+
get pendingEndDate(): string;
|
|
91
97
|
private readonly defaultMonthNames;
|
|
92
98
|
private readonly defaultMonthNamesShort;
|
|
93
99
|
private readonly defaultDayNames;
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Date Adapter Abstraction for Timezone-Safe Date Operations
|
|
3
|
+
*
|
|
4
|
+
* Problem:
|
|
5
|
+
* Using Date natively in calendar/range logic causes enterprise bugs:
|
|
6
|
+
* - Date ranges shift by 1 day due to timezone/DST
|
|
7
|
+
* - Server (UTC) vs Client (local timezone) discrepancies
|
|
8
|
+
* - Inconsistent reporting in BI/ERP/invoicing/hotel systems
|
|
9
|
+
*
|
|
10
|
+
* Solution:
|
|
11
|
+
* All date calculations go through an adapter layer.
|
|
12
|
+
* This allows:
|
|
13
|
+
* - Timezone-safe operations by default (NativeDateAdapter normalizes to day boundaries)
|
|
14
|
+
* - Optional integration with Luxon/DayJS/date-fns for advanced timezone handling
|
|
15
|
+
* - Consistent behavior across SSR and client
|
|
16
|
+
* - Migration path to timezone-aware libraries without breaking changes
|
|
17
|
+
*
|
|
18
|
+
* Architecture:
|
|
19
|
+
* ```
|
|
20
|
+
* DualDateRangeStore
|
|
21
|
+
* ↓ uses
|
|
22
|
+
* DateAdapter ← Inject via DATE_ADAPTER token
|
|
23
|
+
* ↓ implements
|
|
24
|
+
* NativeDateAdapter (default, zero deps)
|
|
25
|
+
* LuxonDateAdapter (optional, full timezone support)
|
|
26
|
+
* DayJSDateAdapter (optional, lightweight)
|
|
27
|
+
* ```
|
|
28
|
+
*
|
|
29
|
+
* Usage:
|
|
30
|
+
* ```typescript
|
|
31
|
+
* // Default (NativeDateAdapter)
|
|
32
|
+
* bootstrapApplication(AppComponent);
|
|
33
|
+
*
|
|
34
|
+
* // Advanced (Luxon with timezone)
|
|
35
|
+
* import { LuxonDateAdapter } from '@oneluiz/dual-datepicker/luxon';
|
|
36
|
+
*
|
|
37
|
+
* bootstrapApplication(AppComponent, {
|
|
38
|
+
* providers: [
|
|
39
|
+
* { provide: DATE_ADAPTER, useClass: LuxonDateAdapter }
|
|
40
|
+
* ]
|
|
41
|
+
* });
|
|
42
|
+
* ```
|
|
43
|
+
*/
|
|
44
|
+
import { InjectionToken } from '@angular/core';
|
|
45
|
+
/**
|
|
46
|
+
* Injection token for DateAdapter
|
|
47
|
+
*
|
|
48
|
+
* Default: NativeDateAdapter (zero dependencies)
|
|
49
|
+
*
|
|
50
|
+
* Override for advanced timezone handling:
|
|
51
|
+
*
|
|
52
|
+
* ```typescript
|
|
53
|
+
* // Luxon with timezone
|
|
54
|
+
* import { LuxonDateAdapter } from '@oneluiz/dual-datepicker/luxon';
|
|
55
|
+
*
|
|
56
|
+
* bootstrapApplication(AppComponent, {
|
|
57
|
+
* providers: [
|
|
58
|
+
* {
|
|
59
|
+
* provide: DATE_ADAPTER,
|
|
60
|
+
* useClass: LuxonDateAdapter
|
|
61
|
+
* }
|
|
62
|
+
* ]
|
|
63
|
+
* });
|
|
64
|
+
* ```
|
|
65
|
+
*
|
|
66
|
+
* Custom implementation:
|
|
67
|
+
*
|
|
68
|
+
* ```typescript
|
|
69
|
+
* class CustomDateAdapter implements DateAdapter {
|
|
70
|
+
* // Your implementation for backend-specific date handling
|
|
71
|
+
* }
|
|
72
|
+
*
|
|
73
|
+
* provide(DATE_ADAPTER, { useClass: CustomDateAdapter });
|
|
74
|
+
* ```
|
|
75
|
+
*/
|
|
76
|
+
export const DATE_ADAPTER = new InjectionToken('DATE_ADAPTER');
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-adapter.js","sourceRoot":"","sources":["../../../src/core/date-adapter.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0CG;AAEH,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAiP/C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8BG;AACH,MAAM,CAAC,MAAM,YAAY,GAAG,IAAI,cAAc,CAAc,cAAc,CAAC,CAAC","sourcesContent":["/**\n * Date Adapter Abstraction for Timezone-Safe Date Operations\n * \n * Problem:\n * Using Date natively in calendar/range logic causes enterprise bugs:\n * - Date ranges shift by 1 day due to timezone/DST\n * - Server (UTC) vs Client (local timezone) discrepancies\n * - Inconsistent reporting in BI/ERP/invoicing/hotel systems\n * \n * Solution:\n * All date calculations go through an adapter layer.\n * This allows:\n * - Timezone-safe operations by default (NativeDateAdapter normalizes to day boundaries)\n * - Optional integration with Luxon/DayJS/date-fns for advanced timezone handling\n * - Consistent behavior across SSR and client\n * - Migration path to timezone-aware libraries without breaking changes\n * \n * Architecture:\n * ```\n * DualDateRangeStore\n *     ↓ uses\n * DateAdapter ← Inject via DATE_ADAPTER token\n *     ↓ implements\n * NativeDateAdapter (default, zero deps)\n * LuxonDateAdapter (optional, full timezone support)\n * DayJSDateAdapter (optional, lightweight)\n * ```\n * \n * Usage:\n * ```typescript\n * // Default (NativeDateAdapter)\n * bootstrapApplication(AppComponent);\n * \n * // Advanced (Luxon with timezone)\n * import { LuxonDateAdapter } from '@oneluiz/dual-datepicker/luxon';\n * \n * bootstrapApplication(AppComponent, {\n *   providers: [\n *     { provide: DATE_ADAPTER, useClass: LuxonDateAdapter }\n *   ]\n * });\n * ```\n */\n\nimport { InjectionToken } from '@angular/core';\n\n/**\n * Date adapter interface for all calendar/range operations\n * \n * All methods operate on \"calendar day\" level, ignoring time components.\n * Implementations must ensure timezone-safe behavior.\n * \n * Implementations:\n * - NativeDateAdapter: Default, zero dependencies, uses Date with normalization\n * - LuxonDateAdapter: Optional, full timezone support with Luxon\n * - DayJSDateAdapter: Optional, lightweight timezone support\n * - Custom: Implement for your specific backend/timezone requirements\n */\nexport interface DateAdapter {\n  /**\n   * Normalize date to start of day (00:00:00.000)\n   * \n   * Critical for timezone-safe comparisons.\n   * \n   * Example:\n   * ```typescript\n   * const date = new Date('2026-02-21T15:30:00');\n   * const normalized = adapter.normalize(date);\n   * // → 2026-02-21T00:00:00.000 (in local timezone)\n   * ```\n   * \n   * @param date - Date to normalize\n   * @returns Date with time set to 00:00:00.000\n   */\n  normalize(date: Date): Date;\n\n  /**\n   * Check if two dates represent the same calendar day\n   * \n   * Ignores time component. Timezone-safe.\n   * \n   * Example:\n   * ```typescript\n   * const a = new Date('2026-02-21T23:59:59');\n   * const b = new Date('2026-02-21T00:00:01');\n   * adapter.isSameDay(a, b); // → true\n   * ```\n   */\n  isSameDay(a: Date, b: Date): boolean;\n\n  /**\n   * Check if date A is before date B (calendar day level)\n   * \n   * Example:\n   * ```typescript\n   * adapter.isBeforeDay(new Date('2026-02-20'), new Date('2026-02-21')); // → true\n   * adapter.isBeforeDay(new Date('2026-02-21'), new Date('2026-02-21')); // → false\n   * ```\n   */\n  isBeforeDay(a: Date, b: Date): boolean;\n\n  /**\n   * Check if date A is after date B (calendar day level)\n   * \n   * Example:\n   * ```typescript\n   * adapter.isAfterDay(new Date('2026-02-22'), new Date('2026-02-21')); // → true\n   * adapter.isAfterDay(new Date('2026-02-21'), new Date('2026-02-21')); // → false\n   * ```\n   */\n  isAfterDay(a: Date, b: Date): boolean;\n\n  /**\n   * Add days to a date\n   * \n   * Must handle DST transitions correctly.\n   * \n   * Example:\n   * ```typescript\n   * const date = new Date('2026-02-21');\n   * const future = adapter.addDays(date, 7);\n   * // → 2026-02-28\n   * ```\n   */\n  addDays(date: Date, days: number): Date;\n\n  /**\n   * Add months to a date\n   * \n   * Must handle month overflow (e.g., Jan 31 + 1 month → Feb 28).\n   * \n   * Example:\n   * ```typescript\n   * const date = new Date('2026-01-31');\n   * const next = adapter.addMonths(date, 1);\n   * // → 2026-02-28 (not March 3rd)\n   * ```\n   */\n  addMonths(date: Date, months: number): Date;\n\n  /**\n   * Get start of day (00:00:00.000)\n   * \n   * Similar to normalize() but explicit intent.\n   */\n  startOfDay(date: Date): Date;\n\n  /**\n   * Get end of day (23:59:59.999)\n   * \n   * Useful for range queries that need to include entire day.\n   * \n   * Example:\n   * ```typescript\n   * const date = new Date('2026-02-21');\n   * const end = adapter.endOfDay(date);\n   * // → 2026-02-21T23:59:59.999\n   * ```\n   */\n  endOfDay(date: Date): Date;\n\n  /**\n   * Get first day of month (00:00:00.000)\n   * \n   * Example:\n   * ```typescript\n   * const date = new Date('2026-02-21');\n   * const start = adapter.startOfMonth(date);\n   * // → 2026-02-01T00:00:00.000\n   * ```\n   */\n  startOfMonth(date: Date): Date;\n\n  /**\n   * Get last day of month (23:59:59.999)\n   * \n   * Example:\n   * ```typescript\n   * const date = new Date('2026-02-21');\n   * const end = adapter.endOfMonth(date);\n   * // → 2026-02-28T23:59:59.999\n   * ```\n   */\n  endOfMonth(date: Date): Date;\n\n  /**\n   * Get year (4-digit)\n   * \n   * Example:\n   * ```typescript\n   * adapter.getYear(new Date('2026-02-21')); // → 2026\n   * ```\n   */\n  getYear(date: Date): number;\n\n  /**\n   * Get month (0-11, where 0 = January)\n   * \n   * Example:\n   * ```typescript\n   * adapter.getMonth(new Date('2026-02-21')); // → 1 (February)\n   * ```\n   */\n  getMonth(date: Date): number;\n\n  /**\n   * Get day of month (1-31)\n   * \n   * Example:\n   * ```typescript\n   * adapter.getDate(new Date('2026-02-21')); // → 21\n   * ```\n   */\n  getDate(date: Date): number;\n\n  /**\n   * Get day of week (0-6, where 0 = Sunday)\n   * \n   * Example:\n   * ```typescript\n   * adapter.getDay(new Date('2026-02-21')); // → 6 (Saturday)\n   * ```\n   */\n  getDay(date: Date): number;\n\n  /**\n   * Convert Date to ISO date string (YYYY-MM-DD)\n   * \n   * CRITICAL: Must be timezone-safe!\n   * DO NOT use date.toISOString() as it converts to UTC.\n   * \n   * Example:\n   * ```typescript\n   * // Local timezone GMT-6 (CST)\n   * const date = new Date('2026-02-21T23:00:00'); // 11 PM CST\n   * \n   * // ❌ WRONG: date.toISOString().split('T')[0]\n   * // Returns \"2026-02-22\" (shifted to UTC!)\n   * \n   * // ✅ CORRECT: adapter.toISODate(date)\n   * // Returns \"2026-02-21\" (local date preserved)\n   * ```\n   * \n   * @param date - Date to format (or null)\n   * @returns ISO date string in YYYY-MM-DD format (local timezone), empty string if null\n   */\n  toISODate(date: Date | null): string;\n\n  /**\n   * Parse ISO date string (YYYY-MM-DD) to Date\n   * \n   * CRITICAL: Must be timezone-safe!\n   * DO NOT use new Date(isoString) as it may parse as UTC.\n   * \n   * Example:\n   * ```typescript\n   * // ❌ WRONG: new Date('2026-02-21')\n   * // May parse as UTC midnight, which is previous day in some timezones\n   * \n   * // ✅ CORRECT: adapter.parseISODate('2026-02-21')\n   * // Returns Date representing 2026-02-21 00:00:00 in local timezone\n   * ```\n   * \n   * @param isoDate - ISO date string (YYYY-MM-DD) or null\n   * @returns Date object or null if invalid\n   */\n  parseISODate(isoDate: string | null): Date | null;\n\n  /**\n   * Get week start day for locale\n   * \n   * 0 = Sunday, 1 = Monday, etc.\n   * \n   * Example:\n   * ```typescript\n   * adapter.getWeekStart('en-US'); // → 0 (Sunday)\n   * adapter.getWeekStart('en-GB'); // → 1 (Monday)\n   * ```\n   * \n   * @param locale - Locale string (e.g., 'en-US', 'es-ES')\n   * @returns Day number (0-6)\n   */\n  getWeekStart(locale?: string): 0 | 1 | 2 | 3 | 4 | 5 | 6;\n}\n\n/**\n * Injection token for DateAdapter\n * \n * Default: NativeDateAdapter (zero dependencies)\n * \n * Override for advanced timezone handling:\n * \n * ```typescript\n * // Luxon with timezone\n * import { LuxonDateAdapter } from '@oneluiz/dual-datepicker/luxon';\n * \n * bootstrapApplication(AppComponent, {\n *   providers: [\n *     {\n *       provide: DATE_ADAPTER,\n *       useClass: LuxonDateAdapter\n *     }\n *   ]\n * });\n * ```\n * \n * Custom implementation:\n * \n * ```typescript\n * class CustomDateAdapter implements DateAdapter {\n *   // Your implementation for backend-specific date handling\n * }\n * \n * provide(DATE_ADAPTER, { useClass: CustomDateAdapter });\n * ```\n */\nexport const DATE_ADAPTER = new InjectionToken<DateAdapter>('DATE_ADAPTER');\n"]}
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Date Clock Abstraction for SSR-Safe Date Resolution
|
|
3
|
+
*
|
|
4
|
+
* Problem:
|
|
5
|
+
* Presets like "Last 7 Days" or "This Month" use new Date() which causes:
|
|
6
|
+
* - SSR hydration mismatch
|
|
7
|
+
* - Server renders "2026-02-14", client renders "2026-02-15"
|
|
8
|
+
* - Different filters in dashboards
|
|
9
|
+
* - Different queries in ERP/BI
|
|
10
|
+
* - Cache inconsistency
|
|
11
|
+
*
|
|
12
|
+
* Solution:
|
|
13
|
+
* Inject a DateClock to control time deterministically:
|
|
14
|
+
*
|
|
15
|
+
* Server (SSR):
|
|
16
|
+
* provide(DATE_CLOCK, {
|
|
17
|
+
* useValue: { now: () => new Date('2026-02-21T00:00:00Z') }
|
|
18
|
+
* })
|
|
19
|
+
*
|
|
20
|
+
* Client (Browser):
|
|
21
|
+
* Uses SystemClock by default (new Date())
|
|
22
|
+
*
|
|
23
|
+
* Testing:
|
|
24
|
+
* provide(DATE_CLOCK, {
|
|
25
|
+
* useValue: { now: () => new Date('2026-01-15T10:30:00Z') }
|
|
26
|
+
* })
|
|
27
|
+
*
|
|
28
|
+
* Architecture:
|
|
29
|
+
* - PresetEngine receives DateClock via DI
|
|
30
|
+
* - All preset calculations use clock.now() instead of new Date()
|
|
31
|
+
* - Deterministic: Same clock.now() → Same preset result
|
|
32
|
+
* - SSR-compatible: Server and client resolve identical ranges
|
|
33
|
+
*/
|
|
34
|
+
import { InjectionToken } from '@angular/core';
|
|
35
|
+
/**
|
|
36
|
+
* Injection token for DateClock
|
|
37
|
+
*
|
|
38
|
+
* Usage:
|
|
39
|
+
* ```typescript
|
|
40
|
+
* // Default (uses SystemClock)
|
|
41
|
+
* bootstrapApplication(AppComponent);
|
|
42
|
+
*
|
|
43
|
+
* // SSR Override
|
|
44
|
+
* bootstrapApplication(AppComponent, {
|
|
45
|
+
* providers: [
|
|
46
|
+
* {
|
|
47
|
+
* provide: DATE_CLOCK,
|
|
48
|
+
* useValue: { now: () => new Date('2026-02-21T00:00:00Z') }
|
|
49
|
+
* }
|
|
50
|
+
* ]
|
|
51
|
+
* });
|
|
52
|
+
*
|
|
53
|
+
* // Testing Override
|
|
54
|
+
* TestBed.configureTestingModule({
|
|
55
|
+
* providers: [
|
|
56
|
+
* {
|
|
57
|
+
* provide: DATE_CLOCK,
|
|
58
|
+
* useValue: { now: () => new Date('2026-01-15T12:00:00Z') }
|
|
59
|
+
* }
|
|
60
|
+
* ]
|
|
61
|
+
* });
|
|
62
|
+
* ```
|
|
63
|
+
*/
|
|
64
|
+
export const DATE_CLOCK = new InjectionToken('DATE_CLOCK');
|
|
65
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0ZS1jbG9jay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb3JlL2RhdGUtY2xvY2sudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBZ0NHO0FBRUgsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQXFCL0M7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0E0Qkc7QUFDSCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQUcsSUFBSSxjQUFjLENBQVksWUFBWSxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIERhdGUgQ2xvY2sgQWJzdHJhY3Rpb24gZm9yIFNTUi1TYWZlIERhdGUgUmVzb2x1dGlvblxuICogXG4gKiBQcm9ibGVtOlxuICogUHJlc2V0cyBsaWtlIFwiTGFzdCA3IERheXNcIiBvciBcIlRoaXMgTW9udGhcIiB1c2UgbmV3IERhdGUoKSB3aGljaCBjYXVzZXM6XG4gKiAtIFNTUiBoeWRyYXRpb24gbWlzbWF0Y2hcbiAqIC0gU2VydmVyIHJlbmRlcnMgXCIyMDI2LTAyLTE0XCIsIGNsaWVudCByZW5kZXJzIFwiMjAyNi0wMi0xNVwiXG4gKiAtIERpZmZlcmVudCBmaWx0ZXJzIGluIGRhc2hib2FyZHNcbiAqIC0gRGlmZmVyZW50IHF1ZXJpZXMgaW4gRVJQL0JJXG4gKiAtIENhY2hlIGluY29uc2lzdGVuY3lcbiAqIFxuICogU29sdXRpb246XG4gKiBJbmplY3QgYSBEYXRlQ2xvY2sgdG8gY29udHJvbCB0aW1lIGRldGVybWluaXN0aWNhbGx5OlxuICogXG4gKiBTZXJ2ZXIgKFNTUik6XG4gKiBwcm92aWRlKERBVEVfQ0xPQ0ssIHtcbiAqICAgdXNlVmFsdWU6IHsgbm93OiAoKSA9PiBuZXcgRGF0ZSgnMjAyNi0wMi0yMVQwMDowMDowMFonKSB9XG4gKiB9KVxuICogXG4gKiBDbGllbnQgKEJyb3dzZXIpOlxuICogVXNlcyBTeXN0ZW1DbG9jayBieSBkZWZhdWx0IChuZXcgRGF0ZSgpKVxuICogXG4gKiBUZXN0aW5nOlxuICogcHJvdmlkZShEQVRFX0NMT0NLLCB7XG4gKiAgIHVzZVZhbHVlOiB7IG5vdzogKCkgPT4gbmV3IERhdGUoJzIwMjYtMDEtMTVUMTA6MzA6MDBaJykgfVxuICogfSlcbiAqIFxuICogQXJjaGl0ZWN0dXJlOlxuICogLSBQcmVzZXRFbmdpbmUgcmVjZWl2ZXMgRGF0ZUNsb2NrIHZpYSBESVxuICogLSBBbGwgcHJlc2V0IGNhbGN1bGF0aW9ucyB1c2UgY2xvY2subm93KCkgaW5zdGVhZCBvZiBuZXcgRGF0ZSgpXG4gKiAtIERldGVybWluaXN0aWM6IFNhbWUgY2xvY2subm93KCkg4oaSIFNhbWUgcHJlc2V0IHJlc3VsdFxuICogLSBTU1ItY29tcGF0aWJsZTogU2VydmVyIGFuZCBjbGllbnQgcmVzb2x2ZSBpZGVudGljYWwgcmFuZ2VzXG4gKi9cblxuaW1wb3J0IHsgSW5qZWN0aW9uVG9rZW4gfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuLyoqXG4gKiBDbG9jayBhYnN0cmFjdGlvbiBmb3IgZGV0ZXJtaW5pc3RpYyBkYXRlIHJlc29sdXRpb25cbiAqIFxuICogVXNlIGNhc2VzOlxuICogLSBTU1I6IEVuc3VyZSBzZXJ2ZXIgYW5kIGNsaWVudCBnZW5lcmF0ZSBpZGVudGljYWwgcHJlc2V0c1xuICogLSBUZXN0aW5nOiBDb250cm9sIHRpbWUgZm9yIHByZWRpY3RhYmxlIHRlc3QgcmVzdWx0c1xuICogLSBSZXBsYXk6IFJlcHJvZHVjZSBleGFjdCB1c2VyIHN0YXRlIGZyb20gcGFzdCBzZXNzaW9uc1xuICogLSBEZW1vOiBGcmVlemUgdGltZSBmb3IgcmVwcm9kdWNpYmxlIGRlbW9zXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgRGF0ZUNsb2NrIHtcbiAgLyoqXG4gICAqIEdldCBjdXJyZW50IGRhdGUvdGltZVxuICAgKiBcbiAgICogRGVmYXVsdCBpbXBsZW1lbnRhdGlvbiByZXR1cm5zIG5ldyBEYXRlKClcbiAgICogT3ZlcnJpZGUgZm9yIFNTUiwgdGVzdGluZywgb3IgdGltZS10cmF2ZWwgc2NlbmFyaW9zXG4gICAqL1xuICBub3coKTogRGF0ZTtcbn1cblxuLyoqXG4gKiBJbmplY3Rpb24gdG9rZW4gZm9yIERhdGVDbG9ja1xuICogXG4gKiBVc2FnZTpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIERlZmF1bHQgKHVzZXMgU3lzdGVtQ2xvY2spXG4gKiBib290c3RyYXBBcHBsaWNhdGlvbihBcHBDb21wb25lbnQpO1xuICogXG4gKiAvLyBTU1IgT3ZlcnJpZGVcbiAqIGJvb3RzdHJhcEFwcGxpY2F0aW9uKEFwcENvbXBvbmVudCwge1xuICogICBwcm92aWRlcnM6IFtcbiAqICAgICB7XG4gKiAgICAgICBwcm92aWRlOiBEQVRFX0NMT0NLLFxuICogICAgICAgdXNlVmFsdWU6IHsgbm93OiAoKSA9PiBuZXcgRGF0ZSgnMjAyNi0wMi0yMVQwMDowMDowMFonKSB9XG4gKiAgICAgfVxuICogICBdXG4gKiB9KTtcbiAqIFxuICogLy8gVGVzdGluZyBPdmVycmlkZVxuICogVGVzdEJlZC5jb25maWd1cmVUZXN0aW5nTW9kdWxlKHtcbiAqICAgcHJvdmlkZXJzOiBbXG4gKiAgICAge1xuICogICAgICAgcHJvdmlkZTogREFURV9DTE9DSyxcbiAqICAgICAgIHVzZVZhbHVlOiB7IG5vdzogKCkgPT4gbmV3IERhdGUoJzIwMjYtMDEtMTVUMTI6MDA6MDBaJykgfVxuICogICAgIH1cbiAqICAgXVxuICogfSk7XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGNvbnN0IERBVEVfQ0xPQ0sgPSBuZXcgSW5qZWN0aW9uVG9rZW48RGF0ZUNsb2NrPignREFURV9DTE9DSycpO1xuIl19
|