@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.
Files changed (36) hide show
  1. package/README.md +26 -2
  2. package/fesm2022/oneluiz-dual-datepicker.mjs +75 -71
  3. package/fesm2022/oneluiz-dual-datepicker.mjs.map +1 -1
  4. package/package.json +8 -9
  5. package/types/oneluiz-dual-datepicker.d.ts +1618 -0
  6. package/dual-datepicker.component.d.ts +0 -243
  7. package/esm2022/core/built-in-presets.mjs +0 -289
  8. package/esm2022/core/calendar-grid/cache.config.mjs +0 -35
  9. package/esm2022/core/calendar-grid/calendar-grid.cache.mjs +0 -98
  10. package/esm2022/core/calendar-grid/calendar-grid.factory.mjs +0 -97
  11. package/esm2022/core/calendar-grid/calendar-grid.types.mjs +0 -8
  12. package/esm2022/core/calendar-grid/range-highlighter.cache.mjs +0 -200
  13. package/esm2022/core/calendar-grid/range-highlighter.mjs +0 -185
  14. package/esm2022/core/calendar-grid/range-highlighter.types.mjs +0 -11
  15. package/esm2022/core/calendar-grid/virtual-weeks.logic.mjs +0 -149
  16. package/esm2022/core/calendar-grid/virtual-weeks.types.mjs +0 -11
  17. package/esm2022/core/date-adapter.mjs +0 -77
  18. package/esm2022/core/date-clock.mjs +0 -65
  19. package/esm2022/core/dual-date-range.store.mjs +0 -329
  20. package/esm2022/core/internal.mjs +0 -92
  21. package/esm2022/core/native-date-adapter.mjs +0 -286
  22. package/esm2022/core/preset-providers.mjs +0 -243
  23. package/esm2022/core/preset-registry.mjs +0 -277
  24. package/esm2022/core/preset.engine.mjs +0 -179
  25. package/esm2022/core/public.mjs +0 -94
  26. package/esm2022/core/range-preset.plugin.mjs +0 -70
  27. package/esm2022/core/range.validator.mjs +0 -105
  28. package/esm2022/core/system-clock.mjs +0 -34
  29. package/esm2022/core/testing/date.helpers.mjs +0 -49
  30. package/esm2022/core/testing/fixed-clock.mjs +0 -30
  31. package/esm2022/core/testing/index.mjs +0 -8
  32. package/esm2022/dual-datepicker.component.mjs +0 -1314
  33. package/esm2022/oneluiz-dual-datepicker.mjs +0 -5
  34. package/esm2022/public-api.mjs +0 -43
  35. package/index.d.ts +0 -5
  36. package/public-api.d.ts +0 -44
