@oneluiz/dual-datepicker 4.0.0 → 4.0.2
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 +26 -2
- package/fesm2022/oneluiz-dual-datepicker.mjs +75 -71
- package/fesm2022/oneluiz-dual-datepicker.mjs.map +1 -1
- package/package.json +8 -9
- package/types/oneluiz-dual-datepicker.d.ts +1618 -0
- package/dual-datepicker.component.d.ts +0 -243
- package/esm2022/core/built-in-presets.mjs +0 -289
- package/esm2022/core/calendar-grid/cache.config.mjs +0 -35
- package/esm2022/core/calendar-grid/calendar-grid.cache.mjs +0 -98
- package/esm2022/core/calendar-grid/calendar-grid.factory.mjs +0 -97
- package/esm2022/core/calendar-grid/calendar-grid.types.mjs +0 -8
- package/esm2022/core/calendar-grid/range-highlighter.cache.mjs +0 -200
- package/esm2022/core/calendar-grid/range-highlighter.mjs +0 -185
- package/esm2022/core/calendar-grid/range-highlighter.types.mjs +0 -11
- package/esm2022/core/calendar-grid/virtual-weeks.logic.mjs +0 -149
- package/esm2022/core/calendar-grid/virtual-weeks.types.mjs +0 -11
- package/esm2022/core/date-adapter.mjs +0 -77
- package/esm2022/core/date-clock.mjs +0 -65
- package/esm2022/core/dual-date-range.store.mjs +0 -329
- package/esm2022/core/internal.mjs +0 -92
- package/esm2022/core/native-date-adapter.mjs +0 -286
- package/esm2022/core/preset-providers.mjs +0 -243
- package/esm2022/core/preset-registry.mjs +0 -277
- package/esm2022/core/preset.engine.mjs +0 -179
- package/esm2022/core/public.mjs +0 -94
- package/esm2022/core/range-preset.plugin.mjs +0 -70
- package/esm2022/core/range.validator.mjs +0 -105
- package/esm2022/core/system-clock.mjs +0 -34
- package/esm2022/core/testing/date.helpers.mjs +0 -49
- package/esm2022/core/testing/fixed-clock.mjs +0 -30
- package/esm2022/core/testing/index.mjs +0 -8
- package/esm2022/dual-datepicker.component.mjs +0 -1314
- package/esm2022/oneluiz-dual-datepicker.mjs +0 -5
- package/esm2022/public-api.mjs +0 -43
- package/index.d.ts +0 -5
- package/public-api.d.ts +0 -44
|
@@ -1,185 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Range Highlighter Service
|
|
3
|
-
*
|
|
4
|
-
* Decorates calendar grids with range highlights, hover previews, and disabled states.
|
|
5
|
-
* Pure computation layer - no caching (see RangeHighlighterCache for memoization).
|
|
6
|
-
*
|
|
7
|
-
* @module core/calendar-grid/range-highlighter
|
|
8
|
-
* @version 3.8.0
|
|
9
|
-
*/
|
|
10
|
-
import { Injectable, Inject } from '@angular/core';
|
|
11
|
-
import { DATE_ADAPTER } from '../date-adapter';
|
|
12
|
-
import * as i0 from "@angular/core";
|
|
13
|
-
/**
|
|
14
|
-
* Range Highlighter
|
|
15
|
-
*
|
|
16
|
-
* Applies decorations to calendar grids:
|
|
17
|
-
* - isSelectedStart, isSelectedEnd (range boundaries)
|
|
18
|
-
* - isInRange (cells within selected range)
|
|
19
|
-
* - isInHoverRange (hover preview)
|
|
20
|
-
* - isDisabled (constraints, custom predicates)
|
|
21
|
-
*
|
|
22
|
-
* Design:
|
|
23
|
-
* - Pure functions (same input = same output)
|
|
24
|
-
* - Uses DateAdapter for all date operations (SSR-safe)
|
|
25
|
-
* - No side effects, no state
|
|
26
|
-
* - Fast (~1ms for 42 cells on mobile)
|
|
27
|
-
*
|
|
28
|
-
* Usage:
|
|
29
|
-
* ```typescript
|
|
30
|
-
* const grid = calendarGridCache.get(monthDate, 0);
|
|
31
|
-
* const decorated = rangeHighlighter.decorate(grid, {
|
|
32
|
-
* start: startDate,
|
|
33
|
-
* end: endDate,
|
|
34
|
-
* hoverDate: '2026-01-20',
|
|
35
|
-
* disabledDates: [...]
|
|
36
|
-
* });
|
|
37
|
-
* ```
|
|
38
|
-
*/
|
|
39
|
-
export class RangeHighlighter {
|
|
40
|
-
adapter;
|
|
41
|
-
constructor(adapter) {
|
|
42
|
-
this.adapter = adapter;
|
|
43
|
-
}
|
|
44
|
-
/**
|
|
45
|
-
* Decorate calendar grid with range highlights
|
|
46
|
-
*
|
|
47
|
-
* @param grid Base calendar grid (from CalendarGridCache)
|
|
48
|
-
* @param params Decoration parameters (start, end, hover, disabled, etc.)
|
|
49
|
-
* @returns Decorated grid with computed properties
|
|
50
|
-
*/
|
|
51
|
-
decorate(grid, params) {
|
|
52
|
-
// Normalize dates (ensure start of day, consistent ISO strings)
|
|
53
|
-
const startISO = params.start ? this.adapter.toISODate(params.start) : null;
|
|
54
|
-
const endISO = params.end ? this.adapter.toISODate(params.end) : null;
|
|
55
|
-
const minISO = params.minDate ? this.adapter.toISODate(params.minDate) : null;
|
|
56
|
-
const maxISO = params.maxDate ? this.adapter.toISODate(params.maxDate) : null;
|
|
57
|
-
const hoverISO = params.hoverDate || null;
|
|
58
|
-
// Compute hover range boundaries (if applicable)
|
|
59
|
-
const hoverRange = this.computeHoverRange(startISO, hoverISO, params.selectingStartDate || false, params.multiRange || false);
|
|
60
|
-
// Decorate all cells
|
|
61
|
-
const decoratedCells = grid.cells.map((cell) => {
|
|
62
|
-
// Padding cells (previous/next month) get default decorations
|
|
63
|
-
if (!cell.inCurrentMonth) {
|
|
64
|
-
return {
|
|
65
|
-
...cell,
|
|
66
|
-
isSelectedStart: false,
|
|
67
|
-
isSelectedEnd: false,
|
|
68
|
-
isInRange: false,
|
|
69
|
-
isInHoverRange: false,
|
|
70
|
-
isDisabled: false
|
|
71
|
-
};
|
|
72
|
-
}
|
|
73
|
-
// Current month cells: apply full decoration logic
|
|
74
|
-
const cellISO = cell.iso;
|
|
75
|
-
return {
|
|
76
|
-
...cell,
|
|
77
|
-
isSelectedStart: startISO === cellISO,
|
|
78
|
-
isSelectedEnd: endISO === cellISO,
|
|
79
|
-
isInRange: this.isInRange(cellISO, startISO, endISO),
|
|
80
|
-
isInHoverRange: this.isInHoverRange(cellISO, hoverRange),
|
|
81
|
-
isDisabled: this.isDisabled(cell.date, minISO, maxISO, params.disabledDates)
|
|
82
|
-
};
|
|
83
|
-
});
|
|
84
|
-
// Organize cells into weeks (6 × 7)
|
|
85
|
-
const weeks = [];
|
|
86
|
-
for (let i = 0; i < decoratedCells.length; i += 7) {
|
|
87
|
-
weeks.push(decoratedCells.slice(i, i + 7));
|
|
88
|
-
}
|
|
89
|
-
return {
|
|
90
|
-
base: grid,
|
|
91
|
-
cells: decoratedCells,
|
|
92
|
-
weeks
|
|
93
|
-
};
|
|
94
|
-
}
|
|
95
|
-
/**
|
|
96
|
-
* Check if cell is within selected range
|
|
97
|
-
*
|
|
98
|
-
* @param cellISO Cell date (ISO format)
|
|
99
|
-
* @param startISO Start date (ISO or null)
|
|
100
|
-
* @param endISO End date (ISO or null)
|
|
101
|
-
* @returns True if cell is in range [start, end] (inclusive)
|
|
102
|
-
*/
|
|
103
|
-
isInRange(cellISO, startISO, endISO) {
|
|
104
|
-
if (!startISO || !endISO)
|
|
105
|
-
return false;
|
|
106
|
-
return cellISO >= startISO && cellISO <= endISO;
|
|
107
|
-
}
|
|
108
|
-
/**
|
|
109
|
-
* Check if cell is within hover preview range
|
|
110
|
-
*
|
|
111
|
-
* @param cellISO Cell date (ISO format)
|
|
112
|
-
* @param hoverRange Hover range boundaries (or null)
|
|
113
|
-
* @returns True if cell is in hover range
|
|
114
|
-
*/
|
|
115
|
-
isInHoverRange(cellISO, hoverRange) {
|
|
116
|
-
if (!hoverRange)
|
|
117
|
-
return false;
|
|
118
|
-
return cellISO >= hoverRange.min && cellISO <= hoverRange.max;
|
|
119
|
-
}
|
|
120
|
-
/**
|
|
121
|
-
* Check if cell is disabled
|
|
122
|
-
*
|
|
123
|
-
* @param date Cell date object
|
|
124
|
-
* @param minISO Minimum allowed date (ISO or null)
|
|
125
|
-
* @param maxISO Maximum allowed date (ISO or null)
|
|
126
|
-
* @param disabledDates Disabled dates (array, function, or null)
|
|
127
|
-
* @returns True if cell is disabled
|
|
128
|
-
*/
|
|
129
|
-
isDisabled(date, minISO, maxISO, disabledDates) {
|
|
130
|
-
const dateISO = this.adapter.toISODate(date);
|
|
131
|
-
// Check min/max constraints
|
|
132
|
-
if (minISO && dateISO < minISO)
|
|
133
|
-
return true;
|
|
134
|
-
if (maxISO && dateISO > maxISO)
|
|
135
|
-
return true;
|
|
136
|
-
// Check disabled dates
|
|
137
|
-
if (!disabledDates)
|
|
138
|
-
return false;
|
|
139
|
-
if (typeof disabledDates === 'function') {
|
|
140
|
-
// Custom predicate
|
|
141
|
-
return disabledDates(date);
|
|
142
|
-
}
|
|
143
|
-
else if (Array.isArray(disabledDates)) {
|
|
144
|
-
// Array of disabled dates (exact day match)
|
|
145
|
-
return disabledDates.some(disabledDate => {
|
|
146
|
-
return this.adapter.getYear(date) === this.adapter.getYear(disabledDate) &&
|
|
147
|
-
this.adapter.getMonth(date) === this.adapter.getMonth(disabledDate) &&
|
|
148
|
-
this.adapter.getDate(date) === this.adapter.getDate(disabledDate);
|
|
149
|
-
});
|
|
150
|
-
}
|
|
151
|
-
return false;
|
|
152
|
-
}
|
|
153
|
-
/**
|
|
154
|
-
* Compute hover preview range
|
|
155
|
-
*
|
|
156
|
-
* When user hovers over a date (and not selecting start):
|
|
157
|
-
* - Show preview range from start to hover
|
|
158
|
-
* - Range is always [min, max] where min <= max
|
|
159
|
-
*
|
|
160
|
-
* @param startISO Current start date (ISO or null)
|
|
161
|
-
* @param hoverISO Hovered date (ISO or null)
|
|
162
|
-
* @param selectingStart True if selecting start date (hover disabled)
|
|
163
|
-
* @param multiRange True if in multi-range mode
|
|
164
|
-
* @returns Hover range boundaries or null
|
|
165
|
-
*/
|
|
166
|
-
computeHoverRange(startISO, hoverISO, selectingStart, multiRange) {
|
|
167
|
-
// No hover preview when selecting start
|
|
168
|
-
if (selectingStart || !hoverISO || !startISO)
|
|
169
|
-
return null;
|
|
170
|
-
// Compute min/max (always normalized order)
|
|
171
|
-
const min = startISO < hoverISO ? startISO : hoverISO;
|
|
172
|
-
const max = startISO > hoverISO ? startISO : hoverISO;
|
|
173
|
-
return { min, max };
|
|
174
|
-
}
|
|
175
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RangeHighlighter, deps: [{ token: DATE_ADAPTER }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
176
|
-
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RangeHighlighter, providedIn: 'root' });
|
|
177
|
-
}
|
|
178
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RangeHighlighter, decorators: [{
|
|
179
|
-
type: Injectable,
|
|
180
|
-
args: [{ providedIn: 'root' }]
|
|
181
|
-
}], ctorParameters: () => [{ type: undefined, decorators: [{
|
|
182
|
-
type: Inject,
|
|
183
|
-
args: [DATE_ADAPTER]
|
|
184
|
-
}] }] });
|
|
185
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"range-highlighter.js","sourceRoot":"","sources":["../../../../src/core/calendar-grid/range-highlighter.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Range Highlighter Service\n * \n * Decorates calendar grids with range highlights, hover previews, and disabled states.\n * Pure computation layer - no caching (see RangeHighlighterCache for memoization).\n * \n * @module core/calendar-grid/range-highlighter\n * @version 3.8.0\n */\n\nimport { Injectable, Inject } from '@angular/core';\nimport { DateAdapter, DATE_ADAPTER } from '../date-adapter';\nimport { CalendarGrid, CalendarCell } from './calendar-grid.types';\nimport { DecoratedGrid, DecoratedCell, RangeDecorationParams } from './range-highlighter.types';\n\n/**\n * Range Highlighter\n * \n * Applies decorations to calendar grids:\n * - isSelectedStart, isSelectedEnd (range boundaries)\n * - isInRange (cells within selected range)\n * - isInHoverRange (hover preview)\n * - isDisabled (constraints, custom predicates)\n * \n * Design:\n * - Pure functions (same input = same output)\n * - Uses DateAdapter for all date operations (SSR-safe)\n * - No side effects, no state\n * - Fast (~1ms for 42 cells on mobile)\n * \n * Usage:\n * ```typescript\n * const grid = calendarGridCache.get(monthDate, 0);\n * const decorated = rangeHighlighter.decorate(grid, {\n *   start: startDate,\n *   end: endDate,\n *   hoverDate: '2026-01-20',\n *   disabledDates: [...]\n * });\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class RangeHighlighter {\n  constructor(\n    @Inject(DATE_ADAPTER) private adapter: DateAdapter\n  ) {}\n\n  /**\n   * Decorate calendar grid with range highlights\n   * \n   * @param grid Base calendar grid (from CalendarGridCache)\n   * @param params Decoration parameters (start, end, hover, disabled, etc.)\n   * @returns Decorated grid with computed properties\n   */\n  decorate(\n    grid: CalendarGrid,\n    params: RangeDecorationParams\n  ): DecoratedGrid {\n    // Normalize dates (ensure start of day, consistent ISO strings)\n    const startISO = params.start ? this.adapter.toISODate(params.start) : null;\n    const endISO = params.end ? this.adapter.toISODate(params.end) : null;\n    const minISO = params.minDate ? this.adapter.toISODate(params.minDate) : null;\n    const maxISO = params.maxDate ? this.adapter.toISODate(params.maxDate) : null;\n    const hoverISO = params.hoverDate || null;\n\n    // Compute hover range boundaries (if applicable)\n    const hoverRange = this.computeHoverRange(\n      startISO,\n      hoverISO,\n      params.selectingStartDate || false,\n      params.multiRange || false\n    );\n\n    // Decorate all cells\n    const decoratedCells = grid.cells.map((cell: CalendarCell): DecoratedCell => {\n      // Padding cells (previous/next month) get default decorations\n      if (!cell.inCurrentMonth) {\n        return {\n          ...cell,\n          isSelectedStart: false,\n          isSelectedEnd: false,\n          isInRange: false,\n          isInHoverRange: false,\n          isDisabled: false\n        };\n      }\n\n      // Current month cells: apply full decoration logic\n      const cellISO = cell.iso;\n\n      return {\n        ...cell,\n        isSelectedStart: startISO === cellISO,\n        isSelectedEnd: endISO === cellISO,\n        isInRange: this.isInRange(cellISO, startISO, endISO),\n        isInHoverRange: this.isInHoverRange(cellISO, hoverRange),\n        isDisabled: this.isDisabled(cell.date, minISO, maxISO, params.disabledDates)\n      };\n    });\n\n    // Organize cells into weeks (6 × 7)\n    const weeks: DecoratedCell[][] = [];\n    for (let i = 0; i < decoratedCells.length; i += 7) {\n      weeks.push(decoratedCells.slice(i, i + 7));\n    }\n\n    return {\n      base: grid,\n      cells: decoratedCells,\n      weeks\n    };\n  }\n\n  /**\n   * Check if cell is within selected range\n   * \n   * @param cellISO Cell date (ISO format)\n   * @param startISO Start date (ISO or null)\n   * @param endISO End date (ISO or null)\n   * @returns True if cell is in range [start, end] (inclusive)\n   */\n  private isInRange(\n    cellISO: string,\n    startISO: string | null,\n    endISO: string | null\n  ): boolean {\n    if (!startISO || !endISO) return false;\n    return cellISO >= startISO && cellISO <= endISO;\n  }\n\n  /**\n   * Check if cell is within hover preview range\n   * \n   * @param cellISO Cell date (ISO format)\n   * @param hoverRange Hover range boundaries (or null)\n   * @returns True if cell is in hover range\n   */\n  private isInHoverRange(\n    cellISO: string,\n    hoverRange: { min: string; max: string } | null\n  ): boolean {\n    if (!hoverRange) return false;\n    return cellISO >= hoverRange.min && cellISO <= hoverRange.max;\n  }\n\n  /**\n   * Check if cell is disabled\n   * \n   * @param date Cell date object\n   * @param minISO Minimum allowed date (ISO or null)\n   * @param maxISO Maximum allowed date (ISO or null)\n   * @param disabledDates Disabled dates (array, function, or null)\n   * @returns True if cell is disabled\n   */\n  private isDisabled(\n    date: Date,\n    minISO: string | null,\n    maxISO: string | null,\n    disabledDates?: Date[] | ((date: Date) => boolean) | null\n  ): boolean {\n    const dateISO = this.adapter.toISODate(date);\n\n    // Check min/max constraints\n    if (minISO && dateISO < minISO) return true;\n    if (maxISO && dateISO > maxISO) return true;\n\n    // Check disabled dates\n    if (!disabledDates) return false;\n\n    if (typeof disabledDates === 'function') {\n      // Custom predicate\n      return disabledDates(date);\n    } else if (Array.isArray(disabledDates)) {\n      // Array of disabled dates (exact day match)\n      return disabledDates.some(disabledDate => {\n        return this.adapter.getYear(date) === this.adapter.getYear(disabledDate) &&\n               this.adapter.getMonth(date) === this.adapter.getMonth(disabledDate) &&\n               this.adapter.getDate(date) === this.adapter.getDate(disabledDate);\n      });\n    }\n\n    return false;\n  }\n\n  /**\n   * Compute hover preview range\n   * \n   * When user hovers over a date (and not selecting start):\n   * - Show preview range from start to hover\n   * - Range is always [min, max] where min <= max\n   * \n   * @param startISO Current start date (ISO or null)\n   * @param hoverISO Hovered date (ISO or null)\n   * @param selectingStart True if selecting start date (hover disabled)\n   * @param multiRange True if in multi-range mode\n   * @returns Hover range boundaries or null\n   */\n  private computeHoverRange(\n    startISO: string | null,\n    hoverISO: string | null,\n    selectingStart: boolean,\n    multiRange: boolean\n  ): { min: string; max: string } | null {\n    // No hover preview when selecting start\n    if (selectingStart || !hoverISO || !startISO) return null;\n\n    // Compute min/max (always normalized order)\n    const min = startISO < hoverISO ? startISO : hoverISO;\n    const max = startISO > hoverISO ? startISO : hoverISO;\n\n    return { min, max };\n  }\n}\n"]}
|
|
@@ -1,11 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Range Highlighter Types
|
|
3
|
-
*
|
|
4
|
-
* Type definitions for calendar cell decorations with range highlighting.
|
|
5
|
-
* Used by RangeHighlighter to cache decorated grids and avoid recomputations.
|
|
6
|
-
*
|
|
7
|
-
* @module core/calendar-grid/range-highlighter.types
|
|
8
|
-
* @version 3.8.0
|
|
9
|
-
*/
|
|
10
|
-
export {};
|
|
11
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmFuZ2UtaGlnaGxpZ2h0ZXIudHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvY29yZS9jYWxlbmRhci1ncmlkL3JhbmdlLWhpZ2hsaWdodGVyLnR5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7OztHQVFHIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBSYW5nZSBIaWdobGlnaHRlciBUeXBlc1xuICogXG4gKiBUeXBlIGRlZmluaXRpb25zIGZvciBjYWxlbmRhciBjZWxsIGRlY29yYXRpb25zIHdpdGggcmFuZ2UgaGlnaGxpZ2h0aW5nLlxuICogVXNlZCBieSBSYW5nZUhpZ2hsaWdodGVyIHRvIGNhY2hlIGRlY29yYXRlZCBncmlkcyBhbmQgYXZvaWQgcmVjb21wdXRhdGlvbnMuXG4gKiBcbiAqIEBtb2R1bGUgY29yZS9jYWxlbmRhci1ncmlkL3JhbmdlLWhpZ2hsaWdodGVyLnR5cGVzXG4gKiBAdmVyc2lvbiAzLjguMFxuICovXG5cbmltcG9ydCB7IENhbGVuZGFyQ2VsbCwgQ2FsZW5kYXJHcmlkIH0gZnJvbSAnLi9jYWxlbmRhci1ncmlkLnR5cGVzJztcblxuLyoqXG4gKiBDYWxlbmRhciBjZWxsIHdpdGggcmFuZ2UgaGlnaGxpZ2h0IGRlY29yYXRpb25zXG4gKiBcbiAqIEV4dGVuZHMgYmFzZSBDYWxlbmRhckNlbGwgd2l0aCBjb21wdXRlZCBwcm9wZXJ0aWVzIGZvcjpcbiAqIC0gU3RhcnQvZW5kIG1hcmtlcnMgKHNlbGVjdGVkIHJhbmdlIGJvdW5kYXJpZXMpXG4gKiAtIFJhbmdlIG1lbWJlcnNoaXAgKGNlbGwgd2l0aGluIHNlbGVjdGVkIHJhbmdlKVxuICogLSBIb3ZlciBwcmV2aWV3ICh0ZW1wb3JhcnkgcmFuZ2UgZHVyaW5nIG1vdXNlIGhvdmVyKVxuICogLSBEaXNhYmxlZCBzdGF0ZSAoY29uc3RyYWludHMsIGN1c3RvbSBwcmVkaWNhdGVzKVxuICovXG5leHBvcnQgaW50ZXJmYWNlIERlY29yYXRlZENlbGwgZXh0ZW5kcyBDYWxlbmRhckNlbGwge1xuICAvKipcbiAgICogVHJ1ZSBpZiB0aGlzIGNlbGwgaXMgdGhlIHNlbGVjdGVkIHN0YXJ0IGRhdGVcbiAgICovXG4gIGlzU2VsZWN0ZWRTdGFydDogYm9vbGVhbjtcblxuICAvKipcbiAgICogVHJ1ZSBpZiB0aGlzIGNlbGwgaXMgdGhlIHNlbGVjdGVkIGVuZCBkYXRlXG4gICAqL1xuICBpc1NlbGVjdGVkRW5kOiBib29sZWFuO1xuXG4gIC8qKlxuICAgKiBUcnVlIGlmIHRoaXMgY2VsbCBpcyB3aXRoaW4gdGhlIHNlbGVjdGVkIHJhbmdlIChpbmNsdXNpdmUpXG4gICAqL1xuICBpc0luUmFuZ2U6IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIFRydWUgaWYgdGhpcyBjZWxsIGlzIHdpdGhpbiB0aGUgaG92ZXIgcHJldmlldyByYW5nZVxuICAgKi9cbiAgaXNJbkhvdmVyUmFuZ2U6IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIFRydWUgaWYgdGhpcyBjZWxsIGlzIGRpc2FibGVkICh2aWEgY29uc3RyYWludHMgb3IgY3VzdG9tIGxvZ2ljKVxuICAgKi9cbiAgaXNEaXNhYmxlZDogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBDYWxlbmRhciBncmlkIHdpdGggZGVjb3JhdGVkIGNlbGxzXG4gKiBcbiAqIENvbnRhaW5zOlxuICogLSBiYXNlOiBPcmlnaW5hbCBDYWxlbmRhckdyaWQgKGZyb20gQ2FsZW5kYXJHcmlkQ2FjaGUpXG4gKiAtIHdlZWtzOiAyRCBhcnJheSBvZiBkZWNvcmF0ZWQgY2VsbHMgKDYgw5cgNylcbiAqIC0gY2VsbHM6IEZsYXQgYXJyYXkgb2YgZGVjb3JhdGVkIGNlbGxzICg0MiBjZWxscylcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBEZWNvcmF0ZWRHcmlkIHtcbiAgLyoqXG4gICAqIEJhc2UgZ3JpZCBzdHJ1Y3R1cmUgKG1lbW9pemVkIGJ5IENhbGVuZGFyR3JpZENhY2hlKVxuICAgKi9cbiAgYmFzZTogQ2FsZW5kYXJHcmlkO1xuXG4gIC8qKlxuICAgKiBEZWNvcmF0ZWQgY2VsbHMgb3JnYW5pemVkIGFzIHdlZWtzICg2IHdlZWtzIMOXIDcgZGF5cylcbiAgICovXG4gIHdlZWtzOiBEZWNvcmF0ZWRDZWxsW11bXTtcblxuICAvKipcbiAgICogRGVjb3JhdGVkIGNlbGxzIGFzIGZsYXQgYXJyYXkgKDQyIGNlbGxzIGZvciBsYXlvdXQgc3RhYmlsaXR5KVxuICAgKi9cbiAgY2VsbHM6IERlY29yYXRlZENlbGxbXTtcbn1cblxuLyoqXG4gKiBQYXJhbWV0ZXJzIGZvciByYW5nZSBkZWNvcmF0aW9uXG4gKiBcbiAqIEFsbCBkYXRlcyBtdXN0IGJlIERhdGUgb2JqZWN0cyAocHJlLXBhcnNlZCBieSBjb21wb25lbnQpLlxuICogQ2FjaGUga2V5IGlzIGJ1aWx0IGZyb20gSVNPIGRhdGVzICh2aWEgRGF0ZUFkYXB0ZXIudG9JU09EYXRlKS5cbiAqIFxuICogQ29uc3RyYWludHM6XG4gKiAtIHN0YXJ0L2VuZDogU2VsZWN0ZWQgcmFuZ2UgYm91bmRhcmllcyAobnVsbCBpZiBub3Qgc2VsZWN0ZWQpXG4gKiAtIG1pbkRhdGUvbWF4RGF0ZTogSGFyZCBjb25zdHJhaW50cyAob3B0aW9uYWwsIGZvciBmdXR1cmUgdXNlKVxuICogLSBob3ZlckRhdGU6IEN1cnJlbnQgaG92ZXIgZGF0ZSAobnVsbCBpZiBub3QgaG92ZXJpbmcpXG4gKiAtIGRpc2FibGVkRGF0ZXM6IEFycmF5IG9mIGRpc2FibGVkIGRhdGVzIE9SIGZ1bmN0aW9uIHByZWRpY2F0ZVxuICovXG5leHBvcnQgaW50ZXJmYWNlIFJhbmdlRGVjb3JhdGlvblBhcmFtcyB7XG4gIC8qKlxuICAgKiBTZWxlY3RlZCBzdGFydCBkYXRlIChudWxsIGlmIG5vdCBzZWxlY3RlZCB5ZXQpXG4gICAqL1xuICBzdGFydDogRGF0ZSB8IG51bGw7XG5cbiAgLyoqXG4gICAqIFNlbGVjdGVkIGVuZCBkYXRlIChudWxsIGlmIG5vdCBzZWxlY3RlZCB5ZXQpXG4gICAqL1xuICBlbmQ6IERhdGUgfCBudWxsO1xuXG4gIC8qKlxuICAgKiBNaW5pbXVtIGFsbG93ZWQgZGF0ZSAob3B0aW9uYWwsIGZvciBmdXR1cmUgY29uc3RyYWludCBzdXBwb3J0KVxuICAgKiBBbGwgZGF0ZXMgYmVmb3JlIHRoaXMgd2lsbCBiZSBkaXNhYmxlZC5cbiAgICovXG4gIG1pbkRhdGU/OiBEYXRlIHwgbnVsbDtcblxuICAvKipcbiAgICogTWF4aW11bSBhbGxvd2VkIGRhdGUgKG9wdGlvbmFsLCBmb3IgZnV0dXJlIGNvbnN0cmFpbnQgc3VwcG9ydClcbiAgICogQWxsIGRhdGVzIGFmdGVyIHRoaXMgd2lsbCBiZSBkaXNhYmxlZC5cbiAgICovXG4gIG1heERhdGU/OiBEYXRlIHwgbnVsbDtcblxuICAvKipcbiAgICogQ3VycmVudCBob3ZlciBkYXRlIChudWxsIGlmIG5vdCBob3ZlcmluZylcbiAgICogVXNlZCB0byBjYWxjdWxhdGUgaG92ZXIgcHJldmlldyByYW5nZS5cbiAgICovXG4gIGhvdmVyRGF0ZT86IHN0cmluZyB8IG51bGw7XG5cbiAgLyoqXG4gICAqIERpc2FibGVkIGRhdGVzIHNwZWNpZmljYXRpb25cbiAgICogXG4gICAqIENhbiBiZSBvbmUgb2Y6XG4gICAqIC0gQXJyYXkgb2YgRGF0ZSBvYmplY3RzIHRvIGRpc2FibGUgKGV4YWN0IG1hdGNoZXMpXG4gICAqIC0gRnVuY3Rpb24gcHJlZGljYXRlIChkYXRlKSA9PiBib29sZWFuXG4gICAqIC0gdW5kZWZpbmVkL251bGwgKG5vIGRhdGVzIGRpc2FibGVkKVxuICAgKiBcbiAgICogTk9URTogRnVuY3Rpb25zIGNhbm5vdCBiZSBjYWNoZWQgcmVsaWFibHkgKG5vIHN0YWJsZSBrZXkpLlxuICAgKiBJZiB1c2luZyBmdW5jdGlvbiwgY2FjaGUgd2lsbCBiZSBieXBhc3NlZCBmb3IgaXNEaXNhYmxlZCBjb21wdXRhdGlvbi5cbiAgICovXG4gIGRpc2FibGVkRGF0ZXM/OiBEYXRlW10gfCAoKGRhdGU6IERhdGUpID0+IGJvb2xlYW4pIHwgbnVsbDtcblxuICAvKipcbiAgICogTXVsdGktcmFuZ2UgbW9kZSBmbGFnIChhZmZlY3RzIGhvdmVyIHJhbmdlIGJlaGF2aW9yKVxuICAgKiBEZWZhdWx0OiBmYWxzZVxuICAgKi9cbiAgbXVsdGlSYW5nZT86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIEN1cnJlbnRseSBzZWxlY3Rpbmcgc3RhcnQgZGF0ZSAoYWZmZWN0cyBob3ZlciBwcmV2aWV3KVxuICAgKiBEZWZhdWx0OiBmYWxzZVxuICAgKi9cbiAgc2VsZWN0aW5nU3RhcnREYXRlPzogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBDYWNoZSBrZXkgZm9yIGRlY29yYXRlZCBncmlkXG4gKiBcbiAqIEJ1aWx0IGZyb206XG4gKiAtIG1vbnRoS2V5OiBgJHt5ZWFyfS0ke21vbnRofS0ke3dlZWtTdGFydH0tJHtsb2NhbGV9YFxuICogLSBzdGFydElTTywgZW5kSVNPOiBJU08gZGF0ZXMgb3IgJ251bGwnXG4gKiAtIG1pbklTTywgbWF4SVNPOiBJU08gZGF0ZXMgb3IgJ251bGwnIChvcHRpb25hbClcbiAqIC0gaG92ZXJJU086IElTTyBkYXRlIG9yICdudWxsJ1xuICogLSBkaXNhYmxlZFNpZ25hdHVyZTogSGFzaCBvZiBkaXNhYmxlZCBkYXRlcyAob3IgJ25vbmUnIC8gJ2Z1bmN0aW9uJylcbiAqIFxuICogRXhhbXBsZTpcbiAqIFwiMjAyNi0xLTAtfDIwMjYtMDEtMTV8MjAyNi0wMS0yNXxudWxsfG51bGx8MjAyNi0wMS0yMHwyMDI2LTAxLTEwLDIwMjYtMDEtMTFcIlxuICogXG4gKiBTZWdtZW50IGJyZWFrZG93bjpcbiAqIDEuIDIwMjYtMS0wLSAobW9udGg6IEphbiAyMDI2LCB3ZWVrU3RhcnQ6IFN1bmRheSwgbm8gbG9jYWxlKVxuICogMi4gMjAyNi0wMS0xNSAoc3RhcnQgZGF0ZSlcbiAqIDMuIDIwMjYtMDEtMjUgKGVuZCBkYXRlKVxuICogNC4gbnVsbCAobm8gbWluRGF0ZSlcbiAqIDUuIG51bGwgKG5vIG1heERhdGUpXG4gKiA2LiAyMDI2LTAxLTIwIChob3ZlcmluZyBvdmVyIEphbiAyMClcbiAqIDcuIDIwMjYtMDEtMTAsMjAyNi0wMS0xMSAoZGlzYWJsZWQgZGF0ZXM6IEphbiAxMCAmIDExKVxuICovXG5leHBvcnQgaW50ZXJmYWNlIERlY29yYXRlZEdyaWRDYWNoZUtleSB7XG4gIC8qKlxuICAgKiBDb21iaW5lZCBtb250aCBrZXkgKGZyb20gYmFzZSBncmlkKVxuICAgKi9cbiAgbW9udGhLZXk6IHN0cmluZztcblxuICAvKipcbiAgICogU3RhcnQgZGF0ZSBJU08gKFlZWVktTU0tREQpIG9yICdudWxsJ1xuICAgKi9cbiAgc3RhcnRJU086IHN0cmluZztcblxuICAvKipcbiAgICogRW5kIGRhdGUgSVNPIChZWVlZLU1NLUREKSBvciAnbnVsbCdcbiAgICovXG4gIGVuZElTTzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBNaW4gZGF0ZSBJU08gKFlZWVktTU0tREQpIG9yICdudWxsJ1xuICAgKi9cbiAgbWluSVNPOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIE1heCBkYXRlIElTTyAoWVlZWS1NTS1ERCkgb3IgJ251bGwnXG4gICAqL1xuICBtYXhJU086IHN0cmluZztcblxuICAvKipcbiAgICogSG92ZXIgZGF0ZSBJU08gKFlZWVktTU0tREQpIG9yICdudWxsJ1xuICAgKi9cbiAgaG92ZXJJU086IHN0cmluZztcblxuICAvKipcbiAgICogRGlzYWJsZWQgZGF0ZXMgc2lnbmF0dXJlXG4gICAqIFxuICAgKiAtICdub25lJzogTm8gZGF0ZXMgZGlzYWJsZWRcbiAgICogLSAnZnVuY3Rpb24nOiBVc2luZyBwcmVkaWNhdGUgZnVuY3Rpb24gKG5vdCBjYWNoZWFibGUpXG4gICAqIC0gc29ydGVkIElTTyBzdHJpbmc6IFwiMjAyNi0wMS0xMCwyMDI2LTAxLTExLC4uLlwiIChmb3IgYXJyYXkpXG4gICAqL1xuICBkaXNhYmxlZFNpZ25hdHVyZTogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBGdWxsIGNhY2hlIGtleSAoY29uY2F0ZW5hdGlvbiBvZiBhbGwgc2VnbWVudHMpXG4gICAqL1xuICBmdWxsOiBzdHJpbmc7XG59XG4iXX0=
|
|
@@ -1,149 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Virtual Weeks Logic (Pure Functions)
|
|
3
|
-
*
|
|
4
|
-
* Pure computation layer for windowed week rendering.
|
|
5
|
-
* No side effects, fully testable with node:test.
|
|
6
|
-
*
|
|
7
|
-
* @module core/calendar-grid/virtual-weeks.logic
|
|
8
|
-
* @version 3.9.0
|
|
9
|
-
*/
|
|
10
|
-
/**
|
|
11
|
-
* Get visible weeks from total weeks array
|
|
12
|
-
*
|
|
13
|
-
* Pure function: Given weeks array and window config, returns visible slice.
|
|
14
|
-
*
|
|
15
|
-
* @param weeks Total weeks array (usually 6 weeks)
|
|
16
|
-
* @param startIndex Start index of visible window (0-based)
|
|
17
|
-
* @param windowSize How many weeks to show
|
|
18
|
-
* @returns Visible weeks slice
|
|
19
|
-
*
|
|
20
|
-
* @example
|
|
21
|
-
* ```typescript
|
|
22
|
-
* const allWeeks = [week0, week1, week2, week3, week4, week5]; // 6 weeks
|
|
23
|
-
* const visible = getVisibleWeeks(allWeeks, 0, 3);
|
|
24
|
-
* // Returns [week0, week1, week2]
|
|
25
|
-
*
|
|
26
|
-
* const visible2 = getVisibleWeeks(allWeeks, 3, 3);
|
|
27
|
-
* // Returns [week3, week4, week5]
|
|
28
|
-
* ```
|
|
29
|
-
*/
|
|
30
|
-
export function getVisibleWeeks(weeks, startIndex, windowSize) {
|
|
31
|
-
if (!weeks || weeks.length === 0) {
|
|
32
|
-
return [];
|
|
33
|
-
}
|
|
34
|
-
// If no windowing (windowSize undefined or >= total weeks), return all
|
|
35
|
-
if (windowSize === undefined || windowSize >= weeks.length) {
|
|
36
|
-
return weeks;
|
|
37
|
-
}
|
|
38
|
-
// Clamp startIndex to valid range
|
|
39
|
-
const clampedStart = clampWeekStart(startIndex, weeks.length, windowSize);
|
|
40
|
-
// Return slice
|
|
41
|
-
return weeks.slice(clampedStart, clampedStart + windowSize);
|
|
42
|
-
}
|
|
43
|
-
/**
|
|
44
|
-
* Clamp week start index to valid range
|
|
45
|
-
*
|
|
46
|
-
* Ensures startIndex is within bounds [0, totalWeeks - windowSize].
|
|
47
|
-
* Prevents scrolling beyond available weeks.
|
|
48
|
-
*
|
|
49
|
-
* @param startIndex Desired start index
|
|
50
|
-
* @param totalWeeks Total weeks available
|
|
51
|
-
* @param windowSize Window size
|
|
52
|
-
* @returns Clamped start index
|
|
53
|
-
*
|
|
54
|
-
* @example
|
|
55
|
-
* ```typescript
|
|
56
|
-
* clampWeekStart(0, 6, 3); // 0 (valid)
|
|
57
|
-
* clampWeekStart(3, 6, 3); // 3 (valid, shows weeks 3-5)
|
|
58
|
-
* clampWeekStart(4, 6, 3); // 3 (clamped, can't show beyond week 5)
|
|
59
|
-
* clampWeekStart(-1, 6, 3); // 0 (clamped, can't go negative)
|
|
60
|
-
* ```
|
|
61
|
-
*/
|
|
62
|
-
export function clampWeekStart(startIndex, totalWeeks, windowSize) {
|
|
63
|
-
if (windowSize >= totalWeeks) {
|
|
64
|
-
return 0; // No windowing needed
|
|
65
|
-
}
|
|
66
|
-
const maxStart = totalWeeks - windowSize;
|
|
67
|
-
return Math.max(0, Math.min(startIndex, maxStart));
|
|
68
|
-
}
|
|
69
|
-
/**
|
|
70
|
-
* Navigate week window (scroll up/down)
|
|
71
|
-
*
|
|
72
|
-
* Returns new start index after navigation.
|
|
73
|
-
* Handles clamping automatically.
|
|
74
|
-
*
|
|
75
|
-
* @param currentStart Current start index
|
|
76
|
-
* @param direction Navigation direction (+1 = down/later, -1 = up/earlier)
|
|
77
|
-
* @param totalWeeks Total weeks available
|
|
78
|
-
* @param windowSize Window size
|
|
79
|
-
* @returns New start index after navigation
|
|
80
|
-
*
|
|
81
|
-
* @example
|
|
82
|
-
* ```typescript
|
|
83
|
-
* // Start at week 0, navigate down
|
|
84
|
-
* navigateWeekWindow(0, 1, 6, 3); // Returns 1 (now showing weeks 1-3)
|
|
85
|
-
*
|
|
86
|
-
* // At week 3 (last valid position), navigate down
|
|
87
|
-
* navigateWeekWindow(3, 1, 6, 3); // Returns 3 (can't go further)
|
|
88
|
-
*
|
|
89
|
-
* // At week 1, navigate up
|
|
90
|
-
* navigateWeekWindow(1, -1, 6, 3); // Returns 0 (now showing weeks 0-2)
|
|
91
|
-
* ```
|
|
92
|
-
*/
|
|
93
|
-
export function navigateWeekWindow(currentStart, direction, totalWeeks, windowSize) {
|
|
94
|
-
const newStart = currentStart + direction;
|
|
95
|
-
return clampWeekStart(newStart, totalWeeks, windowSize);
|
|
96
|
-
}
|
|
97
|
-
/**
|
|
98
|
-
* Get virtual week window state
|
|
99
|
-
*
|
|
100
|
-
* Computes full window state including navigation capabilities.
|
|
101
|
-
*
|
|
102
|
-
* @param startIndex Current start index
|
|
103
|
-
* @param totalWeeks Total weeks available
|
|
104
|
-
* @param windowSize Window size
|
|
105
|
-
* @returns Complete window state
|
|
106
|
-
*
|
|
107
|
-
* @example
|
|
108
|
-
* ```typescript
|
|
109
|
-
* const state = getVirtualWeekWindow(0, 6, 3);
|
|
110
|
-
* // {
|
|
111
|
-
* // startIndex: 0,
|
|
112
|
-
* // windowSize: 3,
|
|
113
|
-
* // totalWeeks: 6,
|
|
114
|
-
* // canNavigateUp: false, // Already at top
|
|
115
|
-
* // canNavigateDown: true // Can scroll down
|
|
116
|
-
* // }
|
|
117
|
-
*
|
|
118
|
-
* const state2 = getVirtualWeekWindow(3, 6, 3);
|
|
119
|
-
* // {
|
|
120
|
-
* // startIndex: 3,
|
|
121
|
-
* // windowSize: 3,
|
|
122
|
-
* // totalWeeks: 6,
|
|
123
|
-
* // canNavigateUp: true, // Can scroll up
|
|
124
|
-
* // canNavigateDown: false // Already at bottom
|
|
125
|
-
* // }
|
|
126
|
-
* ```
|
|
127
|
-
*/
|
|
128
|
-
export function getVirtualWeekWindow(startIndex, totalWeeks, windowSize) {
|
|
129
|
-
const clampedStart = clampWeekStart(startIndex, totalWeeks, windowSize);
|
|
130
|
-
const maxStart = Math.max(0, totalWeeks - windowSize);
|
|
131
|
-
return {
|
|
132
|
-
startIndex: clampedStart,
|
|
133
|
-
windowSize,
|
|
134
|
-
totalWeeks,
|
|
135
|
-
canNavigateUp: clampedStart > 0,
|
|
136
|
-
canNavigateDown: clampedStart < maxStart
|
|
137
|
-
};
|
|
138
|
-
}
|
|
139
|
-
/**
|
|
140
|
-
* Check if virtual weeks mode is enabled
|
|
141
|
-
*
|
|
142
|
-
* @param windowSize Window size from config (undefined = disabled)
|
|
143
|
-
* @param totalWeeks Total weeks available
|
|
144
|
-
* @returns True if windowing should be applied
|
|
145
|
-
*/
|
|
146
|
-
export function isVirtualWeeksEnabled(windowSize, totalWeeks) {
|
|
147
|
-
return windowSize !== undefined && windowSize < totalWeeks;
|
|
148
|
-
}
|
|
149
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"virtual-weeks.logic.js","sourceRoot":"","sources":["../../../../src/core/calendar-grid/virtual-weeks.logic.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAIH;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAM,UAAU,eAAe,CAC7B,KAAU,EACV,UAAkB,EAClB,UAAkB;IAElB,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACjC,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,uEAAuE;IACvE,IAAI,UAAU,KAAK,SAAS,IAAI,UAAU,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;QAC3D,OAAO,KAAK,CAAC;IACf,CAAC;IAED,kCAAkC;IAClC,MAAM,YAAY,GAAG,cAAc,CAAC,UAAU,EAAE,KAAK,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;IAE1E,eAAe;IACf,OAAO,KAAK,CAAC,KAAK,CAAC,YAAY,EAAE,YAAY,GAAG,UAAU,CAAC,CAAC;AAC9D,CAAC;AAED;;;;;;;;;;;;;;;;;;GAkBG;AACH,MAAM,UAAU,cAAc,CAC5B,UAAkB,EAClB,UAAkB,EAClB,UAAkB;IAElB,IAAI,UAAU,IAAI,UAAU,EAAE,CAAC;QAC7B,OAAO,CAAC,CAAC,CAAC,sBAAsB;IAClC,CAAC;IAED,MAAM,QAAQ,GAAG,UAAU,GAAG,UAAU,CAAC;IACzC,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC,CAAC;AACrD,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;GAuBG;AACH,MAAM,UAAU,kBAAkB,CAChC,YAAoB,EACpB,SAAiB,EACjB,UAAkB,EAClB,UAAkB;IAElB,MAAM,QAAQ,GAAG,YAAY,GAAG,SAAS,CAAC;IAC1C,OAAO,cAAc,CAAC,QAAQ,EAAE,UAAU,EAAE,UAAU,CAAC,CAAC;AAC1D,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8BG;AACH,MAAM,UAAU,oBAAoB,CAClC,UAAkB,EAClB,UAAkB,EAClB,UAAkB;IAElB,MAAM,YAAY,GAAG,cAAc,CAAC,UAAU,EAAE,UAAU,EAAE,UAAU,CAAC,CAAC;IACxE,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,UAAU,GAAG,UAAU,CAAC,CAAC;IAEtD,OAAO;QACL,UAAU,EAAE,YAAY;QACxB,UAAU;QACV,UAAU;QACV,aAAa,EAAE,YAAY,GAAG,CAAC;QAC/B,eAAe,EAAE,YAAY,GAAG,QAAQ;KACzC,CAAC;AACJ,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,qBAAqB,CACnC,UAA8B,EAC9B,UAAkB;IAElB,OAAO,UAAU,KAAK,SAAS,IAAI,UAAU,GAAG,UAAU,CAAC;AAC7D,CAAC","sourcesContent":["/**\n * Virtual Weeks Logic (Pure Functions)\n * \n * Pure computation layer for windowed week rendering.\n * No side effects, fully testable with node:test.\n * \n * @module core/calendar-grid/virtual-weeks.logic\n * @version 3.9.0\n */\n\nimport { VirtualWeekWindow } from './virtual-weeks.types';\n\n/**\n * Get visible weeks from total weeks array\n * \n * Pure function: Given weeks array and window config, returns visible slice.\n * \n * @param weeks Total weeks array (usually 6 weeks)\n * @param startIndex Start index of visible window (0-based)\n * @param windowSize How many weeks to show\n * @returns Visible weeks slice\n * \n * @example\n * ```typescript\n * const allWeeks = [week0, week1, week2, week3, week4, week5]; // 6 weeks\n * const visible = getVisibleWeeks(allWeeks, 0, 3);\n * // Returns [week0, week1, week2]\n * \n * const visible2 = getVisibleWeeks(allWeeks, 3, 3);\n * // Returns [week3, week4, week5]\n * ```\n */\nexport function getVisibleWeeks<T>(\n  weeks: T[],\n  startIndex: number,\n  windowSize: number\n): T[] {\n  if (!weeks || weeks.length === 0) {\n    return [];\n  }\n\n  // If no windowing (windowSize undefined or >= total weeks), return all\n  if (windowSize === undefined || windowSize >= weeks.length) {\n    return weeks;\n  }\n\n  // Clamp startIndex to valid range\n  const clampedStart = clampWeekStart(startIndex, weeks.length, windowSize);\n\n  // Return slice\n  return weeks.slice(clampedStart, clampedStart + windowSize);\n}\n\n/**\n * Clamp week start index to valid range\n * \n * Ensures startIndex is within bounds [0, totalWeeks - windowSize].\n * Prevents scrolling beyond available weeks.\n * \n * @param startIndex Desired start index\n * @param totalWeeks Total weeks available\n * @param windowSize Window size\n * @returns Clamped start index\n * \n * @example\n * ```typescript\n * clampWeekStart(0, 6, 3); // 0 (valid)\n * clampWeekStart(3, 6, 3); // 3 (valid, shows weeks 3-5)\n * clampWeekStart(4, 6, 3); // 3 (clamped, can't show beyond week 5)\n * clampWeekStart(-1, 6, 3); // 0 (clamped, can't go negative)\n * ```\n */\nexport function clampWeekStart(\n  startIndex: number,\n  totalWeeks: number,\n  windowSize: number\n): number {\n  if (windowSize >= totalWeeks) {\n    return 0; // No windowing needed\n  }\n\n  const maxStart = totalWeeks - windowSize;\n  return Math.max(0, Math.min(startIndex, maxStart));\n}\n\n/**\n * Navigate week window (scroll up/down)\n * \n * Returns new start index after navigation.\n * Handles clamping automatically.\n * \n * @param currentStart Current start index\n * @param direction Navigation direction (+1 = down/later, -1 = up/earlier)\n * @param totalWeeks Total weeks available\n * @param windowSize Window size\n * @returns New start index after navigation\n * \n * @example\n * ```typescript\n * // Start at week 0, navigate down\n * navigateWeekWindow(0, 1, 6, 3); // Returns 1 (now showing weeks 1-3)\n * \n * // At week 3 (last valid position), navigate down\n * navigateWeekWindow(3, 1, 6, 3); // Returns 3 (can't go further)\n * \n * // At week 1, navigate up\n * navigateWeekWindow(1, -1, 6, 3); // Returns 0 (now showing weeks 0-2)\n * ```\n */\nexport function navigateWeekWindow(\n  currentStart: number,\n  direction: number,\n  totalWeeks: number,\n  windowSize: number\n): number {\n  const newStart = currentStart + direction;\n  return clampWeekStart(newStart, totalWeeks, windowSize);\n}\n\n/**\n * Get virtual week window state\n * \n * Computes full window state including navigation capabilities.\n * \n * @param startIndex Current start index\n * @param totalWeeks Total weeks available\n * @param windowSize Window size\n * @returns Complete window state\n * \n * @example\n * ```typescript\n * const state = getVirtualWeekWindow(0, 6, 3);\n * // {\n * //   startIndex: 0,\n * //   windowSize: 3,\n * //   totalWeeks: 6,\n * //   canNavigateUp: false,    // Already at top\n * //   canNavigateDown: true    // Can scroll down\n * // }\n * \n * const state2 = getVirtualWeekWindow(3, 6, 3);\n * // {\n * //   startIndex: 3,\n * //   windowSize: 3,\n * //   totalWeeks: 6,\n * //   canNavigateUp: true,     // Can scroll up\n * //   canNavigateDown: false   // Already at bottom\n * // }\n * ```\n */\nexport function getVirtualWeekWindow(\n  startIndex: number,\n  totalWeeks: number,\n  windowSize: number\n): VirtualWeekWindow {\n  const clampedStart = clampWeekStart(startIndex, totalWeeks, windowSize);\n  const maxStart = Math.max(0, totalWeeks - windowSize);\n\n  return {\n    startIndex: clampedStart,\n    windowSize,\n    totalWeeks,\n    canNavigateUp: clampedStart > 0,\n    canNavigateDown: clampedStart < maxStart\n  };\n}\n\n/**\n * Check if virtual weeks mode is enabled\n * \n * @param windowSize Window size from config (undefined = disabled)\n * @param totalWeeks Total weeks available\n * @returns True if windowing should be applied\n */\nexport function isVirtualWeeksEnabled(\n  windowSize: number | undefined,\n  totalWeeks: number\n): boolean {\n  return windowSize !== undefined && windowSize < totalWeeks;\n}\n"]}
|
|
@@ -1,11 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Virtual Weeks Types
|
|
3
|
-
*
|
|
4
|
-
* Type definitions for windowed week rendering (virtual scrolling).
|
|
5
|
-
* Reduces DOM nodes by rendering only visible weeks instead of full month.
|
|
6
|
-
*
|
|
7
|
-
* @module core/calendar-grid/virtual-weeks.types
|
|
8
|
-
* @version 3.9.0
|
|
9
|
-
*/
|
|
10
|
-
export {};
|
|
11
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmlydHVhbC13ZWVrcy50eXBlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9jb3JlL2NhbGVuZGFyLWdyaWQvdmlydHVhbC13ZWVrcy50eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7R0FRRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogVmlydHVhbCBXZWVrcyBUeXBlc1xuICogXG4gKiBUeXBlIGRlZmluaXRpb25zIGZvciB3aW5kb3dlZCB3ZWVrIHJlbmRlcmluZyAodmlydHVhbCBzY3JvbGxpbmcpLlxuICogUmVkdWNlcyBET00gbm9kZXMgYnkgcmVuZGVyaW5nIG9ubHkgdmlzaWJsZSB3ZWVrcyBpbnN0ZWFkIG9mIGZ1bGwgbW9udGguXG4gKiBcbiAqIEBtb2R1bGUgY29yZS9jYWxlbmRhci1ncmlkL3ZpcnR1YWwtd2Vla3MudHlwZXNcbiAqIEB2ZXJzaW9uIDMuOS4wXG4gKi9cblxuLyoqXG4gKiBWaXJ0dWFsIHdlZWtzIGNvbmZpZ3VyYXRpb25cbiAqIFxuICogQ29udHJvbHMgaG93IG1hbnkgd2Vla3MgdG8gcmVuZGVyIGF0IG9uY2UgKHdpbmRvd2VkIHJlbmRlcmluZykuXG4gKiBSZWR1Y2VzIERPTSBjb21wbGV4aXR5IGZvciBiZXR0ZXIgbW9iaWxlIHBlcmZvcm1hbmNlLlxuICogXG4gKiBFeGFtcGxlOlxuICogYGBgdHlwZXNjcmlwdFxuICogPG5neC1kdWFsLWRhdGVwaWNrZXJcbiAqICAgW3ZpcnR1YWxXZWVrc109XCJ7IHdpbmRvd1NpemU6IDMgfVwiXG4gKiA8L25neC1kdWFsLWRhdGVwaWNrZXI+XG4gKiBgYGBcbiAqIFxuICogV2l0aCB3aW5kb3dTaXplPTM6XG4gKiAtIFJlbmRlcnMgb25seSAzIHdlZWtzIGF0IGEgdGltZSAoMjEgY2VsbHMgdnMgNDIgY2VsbHMpXG4gKiAtIFVzZXIgY2FuIG5hdmlnYXRlIGJldHdlZW4gd2VlayB3aW5kb3dzXG4gKiAtIH41MCUgcmVkdWN0aW9uIGluIERPTSBub2RlcyBwZXIgY2FsZW5kYXJcbiAqIC0gfjUwJSByZWR1Y3Rpb24gaW4gcmVmbG93L3JlcGFpbnQgY29zdFxuICovXG5leHBvcnQgaW50ZXJmYWNlIFZpcnR1YWxXZWVrc0NvbmZpZyB7XG4gIC8qKlxuICAgKiBOdW1iZXIgb2Ygd2Vla3MgdG8gcmVuZGVyIGF0IG9uY2VcbiAgICogXG4gICAqIERlZmF1bHQ6IHVuZGVmaW5lZCAocmVuZGVyIGFsbCA2IHdlZWtzIC0gYmFja3dhcmQgY29tcGF0aWJsZSlcbiAgICogXG4gICAqIFJlY29tbWVuZGVkIHZhbHVlczpcbiAgICogLSAzOiBHb29kIGJhbGFuY2UgKDIxIGNlbGxzKVxuICAgKiAtIDQ6IE1vcmUgY29udGV4dCAoMjggY2VsbHMpXG4gICAqIC0gMjogTWluaW1hbCAoMTQgY2VsbHMsIG1heSBmZWVsIGNyYW1wZWQpXG4gICAqL1xuICB3aW5kb3dTaXplOiBudW1iZXI7XG59XG5cbi8qKlxuICogVmlydHVhbCB3ZWVrIHdpbmRvdyBzdGF0ZVxuICogXG4gKiBUcmFja3Mgd2hpY2ggd2Vla3MgYXJlIGN1cnJlbnRseSB2aXNpYmxlIGluIHRoZSB3aW5kb3cuXG4gKiBNYW5hZ2VkIGJ5IGNvbXBvbmVudCBzaWduYWxzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFZpcnR1YWxXZWVrV2luZG93IHtcbiAgLyoqXG4gICAqIFN0YXJ0IGluZGV4IG9mIHZpc2libGUgd2VlayByYW5nZSAoMC1iYXNlZClcbiAgICogXG4gICAqIFJhbmdlOiBbMCwgdG90YWxXZWVrcyAtIHdpbmRvd1NpemVdXG4gICAqL1xuICBzdGFydEluZGV4OiBudW1iZXI7XG5cbiAgLyoqXG4gICAqIFdpbmRvdyBzaXplIChob3cgbWFueSB3ZWVrcyB2aXNpYmxlKVxuICAgKi9cbiAgd2luZG93U2l6ZTogbnVtYmVyO1xuXG4gIC8qKlxuICAgKiBUb3RhbCB3ZWVrcyBhdmFpbGFibGUgaW4gbW9udGggKHVzdWFsbHkgNiwgc29tZXRpbWVzIDUpXG4gICAqL1xuICB0b3RhbFdlZWtzOiBudW1iZXI7XG5cbiAgLyoqXG4gICAqIFdoZXRoZXIgdXNlciBjYW4gc2Nyb2xsL25hdmlnYXRlIHVwICh0byBlYXJsaWVyIHdlZWtzKVxuICAgKi9cbiAgY2FuTmF2aWdhdGVVcDogYm9vbGVhbjtcblxuICAvKipcbiAgICogV2hldGhlciB1c2VyIGNhbiBzY3JvbGwvbmF2aWdhdGUgZG93biAodG8gbGF0ZXIgd2Vla3MpXG4gICAqL1xuICBjYW5OYXZpZ2F0ZURvd246IGJvb2xlYW47XG59XG4iXX0=
|
|
@@ -1,77 +0,0 @@
|
|
|
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"]}
|
|
@@ -1,65 +0,0 @@
|
|
|
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
|