@tolle_/tolle-ui 0.0.1-beta

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 (77) hide show
  1. package/README.md +35 -0
  2. package/esm2022/lib/accordion-item.component.mjs +78 -0
  3. package/esm2022/lib/accordion.component.mjs +60 -0
  4. package/esm2022/lib/badge.component.mjs +76 -0
  5. package/esm2022/lib/button-group.component.mjs +25 -0
  6. package/esm2022/lib/button.component.mjs +70 -0
  7. package/esm2022/lib/calendar.component.mjs +315 -0
  8. package/esm2022/lib/card.component.mjs +94 -0
  9. package/esm2022/lib/checkbox.component.mjs +100 -0
  10. package/esm2022/lib/data-table.component.mjs +332 -0
  11. package/esm2022/lib/date-picker.component.mjs +232 -0
  12. package/esm2022/lib/date-range-picker.component.mjs +208 -0
  13. package/esm2022/lib/input.component.mjs +134 -0
  14. package/esm2022/lib/masked-input.component.mjs +179 -0
  15. package/esm2022/lib/modal-ref.mjs +31 -0
  16. package/esm2022/lib/modal-stack.service.mjs +26 -0
  17. package/esm2022/lib/modal.component.mjs +98 -0
  18. package/esm2022/lib/modal.mjs +27 -0
  19. package/esm2022/lib/modal.service.mjs +65 -0
  20. package/esm2022/lib/multi-select.component.mjs +231 -0
  21. package/esm2022/lib/pagination.component.mjs +279 -0
  22. package/esm2022/lib/range-calendar.component.mjs +285 -0
  23. package/esm2022/lib/select-group.component.mjs +28 -0
  24. package/esm2022/lib/select-item.component.mjs +84 -0
  25. package/esm2022/lib/select-separator.component.mjs +24 -0
  26. package/esm2022/lib/select.component.mjs +261 -0
  27. package/esm2022/lib/select.service.mjs +21 -0
  28. package/esm2022/lib/skeleton.component.mjs +34 -0
  29. package/esm2022/lib/switch.component.mjs +133 -0
  30. package/esm2022/lib/toast.service.mjs +59 -0
  31. package/esm2022/lib/tolle-cell.directive.mjs +22 -0
  32. package/esm2022/lib/tolle-config.mjs +11 -0
  33. package/esm2022/lib/tooltip.directive.mjs +71 -0
  34. package/esm2022/lib/types/date-range.mjs +2 -0
  35. package/esm2022/lib/utils/cn.mjs +6 -0
  36. package/esm2022/public-api.mjs +36 -0
  37. package/esm2022/tolle_-tolle-ui.mjs +5 -0
  38. package/fesm2022/tolle_-tolle-ui.mjs +3553 -0
  39. package/fesm2022/tolle_-tolle-ui.mjs.map +1 -0
  40. package/index.d.ts +5 -0
  41. package/lib/accordion-item.component.d.ts +13 -0
  42. package/lib/accordion.component.d.ts +14 -0
  43. package/lib/badge.component.d.ts +14 -0
  44. package/lib/button-group.component.d.ts +8 -0
  45. package/lib/button.component.d.ts +16 -0
  46. package/lib/calendar.component.d.ts +35 -0
  47. package/lib/card.component.d.ts +32 -0
  48. package/lib/checkbox.component.d.ts +23 -0
  49. package/lib/data-table.component.d.ts +45 -0
  50. package/lib/date-picker.component.d.ts +35 -0
  51. package/lib/date-range-picker.component.d.ts +36 -0
  52. package/lib/input.component.d.ts +27 -0
  53. package/lib/masked-input.component.d.ts +36 -0
  54. package/lib/modal-ref.d.ts +16 -0
  55. package/lib/modal-stack.service.d.ts +12 -0
  56. package/lib/modal.component.d.ts +19 -0
  57. package/lib/modal.d.ts +29 -0
  58. package/lib/modal.service.d.ts +18 -0
  59. package/lib/multi-select.component.d.ts +47 -0
  60. package/lib/pagination.component.d.ts +36 -0
  61. package/lib/range-calendar.component.d.ts +37 -0
  62. package/lib/select-group.component.d.ts +8 -0
  63. package/lib/select-item.component.d.ts +18 -0
  64. package/lib/select-separator.component.d.ts +8 -0
  65. package/lib/select.component.d.ts +45 -0
  66. package/lib/select.service.d.ts +10 -0
  67. package/lib/skeleton.component.d.ts +10 -0
  68. package/lib/switch.component.d.ts +39 -0
  69. package/lib/toast.service.d.ts +24 -0
  70. package/lib/tolle-cell.directive.d.ts +9 -0
  71. package/lib/tolle-config.d.ts +9 -0
  72. package/lib/tooltip.directive.d.ts +15 -0
  73. package/lib/types/date-range.d.ts +4 -0
  74. package/lib/utils/cn.d.ts +2 -0
  75. package/package.json +32 -0
  76. package/public-api.d.ts +32 -0
  77. package/theme.css +211 -0
