@oneluiz/dual-datepicker 3.4.0 → 3.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +86 -1
- package/core/dual-date-range.store.d.ts +104 -0
- package/core/index.d.ts +7 -0
- package/core/preset.engine.d.ts +54 -0
- package/core/range.validator.d.ts +37 -0
- package/esm2022/core/dual-date-range.store.mjs +293 -0
- package/esm2022/core/index.mjs +8 -0
- package/esm2022/core/preset.engine.mjs +209 -0
- package/esm2022/core/range.validator.mjs +105 -0
- package/esm2022/public-api.mjs +5 -2
- package/fesm2022/oneluiz-dual-datepicker.mjs +608 -1
- package/fesm2022/oneluiz-dual-datepicker.mjs.map +1 -1
- package/package.json +2 -2
- package/public-api.d.ts +1 -0
package/README.md
CHANGED
|
@@ -2,6 +2,8 @@
|
|
|
2
2
|
|
|
3
3
|
A lightweight, zero-dependency date range picker for Angular 17+. Built with standalone components, Reactive Forms, and Angular Signals. No Angular Material required.
|
|
4
4
|
|
|
5
|
+
> **🆕 NEW in v3.5.0**: [**Headless Architecture**](HEADLESS.md) - Use date range state WITHOUT the UI component. Perfect for SSR, services, and global dashboard filters! 🎯
|
|
6
|
+
|
|
5
7
|
[](https://www.npmjs.com/package/@oneluiz/dual-datepicker)
|
|
6
8
|
[](https://www.npmjs.com/package/@oneluiz/dual-datepicker)
|
|
7
9
|

|
|
@@ -17,6 +19,35 @@ npm install @oneluiz/dual-datepicker
|
|
|
17
19
|
|
|
18
20
|
---
|
|
19
21
|
|
|
22
|
+
## 🌟 What's New
|
|
23
|
+
|
|
24
|
+
### Headless Architecture (v3.5.0)
|
|
25
|
+
|
|
26
|
+
Use date range logic **without the UI component**:
|
|
27
|
+
|
|
28
|
+
```typescript
|
|
29
|
+
// Inject the store anywhere - no UI needed!
|
|
30
|
+
const rangeStore = inject(DualDateRangeStore);
|
|
31
|
+
|
|
32
|
+
// Apply preset
|
|
33
|
+
rangeStore.applyPreset('THIS_MONTH');
|
|
34
|
+
|
|
35
|
+
// Use in API calls
|
|
36
|
+
const range = rangeStore.range();
|
|
37
|
+
http.get(`/api/sales?start=${range.start}&end=${range.end}`);
|
|
38
|
+
```
|
|
39
|
+
|
|
40
|
+
**Perfect for:**
|
|
41
|
+
- 📊 Dashboard filters (control multiple charts)
|
|
42
|
+
- 🏢 SSR applications
|
|
43
|
+
- 🔄 Global state management
|
|
44
|
+
- 🎯 Service-layer filtering
|
|
45
|
+
- 📈 Analytics and BI tools
|
|
46
|
+
|
|
47
|
+
**[📖 Read the Headless Architecture Guide →](HEADLESS.md)**
|
|
48
|
+
|
|
49
|
+
---
|
|
50
|
+
|
|
20
51
|
## 📋 Table of Contents
|
|
21
52
|
|
|
22
53
|
- [Features](#-features)
|
|
@@ -26,6 +57,7 @@ npm install @oneluiz/dual-datepicker
|
|
|
26
57
|
- [Basic Usage](#basic-usage)
|
|
27
58
|
- [Reactive Forms](#with-reactive-forms)
|
|
28
59
|
- [Angular Signals](#with-angular-signals)
|
|
60
|
+
- [Headless Usage](#headless-usage-new) ⭐ NEW
|
|
29
61
|
- [Advanced Features](#-advanced-features)
|
|
30
62
|
- [Multi-Range Selection](#multi-range-support)
|
|
31
63
|
- [Disabled Dates](#disabled-dates)
|
|
@@ -169,6 +201,52 @@ dateRange = signal<DateRange | null>(null);
|
|
|
169
201
|
</ngx-dual-datepicker>
|
|
170
202
|
```
|
|
171
203
|
|
|
204
|
+
### Headless Usage (NEW) ⭐
|
|
205
|
+
|
|
206
|
+
**Use date range state WITHOUT the UI component** - perfect for SSR, services, and global filters!
|
|
207
|
+
|
|
208
|
+
```typescript
|
|
209
|
+
import { Component, inject } from '@angular/core';
|
|
210
|
+
import { DualDateRangeStore } from '@oneluiz/dual-datepicker';
|
|
211
|
+
import { HttpClient } from '@angular/common/http';
|
|
212
|
+
|
|
213
|
+
@Component({
|
|
214
|
+
template: `
|
|
215
|
+
<div class="dashboard">
|
|
216
|
+
<button (click)="setPreset('TODAY')">Today</button>
|
|
217
|
+
<button (click)="setPreset('THIS_MONTH')">This Month</button>
|
|
218
|
+
<p>{{ rangeText() }}</p>
|
|
219
|
+
</div>
|
|
220
|
+
`
|
|
221
|
+
})
|
|
222
|
+
export class DashboardComponent {
|
|
223
|
+
private rangeStore = inject(DualDateRangeStore);
|
|
224
|
+
private http = inject(HttpClient);
|
|
225
|
+
|
|
226
|
+
// Expose signals for template
|
|
227
|
+
rangeText = this.rangeStore.rangeText;
|
|
228
|
+
|
|
229
|
+
setPreset(key: string) {
|
|
230
|
+
this.rangeStore.applyPreset(key);
|
|
231
|
+
|
|
232
|
+
// Use in API call
|
|
233
|
+
const range = this.rangeStore.range();
|
|
234
|
+
this.http.get(`/api/sales`, {
|
|
235
|
+
params: { start: range.start, end: range.end }
|
|
236
|
+
}).subscribe(data => console.log(data));
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
```
|
|
240
|
+
|
|
241
|
+
**Benefits:**
|
|
242
|
+
- ✅ No UI component needed
|
|
243
|
+
- ✅ SSR-compatible
|
|
244
|
+
- ✅ Global state management
|
|
245
|
+
- ✅ Perfect for services and guards
|
|
246
|
+
- ✅ Testable and deterministic
|
|
247
|
+
|
|
248
|
+
**[📖 Full Headless Architecture Guide →](HEADLESS.md)** | **[💻 Code Examples →](HEADLESS_EXAMPLES.ts)**
|
|
249
|
+
|
|
172
250
|
---
|
|
173
251
|
|
|
174
252
|
## 🎯 Advanced Features
|
|
@@ -1998,6 +2076,12 @@ export class ExampleComponent {
|
|
|
1998
2076
|
|
|
1999
2077
|
Recently shipped:
|
|
2000
2078
|
|
|
2079
|
+
**v3.4.0:**
|
|
2080
|
+
- ✅ **Time Picker** - Select precise datetime ranges with 12h/24h format
|
|
2081
|
+
- ✅ **Configurable Minute Steps** - Choose 1, 5, 15, or 30-minute intervals
|
|
2082
|
+
- ✅ **Default Times** - Set default start/end times
|
|
2083
|
+
- ✅ **Full Theme Support** - Works seamlessly with all built-in themes
|
|
2084
|
+
|
|
2001
2085
|
**v3.3.0:**
|
|
2002
2086
|
- ✅ **Theming System** - Pre-built themes for Bootstrap, Bulma, Foundation, Tailwind CSS, and Custom
|
|
2003
2087
|
- ✅ **CSS Variables Support** - 13 customizable variables for branding
|
|
@@ -2025,9 +2109,10 @@ Recently shipped:
|
|
|
2025
2109
|
|
|
2026
2110
|
Planned features:
|
|
2027
2111
|
|
|
2028
|
-
- ⬜ **Time Picker** - Select date + time ranges
|
|
2029
2112
|
- ⬜ **Mobile Optimizations** - Enhanced touch gestures and responsive layout
|
|
2030
2113
|
- ⬜ **Range Shortcuts** - Quick selection buttons (Today, This Week, etc.)
|
|
2114
|
+
- ⬜ **Time Constraints** - Min/max time validation and business hours
|
|
2115
|
+
- ⬜ **Multi-range + Time Picker** - Combined support for multiple datetime ranges
|
|
2031
2116
|
|
|
2032
2117
|
## 📄 License
|
|
2033
2118
|
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import * as i0 from "@angular/core";
|
|
2
|
+
export interface DateRangeState {
|
|
3
|
+
start: string;
|
|
4
|
+
end: string;
|
|
5
|
+
startTime?: string;
|
|
6
|
+
endTime?: string;
|
|
7
|
+
}
|
|
8
|
+
export interface DateRangeConfig {
|
|
9
|
+
minDate?: Date;
|
|
10
|
+
maxDate?: Date;
|
|
11
|
+
disabledDates?: Date[] | ((date: Date) => boolean);
|
|
12
|
+
enableTimePicker?: boolean;
|
|
13
|
+
defaultStartTime?: string;
|
|
14
|
+
defaultEndTime?: string;
|
|
15
|
+
}
|
|
16
|
+
export declare class DualDateRangeStore {
|
|
17
|
+
private config;
|
|
18
|
+
private _startDate;
|
|
19
|
+
private _endDate;
|
|
20
|
+
private _leftMonth;
|
|
21
|
+
private _rightMonth;
|
|
22
|
+
private _selectingStart;
|
|
23
|
+
private _startTime;
|
|
24
|
+
private _endTime;
|
|
25
|
+
private _pendingStart;
|
|
26
|
+
private _pendingEnd;
|
|
27
|
+
private _hasPendingChanges;
|
|
28
|
+
readonly startDate: import("@angular/core").Signal<Date>;
|
|
29
|
+
readonly endDate: import("@angular/core").Signal<Date>;
|
|
30
|
+
readonly leftMonth: import("@angular/core").Signal<Date>;
|
|
31
|
+
readonly rightMonth: import("@angular/core").Signal<Date>;
|
|
32
|
+
readonly selectingStart: import("@angular/core").Signal<boolean>;
|
|
33
|
+
readonly startTime: import("@angular/core").Signal<string>;
|
|
34
|
+
readonly endTime: import("@angular/core").Signal<string>;
|
|
35
|
+
readonly hasPendingChanges: import("@angular/core").Signal<boolean>;
|
|
36
|
+
readonly range: import("@angular/core").Signal<DateRangeState>;
|
|
37
|
+
readonly isValid: import("@angular/core").Signal<boolean>;
|
|
38
|
+
readonly rangeText: import("@angular/core").Signal<string>;
|
|
39
|
+
/**
|
|
40
|
+
* Configure the store
|
|
41
|
+
*/
|
|
42
|
+
configure(config: Partial<DateRangeConfig>): void;
|
|
43
|
+
/**
|
|
44
|
+
* Set start date (with validation)
|
|
45
|
+
*/
|
|
46
|
+
setStart(date: Date | string | null): void;
|
|
47
|
+
/**
|
|
48
|
+
* Set end date (with validation)
|
|
49
|
+
*/
|
|
50
|
+
setEnd(date: Date | string | null): void;
|
|
51
|
+
/**
|
|
52
|
+
* Set complete range at once
|
|
53
|
+
*/
|
|
54
|
+
setRange(start: Date | string | null, end: Date | string | null): void;
|
|
55
|
+
/**
|
|
56
|
+
* Set pending selection (for requireApply mode)
|
|
57
|
+
*/
|
|
58
|
+
setPendingStart(date: Date | string | null): void;
|
|
59
|
+
setPendingEnd(date: Date | string | null): void;
|
|
60
|
+
/**
|
|
61
|
+
* Apply pending changes
|
|
62
|
+
*/
|
|
63
|
+
applyPending(): void;
|
|
64
|
+
/**
|
|
65
|
+
* Cancel pending changes
|
|
66
|
+
*/
|
|
67
|
+
cancelPending(): void;
|
|
68
|
+
private clearPending;
|
|
69
|
+
/**
|
|
70
|
+
* Reset to empty state
|
|
71
|
+
*/
|
|
72
|
+
reset(): void;
|
|
73
|
+
/**
|
|
74
|
+
* Apply a preset by key
|
|
75
|
+
*/
|
|
76
|
+
applyPreset(presetKey: string, now?: Date): void;
|
|
77
|
+
/**
|
|
78
|
+
* Navigate left calendar month
|
|
79
|
+
*/
|
|
80
|
+
setLeftMonth(date: Date): void;
|
|
81
|
+
/**
|
|
82
|
+
* Navigate right calendar month
|
|
83
|
+
*/
|
|
84
|
+
setRightMonth(date: Date): void;
|
|
85
|
+
/**
|
|
86
|
+
* Set time values
|
|
87
|
+
*/
|
|
88
|
+
setStartTime(time: string): void;
|
|
89
|
+
setEndTime(time: string): void;
|
|
90
|
+
/**
|
|
91
|
+
* Get current state as snapshot
|
|
92
|
+
*/
|
|
93
|
+
getSnapshot(): DateRangeState;
|
|
94
|
+
/**
|
|
95
|
+
* Load state from snapshot
|
|
96
|
+
*/
|
|
97
|
+
loadSnapshot(snapshot: DateRangeState): void;
|
|
98
|
+
private parseDate;
|
|
99
|
+
private getNextMonth;
|
|
100
|
+
private getPreviousMonth;
|
|
101
|
+
private formatDateShort;
|
|
102
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<DualDateRangeStore, never>;
|
|
103
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<DualDateRangeStore>;
|
|
104
|
+
}
|
package/core/index.d.ts
ADDED
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Headless Preset Engine
|
|
3
|
+
* Pure functions that resolve date ranges WITHOUT render dependency
|
|
4
|
+
* Perfect for SSR, global state, dashboard filters
|
|
5
|
+
*/
|
|
6
|
+
export interface RangePreset {
|
|
7
|
+
/**
|
|
8
|
+
* Resolve preset to actual date range
|
|
9
|
+
* @param now - Current date for deterministic calculation
|
|
10
|
+
*/
|
|
11
|
+
resolve(now: Date): {
|
|
12
|
+
start: Date;
|
|
13
|
+
end: Date;
|
|
14
|
+
};
|
|
15
|
+
}
|
|
16
|
+
export interface PresetRange {
|
|
17
|
+
start: string;
|
|
18
|
+
end: string;
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Registry of built-in presets
|
|
22
|
+
* Can be extended by consumers
|
|
23
|
+
*/
|
|
24
|
+
export declare class PresetEngine {
|
|
25
|
+
private presets;
|
|
26
|
+
constructor();
|
|
27
|
+
/**
|
|
28
|
+
* Register a custom preset
|
|
29
|
+
*/
|
|
30
|
+
register(key: string, preset: RangePreset): void;
|
|
31
|
+
/**
|
|
32
|
+
* Resolve a preset to date range
|
|
33
|
+
*/
|
|
34
|
+
resolve(key: string, now?: Date): PresetRange | null;
|
|
35
|
+
/**
|
|
36
|
+
* Get all available preset keys
|
|
37
|
+
*/
|
|
38
|
+
getPresetKeys(): string[];
|
|
39
|
+
/**
|
|
40
|
+
* Register all built-in presets
|
|
41
|
+
*/
|
|
42
|
+
private registerBuiltInPresets;
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Create a custom preset from a function
|
|
46
|
+
*/
|
|
47
|
+
export declare function createPreset(resolver: (now: Date) => {
|
|
48
|
+
start: Date;
|
|
49
|
+
end: Date;
|
|
50
|
+
}): RangePreset;
|
|
51
|
+
/**
|
|
52
|
+
* Singleton preset engine instance
|
|
53
|
+
*/
|
|
54
|
+
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,293 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Headless Date Range Store using Angular Signals
|
|
3
|
+
*
|
|
4
|
+
* Architecture:
|
|
5
|
+
* - State lives HERE, not in UI component
|
|
6
|
+
* - Deterministic: no hidden side effects
|
|
7
|
+
* - SSR-compatible: no window/document dependencies
|
|
8
|
+
* - Testable: pure signal-based state
|
|
9
|
+
* - Reusable: inject anywhere (services, components, guards)
|
|
10
|
+
*
|
|
11
|
+
* Usage:
|
|
12
|
+
* ```typescript
|
|
13
|
+
* // In component
|
|
14
|
+
* const rangeStore = inject(DualDateRangeStore);
|
|
15
|
+
* rangeStore.applyPreset('THIS_MONTH');
|
|
16
|
+
*
|
|
17
|
+
* // In service (headless!)
|
|
18
|
+
* const store = inject(DualDateRangeStore);
|
|
19
|
+
* const range = store.range();
|
|
20
|
+
* this.http.get(`/api/sales?start=${range.start}&end=${range.end}`);
|
|
21
|
+
* ```
|
|
22
|
+
*/
|
|
23
|
+
import { Injectable, signal, computed } from '@angular/core';
|
|
24
|
+
import { validateRangeOrder, validateRangeBounds, isDateDisabled, applyBounds, parseISODate, formatISODate } from './range.validator';
|
|
25
|
+
import { presetEngine } from './preset.engine';
|
|
26
|
+
import * as i0 from "@angular/core";
|
|
27
|
+
export class DualDateRangeStore {
|
|
28
|
+
// Configuration
|
|
29
|
+
config = signal({
|
|
30
|
+
enableTimePicker: false,
|
|
31
|
+
defaultStartTime: '00:00',
|
|
32
|
+
defaultEndTime: '23:59'
|
|
33
|
+
});
|
|
34
|
+
// Core state - using signals
|
|
35
|
+
_startDate = signal(null);
|
|
36
|
+
_endDate = signal(null);
|
|
37
|
+
_leftMonth = signal(new Date());
|
|
38
|
+
_rightMonth = signal(this.getNextMonth(new Date()));
|
|
39
|
+
_selectingStart = signal(true);
|
|
40
|
+
// Time state
|
|
41
|
+
_startTime = signal('00:00');
|
|
42
|
+
_endTime = signal('23:59');
|
|
43
|
+
// Pending state for requireApply mode
|
|
44
|
+
_pendingStart = signal(null);
|
|
45
|
+
_pendingEnd = signal(null);
|
|
46
|
+
_hasPendingChanges = signal(false);
|
|
47
|
+
// Public read-only signals
|
|
48
|
+
startDate = this._startDate.asReadonly();
|
|
49
|
+
endDate = this._endDate.asReadonly();
|
|
50
|
+
leftMonth = this._leftMonth.asReadonly();
|
|
51
|
+
rightMonth = this._rightMonth.asReadonly();
|
|
52
|
+
selectingStart = this._selectingStart.asReadonly();
|
|
53
|
+
startTime = this._startTime.asReadonly();
|
|
54
|
+
endTime = this._endTime.asReadonly();
|
|
55
|
+
hasPendingChanges = this._hasPendingChanges.asReadonly();
|
|
56
|
+
// Computed: ISO range for API consumption
|
|
57
|
+
range = computed(() => {
|
|
58
|
+
const start = this._startDate();
|
|
59
|
+
const end = this._endDate();
|
|
60
|
+
const cfg = this.config();
|
|
61
|
+
const result = {
|
|
62
|
+
start: formatISODate(start),
|
|
63
|
+
end: formatISODate(end)
|
|
64
|
+
};
|
|
65
|
+
if (cfg.enableTimePicker) {
|
|
66
|
+
result.startTime = this._startTime();
|
|
67
|
+
result.endTime = this._endTime();
|
|
68
|
+
}
|
|
69
|
+
return result;
|
|
70
|
+
});
|
|
71
|
+
// Computed: validation state
|
|
72
|
+
isValid = computed(() => {
|
|
73
|
+
const start = this._startDate();
|
|
74
|
+
const end = this._endDate();
|
|
75
|
+
const cfg = this.config();
|
|
76
|
+
const orderValidation = validateRangeOrder(start, end);
|
|
77
|
+
if (!orderValidation.valid)
|
|
78
|
+
return false;
|
|
79
|
+
const boundsValidation = validateRangeBounds(start, end, cfg.minDate, cfg.maxDate);
|
|
80
|
+
return boundsValidation.valid;
|
|
81
|
+
});
|
|
82
|
+
// Computed: range text for display
|
|
83
|
+
rangeText = computed(() => {
|
|
84
|
+
const start = this._startDate();
|
|
85
|
+
const end = this._endDate();
|
|
86
|
+
if (!start && !end)
|
|
87
|
+
return '';
|
|
88
|
+
if (!start)
|
|
89
|
+
return `? - ${this.formatDateShort(end)}`;
|
|
90
|
+
if (!end)
|
|
91
|
+
return `${this.formatDateShort(start)}`;
|
|
92
|
+
return `${this.formatDateShort(start)} - ${this.formatDateShort(end)}`;
|
|
93
|
+
});
|
|
94
|
+
/**
|
|
95
|
+
* Configure the store
|
|
96
|
+
*/
|
|
97
|
+
configure(config) {
|
|
98
|
+
this.config.update(current => ({ ...current, ...config }));
|
|
99
|
+
}
|
|
100
|
+
/**
|
|
101
|
+
* Set start date (with validation)
|
|
102
|
+
*/
|
|
103
|
+
setStart(date) {
|
|
104
|
+
const parsedDate = this.parseDate(date);
|
|
105
|
+
if (parsedDate) {
|
|
106
|
+
const cfg = this.config();
|
|
107
|
+
// Apply bounds
|
|
108
|
+
const boundedDate = applyBounds(parsedDate, cfg.minDate, cfg.maxDate);
|
|
109
|
+
// Check if disabled
|
|
110
|
+
if (isDateDisabled(boundedDate, cfg.disabledDates)) {
|
|
111
|
+
console.warn('Cannot select disabled date');
|
|
112
|
+
return;
|
|
113
|
+
}
|
|
114
|
+
this._startDate.set(boundedDate);
|
|
115
|
+
this._selectingStart.set(false);
|
|
116
|
+
// Auto-adjust end if it becomes invalid
|
|
117
|
+
const end = this._endDate();
|
|
118
|
+
if (end && end < boundedDate) {
|
|
119
|
+
this._endDate.set(null);
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
else {
|
|
123
|
+
this._startDate.set(null);
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Set end date (with validation)
|
|
128
|
+
*/
|
|
129
|
+
setEnd(date) {
|
|
130
|
+
const parsedDate = this.parseDate(date);
|
|
131
|
+
if (parsedDate) {
|
|
132
|
+
const cfg = this.config();
|
|
133
|
+
// Apply bounds
|
|
134
|
+
const boundedDate = applyBounds(parsedDate, cfg.minDate, cfg.maxDate);
|
|
135
|
+
// Check if disabled
|
|
136
|
+
if (isDateDisabled(boundedDate, cfg.disabledDates)) {
|
|
137
|
+
console.warn('Cannot select disabled date');
|
|
138
|
+
return;
|
|
139
|
+
}
|
|
140
|
+
// Validate order
|
|
141
|
+
const start = this._startDate();
|
|
142
|
+
if (start && boundedDate < start) {
|
|
143
|
+
console.warn('End date cannot be before start date');
|
|
144
|
+
return;
|
|
145
|
+
}
|
|
146
|
+
this._endDate.set(boundedDate);
|
|
147
|
+
this._selectingStart.set(true); // Ready for next selection
|
|
148
|
+
}
|
|
149
|
+
else {
|
|
150
|
+
this._endDate.set(null);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Set complete range at once
|
|
155
|
+
*/
|
|
156
|
+
setRange(start, end) {
|
|
157
|
+
this.setStart(start);
|
|
158
|
+
this.setEnd(end);
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Set pending selection (for requireApply mode)
|
|
162
|
+
*/
|
|
163
|
+
setPendingStart(date) {
|
|
164
|
+
const parsedDate = this.parseDate(date);
|
|
165
|
+
this._pendingStart.set(parsedDate);
|
|
166
|
+
this._hasPendingChanges.set(true);
|
|
167
|
+
}
|
|
168
|
+
setPendingEnd(date) {
|
|
169
|
+
const parsedDate = this.parseDate(date);
|
|
170
|
+
this._pendingEnd.set(parsedDate);
|
|
171
|
+
this._hasPendingChanges.set(true);
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* Apply pending changes
|
|
175
|
+
*/
|
|
176
|
+
applyPending() {
|
|
177
|
+
const pendingStart = this._pendingStart();
|
|
178
|
+
const pendingEnd = this._pendingEnd();
|
|
179
|
+
if (pendingStart)
|
|
180
|
+
this.setStart(pendingStart);
|
|
181
|
+
if (pendingEnd)
|
|
182
|
+
this.setEnd(pendingEnd);
|
|
183
|
+
this.clearPending();
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* Cancel pending changes
|
|
187
|
+
*/
|
|
188
|
+
cancelPending() {
|
|
189
|
+
this.clearPending();
|
|
190
|
+
}
|
|
191
|
+
clearPending() {
|
|
192
|
+
this._pendingStart.set(null);
|
|
193
|
+
this._pendingEnd.set(null);
|
|
194
|
+
this._hasPendingChanges.set(false);
|
|
195
|
+
}
|
|
196
|
+
/**
|
|
197
|
+
* Reset to empty state
|
|
198
|
+
*/
|
|
199
|
+
reset() {
|
|
200
|
+
this._startDate.set(null);
|
|
201
|
+
this._endDate.set(null);
|
|
202
|
+
this._selectingStart.set(true);
|
|
203
|
+
this.clearPending();
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Apply a preset by key
|
|
207
|
+
*/
|
|
208
|
+
applyPreset(presetKey, now = new Date()) {
|
|
209
|
+
const range = presetEngine.resolve(presetKey, now);
|
|
210
|
+
if (range) {
|
|
211
|
+
this.setRange(range.start, range.end);
|
|
212
|
+
}
|
|
213
|
+
else {
|
|
214
|
+
console.warn(`Preset "${presetKey}" not found`);
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* Navigate left calendar month
|
|
219
|
+
*/
|
|
220
|
+
setLeftMonth(date) {
|
|
221
|
+
this._leftMonth.set(date);
|
|
222
|
+
// Ensure right month is always after left
|
|
223
|
+
const right = this._rightMonth();
|
|
224
|
+
if (right <= date) {
|
|
225
|
+
this._rightMonth.set(this.getNextMonth(date));
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
/**
|
|
229
|
+
* Navigate right calendar month
|
|
230
|
+
*/
|
|
231
|
+
setRightMonth(date) {
|
|
232
|
+
this._rightMonth.set(date);
|
|
233
|
+
// Ensure left month is always before right
|
|
234
|
+
const left = this._leftMonth();
|
|
235
|
+
if (left >= date) {
|
|
236
|
+
this._leftMonth.set(this.getPreviousMonth(date));
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
/**
|
|
240
|
+
* Set time values
|
|
241
|
+
*/
|
|
242
|
+
setStartTime(time) {
|
|
243
|
+
this._startTime.set(time);
|
|
244
|
+
}
|
|
245
|
+
setEndTime(time) {
|
|
246
|
+
this._endTime.set(time);
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* Get current state as snapshot
|
|
250
|
+
*/
|
|
251
|
+
getSnapshot() {
|
|
252
|
+
return this.range();
|
|
253
|
+
}
|
|
254
|
+
/**
|
|
255
|
+
* Load state from snapshot
|
|
256
|
+
*/
|
|
257
|
+
loadSnapshot(snapshot) {
|
|
258
|
+
this.setRange(snapshot.start, snapshot.end);
|
|
259
|
+
if (snapshot.startTime) {
|
|
260
|
+
this.setStartTime(snapshot.startTime);
|
|
261
|
+
}
|
|
262
|
+
if (snapshot.endTime) {
|
|
263
|
+
this.setEndTime(snapshot.endTime);
|
|
264
|
+
}
|
|
265
|
+
}
|
|
266
|
+
// Helper methods
|
|
267
|
+
parseDate(date) {
|
|
268
|
+
if (!date)
|
|
269
|
+
return null;
|
|
270
|
+
if (date instanceof Date)
|
|
271
|
+
return date;
|
|
272
|
+
return parseISODate(date);
|
|
273
|
+
}
|
|
274
|
+
getNextMonth(date) {
|
|
275
|
+
return new Date(date.getFullYear(), date.getMonth() + 1, 1);
|
|
276
|
+
}
|
|
277
|
+
getPreviousMonth(date) {
|
|
278
|
+
return new Date(date.getFullYear(), date.getMonth() - 1, 1);
|
|
279
|
+
}
|
|
280
|
+
formatDateShort(date) {
|
|
281
|
+
const months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'];
|
|
282
|
+
return `${date.getDate()} ${months[date.getMonth()]}`;
|
|
283
|
+
}
|
|
284
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DualDateRangeStore, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
285
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DualDateRangeStore, providedIn: 'root' });
|
|
286
|
+
}
|
|
287
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DualDateRangeStore, decorators: [{
|
|
288
|
+
type: Injectable,
|
|
289
|
+
args: [{
|
|
290
|
+
providedIn: 'root'
|
|
291
|
+
}]
|
|
292
|
+
}] });
|
|
293
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dual-date-range.store.js","sourceRoot":"","sources":["../../../src/core/dual-date-range.store.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Headless Date Range Store using Angular Signals\n * \n * Architecture:\n * - State lives HERE, not in UI component\n * - Deterministic: no hidden side effects\n * - SSR-compatible: no window/document dependencies\n * - Testable: pure signal-based state\n * - Reusable: inject anywhere (services, components, guards)\n * \n * Usage:\n * ```typescript\n * // In component\n * const rangeStore = inject(DualDateRangeStore);\n * rangeStore.applyPreset('THIS_MONTH');\n * \n * // In service (headless!)\n * const store = inject(DualDateRangeStore);\n * const range = store.range();\n * this.http.get(`/api/sales?start=${range.start}&end=${range.end}`);\n * ```\n */\n\nimport { Injectable, signal, computed, effect, inject } from '@angular/core';\nimport { \n  validateRangeOrder, \n  validateRangeBounds, \n  isDateDisabled,\n  applyBounds,\n  parseISODate,\n  formatISODate\n} from './range.validator';\nimport { presetEngine, PresetRange } from './preset.engine';\n\nexport interface DateRangeState {\n  start: string; // ISO date\n  end: string; // ISO date\n  startTime?: string; // HH:mm\n  endTime?: string; // HH:mm\n}\n\nexport interface DateRangeConfig {\n  minDate?: Date;\n  maxDate?: Date;\n  disabledDates?: Date[] | ((date: Date) => boolean);\n  enableTimePicker?: boolean;\n  defaultStartTime?: string;\n  defaultEndTime?: string;\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class DualDateRangeStore {\n  // Configuration\n  private config = signal<DateRangeConfig>({\n    enableTimePicker: false,\n    defaultStartTime: '00:00',\n    defaultEndTime: '23:59'\n  });\n\n  // Core state - using signals\n  private _startDate = signal<Date | null>(null);\n  private _endDate = signal<Date | null>(null);\n  private _leftMonth = signal<Date>(new Date());\n  private _rightMonth = signal<Date>(this.getNextMonth(new Date()));\n  private _selectingStart = signal<boolean>(true);\n\n  // Time state\n  private _startTime = signal<string>('00:00');\n  private _endTime = signal<string>('23:59');\n\n  // Pending state for requireApply mode\n  private _pendingStart = signal<Date | null>(null);\n  private _pendingEnd = signal<Date | null>(null);\n  private _hasPendingChanges = signal<boolean>(false);\n\n  // Public read-only signals\n  readonly startDate = this._startDate.asReadonly();\n  readonly endDate = this._endDate.asReadonly();\n  readonly leftMonth = this._leftMonth.asReadonly();\n  readonly rightMonth = this._rightMonth.asReadonly();\n  readonly selectingStart = this._selectingStart.asReadonly();\n  readonly startTime = this._startTime.asReadonly();\n  readonly endTime = this._endTime.asReadonly();\n  readonly hasPendingChanges = this._hasPendingChanges.asReadonly();\n\n  // Computed: ISO range for API consumption\n  readonly range = computed<DateRangeState>(() => {\n    const start = this._startDate();\n    const end = this._endDate();\n    const cfg = this.config();\n\n    const result: DateRangeState = {\n      start: formatISODate(start),\n      end: formatISODate(end)\n    };\n\n    if (cfg.enableTimePicker) {\n      result.startTime = this._startTime();\n      result.endTime = this._endTime();\n    }\n\n    return result;\n  });\n\n  // Computed: validation state\n  readonly isValid = computed(() => {\n    const start = this._startDate();\n    const end = this._endDate();\n    const cfg = this.config();\n\n    const orderValidation = validateRangeOrder(start, end);\n    if (!orderValidation.valid) return false;\n\n    const boundsValidation = validateRangeBounds(\n      start,\n      end,\n      cfg.minDate,\n      cfg.maxDate\n    );\n    return boundsValidation.valid;\n  });\n\n  // Computed: range text for display\n  readonly rangeText = computed(() => {\n    const start = this._startDate();\n    const end = this._endDate();\n\n    if (!start && !end) return '';\n    if (!start) return `? - ${this.formatDateShort(end!)}`;\n    if (!end) return `${this.formatDateShort(start)}`;\n\n    return `${this.formatDateShort(start)} - ${this.formatDateShort(end)}`;\n  });\n\n  /**\n   * Configure the store\n   */\n  configure(config: Partial<DateRangeConfig>): void {\n    this.config.update(current => ({ ...current, ...config }));\n  }\n\n  /**\n   * Set start date (with validation)\n   */\n  setStart(date: Date | string | null): void {\n    const parsedDate = this.parseDate(date);\n    \n    if (parsedDate) {\n      const cfg = this.config();\n      \n      // Apply bounds\n      const boundedDate = applyBounds(parsedDate, cfg.minDate, cfg.maxDate);\n      \n      // Check if disabled\n      if (isDateDisabled(boundedDate, cfg.disabledDates)) {\n        console.warn('Cannot select disabled date');\n        return;\n      }\n\n      this._startDate.set(boundedDate);\n      this._selectingStart.set(false);\n\n      // Auto-adjust end if it becomes invalid\n      const end = this._endDate();\n      if (end && end < boundedDate) {\n        this._endDate.set(null);\n      }\n    } else {\n      this._startDate.set(null);\n    }\n  }\n\n  /**\n   * Set end date (with validation)\n   */\n  setEnd(date: Date | string | null): void {\n    const parsedDate = this.parseDate(date);\n    \n    if (parsedDate) {\n      const cfg = this.config();\n      \n      // Apply bounds\n      const boundedDate = applyBounds(parsedDate, cfg.minDate, cfg.maxDate);\n      \n      // Check if disabled\n      if (isDateDisabled(boundedDate, cfg.disabledDates)) {\n        console.warn('Cannot select disabled date');\n        return;\n      }\n\n      // Validate order\n      const start = this._startDate();\n      if (start && boundedDate < start) {\n        console.warn('End date cannot be before start date');\n        return;\n      }\n\n      this._endDate.set(boundedDate);\n      this._selectingStart.set(true); // Ready for next selection\n    } else {\n      this._endDate.set(null);\n    }\n  }\n\n  /**\n   * Set complete range at once\n   */\n  setRange(start: Date | string | null, end: Date | string | null): void {\n    this.setStart(start);\n    this.setEnd(end);\n  }\n\n  /**\n   * Set pending selection (for requireApply mode)\n   */\n  setPendingStart(date: Date | string | null): void {\n    const parsedDate = this.parseDate(date);\n    this._pendingStart.set(parsedDate);\n    this._hasPendingChanges.set(true);\n  }\n\n  setPendingEnd(date: Date | string | null): void {\n    const parsedDate = this.parseDate(date);\n    this._pendingEnd.set(parsedDate);\n    this._hasPendingChanges.set(true);\n  }\n\n  /**\n   * Apply pending changes\n   */\n  applyPending(): void {\n    const pendingStart = this._pendingStart();\n    const pendingEnd = this._pendingEnd();\n\n    if (pendingStart) this.setStart(pendingStart);\n    if (pendingEnd) this.setEnd(pendingEnd);\n\n    this.clearPending();\n  }\n\n  /**\n   * Cancel pending changes\n   */\n  cancelPending(): void {\n    this.clearPending();\n  }\n\n  private clearPending(): void {\n    this._pendingStart.set(null);\n    this._pendingEnd.set(null);\n    this._hasPendingChanges.set(false);\n  }\n\n  /**\n   * Reset to empty state\n   */\n  reset(): void {\n    this._startDate.set(null);\n    this._endDate.set(null);\n    this._selectingStart.set(true);\n    this.clearPending();\n  }\n\n  /**\n   * Apply a preset by key\n   */\n  applyPreset(presetKey: string, now: Date = new Date()): void {\n    const range = presetEngine.resolve(presetKey, now);\n    \n    if (range) {\n      this.setRange(range.start, range.end);\n    } else {\n      console.warn(`Preset \"${presetKey}\" not found`);\n    }\n  }\n\n  /**\n   * Navigate left calendar month\n   */\n  setLeftMonth(date: Date): void {\n    this._leftMonth.set(date);\n    \n    // Ensure right month is always after left\n    const right = this._rightMonth();\n    if (right <= date) {\n      this._rightMonth.set(this.getNextMonth(date));\n    }\n  }\n\n  /**\n   * Navigate right calendar month\n   */\n  setRightMonth(date: Date): void {\n    this._rightMonth.set(date);\n    \n    // Ensure left month is always before right\n    const left = this._leftMonth();\n    if (left >= date) {\n      this._leftMonth.set(this.getPreviousMonth(date));\n    }\n  }\n\n  /**\n   * Set time values\n   */\n  setStartTime(time: string): void {\n    this._startTime.set(time);\n  }\n\n  setEndTime(time: string): void {\n    this._endTime.set(time);\n  }\n\n  /**\n   * Get current state as snapshot\n   */\n  getSnapshot(): DateRangeState {\n    return this.range();\n  }\n\n  /**\n   * Load state from snapshot\n   */\n  loadSnapshot(snapshot: DateRangeState): void {\n    this.setRange(snapshot.start, snapshot.end);\n    \n    if (snapshot.startTime) {\n      this.setStartTime(snapshot.startTime);\n    }\n    \n    if (snapshot.endTime) {\n      this.setEndTime(snapshot.endTime);\n    }\n  }\n\n  // Helper methods\n  private parseDate(date: Date | string | null): Date | null {\n    if (!date) return null;\n    if (date instanceof Date) return date;\n    return parseISODate(date);\n  }\n\n  private getNextMonth(date: Date): Date {\n    return new Date(date.getFullYear(), date.getMonth() + 1, 1);\n  }\n\n  private getPreviousMonth(date: Date): Date {\n    return new Date(date.getFullYear(), date.getMonth() - 1, 1);\n  }\n\n  private formatDateShort(date: Date): string {\n    const months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'];\n    return `${date.getDate()} ${months[date.getMonth()]}`;\n  }\n}\n"]}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Core headless date range logic
|
|
3
|
+
* Import from here for clean barrel exports
|
|
4
|
+
*/
|
|
5
|
+
export * from './dual-date-range.store';
|
|
6
|
+
export * from './preset.engine';
|
|
7
|
+
export * from './range.validator';
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29yZS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxjQUFjLHlCQUF5QixDQUFDO0FBQ3hDLGNBQWMsaUJBQWlCLENBQUM7QUFDaEMsY0FBYyxtQkFBbUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ29yZSBoZWFkbGVzcyBkYXRlIHJhbmdlIGxvZ2ljXG4gKiBJbXBvcnQgZnJvbSBoZXJlIGZvciBjbGVhbiBiYXJyZWwgZXhwb3J0c1xuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vZHVhbC1kYXRlLXJhbmdlLnN0b3JlJztcbmV4cG9ydCAqIGZyb20gJy4vcHJlc2V0LmVuZ2luZSc7XG5leHBvcnQgKiBmcm9tICcuL3JhbmdlLnZhbGlkYXRvcic7XG4iXX0=
|