@@ -1,98 +0,0 @@
1
- /**
2
- * Calendar Grid Cache
3
- *
4
- * LRU cache for calendar month grids to avoid recomputing the same month grid
5
- * multiple times when only decorations (selected dates, hover, etc.) change.
6
- *
7
- * Strategy:
8
- * - Key: year-month-weekStart-locale
9
- * - LRU eviction (least recently used) when limit is reached
10
- * - Default limit: 48 months (covers 2 years of navigation)
11
- *
12
- * Performance impact:
13
- * - Eliminates ~90% of grid recalculations in typical usage
14
- * - Memory footprint: ~10KB per cached month (negligible)
15
- *
16
- * Memory safety (v3.9.2):
17
- * - MAX_CACHE_ENTRIES prevents unbounded growth in long-running sessions
18
- * - Critical for: ERP, BI dashboards, hotel reservation systems
19
- * - FIFO eviction when limit exceeded
20
- */
21
- import { Injectable } from '@angular/core';
22
- import { MAX_CACHE_ENTRIES } from './cache.config';
23
- import * as i0 from "@angular/core";
24
- import * as i1 from "./calendar-grid.factory";
25
- export class CalendarGridCache {
26
- factory;
27
- cache = new Map();
28
- maxSize = MAX_CACHE_ENTRIES;
29
- constructor(factory) {
30
- this.factory = factory;
31
- }
32
- /**
33
- * Get or create a calendar grid
34
- *
35
- * @param monthDate - Any date within the target month
36
- * @param weekStart - First day of week (0 = Sunday, 1 = Monday, etc.)
37
- * @param locale - Locale identifier (optional)
38
- * @returns CalendarGrid - cached or newly created
39
- */
40
- get(monthDate, weekStart = 0, locale) {
41
- const key = this.buildKey(monthDate, weekStart, locale);
42
- // Check cache
43
- const cached = this.cache.get(key);
44
- if (cached) {
45
- // Move to end (LRU)
46
- this.cache.delete(key);
47
- this.cache.set(key, cached);
48
- return cached;
49
- }
50
- // Generate new grid
51
- const grid = this.factory.createGrid(monthDate, weekStart, locale);
52
- // Store in cache
53
- this.cache.set(key, grid);
54
- // Evict oldest if over limit (LRU)
55
- if (this.cache.size > this.maxSize) {
56
- const firstKey = this.cache.keys().next().value;
57
- this.cache.delete(firstKey);
58
- }
59
- return grid;
60
- }
61
- /**
62
- * Build cache key from month parameters
63
- *
64
- * Format: "year-month-weekStart-locale"
65
- * Example: "2026-1-0-en" (Feb 2026, Sunday start, English)
66
- */
67
- buildKey(monthDate, weekStart, locale) {
68
- const year = monthDate.getFullYear();
69
- const month = monthDate.getMonth();
70
- return `${year}-${month}-${weekStart}${locale ? '-' + locale : ''}`;
71
- }
72
- /**
73
- * Clear entire cache (for testing or manual reset)
74
- */
75
- clear() {
76
- this.cache.clear();
77
- }
78
- /**
79
- * Get cache size (for debugging/testing)
80
- */
81
- size() {
82
- return this.cache.size;
83
- }
84
- /**
85
- * Check if a specific month is cached
86
- */
87
- has(monthDate, weekStart = 0, locale) {
88
- const key = this.buildKey(monthDate, weekStart, locale);
89
- return this.cache.has(key);
90
- }
91
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CalendarGridCache, deps: [{ token: i1.CalendarGridFactory }], target: i0.ɵɵFactoryTarget.Injectable });
92
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CalendarGridCache, providedIn: 'root' });
93
- }
94
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CalendarGridCache, decorators: [{
95
- type: Injectable,
96
- args: [{ providedIn: 'root' }]
97
- }], ctorParameters: () => [{ type: i1.CalendarGridFactory }] });
98
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXItZ3JpZC5jYWNoZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9jb3JlL2NhbGVuZGFyLWdyaWQvY2FsZW5kYXItZ3JpZC5jYWNoZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW1CRztBQUVILE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFHM0MsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sZ0JBQWdCLENBQUM7OztBQUduRCxNQUFNLE9BQU8saUJBQWlCO0lBSVI7SUFIWixLQUFLLEdBQUcsSUFBSSxHQUFHLEVBQXdCLENBQUM7SUFDL0IsT0FBTyxHQUFXLGlCQUFpQixDQUFDO0lBRXJELFlBQW9CLE9BQTRCO1FBQTVCLFlBQU8sR0FBUCxPQUFPLENBQXFCO0lBQUcsQ0FBQztJQUVwRDs7Ozs7OztPQU9HO0lBQ0gsR0FBRyxDQUFDLFNBQWUsRUFBRSxZQUFvQixDQUFDLEVBQUUsTUFBZTtRQUN6RCxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsUUFBUSxDQUFDLFNBQVMsRUFBRSxTQUFTLEVBQUUsTUFBTSxDQUFDLENBQUM7UUFFeEQsY0FBYztRQUNkLE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQ25DLElBQUksTUFBTSxFQUFFLENBQUM7WUFDWCxvQkFBb0I7WUFDcEIsSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDdkIsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsR0FBRyxFQUFFLE1BQU0sQ0FBQyxDQUFDO1lBQzVCLE9BQU8sTUFBTSxDQUFDO1FBQ2hCLENBQUM7UUFFRCxvQkFBb0I7UUFDcEIsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsU0FBUyxFQUFFLFNBQVMsRUFBRSxNQUFNLENBQUMsQ0FBQztRQUVuRSxpQkFBaUI7UUFDakIsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsR0FBRyxFQUFFLElBQUksQ0FBQyxDQUFDO1FBRTFCLG1DQUFtQztRQUNuQyxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQztZQUNuQyxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksRUFBRSxDQUFDLElBQUksRUFBRSxDQUFDLEtBQUssQ0FBQztZQUNoRCxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUM5QixDQUFDO1FBRUQsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSyxRQUFRLENBQUMsU0FBZSxFQUFFLFNBQWlCLEVBQUUsTUFBZTtRQUNsRSxNQUFNLElBQUksR0FBRyxTQUFTLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDckMsTUFBTSxLQUFLLEdBQUcsU0FBUyxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQ25DLE9BQU8sR0FBRyxJQUFJLElBQUksS0FBSyxJQUFJLFNBQVMsR0FBRyxNQUFNLENBQUMsQ0FBQyxDQUFDLEdBQUcsR0FBRyxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDO0lBQ3RFLENBQUM7SUFFRDs7T0FFRztJQUNILEtBQUs7UUFDSCxJQUFJLENBQUMsS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQ3JCLENBQUM7SUFFRDs7T0FFRztJQUNILElBQUk7UUFDRixPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDO0lBQ3pCLENBQUM7SUFFRDs7T0FFRztJQUNILEdBQUcsQ0FBQyxTQUFlLEVBQUUsWUFBb0IsQ0FBQyxFQUFFLE1BQWU7UUFDekQsTUFBTSxHQUFHLEdBQUcsSUFBSSxDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsU0FBUyxFQUFFLE1BQU0sQ0FBQyxDQUFDO1FBQ3hELE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDN0IsQ0FBQzt3R0F6RVUsaUJBQWlCOzRHQUFqQixpQkFBaUIsY0FESixNQUFNOzs0RkFDbkIsaUJBQWlCO2tCQUQ3QixVQUFVO21CQUFDLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ2FsZW5kYXIgR3JpZCBDYWNoZVxuICpcbiAqIExSVSBjYWNoZSBmb3IgY2FsZW5kYXIgbW9udGggZ3JpZHMgdG8gYXZvaWQgcmVjb21wdXRpbmcgdGhlIHNhbWUgbW9udGggZ3JpZFxuICogbXVsdGlwbGUgdGltZXMgd2hlbiBvbmx5IGRlY29yYXRpb25zIChzZWxlY3RlZCBkYXRlcywgaG92ZXIsIGV0Yy4pIGNoYW5nZS5cbiAqXG4gKiBTdHJhdGVneTpcbiAqIC0gS2V5OiB5ZWFyLW1vbnRoLXdlZWtTdGFydC1sb2NhbGVcbiAqIC0gTFJVIGV2aWN0aW9uIChsZWFzdCByZWNlbnRseSB1c2VkKSB3aGVuIGxpbWl0IGlzIHJlYWNoZWRcbiAqIC0gRGVmYXVsdCBsaW1pdDogNDggbW9udGhzIChjb3ZlcnMgMiB5ZWFycyBvZiBuYXZpZ2F0aW9uKVxuICpcbiAqIFBlcmZvcm1hbmNlIGltcGFjdDpcbiAqIC0gRWxpbWluYXRlcyB+OTAlIG9mIGdyaWQgcmVjYWxjdWxhdGlvbnMgaW4gdHlwaWNhbCB1c2FnZVxuICogLSBNZW1vcnkgZm9vdHByaW50OiB+MTBLQiBwZXIgY2FjaGVkIG1vbnRoIChuZWdsaWdpYmxlKVxuICogXG4gKiBNZW1vcnkgc2FmZXR5ICh2My45LjIpOlxuICogLSBNQVhfQ0FDSEVfRU5UUklFUyBwcmV2ZW50cyB1bmJvdW5kZWQgZ3Jvd3RoIGluIGxvbmctcnVubmluZyBzZXNzaW9uc1xuICogLSBDcml0aWNhbCBmb3I6IEVSUCwgQkkgZGFzaGJvYXJkcywgaG90ZWwgcmVzZXJ2YXRpb24gc3lzdGVtc1xuICogLSBGSUZPIGV2aWN0aW9uIHdoZW4gbGltaXQgZXhjZWVkZWRcbiAqL1xuXG5pbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDYWxlbmRhckdyaWRGYWN0b3J5IH0gZnJvbSAnLi9jYWxlbmRhci1ncmlkLmZhY3RvcnknO1xuaW1wb3J0IHsgQ2FsZW5kYXJHcmlkLCBDYWxlbmRhckdyaWRDYWNoZUtleSB9IGZyb20gJy4vY2FsZW5kYXItZ3JpZC50eXBlcyc7XG5pbXBvcnQgeyBNQVhfQ0FDSEVfRU5UUklFUyB9IGZyb20gJy4vY2FjaGUuY29uZmlnJztcblxuQEluamVjdGFibGUoeyBwcm92aWRlZEluOiAncm9vdCcgfSlcbmV4cG9ydCBjbGFzcyBDYWxlbmRhckdyaWRDYWNoZSB7XG4gIHByaXZhdGUgY2FjaGUgPSBuZXcgTWFwPHN0cmluZywgQ2FsZW5kYXJHcmlkPigpO1xuICBwcml2YXRlIHJlYWRvbmx5IG1heFNpemU6IG51bWJlciA9IE1BWF9DQUNIRV9FTlRSSUVTO1xuXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgZmFjdG9yeTogQ2FsZW5kYXJHcmlkRmFjdG9yeSkge31cblxuICAvKipcbiAgICogR2V0IG9yIGNyZWF0ZSBhIGNhbGVuZGFyIGdyaWRcbiAgICpcbiAgICogQHBhcmFtIG1vbnRoRGF0ZSAtIEFueSBkYXRlIHdpdGhpbiB0aGUgdGFyZ2V0IG1vbnRoXG4gICAqIEBwYXJhbSB3ZWVrU3RhcnQgLSBGaXJzdCBkYXkgb2Ygd2VlayAoMCA9IFN1bmRheSwgMSA9IE1vbmRheSwgZXRjLilcbiAgICogQHBhcmFtIGxvY2FsZSAtIExvY2FsZSBpZGVudGlmaWVyIChvcHRpb25hbClcbiAgICogQHJldHVybnMgQ2FsZW5kYXJHcmlkIC0gY2FjaGVkIG9yIG5ld2x5IGNyZWF0ZWRcbiAgICovXG4gIGdldChtb250aERhdGU6IERhdGUsIHdlZWtTdGFydDogbnVtYmVyID0gMCwgbG9jYWxlPzogc3RyaW5nKTogQ2FsZW5kYXJHcmlkIHtcbiAgICBjb25zdCBrZXkgPSB0aGlzLmJ1aWxkS2V5KG1vbnRoRGF0ZSwgd2Vla1N0YXJ0LCBsb2NhbGUpO1xuXG4gICAgLy8gQ2hlY2sgY2FjaGVcbiAgICBjb25zdCBjYWNoZWQgPSB0aGlzLmNhY2hlLmdldChrZXkpO1xuICAgIGlmIChjYWNoZWQpIHtcbiAgICAgIC8vIE1vdmUgdG8gZW5kIChMUlUpXG4gICAgICB0aGlzLmNhY2hlLmRlbGV0ZShrZXkpO1xuICAgICAgdGhpcy5jYWNoZS5zZXQoa2V5LCBjYWNoZWQpO1xuICAgICAgcmV0dXJuIGNhY2hlZDtcbiAgICB9XG5cbiAgICAvLyBHZW5lcmF0ZSBuZXcgZ3JpZFxuICAgIGNvbnN0IGdyaWQgPSB0aGlzLmZhY3RvcnkuY3JlYXRlR3JpZChtb250aERhdGUsIHdlZWtTdGFydCwgbG9jYWxlKTtcblxuICAgIC8vIFN0b3JlIGluIGNhY2hlXG4gICAgdGhpcy5jYWNoZS5zZXQoa2V5LCBncmlkKTtcblxuICAgIC8vIEV2aWN0IG9sZGVzdCBpZiBvdmVyIGxpbWl0IChMUlUpXG4gICAgaWYgKHRoaXMuY2FjaGUuc2l6ZSA+IHRoaXMubWF4U2l6ZSkge1xuICAgICAgY29uc3QgZmlyc3RLZXkgPSB0aGlzLmNhY2hlLmtleXMoKS5uZXh0KCkudmFsdWU7XG4gICAgICB0aGlzLmNhY2hlLmRlbGV0ZShmaXJzdEtleSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGdyaWQ7XG4gIH1cblxuICAvKipcbiAgICogQnVpbGQgY2FjaGUga2V5IGZyb20gbW9udGggcGFyYW1ldGVyc1xuICAgKlxuICAgKiBGb3JtYXQ6IFwieWVhci1tb250aC13ZWVrU3RhcnQtbG9jYWxlXCJcbiAgICogRXhhbXBsZTogXCIyMDI2LTEtMC1lblwiIChGZWIgMjAyNiwgU3VuZGF5IHN0YXJ0LCBFbmdsaXNoKVxuICAgKi9cbiAgcHJpdmF0ZSBidWlsZEtleShtb250aERhdGU6IERhdGUsIHdlZWtTdGFydDogbnVtYmVyLCBsb2NhbGU/OiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIGNvbnN0IHllYXIgPSBtb250aERhdGUuZ2V0RnVsbFllYXIoKTtcbiAgICBjb25zdCBtb250aCA9IG1vbnRoRGF0ZS5nZXRNb250aCgpO1xuICAgIHJldHVybiBgJHt5ZWFyfS0ke21vbnRofS0ke3dlZWtTdGFydH0ke2xvY2FsZSA/ICctJyArIGxvY2FsZSA6ICcnfWA7XG4gIH1cblxuICAvKipcbiAgICogQ2xlYXIgZW50aXJlIGNhY2hlIChmb3IgdGVzdGluZyBvciBtYW51YWwgcmVzZXQpXG4gICAqL1xuICBjbGVhcigpOiB2b2lkIHtcbiAgICB0aGlzLmNhY2hlLmNsZWFyKCk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IGNhY2hlIHNpemUgKGZvciBkZWJ1Z2dpbmcvdGVzdGluZylcbiAgICovXG4gIHNpemUoKTogbnVtYmVyIHtcbiAgICByZXR1cm4gdGhpcy5jYWNoZS5zaXplO1xuICB9XG5cbiAgLyoqXG4gICAqIENoZWNrIGlmIGEgc3BlY2lmaWMgbW9udGggaXMgY2FjaGVkXG4gICAqL1xuICBoYXMobW9udGhEYXRlOiBEYXRlLCB3ZWVrU3RhcnQ6IG51bWJlciA9IDAsIGxvY2FsZT86IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IGtleSA9IHRoaXMuYnVpbGRLZXkobW9udGhEYXRlLCB3ZWVrU3RhcnQsIGxvY2FsZSk7XG4gICAgcmV0dXJuIHRoaXMuY2FjaGUuaGFzKGtleSk7XG4gIH1cbn1cbiJdfQ==
@@ -1,97 +0,0 @@
1
- /**
2
- * Calendar Grid Factory
3
- *
4
- * Generates calendar month grids using DateAdapter for deterministic,
5
- * timezone-safe date operations.
6
- *
7
- * Grid structure:
8
- * - Always 6 weeks x 7 days (42 cells) for layout stability
9
- * - Includes padding days from previous/next month
10
- * - No decorations (selected, disabled, etc.) - those are applied separately
11
- */
12
- import { Injectable, Inject } from '@angular/core';
13
- import { DATE_ADAPTER } from '../date-adapter';
14
- import * as i0 from "@angular/core";
15
- export class CalendarGridFactory {
16
- adapter;
17
- constructor(adapter) {
18
- this.adapter = adapter;
19
- }
20
- /**
21
- * Create a calendar grid for a given month
22
- *
23
- * @param monthDate - Any date within the target month (will be normalized to start of month)
24
- * @param weekStart - First day of week (0 = Sunday, 1 = Monday, etc.)
25
- * @param locale - Locale identifier (optional, for future use)
26
- * @returns CalendarGrid - 6 weeks x 7 days grid
27
- */
28
- createGrid(monthDate, weekStart = 0, locale) {
29
- // Normalize to start of month
30
- const year = this.adapter.getYear(monthDate);
31
- const month = this.adapter.getMonth(monthDate);
32
- const firstDayOfMonth = new Date(year, month, 1);
33
- const normalizedFirst = this.adapter.normalize(firstDayOfMonth);
34
- // Get days in month (day 0 of next month = last day of current month)
35
- const lastDayOfMonth = new Date(year, month + 1, 0);
36
- const daysInMonth = this.adapter.getDate(lastDayOfMonth);
37
- // Get first day of week offset
38
- const firstDayOfWeek = this.adapter.getDay(normalizedFirst);
39
- const offset = this.calculateOffset(firstDayOfWeek, weekStart);
40
- // Generate 42 cells (6 weeks x 7 days)
41
- const cells = [];
42
- let currentDate = this.adapter.addDays(normalizedFirst, -offset);
43
- for (let i = 0; i < 42; i++) {
44
- const cellDate = this.adapter.normalize(currentDate);
45
- const cellYear = this.adapter.getYear(cellDate);
46
- const cellMonth = this.adapter.getMonth(cellDate);
47
- const cellDay = this.adapter.getDate(cellDate);
48
- const cellDayOfWeek = this.adapter.getDay(cellDate);
49
- cells.push({
50
- date: cellDate,
51
- inCurrentMonth: cellYear === year && cellMonth === month,
52
- iso: this.adapter.toISODate(cellDate),
53
- day: cellDay,
54
- month: cellMonth,
55
- year: cellYear,
56
- dayOfWeek: cellDayOfWeek
57
- });
58
- currentDate = this.adapter.addDays(currentDate, 1);
59
- }
60
- // Split into weeks
61
- const weeks = [];
62
- for (let i = 0; i < 6; i++) {
63
- weeks.push(cells.slice(i * 7, (i + 1) * 7));
64
- }
65
- return {
66
- month: { year, month },
67
- weekStart,
68
- locale,
69
- weeks,
70
- cells
71
- };
72
- }
73
- /**
74
- * Calculate offset (number of padding days from previous month)
75
- *
76
- * @param firstDayOfWeek - Day of week for first day of month (0-6)
77
- * @param weekStart - Desired week start (0-6)
78
- * @returns Number of padding days needed
79
- */
80
- calculateOffset(firstDayOfWeek, weekStart) {
81
- let offset = firstDayOfWeek - weekStart;
82
- if (offset < 0) {
83
- offset += 7;
84
- }
85
- return offset;
86
- }
87
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CalendarGridFactory, deps: [{ token: DATE_ADAPTER }], target: i0.ɵɵFactoryTarget.Injectable });
88
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CalendarGridFactory, providedIn: 'root' });
89
- }
90
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CalendarGridFactory, decorators: [{
91
- type: Injectable,
92
- args: [{ providedIn: 'root' }]
93
- }], ctorParameters: () => [{ type: undefined, decorators: [{
94
- type: Inject,
95
- args: [DATE_ADAPTER]
96
- }] }] });
97
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXItZ3JpZC5mYWN0b3J5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvcmUvY2FsZW5kYXItZ3JpZC9jYWxlbmRhci1ncmlkLmZhY3RvcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7R0FVRztBQUVILE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ25ELE9BQU8sRUFBZSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQzs7QUFJNUQsTUFBTSxPQUFPLG1CQUFtQjtJQUVFO0lBRGhDLFlBQ2dDLE9BQW9CO1FBQXBCLFlBQU8sR0FBUCxPQUFPLENBQWE7SUFDakQsQ0FBQztJQUVKOzs7Ozs7O09BT0c7SUFDSCxVQUFVLENBQUMsU0FBZSxFQUFFLFlBQW9CLENBQUMsRUFBRSxNQUFlO1FBQ2hFLDhCQUE4QjtRQUM5QixNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUM3QyxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUMvQyxNQUFNLGVBQWUsR0FBRyxJQUFJLElBQUksQ0FBQyxJQUFJLEVBQUUsS0FBSyxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQ2pELE1BQU0sZUFBZSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsU0FBUyxDQUFDLGVBQWUsQ0FBQyxDQUFDO1FBRWhFLHNFQUFzRTtRQUN0RSxNQUFNLGNBQWMsR0FBRyxJQUFJLElBQUksQ0FBQyxJQUFJLEVBQUUsS0FBSyxHQUFHLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQztRQUNwRCxNQUFNLFdBQVcsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxjQUFjLENBQUMsQ0FBQztRQUV6RCwrQkFBK0I7UUFDL0IsTUFBTSxjQUFjLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsZUFBZSxDQUFDLENBQUM7UUFDNUQsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLGVBQWUsQ0FBQyxjQUFjLEVBQUUsU0FBUyxDQUFDLENBQUM7UUFFL0QsdUNBQXVDO1FBQ3ZDLE1BQU0sS0FBSyxHQUFtQixFQUFFLENBQUM7UUFDakMsSUFBSSxXQUFXLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsZUFBZSxFQUFFLENBQUMsTUFBTSxDQUFDLENBQUM7UUFFakUsS0FBSyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDO1lBQzVCLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsU0FBUyxDQUFDLFdBQVcsQ0FBQyxDQUFDO1lBQ3JELE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxDQUFDO1lBQ2hELE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1lBQ2xELE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxDQUFDO1lBQy9DLE1BQU0sYUFBYSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1lBRXBELEtBQUssQ0FBQyxJQUFJLENBQUM7Z0JBQ1QsSUFBSSxFQUFFLFFBQVE7Z0JBQ2QsY0FBYyxFQUFFLFFBQVEsS0FBSyxJQUFJLElBQUksU0FBUyxLQUFLLEtBQUs7Z0JBQ3hELEdBQUcsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLFNBQVMsQ0FBQyxRQUFRLENBQUM7Z0JBQ3JDLEdBQUcsRUFBRSxPQUFPO2dCQUNaLEtBQUssRUFBRSxTQUFTO2dCQUNoQixJQUFJLEVBQUUsUUFBUTtnQkFDZCxTQUFTLEVBQUUsYUFBYTthQUN6QixDQUFDLENBQUM7WUFFSCxXQUFXLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsV0FBVyxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQ3JELENBQUM7UUFFRCxtQkFBbUI7UUFDbkIsTUFBTSxLQUFLLEdBQXFCLEVBQUUsQ0FBQztRQUNuQyxLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDM0IsS0FBSyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUM5QyxDQUFDO1FBRUQsT0FBTztZQUNMLEtBQUssRUFBRSxFQUFFLElBQUksRUFBRSxLQUFLLEVBQUU7WUFDdEIsU0FBUztZQUNULE1BQU07WUFDTixLQUFLO1lBQ0wsS0FBSztTQUNOLENBQUM7SUFDSixDQUFDO0lBRUQ7Ozs7OztPQU1HO0lBQ0ssZUFBZSxDQUFDLGNBQXNCLEVBQUUsU0FBaUI7UUFDL0QsSUFBSSxNQUFNLEdBQUcsY0FBYyxHQUFHLFNBQVMsQ0FBQztRQUN4QyxJQUFJLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUNmLE1BQU0sSUFBSSxDQUFDLENBQUM7UUFDZCxDQUFDO1FBQ0QsT0FBTyxNQUFNLENBQUM7SUFDaEIsQ0FBQzt3R0FoRlUsbUJBQW1CLGtCQUVwQixZQUFZOzRHQUZYLG1CQUFtQixjQUROLE1BQU07OzRGQUNuQixtQkFBbUI7a0JBRC9CLFVBQVU7bUJBQUMsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFOzswQkFHN0IsTUFBTTsyQkFBQyxZQUFZIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDYWxlbmRhciBHcmlkIEZhY3RvcnlcbiAqXG4gKiBHZW5lcmF0ZXMgY2FsZW5kYXIgbW9udGggZ3JpZHMgdXNpbmcgRGF0ZUFkYXB0ZXIgZm9yIGRldGVybWluaXN0aWMsXG4gKiB0aW1lem9uZS1zYWZlIGRhdGUgb3BlcmF0aW9ucy5cbiAqXG4gKiBHcmlkIHN0cnVjdHVyZTpcbiAqIC0gQWx3YXlzIDYgd2Vla3MgeCA3IGRheXMgKDQyIGNlbGxzKSBmb3IgbGF5b3V0IHN0YWJpbGl0eVxuICogLSBJbmNsdWRlcyBwYWRkaW5nIGRheXMgZnJvbSBwcmV2aW91cy9uZXh0IG1vbnRoXG4gKiAtIE5vIGRlY29yYXRpb25zIChzZWxlY3RlZCwgZGlzYWJsZWQsIGV0Yy4pIC0gdGhvc2UgYXJlIGFwcGxpZWQgc2VwYXJhdGVseVxuICovXG5cbmltcG9ydCB7IEluamVjdGFibGUsIEluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRGF0ZUFkYXB0ZXIsIERBVEVfQURBUFRFUiB9IGZyb20gJy4uL2RhdGUtYWRhcHRlcic7XG5pbXBvcnQgeyBDYWxlbmRhckNlbGwsIENhbGVuZGFyR3JpZCB9IGZyb20gJy4vY2FsZW5kYXItZ3JpZC50eXBlcyc7XG5cbkBJbmplY3RhYmxlKHsgcHJvdmlkZWRJbjogJ3Jvb3QnIH0pXG5leHBvcnQgY2xhc3MgQ2FsZW5kYXJHcmlkRmFjdG9yeSB7XG4gIGNvbnN0cnVjdG9yKFxuICAgIEBJbmplY3QoREFURV9BREFQVEVSKSBwcml2YXRlIGFkYXB0ZXI6IERhdGVBZGFwdGVyXG4gICkge31cblxuICAvKipcbiAgICogQ3JlYXRlIGEgY2FsZW5kYXIgZ3JpZCBmb3IgYSBnaXZlbiBtb250aFxuICAgKlxuICAgKiBAcGFyYW0gbW9udGhEYXRlIC0gQW55IGRhdGUgd2l0aGluIHRoZSB0YXJnZXQgbW9udGggKHdpbGwgYmUgbm9ybWFsaXplZCB0byBzdGFydCBvZiBtb250aClcbiAgICogQHBhcmFtIHdlZWtTdGFydCAtIEZpcnN0IGRheSBvZiB3ZWVrICgwID0gU3VuZGF5LCAxID0gTW9uZGF5LCBldGMuKVxuICAgKiBAcGFyYW0gbG9jYWxlIC0gTG9jYWxlIGlkZW50aWZpZXIgKG9wdGlvbmFsLCBmb3IgZnV0dXJlIHVzZSlcbiAgICogQHJldHVybnMgQ2FsZW5kYXJHcmlkIC0gNiB3ZWVrcyB4IDcgZGF5cyBncmlkXG4gICAqL1xuICBjcmVhdGVHcmlkKG1vbnRoRGF0ZTogRGF0ZSwgd2Vla1N0YXJ0OiBudW1iZXIgPSAwLCBsb2NhbGU/OiBzdHJpbmcpOiBDYWxlbmRhckdyaWQge1xuICAgIC8vIE5vcm1hbGl6ZSB0byBzdGFydCBvZiBtb250aFxuICAgIGNvbnN0IHllYXIgPSB0aGlzLmFkYXB0ZXIuZ2V0WWVhcihtb250aERhdGUpO1xuICAgIGNvbnN0IG1vbnRoID0gdGhpcy5hZGFwdGVyLmdldE1vbnRoKG1vbnRoRGF0ZSk7XG4gICAgY29uc3QgZmlyc3REYXlPZk1vbnRoID0gbmV3IERhdGUoeWVhciwgbW9udGgsIDEpO1xuICAgIGNvbnN0IG5vcm1hbGl6ZWRGaXJzdCA9IHRoaXMuYWRhcHRlci5ub3JtYWxpemUoZmlyc3REYXlPZk1vbnRoKTtcblxuICAgIC8vIEdldCBkYXlzIGluIG1vbnRoIChkYXkgMCBvZiBuZXh0IG1vbnRoID0gbGFzdCBkYXkgb2YgY3VycmVudCBtb250aClcbiAgICBjb25zdCBsYXN0RGF5T2ZNb250aCA9IG5ldyBEYXRlKHllYXIsIG1vbnRoICsgMSwgMCk7XG4gICAgY29uc3QgZGF5c0luTW9udGggPSB0aGlzLmFkYXB0ZXIuZ2V0RGF0ZShsYXN0RGF5T2ZNb250aCk7XG5cbiAgICAvLyBHZXQgZmlyc3QgZGF5IG9mIHdlZWsgb2Zmc2V0XG4gICAgY29uc3QgZmlyc3REYXlPZldlZWsgPSB0aGlzLmFkYXB0ZXIuZ2V0RGF5KG5vcm1hbGl6ZWRGaXJzdCk7XG4gICAgY29uc3Qgb2Zmc2V0ID0gdGhpcy5jYWxjdWxhdGVPZmZzZXQoZmlyc3REYXlPZldlZWssIHdlZWtTdGFydCk7XG5cbiAgICAvLyBHZW5lcmF0ZSA0MiBjZWxscyAoNiB3ZWVrcyB4IDcgZGF5cylcbiAgICBjb25zdCBjZWxsczogQ2FsZW5kYXJDZWxsW10gPSBbXTtcbiAgICBsZXQgY3VycmVudERhdGUgPSB0aGlzLmFkYXB0ZXIuYWRkRGF5cyhub3JtYWxpemVkRmlyc3QsIC1vZmZzZXQpO1xuXG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCA0MjsgaSsrKSB7XG4gICAgICBjb25zdCBjZWxsRGF0ZSA9IHRoaXMuYWRhcHRlci5ub3JtYWxpemUoY3VycmVudERhdGUpO1xuICAgICAgY29uc3QgY2VsbFllYXIgPSB0aGlzLmFkYXB0ZXIuZ2V0WWVhcihjZWxsRGF0ZSk7XG4gICAgICBjb25zdCBjZWxsTW9udGggPSB0aGlzLmFkYXB0ZXIuZ2V0TW9udGgoY2VsbERhdGUpO1xuICAgICAgY29uc3QgY2VsbERheSA9IHRoaXMuYWRhcHRlci5nZXREYXRlKGNlbGxEYXRlKTtcbiAgICAgIGNvbnN0IGNlbGxEYXlPZldlZWsgPSB0aGlzLmFkYXB0ZXIuZ2V0RGF5KGNlbGxEYXRlKTtcblxuICAgICAgY2VsbHMucHVzaCh7XG4gICAgICAgIGRhdGU6IGNlbGxEYXRlLFxuICAgICAgICBpbkN1cnJlbnRNb250aDogY2VsbFllYXIgPT09IHllYXIgJiYgY2VsbE1vbnRoID09PSBtb250aCxcbiAgICAgICAgaXNvOiB0aGlzLmFkYXB0ZXIudG9JU09EYXRlKGNlbGxEYXRlKSxcbiAgICAgICAgZGF5OiBjZWxsRGF5LFxuICAgICAgICBtb250aDogY2VsbE1vbnRoLFxuICAgICAgICB5ZWFyOiBjZWxsWWVhcixcbiAgICAgICAgZGF5T2ZXZWVrOiBjZWxsRGF5T2ZXZWVrXG4gICAgICB9KTtcblxuICAgICAgY3VycmVudERhdGUgPSB0aGlzLmFkYXB0ZXIuYWRkRGF5cyhjdXJyZW50RGF0ZSwgMSk7XG4gICAgfVxuXG4gICAgLy8gU3BsaXQgaW50byB3ZWVrc1xuICAgIGNvbnN0IHdlZWtzOiBDYWxlbmRhckNlbGxbXVtdID0gW107XG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCA2OyBpKyspIHtcbiAgICAgIHdlZWtzLnB1c2goY2VsbHMuc2xpY2UoaSAqIDcsIChpICsgMSkgKiA3KSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHtcbiAgICAgIG1vbnRoOiB7IHllYXIsIG1vbnRoIH0sXG4gICAgICB3ZWVrU3RhcnQsXG4gICAgICBsb2NhbGUsXG4gICAgICB3ZWVrcyxcbiAgICAgIGNlbGxzXG4gICAgfTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDYWxjdWxhdGUgb2Zmc2V0IChudW1iZXIgb2YgcGFkZGluZyBkYXlzIGZyb20gcHJldmlvdXMgbW9udGgpXG4gICAqXG4gICAqIEBwYXJhbSBmaXJzdERheU9mV2VlayAtIERheSBvZiB3ZWVrIGZvciBmaXJzdCBkYXkgb2YgbW9udGggKDAtNilcbiAgICogQHBhcmFtIHdlZWtTdGFydCAtIERlc2lyZWQgd2VlayBzdGFydCAoMC02KVxuICAgKiBAcmV0dXJucyBOdW1iZXIgb2YgcGFkZGluZyBkYXlzIG5lZWRlZFxuICAgKi9cbiAgcHJpdmF0ZSBjYWxjdWxhdGVPZmZzZXQoZmlyc3REYXlPZldlZWs6IG51bWJlciwgd2Vla1N0YXJ0OiBudW1iZXIpOiBudW1iZXIge1xuICAgIGxldCBvZmZzZXQgPSBmaXJzdERheU9mV2VlayAtIHdlZWtTdGFydDtcbiAgICBpZiAob2Zmc2V0IDwgMCkge1xuICAgICAgb2Zmc2V0ICs9IDc7XG4gICAgfVxuICAgIHJldHVybiBvZmZzZXQ7XG4gIH1cbn1cbiJdfQ==
@@ -1,8 +0,0 @@
1
- /**
2
- * Calendar Grid Types
3
- *
4
- * Defines the structure for memoized calendar month grids.
5
- * Separates base grid structure (cacheable) from decorations (dynamic).
6
- */
7
- export {};
8
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXItZ3JpZC50eXBlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9jb3JlL2NhbGVuZGFyLWdyaWQvY2FsZW5kYXItZ3JpZC50eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7R0FLRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ2FsZW5kYXIgR3JpZCBUeXBlc1xuICpcbiAqIERlZmluZXMgdGhlIHN0cnVjdHVyZSBmb3IgbWVtb2l6ZWQgY2FsZW5kYXIgbW9udGggZ3JpZHMuXG4gKiBTZXBhcmF0ZXMgYmFzZSBncmlkIHN0cnVjdHVyZSAoY2FjaGVhYmxlKSBmcm9tIGRlY29yYXRpb25zIChkeW5hbWljKS5cbiAqL1xuXG4vKipcbiAqIEEgc2luZ2xlIGNhbGVuZGFyIGNlbGwgcmVwcmVzZW50aW5nIG9uZSBkYXlcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDYWxlbmRhckNlbGwge1xuICAvKiogRGF0ZSBvYmplY3QgKG5vcm1hbGl6ZWQgdG8gc3RhcnQgb2YgZGF5KSAqL1xuICBkYXRlOiBEYXRlO1xuXG4gIC8qKiBXaGV0aGVyIHRoaXMgZGF5IGJlbG9uZ3MgdG8gdGhlIGN1cnJlbnQgcmVuZGVyZWQgbW9udGggKi9cbiAgaW5DdXJyZW50TW9udGg6IGJvb2xlYW47XG5cbiAgLyoqIElTTyBzdHJpbmcgJ1lZWVktTU0tREQnICh0aW1lem9uZS1zYWZlIHZpYSBEYXRlQWRhcHRlcikgKi9cbiAgaXNvOiBzdHJpbmc7XG5cbiAgLyoqIERheSBvZiBtb250aCAoMS0zMSkgKi9cbiAgZGF5OiBudW1iZXI7XG5cbiAgLyoqIE1vbnRoICgwLTExLCBKYXZhU2NyaXB0IGNvbnZlbnRpb24pICovXG4gIG1vbnRoOiBudW1iZXI7XG5cbiAgLyoqIEZ1bGwgeWVhciAqL1xuICB5ZWFyOiBudW1iZXI7XG5cbiAgLyoqIERheSBvZiB3ZWVrICgwID0gU3VuZGF5LCA2ID0gU2F0dXJkYXkpICovXG4gIGRheU9mV2VlazogbnVtYmVyO1xufVxuXG4vKipcbiAqIEEgY2FsZW5kYXIgZ3JpZCByZXByZXNlbnRpbmcgb25lIG1vbnRoXG4gKlxuICogR3JpZCBpcyBhbHdheXMgNiB3ZWVrcyB4IDcgZGF5cyAoNDIgY2VsbHMpIGZvciBsYXlvdXQgc3RhYmlsaXR5LlxuICogSW5jbHVkZXMgcGFkZGluZyBkYXlzIGZyb20gcHJldmlvdXMvbmV4dCBtb250aC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDYWxlbmRhckdyaWQge1xuICAvKiogTW9udGggaWRlbnRpZmllciAqL1xuICBtb250aDoge1xuICAgIC8qKiBZZWFyICovXG4gICAgeWVhcjogbnVtYmVyO1xuICAgIC8qKiBNb250aCAwLTExICovXG4gICAgbW9udGg6IG51bWJlcjtcbiAgfTtcblxuICAvKiogV2VlayBzdGFydCBkYXkgKDAgPSBTdW5kYXksIDEgPSBNb25kYXksIGV0Yy4pICovXG4gIHdlZWtTdGFydDogbnVtYmVyO1xuXG4gIC8qKiBMb2NhbGUgKGZvciBmdXR1cmUgaTE4biwgb3B0aW9uYWwpICovXG4gIGxvY2FsZT86IHN0cmluZztcblxuICAvKiogNiB3ZWVrcyB4IDcgZGF5cyBtYXRyaXggKi9cbiAgd2Vla3M6IENhbGVuZGFyQ2VsbFtdW107XG5cbiAgLyoqIEZsYXQgYXJyYXkgb2YgYWxsIGNlbGxzIChjb252ZW5pZW5jZSBhY2Nlc3NvcikgKi9cbiAgY2VsbHM6IENhbGVuZGFyQ2VsbFtdO1xufVxuXG4vKipcbiAqIENhY2hlIGtleSBmb3IgY2FsZW5kYXIgZ3JpZHNcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDYWxlbmRhckdyaWRDYWNoZUtleSB7XG4gIHllYXI6IG51bWJlcjtcbiAgbW9udGg6IG51bWJlcjtcbiAgd2Vla1N0YXJ0OiBudW1iZXI7XG4gIGxvY2FsZT86IHN0cmluZztcbn1cbiJdfQ==
@@ -1,200 +0,0 @@
1
- /**
2
- * Range Highlighter Cache
3
- *
4
- * LRU cache for decorated calendar grids.
5
- * Eliminates redundant decoration computations when range/constraints don't change.
6
- *
7
- * @module core/calendar-grid/range-highlighter.cache
8
- * @version 3.9.2
9
- */
10
- import { Injectable, Inject } from '@angular/core';
11
- import { DATE_ADAPTER } from '../date-adapter';
12
- import { MAX_CACHE_ENTRIES } from './cache.config';
13
- import * as i0 from "@angular/core";
14
- import * as i1 from "./range-highlighter";
15
- /**
16
- * Range Highlighter Cache
17
- *
18
- * Memoizes decorated grids to avoid recomputing decorations.
19
- *
20
- * Cache strategy:
21
- * - LRU eviction (Map preserves insertion order)
22
- * - Max 48 entries (v3.9.2: Memory safety for long-running sessions)
23
- * - Key: month + start + end + hover + disabled signature
24
- *
25
- * Performance:
26
- * - Cache hit: ~0.1ms (object lookup)
27
- * - Cache miss: ~1ms (decoration + store)
28
- * - Hit rate: Expected >80% in typical usage
29
- *
30
- * Memory safety (v3.9.2):
31
- * - MAX_CACHE_ENTRIES prevents unbounded growth
32
- * - Critical for: ERP, BI dashboards, hotel systems
33
- * - FIFO eviction when limit exceeded
34
- * - ~720KB max memory footprint
35
- *
36
- * Edge cases handled:
37
- * - Function predicates: Not cacheable (recompute every time)
38
- * - Large disabled arrays: Sorted + joined for stable key
39
- * - Hover changes: Separate cache entries
40
- * - Multi-range mode: Included in key
41
- *
42
- * Usage:
43
- * ```typescript
44
- * const grid = calendarGridCache.get(monthDate, 0);
45
- * const decorated = rangeHighlighterCache.get(grid, {
46
- * start: startDate,
47
- * end: endDate,
48
- * hoverDate: '2026-01-20',
49
- * disabledDates: [...]
50
- * });
51
- * // Same params = same object reference (===)
52
- * ```
53
- */
54
- export class RangeHighlighterCache {
55
- highlighter;
56
- adapter;
57
- cache = new Map();
58
- maxSize = MAX_CACHE_ENTRIES;
59
- constructor(highlighter, adapter) {
60
- this.highlighter = highlighter;
61
- this.adapter = adapter;
62
- }
63
- /**
64
- * Get decorated grid (cached or computed)
65
- *
66
- * If cache hit: Returns existing DecoratedGrid (same object reference)
67
- * If cache miss: Computes decorations, stores in cache, returns new grid
68
- *
69
- * @param grid Base calendar grid (from CalendarGridCache)
70
- * @param params Decoration parameters
71
- * @returns Decorated grid with range highlights
72
- */
73
- get(grid, params) {
74
- // Build cache key
75
- const cacheKey = this.buildKey(grid, params);
76
- // Check cache (function predicates can't be cached reliably)
77
- const hasFunction = typeof params.disabledDates === 'function';
78
- if (!hasFunction && this.cache.has(cacheKey.full)) {
79
- // Cache hit: Return existing grid
80
- const cached = this.cache.get(cacheKey.full);
81
- // LRU update: Delete and re-insert (moves to end)
82
- this.cache.delete(cacheKey.full);
83
- this.cache.set(cacheKey.full, cached);
84
- return cached;
85
- }
86
- // Cache miss: Compute decorations
87
- const decorated = this.highlighter.decorate(grid, params);
88
- // Store in cache (only if not using function predicate)
89
- if (!hasFunction) {
90
- this.cache.set(cacheKey.full, decorated);
91
- // Evict oldest if over limit
92
- if (this.cache.size > this.maxSize) {
93
- const oldestKey = this.cache.keys().next().value;
94
- this.cache.delete(oldestKey);
95
- }
96
- }
97
- return decorated;
98
- }
99
- /**
100
- * Build cache key from grid and params
101
- *
102
- * Key structure:
103
- * `${monthKey}|${startISO}|${endISO}|${minISO}|${maxISO}|${hoverISO}|${disabledSig}`
104
- *
105
- * Example:
106
- * "2026-1-0-|2026-01-15|2026-01-25|null|null|2026-01-20|2026-01-10,2026-01-11"
107
- *
108
- * @param grid Base calendar grid
109
- * @param params Decoration parameters
110
- * @returns Cache key components
111
- */
112
- buildKey(grid, params) {
113
- // Month key (from base grid)
114
- const monthKey = `${grid.month.year}-${grid.month.month}-${grid.weekStart}-${grid.locale || ''}`;
115
- // Date ISOs (null if not provided)
116
- const startISO = params.start ? this.adapter.toISODate(params.start) : 'null';
117
- const endISO = params.end ? this.adapter.toISODate(params.end) : 'null';
118
- const minISO = params.minDate ? this.adapter.toISODate(params.minDate) : 'null';
119
- const maxISO = params.maxDate ? this.adapter.toISODate(params.maxDate) : 'null';
120
- const hoverISO = params.hoverDate || 'null';
121
- // Disabled dates signature
122
- const disabledSignature = this.buildDisabledSignature(params.disabledDates);
123
- // Multi-range flag (affects hover behavior)
124
- const multiRangeFlag = params.multiRange ? '1' : '0';
125
- const selectingStartFlag = params.selectingStartDate ? '1' : '0';
126
- // Full key (pipe-separated for readability)
127
- const full = `${monthKey}|${startISO}|${endISO}|${minISO}|${maxISO}|${hoverISO}|${disabledSignature}|${multiRangeFlag}|${selectingStartFlag}`;
128
- return {
129
- monthKey,
130
- startISO,
131
- endISO,
132
- minISO,
133
- maxISO,
134
- hoverISO,
135
- disabledSignature,
136
- full
137
- };
138
- }
139
- /**
140
- * Build signature for disabled dates
141
- *
142
- * Strategies:
143
- * - null/undefined: 'none'
144
- * - Function: 'function' (not cacheable)
145
- * - Array: Sorted ISO dates joined with commas
146
- *
147
- * @param disabledDates Disabled dates specification
148
- * @returns Signature string for cache key
149
- */
150
- buildDisabledSignature(disabledDates) {
151
- if (!disabledDates)
152
- return 'none';
153
- if (typeof disabledDates === 'function') {
154
- return 'function';
155
- }
156
- if (Array.isArray(disabledDates)) {
157
- if (disabledDates.length === 0)
158
- return 'none';
159
- // Convert to ISO dates, sort, join
160
- const isoArray = disabledDates
161
- .map(date => this.adapter.toISODate(date))
162
- .sort();
163
- return isoArray.join(',');
164
- }
165
- return 'none';
166
- }
167
- /**
168
- * Clear all cached grids
169
- */
170
- clear() {
171
- this.cache.clear();
172
- }
173
- /**
174
- * Get current cache size
175
- */
176
- size() {
177
- return this.cache.size;
178
- }
179
- /**
180
- * Check if key is cached
181
- *
182
- * @param grid Base calendar grid
183
- * @param params Decoration parameters
184
- * @returns True if this grid+params is cached
185
- */
186
- has(grid, params) {
187
- const key = this.buildKey(grid, params);
188
- return this.cache.has(key.full);
189
- }
190
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RangeHighlighterCache, deps: [{ token: i1.RangeHighlighter }, { token: DATE_ADAPTER }], target: i0.ɵɵFactoryTarget.Injectable });
191
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RangeHighlighterCache, providedIn: 'root' });
192
- }
193
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RangeHighlighterCache, decorators: [{
194
- type: Injectable,
195
- args: [{ providedIn: 'root' }]
196
- }], ctorParameters: () => [{ type: i1.RangeHighlighter }, { type: undefined, decorators: [{
197
- type: Inject,
198
- args: [DATE_ADAPTER]
199
- }] }] });
200
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"range-highlighter.cache.js","sourceRoot":"","sources":["../../../../src/core/calendar-grid/range-highlighter.cache.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Range Highlighter Cache\n * \n * LRU cache for decorated calendar grids.\n * Eliminates redundant decoration computations when range/constraints don't change.\n * \n * @module core/calendar-grid/range-highlighter.cache\n * @version 3.9.2\n */\n\nimport { Injectable, Inject } from '@angular/core';\nimport { DateAdapter, DATE_ADAPTER } from '../date-adapter';\nimport { CalendarGrid } from './calendar-grid.types';\nimport { DecoratedGrid, RangeDecorationParams, DecoratedGridCacheKey } from './range-highlighter.types';\nimport { RangeHighlighter } from './range-highlighter';\nimport { MAX_CACHE_ENTRIES } from './cache.config';\n\n/**\n * Range Highlighter Cache\n * \n * Memoizes decorated grids to avoid recomputing decorations.\n * \n * Cache strategy:\n * - LRU eviction (Map preserves insertion order)\n * - Max 48 entries (v3.9.2: Memory safety for long-running sessions)\n * - Key: month + start + end + hover + disabled signature\n * \n * Performance:\n * - Cache hit: ~0.1ms (object lookup)\n * - Cache miss: ~1ms (decoration + store)\n * - Hit rate: Expected >80% in typical usage\n * \n * Memory safety (v3.9.2):\n * - MAX_CACHE_ENTRIES prevents unbounded growth\n * - Critical for: ERP, BI dashboards, hotel systems\n * - FIFO eviction when limit exceeded\n * - ~720KB max memory footprint\n * \n * Edge cases handled:\n * - Function predicates: Not cacheable (recompute every time)\n * - Large disabled arrays: Sorted + joined for stable key\n * - Hover changes: Separate cache entries\n * - Multi-range mode: Included in key\n * \n * Usage:\n * ```typescript\n * const grid = calendarGridCache.get(monthDate, 0);\n * const decorated = rangeHighlighterCache.get(grid, {\n *   start: startDate,\n *   end: endDate,\n *   hoverDate: '2026-01-20',\n *   disabledDates: [...]\n * });\n * // Same params = same object reference (===)\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class RangeHighlighterCache {\n  private cache = new Map<string, DecoratedGrid>();\n  private maxSize = MAX_CACHE_ENTRIES;\n\n  constructor(\n    private highlighter: RangeHighlighter,\n    @Inject(DATE_ADAPTER) private adapter: DateAdapter\n  ) {}\n\n  /**\n   * Get decorated grid (cached or computed)\n   * \n   * If cache hit: Returns existing DecoratedGrid (same object reference)\n   * If cache miss: Computes decorations, stores in cache, returns new grid\n   * \n   * @param grid Base calendar grid (from CalendarGridCache)\n   * @param params Decoration parameters\n   * @returns Decorated grid with range highlights\n   */\n  get(\n    grid: CalendarGrid,\n    params: RangeDecorationParams\n  ): DecoratedGrid {\n    // Build cache key\n    const cacheKey = this.buildKey(grid, params);\n\n    // Check cache (function predicates can't be cached reliably)\n    const hasFunction = typeof params.disabledDates === 'function';\n    if (!hasFunction && this.cache.has(cacheKey.full)) {\n      // Cache hit: Return existing grid\n      const cached = this.cache.get(cacheKey.full)!;\n      \n      // LRU update: Delete and re-insert (moves to end)\n      this.cache.delete(cacheKey.full);\n      this.cache.set(cacheKey.full, cached);\n      \n      return cached;\n    }\n\n    // Cache miss: Compute decorations\n    const decorated = this.highlighter.decorate(grid, params);\n\n    // Store in cache (only if not using function predicate)\n    if (!hasFunction) {\n      this.cache.set(cacheKey.full, decorated);\n\n      // Evict oldest if over limit\n      if (this.cache.size > this.maxSize) {\n        const oldestKey = this.cache.keys().next().value;\n        this.cache.delete(oldestKey);\n      }\n    }\n\n    return decorated;\n  }\n\n  /**\n   * Build cache key from grid and params\n   * \n   * Key structure:\n   * `${monthKey}|${startISO}|${endISO}|${minISO}|${maxISO}|${hoverISO}|${disabledSig}`\n   * \n   * Example:\n   * \"2026-1-0-|2026-01-15|2026-01-25|null|null|2026-01-20|2026-01-10,2026-01-11\"\n   * \n   * @param grid Base calendar grid\n   * @param params Decoration parameters\n   * @returns Cache key components\n   */\n  private buildKey(\n    grid: CalendarGrid,\n    params: RangeDecorationParams\n  ): DecoratedGridCacheKey {\n    // Month key (from base grid)\n    const monthKey = `${grid.month.year}-${grid.month.month}-${grid.weekStart}-${grid.locale || ''}`;\n\n    // Date ISOs (null if not provided)\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    // Disabled dates signature\n    const disabledSignature = this.buildDisabledSignature(params.disabledDates);\n\n    // Multi-range flag (affects hover behavior)\n    const multiRangeFlag = params.multiRange ? '1' : '0';\n    const selectingStartFlag = params.selectingStartDate ? '1' : '0';\n\n    // Full key (pipe-separated for readability)\n    const full = `${monthKey}|${startISO}|${endISO}|${minISO}|${maxISO}|${hoverISO}|${disabledSignature}|${multiRangeFlag}|${selectingStartFlag}`;\n\n    return {\n      monthKey,\n      startISO,\n      endISO,\n      minISO,\n      maxISO,\n      hoverISO,\n      disabledSignature,\n      full\n    };\n  }\n\n  /**\n   * Build signature for disabled dates\n   * \n   * Strategies:\n   * - null/undefined: 'none'\n   * - Function: 'function' (not cacheable)\n   * - Array: Sorted ISO dates joined with commas\n   * \n   * @param disabledDates Disabled dates specification\n   * @returns Signature string for cache key\n   */\n  private buildDisabledSignature(\n    disabledDates?: Date[] | ((date: Date) => boolean) | null\n  ): string {\n    if (!disabledDates) return 'none';\n\n    if (typeof disabledDates === 'function') {\n      return 'function';\n    }\n\n    if (Array.isArray(disabledDates)) {\n      if (disabledDates.length === 0) return 'none';\n\n      // Convert to ISO dates, sort, join\n      const isoArray = disabledDates\n        .map(date => this.adapter.toISODate(date))\n        .sort();\n\n      return isoArray.join(',');\n    }\n\n    return 'none';\n  }\n\n  /**\n   * Clear all cached grids\n   */\n  clear(): void {\n    this.cache.clear();\n  }\n\n  /**\n   * Get current cache size\n   */\n  size(): number {\n    return this.cache.size;\n  }\n\n  /**\n   * Check if key is cached\n   * \n   * @param grid Base calendar grid\n   * @param params Decoration parameters\n   * @returns True if this grid+params is cached\n   */\n  has(grid: CalendarGrid, params: RangeDecorationParams): boolean {\n    const key = this.buildKey(grid, params);\n    return this.cache.has(key.full);\n  }\n}\n"]}