@@ -0,0 +1,279 @@
1
+ import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, inject, Input, Output } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { cn } from './utils/cn';
4
+ import { SelectItemComponent, SelectComponent } from '@tolle/ui';
5
+ import { FormsModule } from '@angular/forms';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/common";
8
+ import * as i2 from "@angular/forms";
9
+ export class PaginationComponent {
10
+ class = '';
11
+ showPageLinks = true;
12
+ showPageOptions = true;
13
+ showCurrentPageInfo = true;
14
+ currentPageInfoTemplate;
15
+ totalRecords = 0;
16
+ currentPageSize = 10;
17
+ currentPage = 1;
18
+ pageSizeOptions = [10, 20, 30, 50];
19
+ onPageNumberChange = new EventEmitter();
20
+ onPageSizeChange = new EventEmitter();
21
+ totalPages = 0;
22
+ first = 0;
23
+ last = 0;
24
+ displayPageIndex = [];
25
+ pageReport = '';
26
+ initialized = false;
27
+ cd = inject(ChangeDetectorRef);
28
+ cn = cn;
29
+ navBtnClass = 'h-8 w-8 p-0 flex items-center justify-center rounded-md border border-input bg-background hover:bg-accent hover:text-accent-foreground disabled:opacity-50 disabled:cursor-not-allowed';
30
+ ngOnInit() {
31
+ this.initializePagination();
32
+ }
33
+ ngOnChanges(changes) {
34
+ // Only re-init if meaningful data changes
35
+ if (changes['totalRecords'] || changes['currentPage'] || changes['currentPageSize']) {
36
+ this.initializePagination();
37
+ }
38
+ }
39
+ initializePagination() {
40
+ this.calcPagination();
41
+ if (!this.initialized) {
42
+ this.initialized = true;
43
+ }
44
+ this.cd.detectChanges();
45
+ }
46
+ nextPage() {
47
+ if (this.totalPages > this.currentPage) {
48
+ this.currentPage++;
49
+ this.emitChange();
50
+ }
51
+ }
52
+ previousPage() {
53
+ if (this.currentPage > 1) {
54
+ this.currentPage--;
55
+ this.emitChange();
56
+ }
57
+ }
58
+ selectPage(page) {
59
+ if (this.currentPage === page)
60
+ return;
61
+ this.currentPage = page;
62
+ this.emitChange();
63
+ }
64
+ sizeChange(size) {
65
+ this.currentPageSize = size;
66
+ this.currentPage = 1; // Reset to page 1 on size change
67
+ this.emitChange();
68
+ }
69
+ emitChange() {
70
+ this.calcPagination();
71
+ this.onPageNumberChange.emit(this.currentPage);
72
+ this.onPageSizeChange.emit(this.currentPageSize);
73
+ this.cd.detectChanges();
74
+ }
75
+ calcPagination() {
76
+ this.totalPages = Math.ceil(this.totalRecords / this.currentPageSize) || 0;
77
+ // Bounds check
78
+ if (this.currentPage > this.totalPages && this.totalPages > 0) {
79
+ this.currentPage = this.totalPages;
80
+ }
81
+ this.first = this.totalRecords === 0 ? 0 : (this.currentPage - 1) * this.currentPageSize + 1;
82
+ this.last = Math.min(this.totalRecords, this.currentPage * this.currentPageSize);
83
+ // Calculate Sliding Window for Page Numbers (Max 5 visible)
84
+ const pages = Array.from({ length: this.totalPages }, (_, i) => i + 1);
85
+ if (this.totalPages <= 5) {
86
+ this.displayPageIndex = pages;
87
+ }
88
+ else {
89
+ let start = Math.max(0, this.currentPage - 3);
90
+ let end = start + 5;
91
+ if (end > this.totalPages) {
92
+ end = this.totalPages;
93
+ start = end - 5;
94
+ }
95
+ this.displayPageIndex = pages.slice(start, end);
96
+ }
97
+ // Template Parsing
98
+ if (this.currentPageInfoTemplate) {
99
+ this.pageReport = this.currentPageInfoTemplate.replace(/{first}|{last}|{totalRecords}|{currentPage}|{currentPageSize}|{totalPages}/g, (match) => {
100
+ switch (match) {
101
+ case '{first}': return `${this.first}`;
102
+ case '{last}': return `${this.last}`;
103
+ case '{totalRecords}': return `${this.totalRecords}`;
104
+ case '{totalPages}': return `${this.totalPages}`;
105
+ case '{currentPage}': return `${this.currentPage}`;
106
+ case '{currentPageSize}': return `${this.currentPageSize}`;
107
+ default: return match;
108
+ }
109
+ });
110
+ }
111
+ }
112
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PaginationComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
113
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: PaginationComponent, isStandalone: true, selector: "tolle-pagination", inputs: { class: "class", showPageLinks: "showPageLinks", showPageOptions: "showPageOptions", showCurrentPageInfo: "showCurrentPageInfo", currentPageInfoTemplate: "currentPageInfoTemplate", totalRecords: "totalRecords", currentPageSize: "currentPageSize", currentPage: "currentPage", pageSizeOptions: "pageSizeOptions" }, outputs: { onPageNumberChange: "onPageNumberChange", onPageSizeChange: "onPageSizeChange" }, usesOnChanges: true, ngImport: i0, template: `
114
+ <div [class]="cn('flex items-center justify-between px-2 py-4', class)">
115
+
116
+ <div *ngIf="showCurrentPageInfo" class="text-sm text-muted-foreground">
117
+ <ng-container *ngIf="currentPageInfoTemplate; else defaultReport">
118
+ {{ pageReport }}
119
+ </ng-container>
120
+ <ng-template #defaultReport>
121
+ Showing {{ first }} to {{ last }} of {{ totalRecords }} entries
122
+ </ng-template>
123
+ </div>
124
+
125
+ <div class="flex items-center space-x-6 lg:space-x-8">
126
+
127
+ <div *ngIf="showPageOptions" class="flex items-center space-x-2">
128
+ <p class="text-sm font-medium">Rows per page</p>
129
+ <tolle-select
130
+ class="w-[70px]"
131
+ size="sm"
132
+ [ngModel]="currentPageSize"
133
+ (ngModelChange)="sizeChange($event)"
134
+ >
135
+ <tolle-select-item *ngFor="let opt of pageSizeOptions" [value]="opt">
136
+ {{ opt }}
137
+ </tolle-select-item>
138
+ </tolle-select>
139
+ </div>
140
+
141
+ <div class="flex items-center space-x-2">
142
+ <div *ngIf="!showPageLinks" class="flex w-[100px] items-center justify-center text-sm font-medium">
143
+ Page {{ currentPage }} of {{ totalPages }}
144
+ </div>
145
+
146
+ <div *ngIf="showPageLinks" class="flex items-center space-x-1">
147
+ <button
148
+ (click)="previousPage()"
149
+ [disabled]="currentPage === 1"
150
+ [class]="navBtnClass"
151
+ >
152
+ <i class="ri-arrow-left-s-line"></i>
153
+ </button>
154
+
155
+ <button
156
+ *ngFor="let page of displayPageIndex"
157
+ (click)="selectPage(page)"
158
+ [class]="cn(
159
+ 'h-8 w-8 text-sm rounded-md flex items-center justify-center transition-colors',
160
+ currentPage === page
161
+ ? 'bg-primary text-primary-foreground font-medium'
162
+ : 'hover:bg-accent hover:text-accent-foreground'
163
+ )"
164
+ >
165
+ {{ page }}
166
+ </button>
167
+
168
+ <button
169
+ (click)="nextPage()"
170
+ [disabled]="currentPage === totalPages || totalPages === 0"
171
+ [class]="navBtnClass"
172
+ >
173
+ <i class="ri-arrow-right-s-line"></i>
174
+ </button>
175
+ </div>
176
+ </div>
177
+ </div>
178
+ </div>
179
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: SelectComponent, selector: "tolle-select", inputs: ["placeholder", "class", "disabled", "searchable", "size"] }, { kind: "component", type: SelectItemComponent, selector: "tolle-select-item", inputs: ["value", "class", "selected"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
180
+ }
181
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PaginationComponent, decorators: [{
182
+ type: Component,
183
+ args: [{
184
+ selector: 'tolle-pagination',
185
+ standalone: true,
186
+ imports: [CommonModule, SelectComponent, SelectItemComponent, FormsModule],
187
+ template: `
188
+ <div [class]="cn('flex items-center justify-between px-2 py-4', class)">
189
+
190
+ <div *ngIf="showCurrentPageInfo" class="text-sm text-muted-foreground">
191
+ <ng-container *ngIf="currentPageInfoTemplate; else defaultReport">
192
+ {{ pageReport }}
193
+ </ng-container>
194
+ <ng-template #defaultReport>
195
+ Showing {{ first }} to {{ last }} of {{ totalRecords }} entries
196
+ </ng-template>
197
+ </div>
198
+
199
+ <div class="flex items-center space-x-6 lg:space-x-8">
200
+
201
+ <div *ngIf="showPageOptions" class="flex items-center space-x-2">
202
+ <p class="text-sm font-medium">Rows per page</p>
203
+ <tolle-select
204
+ class="w-[70px]"
205
+ size="sm"
206
+ [ngModel]="currentPageSize"
207
+ (ngModelChange)="sizeChange($event)"
208
+ >
209
+ <tolle-select-item *ngFor="let opt of pageSizeOptions" [value]="opt">
210
+ {{ opt }}
211
+ </tolle-select-item>
212
+ </tolle-select>
213
+ </div>
214
+
215
+ <div class="flex items-center space-x-2">
216
+ <div *ngIf="!showPageLinks" class="flex w-[100px] items-center justify-center text-sm font-medium">
217
+ Page {{ currentPage }} of {{ totalPages }}
218
+ </div>
219
+
220
+ <div *ngIf="showPageLinks" class="flex items-center space-x-1">
221
+ <button
222
+ (click)="previousPage()"
223
+ [disabled]="currentPage === 1"
224
+ [class]="navBtnClass"
225
+ >
226
+ <i class="ri-arrow-left-s-line"></i>
227
+ </button>
228
+
229
+ <button
230
+ *ngFor="let page of displayPageIndex"
231
+ (click)="selectPage(page)"
232
+ [class]="cn(
233
+ 'h-8 w-8 text-sm rounded-md flex items-center justify-center transition-colors',
234
+ currentPage === page
235
+ ? 'bg-primary text-primary-foreground font-medium'
236
+ : 'hover:bg-accent hover:text-accent-foreground'
237
+ )"
238
+ >
239
+ {{ page }}
240
+ </button>
241
+
242
+ <button
243
+ (click)="nextPage()"
244
+ [disabled]="currentPage === totalPages || totalPages === 0"
245
+ [class]="navBtnClass"
246
+ >
247
+ <i class="ri-arrow-right-s-line"></i>
248
+ </button>
249
+ </div>
250
+ </div>
251
+ </div>
252
+ </div>
253
+ `,
254
+ changeDetection: ChangeDetectionStrategy.OnPush
255
+ }]
256
+ }], propDecorators: { class: [{
257
+ type: Input
258
+ }], showPageLinks: [{
259
+ type: Input
260
+ }], showPageOptions: [{
261
+ type: Input
262
+ }], showCurrentPageInfo: [{
263
+ type: Input
264
+ }], currentPageInfoTemplate: [{
265
+ type: Input
266
+ }], totalRecords: [{
267
+ type: Input
268
+ }], currentPageSize: [{
269
+ type: Input
270
+ }], currentPage: [{
271
+ type: Input
272
+ }], pageSizeOptions: [{
273
+ type: Input
274
+ }], onPageNumberChange: [{
275
+ type: Output
276
+ }], onPageSizeChange: [{
277
+ type: Output
278
+ }] } });
279
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pagination.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/pagination.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  inject,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { cn } from './utils/cn';\nimport {SelectItemComponent, SelectComponent} from '@tolle/ui';\nimport {FormsModule} from '@angular/forms';\n\n@Component({\n  selector: 'tolle-pagination',\n  standalone: true,\n  imports: [CommonModule, SelectComponent, SelectItemComponent, FormsModule],\n  template: `\n    <div [class]=\"cn('flex items-center justify-between px-2 py-4', class)\">\n\n      <div *ngIf=\"showCurrentPageInfo\" class=\"text-sm text-muted-foreground\">\n        <ng-container *ngIf=\"currentPageInfoTemplate; else defaultReport\">\n          {{ pageReport }}\n        </ng-container>\n        <ng-template #defaultReport>\n          Showing {{ first }} to {{ last }} of {{ totalRecords }} entries\n        </ng-template>\n      </div>\n\n      <div class=\"flex items-center space-x-6 lg:space-x-8\">\n\n        <div *ngIf=\"showPageOptions\" class=\"flex items-center space-x-2\">\n          <p class=\"text-sm font-medium\">Rows per page</p>\n          <tolle-select\n            class=\"w-[70px]\"\n            size=\"sm\"\n            [ngModel]=\"currentPageSize\"\n            (ngModelChange)=\"sizeChange($event)\"\n          >\n            <tolle-select-item *ngFor=\"let opt of pageSizeOptions\" [value]=\"opt\">\n              {{ opt }}\n            </tolle-select-item>\n          </tolle-select>\n        </div>\n\n        <div class=\"flex items-center space-x-2\">\n          <div *ngIf=\"!showPageLinks\" class=\"flex w-[100px] items-center justify-center text-sm font-medium\">\n            Page {{ currentPage }} of {{ totalPages }}\n          </div>\n\n          <div *ngIf=\"showPageLinks\" class=\"flex items-center space-x-1\">\n            <button\n              (click)=\"previousPage()\"\n              [disabled]=\"currentPage === 1\"\n              [class]=\"navBtnClass\"\n            >\n              <i class=\"ri-arrow-left-s-line\"></i>\n            </button>\n\n            <button\n              *ngFor=\"let page of displayPageIndex\"\n              (click)=\"selectPage(page)\"\n              [class]=\"cn(\n                'h-8 w-8 text-sm rounded-md flex items-center justify-center transition-colors',\n                currentPage === page\n                  ? 'bg-primary text-primary-foreground font-medium'\n                  : 'hover:bg-accent hover:text-accent-foreground'\n              )\"\n            >\n              {{ page }}\n            </button>\n\n            <button\n              (click)=\"nextPage()\"\n              [disabled]=\"currentPage === totalPages || totalPages === 0\"\n              [class]=\"navBtnClass\"\n            >\n              <i class=\"ri-arrow-right-s-line\"></i>\n            </button>\n          </div>\n        </div>\n      </div>\n    </div>\n  `,\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PaginationComponent implements OnInit, OnChanges {\n  @Input() class = '';\n  @Input() showPageLinks = true;\n  @Input() showPageOptions = true;\n  @Input() showCurrentPageInfo = true;\n  @Input() currentPageInfoTemplate?: string;\n\n  @Input() totalRecords = 0;\n  @Input() currentPageSize = 10;\n  @Input() currentPage = 1;\n  @Input() pageSizeOptions: number[] = [10, 20, 30, 50];\n\n  @Output() onPageNumberChange = new EventEmitter<number>();\n  @Output() onPageSizeChange = new EventEmitter<number>();\n\n  totalPages = 0;\n  first = 0;\n  last = 0;\n  displayPageIndex: number[] = [];\n  pageReport = '';\n\n  private initialized = false;\n  private cd = inject(ChangeDetectorRef);\n  protected cn = cn;\n\n  navBtnClass = 'h-8 w-8 p-0 flex items-center justify-center rounded-md border border-input bg-background hover:bg-accent hover:text-accent-foreground disabled:opacity-50 disabled:cursor-not-allowed';\n\n  ngOnInit(): void {\n    this.initializePagination();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Only re-init if meaningful data changes\n    if (changes['totalRecords'] || changes['currentPage'] || changes['currentPageSize']) {\n      this.initializePagination();\n    }\n  }\n\n  private initializePagination(): void {\n    this.calcPagination();\n    if (!this.initialized) {\n      this.initialized = true;\n    }\n    this.cd.detectChanges();\n  }\n\n  nextPage() {\n    if (this.totalPages > this.currentPage) {\n      this.currentPage++;\n      this.emitChange();\n    }\n  }\n\n  previousPage() {\n    if (this.currentPage > 1) {\n      this.currentPage--;\n      this.emitChange();\n    }\n  }\n\n  selectPage(page: number) {\n    if (this.currentPage === page) return;\n    this.currentPage = page;\n    this.emitChange();\n  }\n\n  sizeChange(size: number) {\n    this.currentPageSize = size;\n    this.currentPage = 1; // Reset to page 1 on size change\n    this.emitChange();\n  }\n\n  private emitChange(): void {\n    this.calcPagination();\n    this.onPageNumberChange.emit(this.currentPage);\n    this.onPageSizeChange.emit(this.currentPageSize);\n    this.cd.detectChanges();\n  }\n\n  private calcPagination(): void {\n    this.totalPages = Math.ceil(this.totalRecords / this.currentPageSize) || 0;\n\n    // Bounds check\n    if (this.currentPage > this.totalPages && this.totalPages > 0) {\n      this.currentPage = this.totalPages;\n    }\n\n    this.first = this.totalRecords === 0 ? 0 : (this.currentPage - 1) * this.currentPageSize + 1;\n    this.last = Math.min(this.totalRecords, this.currentPage * this.currentPageSize);\n\n    // Calculate Sliding Window for Page Numbers (Max 5 visible)\n    const pages: number[] = Array.from({ length: this.totalPages }, (_, i) => i + 1);\n\n    if (this.totalPages <= 5) {\n      this.displayPageIndex = pages;\n    } else {\n      let start = Math.max(0, this.currentPage - 3);\n      let end = start + 5;\n\n      if (end > this.totalPages) {\n        end = this.totalPages;\n        start = end - 5;\n      }\n      this.displayPageIndex = pages.slice(start, end);\n    }\n\n    // Template Parsing\n    if (this.currentPageInfoTemplate) {\n      this.pageReport = this.currentPageInfoTemplate.replace(\n        /{first}|{last}|{totalRecords}|{currentPage}|{currentPageSize}|{totalPages}/g,\n        (match) => {\n          switch (match) {\n            case '{first}': return `${this.first}`;\n            case '{last}': return `${this.last}`;\n            case '{totalRecords}': return `${this.totalRecords}`;\n            case '{totalPages}': return `${this.totalPages}`;\n            case '{currentPage}': return `${this.currentPage}`;\n            case '{currentPageSize}': return `${this.currentPageSize}`;\n            default: return match;\n          }\n        }\n      );\n    }\n  }\n}\n"]}
@@ -0,0 +1,285 @@
1
+ import { Component, Input, forwardRef, Output, EventEmitter } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { NG_VALUE_ACCESSOR } from '@angular/forms';
4
+ import { addMonths, subMonths, startOfMonth, endOfMonth, startOfWeek, endOfWeek, eachDayOfInterval, isSameMonth, isSameDay, isToday, setMonth, setYear, addYears, subYears, isBefore, startOfDay, isWithinInterval } from 'date-fns';
5
+ import { cn } from './utils/cn';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/common";
8
+ export class RangeCalendarComponent {
9
+ class = '';
10
+ disablePastDates = false;
11
+ rangeSelect = new EventEmitter(); // Emits whenever selection changes
12
+ currentView = 'date';
13
+ viewDate = new Date();
14
+ // The Range Value
15
+ value = { start: null, end: null };
16
+ weekDays = ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'];
17
+ daysInMonth = [];
18
+ months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'];
19
+ years = [];
20
+ navBtnClass = cn('h-7 w-7 bg-transparent p-0 opacity-50 hover:opacity-100 border border-input rounded-md flex items-center justify-center hover:bg-accent hover:text-accent-foreground transition-all');
21
+ onTouched = () => { };
22
+ onChange = () => { };
23
+ cn = cn;
24
+ ngOnInit() {
25
+ this.generateDays();
26
+ this.generateYears();
27
+ }
28
+ // --- Date Generation Logic (Same as Calendar) ---
29
+ generateDays() {
30
+ const start = startOfWeek(startOfMonth(this.viewDate));
31
+ const end = endOfWeek(endOfMonth(this.viewDate));
32
+ this.daysInMonth = eachDayOfInterval({ start, end });
33
+ }
34
+ generateYears() {
35
+ const currentYear = this.viewDate.getFullYear();
36
+ this.years = Array.from({ length: 16 }, (_, i) => currentYear - 6 + i);
37
+ }
38
+ setView(view) {
39
+ this.currentView = view;
40
+ if (view === 'year')
41
+ this.generateYears();
42
+ }
43
+ prev() {
44
+ if (this.currentView === 'date') {
45
+ this.viewDate = subMonths(this.viewDate, 1);
46
+ this.generateDays();
47
+ }
48
+ else if (this.currentView === 'year') {
49
+ this.viewDate = subYears(this.viewDate, 16);
50
+ this.generateYears();
51
+ }
52
+ else if (this.currentView === 'month') {
53
+ this.viewDate = subYears(this.viewDate, 1);
54
+ }
55
+ }
56
+ next() {
57
+ if (this.currentView === 'date') {
58
+ this.viewDate = addMonths(this.viewDate, 1);
59
+ this.generateDays();
60
+ }
61
+ else if (this.currentView === 'year') {
62
+ this.viewDate = addYears(this.viewDate, 16);
63
+ this.generateYears();
64
+ }
65
+ else if (this.currentView === 'month') {
66
+ this.viewDate = addYears(this.viewDate, 1);
67
+ }
68
+ }
69
+ // --- Range Selection Logic ---
70
+ selectDate(date) {
71
+ if (this.isDateDisabled(date))
72
+ return;
73
+ const { start, end } = this.value;
74
+ // 1. If start exists but end doesn't
75
+ if (start && !end) {
76
+ if (isBefore(date, start)) {
77
+ // User clicked earlier date -> Reset start
78
+ this.value = { start: date, end: null };
79
+ }
80
+ else {
81
+ // User clicked later date -> Complete range
82
+ this.value = { start, end: date };
83
+ }
84
+ }
85
+ // 2. If neither exist OR both exist (reset)
86
+ else {
87
+ this.value = { start: date, end: null };
88
+ }
89
+ if (!isSameMonth(date, this.viewDate)) {
90
+ this.viewDate = date;
91
+ this.generateDays();
92
+ }
93
+ this.onChange(this.value);
94
+ this.rangeSelect.emit(this.value);
95
+ this.onTouched();
96
+ }
97
+ selectMonth(monthIndex) {
98
+ this.viewDate = setMonth(this.viewDate, monthIndex);
99
+ this.currentView = 'date';
100
+ this.generateDays();
101
+ }
102
+ selectYear(year) {
103
+ this.viewDate = setYear(this.viewDate, year);
104
+ this.currentView = 'date';
105
+ this.generateDays();
106
+ }
107
+ // --- Visual Styling for Range ---
108
+ getDayClass(date) {
109
+ const { start, end } = this.value;
110
+ const isOutside = !isSameMonth(date, this.viewDate);
111
+ const isDisabled = this.isDateDisabled(date);
112
+ // Range Checks
113
+ const isStart = start && isSameDay(date, start);
114
+ const isEnd = end && isSameDay(date, end);
115
+ const isInside = start && end && isWithinInterval(date, { start, end });
116
+ const isTodayDate = isToday(date);
117
+ return cn(
118
+ // Base: h-9 w-9, but we remove margins/rounding for the 'strip' effect
119
+ 'h-9 w-9 p-0 font-normal text-sm transition-all flex items-center justify-center relative z-10',
120
+ // Default State (Not selected, Not disabled)
121
+ !isInside && !isStart && !isEnd && !isDisabled && 'hover:bg-accent hover:text-accent-foreground rounded-md',
122
+ // The "Caps": Start and End
123
+ (isStart || isEnd) && 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground focus:bg-primary focus:text-primary-foreground',
124
+ // The "Strip": Dates between start and end
125
+ isInside && !isStart && !isEnd && 'bg-accent text-accent-foreground rounded-none',
126
+ // Connecting the Caps to the Strip (Rectangular inside edges)
127
+ isStart && end && 'rounded-l-md rounded-r-none', isEnd && start && 'rounded-r-md rounded-l-none',
128
+ // If no end date yet, Start should be fully rounded
129
+ isStart && !end && 'rounded-md',
130
+ // Muted/Disabled logic
131
+ !isInside && isTodayDate && !isStart && !isEnd && 'bg-accent/50 text-accent-foreground rounded-md', (isOutside || isDisabled) && 'text-muted-foreground opacity-50', isDisabled && 'cursor-not-allowed');
132
+ }
133
+ isDateDisabled(date) {
134
+ return this.disablePastDates ? isBefore(date, startOfDay(new Date())) : false;
135
+ }
136
+ // --- CVA Implementation ---
137
+ writeValue(val) {
138
+ if (val) {
139
+ this.value = val;
140
+ if (val.start)
141
+ this.viewDate = val.start;
142
+ this.generateDays();
143
+ this.generateYears();
144
+ }
145
+ }
146
+ registerOnChange(fn) { this.onChange = fn; }
147
+ registerOnTouched(fn) { this.onTouched = fn; }
148
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RangeCalendarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
149
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: RangeCalendarComponent, isStandalone: true, selector: "tolle-range-calendar", inputs: { class: "class", disablePastDates: "disablePastDates" }, outputs: { rangeSelect: "rangeSelect" }, providers: [
150
+ {
151
+ provide: NG_VALUE_ACCESSOR,
152
+ useExisting: forwardRef(() => RangeCalendarComponent),
153
+ multi: true
154
+ }
155
+ ], ngImport: i0, template: `
156
+ <div [class]="cn('p-3 border rounded-md bg-background text-popover-foreground shadow-sm inline-block min-w-fit', class)">
157
+
158
+ <div class="flex items-center justify-between pt-1 pb-4 gap-2">
159
+ <div class="flex items-center gap-1">
160
+ <button type="button" (click)="setView('month')"
161
+ [class]="cn('text-sm font-semibold px-2 py-1 rounded transition-colors', currentView === 'month' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground')">
162
+ {{ viewDate | date: 'MMMM' }}
163
+ </button>
164
+ <button type="button" (click)="setView('year')"
165
+ [class]="cn('text-sm font-semibold px-2 py-1 rounded transition-colors', currentView === 'year' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground')">
166
+ {{ viewDate | date: 'yyyy' }}
167
+ </button>
168
+ </div>
169
+ <div class="flex items-center space-x-1">
170
+ <button type="button" (click)="prev()" [class]="navBtnClass"><i class="ri-arrow-left-s-line text-lg"></i></button>
171
+ <button type="button" (click)="next()" [class]="navBtnClass"><i class="ri-arrow-right-s-line text-lg"></i></button>
172
+ </div>
173
+ </div>
174
+
175
+ <div *ngIf="currentView === 'date'" class="space-y-2 animate-in fade-in zoom-in-95 duration-200">
176
+ <div class="grid grid-cols-7 gap-y-1 w-full">
177
+ <span *ngFor="let day of weekDays" class="text-[0.8rem] text-muted-foreground font-normal text-center w-9">
178
+ {{ day }}
179
+ </span>
180
+ </div>
181
+ <div class="grid grid-cols-7 gap-y-1 w-full">
182
+ <button
183
+ *ngFor="let date of daysInMonth"
184
+ type="button"
185
+ (click)="selectDate(date)"
186
+ [disabled]="isDateDisabled(date)"
187
+ [class]="getDayClass(date)"
188
+ >
189
+ {{ date | date: 'd' }}
190
+ </button>
191
+ </div>
192
+ </div>
193
+
194
+ <div *ngIf="currentView === 'month'" class="grid grid-cols-3 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200">
195
+ <button *ngFor="let month of months; let i = index" type="button" (click)="selectMonth(i)"
196
+ [class]="cn('text-sm py-2.5 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors', i === viewDate.getMonth() ? 'bg-primary text-primary-foreground' : '')">
197
+ {{ month }}
198
+ </button>
199
+ </div>
200
+
201
+ <div *ngIf="currentView === 'year'" class="grid grid-cols-4 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200">
202
+ <button *ngFor="let year of years" type="button" (click)="selectYear(year)"
203
+ [class]="cn('text-sm py-2 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors', year === viewDate.getFullYear() ? 'bg-primary text-primary-foreground' : '')">
204
+ {{ year }}
205
+ </button>
206
+ </div>
207
+ </div>
208
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i1.DatePipe, name: "date" }] });
209
+ }
210
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RangeCalendarComponent, decorators: [{
211
+ type: Component,
212
+ args: [{
213
+ selector: 'tolle-range-calendar',
214
+ standalone: true,
215
+ imports: [CommonModule],
216
+ providers: [
217
+ {
218
+ provide: NG_VALUE_ACCESSOR,
219
+ useExisting: forwardRef(() => RangeCalendarComponent),
220
+ multi: true
221
+ }
222
+ ],
223
+ template: `
224
+ <div [class]="cn('p-3 border rounded-md bg-background text-popover-foreground shadow-sm inline-block min-w-fit', class)">
225
+
226
+ <div class="flex items-center justify-between pt-1 pb-4 gap-2">
227
+ <div class="flex items-center gap-1">
228
+ <button type="button" (click)="setView('month')"
229
+ [class]="cn('text-sm font-semibold px-2 py-1 rounded transition-colors', currentView === 'month' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground')">
230
+ {{ viewDate | date: 'MMMM' }}
231
+ </button>
232
+ <button type="button" (click)="setView('year')"
233
+ [class]="cn('text-sm font-semibold px-2 py-1 rounded transition-colors', currentView === 'year' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground')">
234
+ {{ viewDate | date: 'yyyy' }}
235
+ </button>
236
+ </div>
237
+ <div class="flex items-center space-x-1">
238
+ <button type="button" (click)="prev()" [class]="navBtnClass"><i class="ri-arrow-left-s-line text-lg"></i></button>
239
+ <button type="button" (click)="next()" [class]="navBtnClass"><i class="ri-arrow-right-s-line text-lg"></i></button>
240
+ </div>
241
+ </div>
242
+
243
+ <div *ngIf="currentView === 'date'" class="space-y-2 animate-in fade-in zoom-in-95 duration-200">
244
+ <div class="grid grid-cols-7 gap-y-1 w-full">
245
+ <span *ngFor="let day of weekDays" class="text-[0.8rem] text-muted-foreground font-normal text-center w-9">
246
+ {{ day }}
247
+ </span>
248
+ </div>
249
+ <div class="grid grid-cols-7 gap-y-1 w-full">
250
+ <button
251
+ *ngFor="let date of daysInMonth"
252
+ type="button"
253
+ (click)="selectDate(date)"
254
+ [disabled]="isDateDisabled(date)"
255
+ [class]="getDayClass(date)"
256
+ >
257
+ {{ date | date: 'd' }}
258
+ </button>
259
+ </div>
260
+ </div>
261
+
262
+ <div *ngIf="currentView === 'month'" class="grid grid-cols-3 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200">
263
+ <button *ngFor="let month of months; let i = index" type="button" (click)="selectMonth(i)"
264
+ [class]="cn('text-sm py-2.5 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors', i === viewDate.getMonth() ? 'bg-primary text-primary-foreground' : '')">
265
+ {{ month }}
266
+ </button>
267
+ </div>
268
+
269
+ <div *ngIf="currentView === 'year'" class="grid grid-cols-4 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200">
270
+ <button *ngFor="let year of years" type="button" (click)="selectYear(year)"
271
+ [class]="cn('text-sm py-2 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors', year === viewDate.getFullYear() ? 'bg-primary text-primary-foreground' : '')">
272
+ {{ year }}
273
+ </button>
274
+ </div>
275
+ </div>
276
+ `
277
+ }]
278
+ }], propDecorators: { class: [{
279
+ type: Input
280
+ }], disablePastDates: [{
281
+ type: Input
282
+ }], rangeSelect: [{
283
+ type: Output
284
+ }] } });
285
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"range-calendar.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/range-calendar.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit, forwardRef, Output, EventEmitter } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport {\n  addMonths, subMonths, startOfMonth, endOfMonth,\n  startOfWeek, endOfWeek, eachDayOfInterval, isSameMonth,\n  isSameDay, isToday, setMonth, setYear, addYears, subYears,\n  isBefore, startOfDay, isWithinInterval, isAfter\n} from 'date-fns';\nimport { cn } from './utils/cn';\nimport {DateRange} from '@tolle/ui/types/date-range';\n\n@Component({\n  selector: 'tolle-range-calendar',\n  standalone: true,\n  imports: [CommonModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => RangeCalendarComponent),\n      multi: true\n    }\n  ],\n  template: `\n    <div [class]=\"cn('p-3 border rounded-md bg-background text-popover-foreground shadow-sm inline-block min-w-fit', class)\">\n\n      <div class=\"flex items-center justify-between pt-1 pb-4 gap-2\">\n        <div class=\"flex items-center gap-1\">\n          <button type=\"button\" (click)=\"setView('month')\"\n            [class]=\"cn('text-sm font-semibold px-2 py-1 rounded transition-colors', currentView === 'month' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground')\">\n            {{ viewDate | date: 'MMMM' }}\n          </button>\n          <button type=\"button\" (click)=\"setView('year')\"\n            [class]=\"cn('text-sm font-semibold px-2 py-1 rounded transition-colors', currentView === 'year' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground')\">\n            {{ viewDate | date: 'yyyy' }}\n          </button>\n        </div>\n        <div class=\"flex items-center space-x-1\">\n          <button type=\"button\" (click)=\"prev()\" [class]=\"navBtnClass\"><i class=\"ri-arrow-left-s-line text-lg\"></i></button>\n          <button type=\"button\" (click)=\"next()\" [class]=\"navBtnClass\"><i class=\"ri-arrow-right-s-line text-lg\"></i></button>\n        </div>\n      </div>\n\n      <div *ngIf=\"currentView === 'date'\" class=\"space-y-2 animate-in fade-in zoom-in-95 duration-200\">\n        <div class=\"grid grid-cols-7 gap-y-1 w-full\">\n          <span *ngFor=\"let day of weekDays\" class=\"text-[0.8rem] text-muted-foreground font-normal text-center w-9\">\n            {{ day }}\n          </span>\n        </div>\n        <div class=\"grid grid-cols-7 gap-y-1 w-full\">\n          <button\n            *ngFor=\"let date of daysInMonth\"\n            type=\"button\"\n            (click)=\"selectDate(date)\"\n            [disabled]=\"isDateDisabled(date)\"\n            [class]=\"getDayClass(date)\"\n          >\n            {{ date | date: 'd' }}\n          </button>\n        </div>\n      </div>\n\n      <div *ngIf=\"currentView === 'month'\" class=\"grid grid-cols-3 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200\">\n        <button *ngFor=\"let month of months; let i = index\" type=\"button\" (click)=\"selectMonth(i)\"\n          [class]=\"cn('text-sm py-2.5 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors', i === viewDate.getMonth() ? 'bg-primary text-primary-foreground' : '')\">\n          {{ month }}\n        </button>\n      </div>\n\n      <div *ngIf=\"currentView === 'year'\" class=\"grid grid-cols-4 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200\">\n        <button *ngFor=\"let year of years\" type=\"button\" (click)=\"selectYear(year)\"\n          [class]=\"cn('text-sm py-2 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors', year === viewDate.getFullYear() ? 'bg-primary text-primary-foreground' : '')\">\n          {{ year }}\n        </button>\n      </div>\n    </div>\n  `\n})\nexport class RangeCalendarComponent implements OnInit, ControlValueAccessor {\n  @Input() class = '';\n  @Input() disablePastDates = false;\n  @Output() rangeSelect = new EventEmitter<DateRange>(); // Emits whenever selection changes\n\n  currentView: 'date' | 'month' | 'year' = 'date';\n  viewDate: Date = new Date();\n\n  // The Range Value\n  value: DateRange = { start: null, end: null };\n\n  weekDays = ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'];\n  daysInMonth: Date[] = [];\n  months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'];\n  years: number[] = [];\n\n  navBtnClass = cn('h-7 w-7 bg-transparent p-0 opacity-50 hover:opacity-100 border border-input rounded-md flex items-center justify-center hover:bg-accent hover:text-accent-foreground transition-all');\n\n  onTouched: () => void = () => {};\n  onChange: (value: DateRange) => void = () => {};\n  protected cn = cn;\n\n  ngOnInit() {\n    this.generateDays();\n    this.generateYears();\n  }\n\n  // --- Date Generation Logic (Same as Calendar) ---\n  generateDays() {\n    const start = startOfWeek(startOfMonth(this.viewDate));\n    const end = endOfWeek(endOfMonth(this.viewDate));\n    this.daysInMonth = eachDayOfInterval({ start, end });\n  }\n\n  generateYears() {\n    const currentYear = this.viewDate.getFullYear();\n    this.years = Array.from({ length: 16 }, (_, i) => currentYear - 6 + i);\n  }\n\n  setView(view: 'date' | 'month' | 'year') {\n    this.currentView = view;\n    if (view === 'year') this.generateYears();\n  }\n\n  prev() {\n    if (this.currentView === 'date') {\n      this.viewDate = subMonths(this.viewDate, 1);\n      this.generateDays();\n    } else if (this.currentView === 'year') {\n      this.viewDate = subYears(this.viewDate, 16);\n      this.generateYears();\n    } else if (this.currentView === 'month') {\n      this.viewDate = subYears(this.viewDate, 1);\n    }\n  }\n\n  next() {\n    if (this.currentView === 'date') {\n      this.viewDate = addMonths(this.viewDate, 1);\n      this.generateDays();\n    } else if (this.currentView === 'year') {\n      this.viewDate = addYears(this.viewDate, 16);\n      this.generateYears();\n    } else if (this.currentView === 'month') {\n      this.viewDate = addYears(this.viewDate, 1);\n    }\n  }\n\n  // --- Range Selection Logic ---\n\n  selectDate(date: Date) {\n    if (this.isDateDisabled(date)) return;\n\n    const { start, end } = this.value;\n\n    // 1. If start exists but end doesn't\n    if (start && !end) {\n      if (isBefore(date, start)) {\n        // User clicked earlier date -> Reset start\n        this.value = { start: date, end: null };\n      } else {\n        // User clicked later date -> Complete range\n        this.value = { start, end: date };\n      }\n    }\n    // 2. If neither exist OR both exist (reset)\n    else {\n      this.value = { start: date, end: null };\n    }\n\n    if (!isSameMonth(date, this.viewDate)) {\n      this.viewDate = date;\n      this.generateDays();\n    }\n\n    this.onChange(this.value);\n    this.rangeSelect.emit(this.value);\n    this.onTouched();\n  }\n\n  selectMonth(monthIndex: number) {\n    this.viewDate = setMonth(this.viewDate, monthIndex);\n    this.currentView = 'date';\n    this.generateDays();\n  }\n\n  selectYear(year: number) {\n    this.viewDate = setYear(this.viewDate, year);\n    this.currentView = 'date';\n    this.generateDays();\n  }\n\n  // --- Visual Styling for Range ---\n\n  getDayClass(date: Date) {\n    const { start, end } = this.value;\n    const isOutside = !isSameMonth(date, this.viewDate);\n    const isDisabled = this.isDateDisabled(date);\n\n    // Range Checks\n    const isStart = start && isSameDay(date, start);\n    const isEnd = end && isSameDay(date, end);\n    const isInside = start && end && isWithinInterval(date, { start, end });\n    const isTodayDate = isToday(date);\n\n    return cn(\n      // Base: h-9 w-9, but we remove margins/rounding for the 'strip' effect\n      'h-9 w-9 p-0 font-normal text-sm transition-all flex items-center justify-center relative z-10',\n\n      // Default State (Not selected, Not disabled)\n      !isInside && !isStart && !isEnd && !isDisabled && 'hover:bg-accent hover:text-accent-foreground rounded-md',\n\n      // The \"Caps\": Start and End\n      (isStart || isEnd) && 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground focus:bg-primary focus:text-primary-foreground',\n\n      // The \"Strip\": Dates between start and end\n      isInside && !isStart && !isEnd && 'bg-accent text-accent-foreground rounded-none',\n\n      // Connecting the Caps to the Strip (Rectangular inside edges)\n      isStart && end && 'rounded-l-md rounded-r-none',\n      isEnd && start && 'rounded-r-md rounded-l-none',\n      // If no end date yet, Start should be fully rounded\n      isStart && !end && 'rounded-md',\n\n      // Muted/Disabled logic\n      !isInside && isTodayDate && !isStart && !isEnd && 'bg-accent/50 text-accent-foreground rounded-md',\n      (isOutside || isDisabled) && 'text-muted-foreground opacity-50',\n      isDisabled && 'cursor-not-allowed'\n    );\n  }\n\n  isDateDisabled(date: Date): boolean {\n    return this.disablePastDates ? isBefore(date, startOfDay(new Date())) : false;\n  }\n\n  // --- CVA Implementation ---\n  writeValue(val: DateRange | null): void {\n    if (val) {\n      this.value = val;\n      if (val.start) this.viewDate = val.start;\n      this.generateDays();\n      this.generateYears();\n    }\n  }\n  registerOnChange(fn: any): void { this.onChange = fn; }\n  registerOnTouched(fn: any): void { this.onTouched = fn; }\n}\n"]}