@tolle_/tolle-ui 0.0.27-beta → 0.0.29-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.
- package/esm2022/lib/calendar.component.mjs +278 -81
- package/esm2022/lib/card.component.mjs +3 -3
- package/esm2022/lib/data-table.component.mjs +3 -3
- package/esm2022/lib/date-picker.component.mjs +126 -29
- package/esm2022/lib/date-range-picker.component.mjs +76 -24
- package/esm2022/lib/multi-select.component.mjs +220 -26
- package/esm2022/lib/pagination.component.mjs +1 -1
- package/esm2022/lib/select-item.component.mjs +98 -14
- package/esm2022/lib/select.component.mjs +29 -36
- package/fesm2022/tolle-ui.mjs +821 -205
- package/fesm2022/tolle-ui.mjs.map +1 -1
- package/lib/button.component.d.ts +1 -1
- package/lib/calendar.component.d.ts +23 -2
- package/lib/date-picker.component.d.ts +16 -2
- package/lib/date-range-picker.component.d.ts +3 -1
- package/lib/multi-select.component.d.ts +13 -1
- package/lib/select-item.component.d.ts +4 -1
- package/package.json +1 -1
- package/theme.css +1 -1
|
@@ -1,13 +1,21 @@
|
|
|
1
|
-
import { Component, Input, forwardRef } from '@angular/core';
|
|
1
|
+
import { Component, Input, forwardRef, Output, EventEmitter } from '@angular/core';
|
|
2
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 } from 'date-fns';
|
|
3
|
+
import { NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';
|
|
4
|
+
import { addMonths, subMonths, startOfMonth, endOfMonth, startOfWeek, endOfWeek, eachDayOfInterval, isSameMonth, isSameDay, isToday, setMonth, setYear, addYears, subYears, isBefore, startOfDay, format } from 'date-fns';
|
|
5
5
|
import { cn } from './utils/cn';
|
|
6
6
|
import * as i0 from "@angular/core";
|
|
7
7
|
import * as i1 from "@angular/common";
|
|
8
8
|
export class CalendarComponent {
|
|
9
9
|
class = '';
|
|
10
|
+
mode = 'date';
|
|
10
11
|
disablePastDates = false;
|
|
12
|
+
showQuickActions = true;
|
|
13
|
+
minDate;
|
|
14
|
+
maxDate;
|
|
15
|
+
formatMonthFn;
|
|
16
|
+
formatYearFn;
|
|
17
|
+
formatDateFn;
|
|
18
|
+
dateSelect = new EventEmitter();
|
|
11
19
|
currentView = 'date';
|
|
12
20
|
viewDate = new Date();
|
|
13
21
|
selectedDate = null;
|
|
@@ -15,87 +23,222 @@ export class CalendarComponent {
|
|
|
15
23
|
daysInMonth = [];
|
|
16
24
|
months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'];
|
|
17
25
|
years = [];
|
|
26
|
+
yearRangeStart;
|
|
18
27
|
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');
|
|
28
|
+
quickActionBtnClass = cn('px-3 py-1.5 text-sm rounded hover:bg-accent text-muted-foreground hover:text-foreground transition-colors');
|
|
19
29
|
onTouched = () => { };
|
|
20
30
|
onChange = () => { };
|
|
21
31
|
cn = cn;
|
|
32
|
+
constructor() {
|
|
33
|
+
this.yearRangeStart = new Date().getFullYear() - 6;
|
|
34
|
+
}
|
|
22
35
|
ngOnInit() {
|
|
36
|
+
// Initialize based on mode
|
|
37
|
+
if (this.mode === 'month') {
|
|
38
|
+
this.currentView = 'month';
|
|
39
|
+
}
|
|
40
|
+
else if (this.mode === 'year') {
|
|
41
|
+
this.currentView = 'year';
|
|
42
|
+
}
|
|
23
43
|
this.generateDays();
|
|
24
44
|
this.generateYears();
|
|
25
45
|
}
|
|
46
|
+
// Format helpers
|
|
47
|
+
formatMonthYear(date, type) {
|
|
48
|
+
if (type === 'month' && this.formatMonthFn) {
|
|
49
|
+
return this.formatMonthFn(date);
|
|
50
|
+
}
|
|
51
|
+
if (type === 'year' && this.formatYearFn) {
|
|
52
|
+
return this.formatYearFn(date);
|
|
53
|
+
}
|
|
54
|
+
return type === 'month' ? format(date, 'MMMM') : format(date, 'yyyy');
|
|
55
|
+
}
|
|
56
|
+
formatDate(date, type) {
|
|
57
|
+
if (type === 'day' && this.formatDateFn) {
|
|
58
|
+
return this.formatDateFn(date);
|
|
59
|
+
}
|
|
60
|
+
return format(date, type === 'day' ? 'd' : type === 'month' ? 'MMM' : 'yyyy');
|
|
61
|
+
}
|
|
26
62
|
generateDays() {
|
|
63
|
+
if (this.mode !== 'date')
|
|
64
|
+
return;
|
|
27
65
|
const start = startOfWeek(startOfMonth(this.viewDate));
|
|
28
66
|
const end = endOfWeek(endOfMonth(this.viewDate));
|
|
29
67
|
this.daysInMonth = eachDayOfInterval({ start, end });
|
|
30
68
|
}
|
|
31
69
|
generateYears() {
|
|
32
70
|
const currentYear = this.viewDate.getFullYear();
|
|
33
|
-
|
|
34
|
-
|
|
71
|
+
if (this.mode === 'year') {
|
|
72
|
+
// For year picker, show a 12-year grid
|
|
73
|
+
this.years = Array.from({ length: 12 }, (_, i) => this.yearRangeStart + i);
|
|
74
|
+
}
|
|
75
|
+
else {
|
|
76
|
+
// For date mode year selector, show 16 years centered on current
|
|
77
|
+
this.years = Array.from({ length: 16 }, (_, i) => currentYear - 6 + i);
|
|
78
|
+
}
|
|
35
79
|
}
|
|
36
80
|
setView(view) {
|
|
37
81
|
this.currentView = view;
|
|
38
|
-
// If switching to year view, ensure the year grid is centered on current view year
|
|
39
82
|
if (view === 'year') {
|
|
40
83
|
this.generateYears();
|
|
41
84
|
}
|
|
42
85
|
}
|
|
43
86
|
prev() {
|
|
44
|
-
if (this.
|
|
45
|
-
|
|
46
|
-
|
|
87
|
+
if (this.mode === 'date') {
|
|
88
|
+
if (this.currentView === 'date') {
|
|
89
|
+
this.viewDate = subMonths(this.viewDate, 1);
|
|
90
|
+
this.generateDays();
|
|
91
|
+
}
|
|
92
|
+
else if (this.currentView === 'year') {
|
|
93
|
+
this.viewDate = subYears(this.viewDate, 16);
|
|
94
|
+
this.generateYears();
|
|
95
|
+
}
|
|
96
|
+
else if (this.currentView === 'month') {
|
|
97
|
+
this.viewDate = subYears(this.viewDate, 1);
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
else if (this.mode === 'month') {
|
|
101
|
+
this.viewDate = subYears(this.viewDate, 1);
|
|
47
102
|
}
|
|
48
|
-
else if (this.
|
|
49
|
-
this.
|
|
103
|
+
else if (this.mode === 'year') {
|
|
104
|
+
this.yearRangeStart -= 12;
|
|
50
105
|
this.generateYears();
|
|
51
106
|
}
|
|
52
|
-
else if (this.
|
|
53
|
-
|
|
107
|
+
else if (this.mode === 'month-year') {
|
|
108
|
+
if (this.currentView === 'month') {
|
|
109
|
+
this.viewDate = subYears(this.viewDate, 1);
|
|
110
|
+
}
|
|
111
|
+
else {
|
|
112
|
+
this.yearRangeStart -= 12;
|
|
113
|
+
this.generateYears();
|
|
114
|
+
}
|
|
54
115
|
}
|
|
55
116
|
}
|
|
56
117
|
next() {
|
|
57
|
-
if (this.
|
|
58
|
-
|
|
59
|
-
|
|
118
|
+
if (this.mode === 'date') {
|
|
119
|
+
if (this.currentView === 'date') {
|
|
120
|
+
this.viewDate = addMonths(this.viewDate, 1);
|
|
121
|
+
this.generateDays();
|
|
122
|
+
}
|
|
123
|
+
else if (this.currentView === 'year') {
|
|
124
|
+
this.viewDate = addYears(this.viewDate, 16);
|
|
125
|
+
this.generateYears();
|
|
126
|
+
}
|
|
127
|
+
else if (this.currentView === 'month') {
|
|
128
|
+
this.viewDate = addYears(this.viewDate, 1);
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
else if (this.mode === 'month') {
|
|
132
|
+
this.viewDate = addYears(this.viewDate, 1);
|
|
60
133
|
}
|
|
61
|
-
else if (this.
|
|
62
|
-
this.
|
|
134
|
+
else if (this.mode === 'year') {
|
|
135
|
+
this.yearRangeStart += 12;
|
|
63
136
|
this.generateYears();
|
|
64
137
|
}
|
|
65
|
-
else if (this.
|
|
66
|
-
|
|
138
|
+
else if (this.mode === 'month-year') {
|
|
139
|
+
if (this.currentView === 'month') {
|
|
140
|
+
this.viewDate = addYears(this.viewDate, 1);
|
|
141
|
+
}
|
|
142
|
+
else {
|
|
143
|
+
this.yearRangeStart += 12;
|
|
144
|
+
this.generateYears();
|
|
145
|
+
}
|
|
67
146
|
}
|
|
68
147
|
}
|
|
148
|
+
prevYears() {
|
|
149
|
+
this.yearRangeStart -= 12;
|
|
150
|
+
this.generateYears();
|
|
151
|
+
}
|
|
152
|
+
nextYears() {
|
|
153
|
+
this.yearRangeStart += 12;
|
|
154
|
+
this.generateYears();
|
|
155
|
+
}
|
|
69
156
|
selectDate(date) {
|
|
70
157
|
if (this.isDateDisabled(date))
|
|
71
158
|
return;
|
|
72
159
|
this.selectedDate = date;
|
|
73
|
-
if (!isSameMonth(date, this.viewDate)) {
|
|
74
|
-
this.viewDate = date;
|
|
75
|
-
this.generateDays();
|
|
76
|
-
}
|
|
77
160
|
this.onChange(date);
|
|
78
161
|
this.onTouched();
|
|
162
|
+
this.dateSelect.emit(date);
|
|
79
163
|
}
|
|
80
164
|
selectMonth(monthIndex) {
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
165
|
+
if (this.mode === 'date') {
|
|
166
|
+
this.viewDate = setMonth(this.viewDate, monthIndex);
|
|
167
|
+
this.currentView = 'date';
|
|
168
|
+
this.generateDays();
|
|
169
|
+
}
|
|
170
|
+
else if (this.mode === 'month') {
|
|
171
|
+
this.viewDate = setMonth(this.viewDate, monthIndex);
|
|
172
|
+
this.selectedDate = this.viewDate;
|
|
173
|
+
this.onChange(this.viewDate);
|
|
174
|
+
this.onTouched();
|
|
175
|
+
this.dateSelect.emit(this.viewDate);
|
|
176
|
+
}
|
|
84
177
|
}
|
|
85
178
|
selectYear(year) {
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
179
|
+
if (this.mode === 'date') {
|
|
180
|
+
this.viewDate = setYear(this.viewDate, year);
|
|
181
|
+
this.currentView = 'date';
|
|
182
|
+
this.generateDays();
|
|
183
|
+
}
|
|
184
|
+
else if (this.mode === 'year' || this.mode === 'month') {
|
|
185
|
+
this.viewDate = setYear(this.viewDate, year);
|
|
186
|
+
this.selectedDate = this.viewDate;
|
|
187
|
+
this.onChange(this.viewDate);
|
|
188
|
+
this.onTouched();
|
|
189
|
+
this.dateSelect.emit(this.viewDate);
|
|
190
|
+
}
|
|
89
191
|
}
|
|
90
192
|
getDayClass(date) {
|
|
91
193
|
const isSelected = this.selectedDate && isSameDay(date, this.selectedDate);
|
|
92
194
|
const isTodayDate = isToday(date);
|
|
93
195
|
const isOutside = !isSameMonth(date, this.viewDate);
|
|
94
196
|
const isDisabled = this.isDateDisabled(date);
|
|
95
|
-
return cn('h-9 w-9 p-0 font-normal text-sm rounded-md transition-all flex items-center justify-center', !isSelected && !isDisabled && 'hover:bg-accent hover:text-accent-foreground', isSelected && 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground
|
|
197
|
+
return cn('h-9 w-9 p-0 font-normal text-sm rounded-md transition-all flex items-center justify-center', !isSelected && !isDisabled && 'hover:bg-accent hover:text-accent-foreground', isSelected && 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground', !isSelected && isTodayDate && 'bg-accent text-accent-foreground', (isOutside || isDisabled) && 'text-muted-foreground opacity-50', isDisabled && 'cursor-not-allowed');
|
|
198
|
+
}
|
|
199
|
+
getMonthClass(monthIndex) {
|
|
200
|
+
const isSelected = this.selectedDate &&
|
|
201
|
+
this.selectedDate.getMonth() === monthIndex &&
|
|
202
|
+
this.selectedDate.getFullYear() === this.viewDate.getFullYear();
|
|
203
|
+
const isCurrent = new Date().getMonth() === monthIndex &&
|
|
204
|
+
new Date().getFullYear() === this.viewDate.getFullYear();
|
|
205
|
+
return cn('text-sm py-2.5 rounded-md transition-colors', isSelected ? 'bg-primary text-primary-foreground hover:bg-primary' :
|
|
206
|
+
isCurrent ? 'border border-primary/30 text-primary' :
|
|
207
|
+
'hover:bg-accent hover:text-accent-foreground');
|
|
208
|
+
}
|
|
209
|
+
getYearClass(year) {
|
|
210
|
+
const isSelected = this.selectedDate &&
|
|
211
|
+
this.selectedDate.getFullYear() === year;
|
|
212
|
+
const isCurrent = new Date().getFullYear() === year;
|
|
213
|
+
return cn('text-sm py-2 rounded-md transition-colors', isSelected ? 'bg-primary text-primary-foreground hover:bg-primary' :
|
|
214
|
+
isCurrent ? 'border border-primary/30 text-primary' :
|
|
215
|
+
'hover:bg-accent hover:text-accent-foreground');
|
|
96
216
|
}
|
|
97
217
|
isDateDisabled(date) {
|
|
98
|
-
|
|
218
|
+
if (this.disablePastDates && isBefore(date, startOfDay(new Date()))) {
|
|
219
|
+
return true;
|
|
220
|
+
}
|
|
221
|
+
if (this.minDate && isBefore(date, this.minDate)) {
|
|
222
|
+
return true;
|
|
223
|
+
}
|
|
224
|
+
if (this.maxDate && isBefore(this.maxDate, date)) {
|
|
225
|
+
return true;
|
|
226
|
+
}
|
|
227
|
+
return false;
|
|
228
|
+
}
|
|
229
|
+
isTodayDisabled() {
|
|
230
|
+
return this.isDateDisabled(new Date());
|
|
231
|
+
}
|
|
232
|
+
selectToday() {
|
|
233
|
+
if (!this.isTodayDisabled()) {
|
|
234
|
+
this.selectDate(new Date());
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
clear() {
|
|
238
|
+
this.selectedDate = null;
|
|
239
|
+
this.onChange(null);
|
|
240
|
+
this.onTouched();
|
|
241
|
+
this.dateSelect.emit(null);
|
|
99
242
|
}
|
|
100
243
|
// CVA Implementation
|
|
101
244
|
writeValue(obj) {
|
|
@@ -112,27 +255,26 @@ export class CalendarComponent {
|
|
|
112
255
|
registerOnChange(fn) { this.onChange = fn; }
|
|
113
256
|
registerOnTouched(fn) { this.onTouched = fn; }
|
|
114
257
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CalendarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
115
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: CalendarComponent, isStandalone: true, selector: "tolle-calendar", inputs: { class: "class", disablePastDates: "disablePastDates" }, providers: [
|
|
258
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: CalendarComponent, isStandalone: true, selector: "tolle-calendar", inputs: { class: "class", mode: "mode", disablePastDates: "disablePastDates", showQuickActions: "showQuickActions", minDate: "minDate", maxDate: "maxDate", formatMonthFn: "formatMonthFn", formatYearFn: "formatYearFn", formatDateFn: "formatDateFn" }, outputs: { dateSelect: "dateSelect" }, providers: [
|
|
116
259
|
{
|
|
117
260
|
provide: NG_VALUE_ACCESSOR,
|
|
118
261
|
useExisting: forwardRef(() => CalendarComponent),
|
|
119
262
|
multi: true
|
|
120
263
|
}
|
|
121
264
|
], ngImport: i0, template: `
|
|
122
|
-
<div [class]="cn('p-3 border rounded-md bg-background text-popover-foreground shadow-sm inline-block w-fit', class)">
|
|
123
|
-
|
|
265
|
+
<div [class]="cn('p-3 border rounded-md bg-background text-popover-foreground shadow-sm inline-block min-w-fit', class)">
|
|
266
|
+
<!-- Header with Navigation -->
|
|
124
267
|
<div class="flex items-center justify-between pt-1 pb-4 gap-2">
|
|
125
|
-
|
|
268
|
+
<!-- View Selector -->
|
|
126
269
|
<div class="flex items-center gap-1">
|
|
127
|
-
<button
|
|
270
|
+
<button *ngIf="mode !== 'year'"
|
|
128
271
|
type="button"
|
|
129
272
|
(click)="setView('month')"
|
|
130
273
|
[class]="cn(
|
|
131
274
|
'text-sm font-semibold px-2 py-1 rounded transition-colors',
|
|
132
275
|
currentView === 'month' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground'
|
|
133
|
-
)"
|
|
134
|
-
|
|
135
|
-
{{ viewDate | date: 'MMMM' }}
|
|
276
|
+
)">
|
|
277
|
+
{{ formatMonthYear(viewDate, 'month') }}
|
|
136
278
|
</button>
|
|
137
279
|
|
|
138
280
|
<button
|
|
@@ -141,12 +283,12 @@ export class CalendarComponent {
|
|
|
141
283
|
[class]="cn(
|
|
142
284
|
'text-sm font-semibold px-2 py-1 rounded transition-colors',
|
|
143
285
|
currentView === 'year' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground'
|
|
144
|
-
)"
|
|
145
|
-
|
|
146
|
-
{{ viewDate | date: 'yyyy' }}
|
|
286
|
+
)">
|
|
287
|
+
{{ formatMonthYear(viewDate, 'year') }}
|
|
147
288
|
</button>
|
|
148
289
|
</div>
|
|
149
290
|
|
|
291
|
+
<!-- Navigation Buttons -->
|
|
150
292
|
<div class="flex items-center space-x-1">
|
|
151
293
|
<button type="button" (click)="prev()" [class]="navBtnClass">
|
|
152
294
|
<i class="ri-arrow-left-s-line text-lg"></i>
|
|
@@ -157,7 +299,8 @@ export class CalendarComponent {
|
|
|
157
299
|
</div>
|
|
158
300
|
</div>
|
|
159
301
|
|
|
160
|
-
|
|
302
|
+
<!-- DATE MODE -->
|
|
303
|
+
<div *ngIf="currentView === 'date' && mode === 'date'" class="space-y-2 animate-in fade-in zoom-in-95 duration-200">
|
|
161
304
|
<div class="grid grid-cols-7 gap-1 w-full">
|
|
162
305
|
<span *ngFor="let day of weekDays" class="text-[0.8rem] text-muted-foreground font-normal text-center w-9">
|
|
163
306
|
{{ day }}
|
|
@@ -171,47 +314,66 @@ export class CalendarComponent {
|
|
|
171
314
|
[disabled]="isDateDisabled(date)"
|
|
172
315
|
[class]="getDayClass(date)"
|
|
173
316
|
>
|
|
174
|
-
{{ date
|
|
317
|
+
{{ formatDate(date, 'day') }}
|
|
175
318
|
</button>
|
|
176
319
|
</div>
|
|
177
320
|
</div>
|
|
178
321
|
|
|
179
|
-
|
|
322
|
+
<!-- MONTH SELECTOR (for date mode and month mode) -->
|
|
323
|
+
<div *ngIf="(currentView === 'month')"
|
|
324
|
+
class="grid grid-cols-3 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200">
|
|
180
325
|
<button
|
|
181
326
|
*ngFor="let month of months; let i = index"
|
|
182
327
|
type="button"
|
|
183
328
|
(click)="selectMonth(i)"
|
|
184
|
-
[class]="
|
|
185
|
-
'text-sm py-2.5 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors',
|
|
186
|
-
i === viewDate.getMonth() ? 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground' : ''
|
|
187
|
-
)"
|
|
329
|
+
[class]="getMonthClass(i)"
|
|
188
330
|
>
|
|
189
331
|
{{ month }}
|
|
190
332
|
</button>
|
|
191
333
|
</div>
|
|
192
334
|
|
|
193
|
-
|
|
335
|
+
<!-- YEAR SELECTOR (for date mode and year mode) -->
|
|
336
|
+
<div *ngIf="(currentView === 'year') "
|
|
337
|
+
class="grid grid-cols-4 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200">
|
|
194
338
|
<button
|
|
195
339
|
*ngFor="let year of years"
|
|
196
340
|
type="button"
|
|
197
341
|
(click)="selectYear(year)"
|
|
198
|
-
[class]="
|
|
199
|
-
'text-sm py-2 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors',
|
|
200
|
-
year === viewDate.getFullYear() ? 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground' : ''
|
|
201
|
-
)"
|
|
342
|
+
[class]="getYearClass(year)"
|
|
202
343
|
>
|
|
203
344
|
{{ year }}
|
|
204
345
|
</button>
|
|
205
346
|
</div>
|
|
347
|
+
|
|
348
|
+
<!-- Quick Actions -->
|
|
349
|
+
<div *ngIf="showQuickActions" class="border-t pt-3 mt-3">
|
|
350
|
+
<div class="flex items-center justify-between gap-2">
|
|
351
|
+
<button
|
|
352
|
+
type="button"
|
|
353
|
+
(click)="selectToday()"
|
|
354
|
+
[class]="quickActionBtnClass"
|
|
355
|
+
[disabled]="isTodayDisabled()"
|
|
356
|
+
>
|
|
357
|
+
Today
|
|
358
|
+
</button>
|
|
359
|
+
<button
|
|
360
|
+
type="button"
|
|
361
|
+
(click)="clear()"
|
|
362
|
+
[class]="quickActionBtnClass"
|
|
363
|
+
>
|
|
364
|
+
Clear
|
|
365
|
+
</button>
|
|
366
|
+
</div>
|
|
367
|
+
</div>
|
|
206
368
|
</div>
|
|
207
|
-
`, 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: "
|
|
369
|
+
`, 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: "ngmodule", type: FormsModule }] });
|
|
208
370
|
}
|
|
209
371
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CalendarComponent, decorators: [{
|
|
210
372
|
type: Component,
|
|
211
373
|
args: [{
|
|
212
374
|
selector: 'tolle-calendar',
|
|
213
375
|
standalone: true,
|
|
214
|
-
imports: [CommonModule],
|
|
376
|
+
imports: [CommonModule, FormsModule],
|
|
215
377
|
providers: [
|
|
216
378
|
{
|
|
217
379
|
provide: NG_VALUE_ACCESSOR,
|
|
@@ -220,20 +382,19 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
220
382
|
}
|
|
221
383
|
],
|
|
222
384
|
template: `
|
|
223
|
-
<div [class]="cn('p-3 border rounded-md bg-background text-popover-foreground shadow-sm inline-block w-fit', class)">
|
|
224
|
-
|
|
385
|
+
<div [class]="cn('p-3 border rounded-md bg-background text-popover-foreground shadow-sm inline-block min-w-fit', class)">
|
|
386
|
+
<!-- Header with Navigation -->
|
|
225
387
|
<div class="flex items-center justify-between pt-1 pb-4 gap-2">
|
|
226
|
-
|
|
388
|
+
<!-- View Selector -->
|
|
227
389
|
<div class="flex items-center gap-1">
|
|
228
|
-
<button
|
|
390
|
+
<button *ngIf="mode !== 'year'"
|
|
229
391
|
type="button"
|
|
230
392
|
(click)="setView('month')"
|
|
231
393
|
[class]="cn(
|
|
232
394
|
'text-sm font-semibold px-2 py-1 rounded transition-colors',
|
|
233
395
|
currentView === 'month' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground'
|
|
234
|
-
)"
|
|
235
|
-
|
|
236
|
-
{{ viewDate | date: 'MMMM' }}
|
|
396
|
+
)">
|
|
397
|
+
{{ formatMonthYear(viewDate, 'month') }}
|
|
237
398
|
</button>
|
|
238
399
|
|
|
239
400
|
<button
|
|
@@ -242,12 +403,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
242
403
|
[class]="cn(
|
|
243
404
|
'text-sm font-semibold px-2 py-1 rounded transition-colors',
|
|
244
405
|
currentView === 'year' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground'
|
|
245
|
-
)"
|
|
246
|
-
|
|
247
|
-
{{ viewDate | date: 'yyyy' }}
|
|
406
|
+
)">
|
|
407
|
+
{{ formatMonthYear(viewDate, 'year') }}
|
|
248
408
|
</button>
|
|
249
409
|
</div>
|
|
250
410
|
|
|
411
|
+
<!-- Navigation Buttons -->
|
|
251
412
|
<div class="flex items-center space-x-1">
|
|
252
413
|
<button type="button" (click)="prev()" [class]="navBtnClass">
|
|
253
414
|
<i class="ri-arrow-left-s-line text-lg"></i>
|
|
@@ -258,7 +419,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
258
419
|
</div>
|
|
259
420
|
</div>
|
|
260
421
|
|
|
261
|
-
|
|
422
|
+
<!-- DATE MODE -->
|
|
423
|
+
<div *ngIf="currentView === 'date' && mode === 'date'" class="space-y-2 animate-in fade-in zoom-in-95 duration-200">
|
|
262
424
|
<div class="grid grid-cols-7 gap-1 w-full">
|
|
263
425
|
<span *ngFor="let day of weekDays" class="text-[0.8rem] text-muted-foreground font-normal text-center w-9">
|
|
264
426
|
{{ day }}
|
|
@@ -272,44 +434,79 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
272
434
|
[disabled]="isDateDisabled(date)"
|
|
273
435
|
[class]="getDayClass(date)"
|
|
274
436
|
>
|
|
275
|
-
{{ date
|
|
437
|
+
{{ formatDate(date, 'day') }}
|
|
276
438
|
</button>
|
|
277
439
|
</div>
|
|
278
440
|
</div>
|
|
279
441
|
|
|
280
|
-
|
|
442
|
+
<!-- MONTH SELECTOR (for date mode and month mode) -->
|
|
443
|
+
<div *ngIf="(currentView === 'month')"
|
|
444
|
+
class="grid grid-cols-3 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200">
|
|
281
445
|
<button
|
|
282
446
|
*ngFor="let month of months; let i = index"
|
|
283
447
|
type="button"
|
|
284
448
|
(click)="selectMonth(i)"
|
|
285
|
-
[class]="
|
|
286
|
-
'text-sm py-2.5 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors',
|
|
287
|
-
i === viewDate.getMonth() ? 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground' : ''
|
|
288
|
-
)"
|
|
449
|
+
[class]="getMonthClass(i)"
|
|
289
450
|
>
|
|
290
451
|
{{ month }}
|
|
291
452
|
</button>
|
|
292
453
|
</div>
|
|
293
454
|
|
|
294
|
-
|
|
455
|
+
<!-- YEAR SELECTOR (for date mode and year mode) -->
|
|
456
|
+
<div *ngIf="(currentView === 'year') "
|
|
457
|
+
class="grid grid-cols-4 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200">
|
|
295
458
|
<button
|
|
296
459
|
*ngFor="let year of years"
|
|
297
460
|
type="button"
|
|
298
461
|
(click)="selectYear(year)"
|
|
299
|
-
[class]="
|
|
300
|
-
'text-sm py-2 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors',
|
|
301
|
-
year === viewDate.getFullYear() ? 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground' : ''
|
|
302
|
-
)"
|
|
462
|
+
[class]="getYearClass(year)"
|
|
303
463
|
>
|
|
304
464
|
{{ year }}
|
|
305
465
|
</button>
|
|
306
466
|
</div>
|
|
467
|
+
|
|
468
|
+
<!-- Quick Actions -->
|
|
469
|
+
<div *ngIf="showQuickActions" class="border-t pt-3 mt-3">
|
|
470
|
+
<div class="flex items-center justify-between gap-2">
|
|
471
|
+
<button
|
|
472
|
+
type="button"
|
|
473
|
+
(click)="selectToday()"
|
|
474
|
+
[class]="quickActionBtnClass"
|
|
475
|
+
[disabled]="isTodayDisabled()"
|
|
476
|
+
>
|
|
477
|
+
Today
|
|
478
|
+
</button>
|
|
479
|
+
<button
|
|
480
|
+
type="button"
|
|
481
|
+
(click)="clear()"
|
|
482
|
+
[class]="quickActionBtnClass"
|
|
483
|
+
>
|
|
484
|
+
Clear
|
|
485
|
+
</button>
|
|
486
|
+
</div>
|
|
487
|
+
</div>
|
|
307
488
|
</div>
|
|
308
489
|
`
|
|
309
490
|
}]
|
|
310
|
-
}], propDecorators: { class: [{
|
|
491
|
+
}], ctorParameters: () => [], propDecorators: { class: [{
|
|
492
|
+
type: Input
|
|
493
|
+
}], mode: [{
|
|
311
494
|
type: Input
|
|
312
495
|
}], disablePastDates: [{
|
|
313
496
|
type: Input
|
|
497
|
+
}], showQuickActions: [{
|
|
498
|
+
type: Input
|
|
499
|
+
}], minDate: [{
|
|
500
|
+
type: Input
|
|
501
|
+
}], maxDate: [{
|
|
502
|
+
type: Input
|
|
503
|
+
}], formatMonthFn: [{
|
|
504
|
+
type: Input
|
|
505
|
+
}], formatYearFn: [{
|
|
506
|
+
type: Input
|
|
507
|
+
}], formatDateFn: [{
|
|
508
|
+
type: Input
|
|
509
|
+
}], dateSelect: [{
|
|
510
|
+
type: Output
|
|
314
511
|
}] } });
|
|
315
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/calendar.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, Input, OnInit, forwardRef\n} 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, isBefore, startOfDay\n} from 'date-fns';\nimport { cn } from './utils/cn';\n\n@Component({\n  selector: 'tolle-calendar',\n  standalone: true,\n  imports: [CommonModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => CalendarComponent),\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 w-fit', class)\">\n\n      <div class=\"flex items-center justify-between pt-1 pb-4 gap-2\">\n\n        <div class=\"flex items-center gap-1\">\n          <button\n            type=\"button\"\n            (click)=\"setView('month')\"\n            [class]=\"cn(\n              'text-sm font-semibold px-2 py-1 rounded transition-colors',\n              currentView === 'month' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground'\n            )\"\n          >\n            {{ viewDate | date: 'MMMM' }}\n          </button>\n\n          <button\n            type=\"button\"\n            (click)=\"setView('year')\"\n            [class]=\"cn(\n              'text-sm font-semibold px-2 py-1 rounded transition-colors',\n              currentView === 'year' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground'\n            )\"\n          >\n            {{ viewDate | date: 'yyyy' }}\n          </button>\n        </div>\n\n        <div class=\"flex items-center space-x-1\">\n          <button type=\"button\" (click)=\"prev()\" [class]=\"navBtnClass\">\n            <i class=\"ri-arrow-left-s-line text-lg\"></i>\n          </button>\n          <button type=\"button\" (click)=\"next()\" [class]=\"navBtnClass\">\n            <i class=\"ri-arrow-right-s-line text-lg\"></i>\n          </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-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-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\n          *ngFor=\"let month of months; let i = index\"\n          type=\"button\"\n          (click)=\"selectMonth(i)\"\n          [class]=\"cn(\n            'text-sm py-2.5 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors',\n            i === viewDate.getMonth() ? 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground' : ''\n          )\"\n        >\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\n          *ngFor=\"let year of years\"\n          type=\"button\"\n          (click)=\"selectYear(year)\"\n          [class]=\"cn(\n            'text-sm py-2 rounded-md hover:bg-accent hover:text-accent-foreground transition-colors',\n            year === viewDate.getFullYear() ? 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground' : ''\n          )\"\n        >\n          {{ year }}\n        </button>\n      </div>\n    </div>\n  `\n})\nexport class CalendarComponent implements OnInit, ControlValueAccessor {\n  @Input() class = '';\n  @Input() disablePastDates = false;\n\n  currentView: 'date' | 'month' | 'year' = 'date';\n  viewDate: Date = new Date();\n  selectedDate: Date | null = 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: any) => void = () => {};\n\n  protected cn = cn;\n\n  ngOnInit() {\n    this.generateDays();\n    this.generateYears();\n  }\n\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    // Generates a 16-year window centered roughly on current view\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 switching to year view, ensure the year grid is centered on current view year\n    if (view === 'year') {\n      this.generateYears();\n    }\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  selectDate(date: Date) {\n    if (this.isDateDisabled(date)) return;\n\n    this.selectedDate = date;\n    if (!isSameMonth(date, this.viewDate)) {\n      this.viewDate = date;\n      this.generateDays();\n    }\n\n    this.onChange(date);\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'; // Jump straight back to date view for efficiency\n    this.generateDays();\n  }\n\n  getDayClass(date: Date) {\n    const isSelected = this.selectedDate && isSameDay(date, this.selectedDate);\n    const isTodayDate = isToday(date);\n    const isOutside = !isSameMonth(date, this.viewDate);\n    const isDisabled = this.isDateDisabled(date);\n\n    return cn(\n      'h-9 w-9 p-0 font-normal text-sm rounded-md transition-all flex items-center justify-center',\n      !isSelected && !isDisabled && 'hover:bg-accent hover:text-accent-foreground',\n      isSelected && 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground focus:bg-primary focus:text-primary-foreground',\n      !isSelected && isTodayDate && 'bg-accent text-accent-foreground',\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(obj: any): void {\n    if (obj) {\n      const date = new Date(obj);\n      if (!isNaN(date.getTime())) {\n        this.selectedDate = date;\n        this.viewDate = date;\n        this.generateDays();\n        this.generateYears();\n      }\n    }\n  }\n  registerOnChange(fn: any): void { this.onChange = fn; }\n  registerOnTouched(fn: any): void { this.onTouched = fn; }\n}\n"]}
|
|
512
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.component.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/calendar.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, Input, OnInit, forwardRef, Output, EventEmitter\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\nimport {\n  addMonths, subMonths, startOfMonth, endOfMonth,\n  startOfWeek, endOfWeek, eachDayOfInterval, isSameMonth,\n  isSameDay, isToday, setMonth, setYear, addYears, subYears,\n  isBefore, startOfDay, format, parseISO, isValid, isSameYear\n} from 'date-fns';\nimport { cn } from './utils/cn';\n\nexport type CalendarMode = 'date' | 'month' | 'year';\n\n@Component({\n  selector: 'tolle-calendar',\n  standalone: true,\n  imports: [CommonModule, FormsModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => CalendarComponent),\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      <!-- Header with Navigation -->\n      <div class=\"flex items-center justify-between pt-1 pb-4 gap-2\">\n        <!-- View Selector -->\n        <div class=\"flex items-center gap-1\">\n          <button *ngIf=\"mode !== 'year'\"\n            type=\"button\"\n            (click)=\"setView('month')\"\n            [class]=\"cn(\n              'text-sm font-semibold px-2 py-1 rounded transition-colors',\n              currentView === 'month' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground'\n            )\">\n            {{ formatMonthYear(viewDate, 'month') }}\n          </button>\n\n          <button\n            type=\"button\"\n            (click)=\"setView('year')\"\n            [class]=\"cn(\n              'text-sm font-semibold px-2 py-1 rounded transition-colors',\n              currentView === 'year' ? 'bg-secondary text-secondary-foreground' : 'hover:bg-accent hover:text-accent-foreground'\n            )\">\n            {{ formatMonthYear(viewDate, 'year') }}\n          </button>\n        </div>\n\n        <!-- Navigation Buttons -->\n        <div class=\"flex items-center space-x-1\">\n          <button type=\"button\" (click)=\"prev()\" [class]=\"navBtnClass\">\n            <i class=\"ri-arrow-left-s-line text-lg\"></i>\n          </button>\n          <button type=\"button\" (click)=\"next()\" [class]=\"navBtnClass\">\n            <i class=\"ri-arrow-right-s-line text-lg\"></i>\n          </button>\n        </div>\n      </div>\n\n      <!-- DATE MODE -->\n      <div *ngIf=\"currentView === 'date' && mode === 'date'\" class=\"space-y-2 animate-in fade-in zoom-in-95 duration-200\">\n        <div class=\"grid grid-cols-7 gap-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-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            {{ formatDate(date, 'day') }}\n          </button>\n        </div>\n      </div>\n\n      <!-- MONTH SELECTOR (for date mode and month mode) -->\n      <div *ngIf=\"(currentView === 'month')\"\n           class=\"grid grid-cols-3 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200\">\n        <button\n          *ngFor=\"let month of months; let i = index\"\n          type=\"button\"\n          (click)=\"selectMonth(i)\"\n          [class]=\"getMonthClass(i)\"\n        >\n          {{ month }}\n        </button>\n      </div>\n\n      <!-- YEAR SELECTOR (for date mode and year mode) -->\n      <div *ngIf=\"(currentView === 'year') \"\n           class=\"grid grid-cols-4 gap-2 w-64 animate-in fade-in zoom-in-95 duration-200\">\n        <button\n          *ngFor=\"let year of years\"\n          type=\"button\"\n          (click)=\"selectYear(year)\"\n          [class]=\"getYearClass(year)\"\n        >\n          {{ year }}\n        </button>\n      </div>\n\n      <!-- Quick Actions -->\n      <div *ngIf=\"showQuickActions\" class=\"border-t pt-3 mt-3\">\n        <div class=\"flex items-center justify-between gap-2\">\n          <button\n            type=\"button\"\n            (click)=\"selectToday()\"\n            [class]=\"quickActionBtnClass\"\n            [disabled]=\"isTodayDisabled()\"\n          >\n            Today\n          </button>\n          <button\n            type=\"button\"\n            (click)=\"clear()\"\n            [class]=\"quickActionBtnClass\"\n          >\n            Clear\n          </button>\n        </div>\n      </div>\n    </div>\n  `\n})\nexport class CalendarComponent implements OnInit, ControlValueAccessor {\n  @Input() class = '';\n  @Input() mode: CalendarMode = 'date';\n  @Input() disablePastDates = false;\n  @Input() showQuickActions = true;\n  @Input() minDate?: Date;\n  @Input() maxDate?: Date;\n  @Input() formatMonthFn?: (date: Date) => string;\n  @Input() formatYearFn?: (date: Date) => string;\n  @Input() formatDateFn?: (date: Date) => string;\n\n  @Output() dateSelect = new EventEmitter<Date | null>();\n\n  currentView: 'date' | 'month' | 'year' = 'date';\n  viewDate: Date = new Date();\n  selectedDate: Date | null = 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  yearRangeStart: 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  quickActionBtnClass = cn('px-3 py-1.5 text-sm rounded hover:bg-accent text-muted-foreground hover:text-foreground transition-colors');\n\n  onTouched: () => void = () => {};\n  onChange: (value: any) => void = () => {};\n\n  protected cn = cn;\n\n  constructor() {\n    this.yearRangeStart = new Date().getFullYear() - 6;\n  }\n\n  ngOnInit() {\n    // Initialize based on mode\n    if (this.mode === 'month') {\n      this.currentView = 'month';\n    } else if (this.mode === 'year') {\n      this.currentView = 'year';\n    }\n\n    this.generateDays();\n    this.generateYears();\n  }\n\n  // Format helpers\n  formatMonthYear(date: Date, type: 'month' | 'year'): string {\n    if (type === 'month' && this.formatMonthFn) {\n      return this.formatMonthFn(date);\n    }\n    if (type === 'year' && this.formatYearFn) {\n      return this.formatYearFn(date);\n    }\n    return type === 'month' ? format(date, 'MMMM') : format(date, 'yyyy');\n  }\n\n  formatDate(date: Date, type: 'day' | 'month' | 'year'): string {\n    if (type === 'day' && this.formatDateFn) {\n      return this.formatDateFn(date);\n    }\n    return format(date, type === 'day' ? 'd' : type === 'month' ? 'MMM' : 'yyyy');\n  }\n\n  generateDays() {\n    if (this.mode !== 'date') return;\n\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\n    if (this.mode === 'year') {\n      // For year picker, show a 12-year grid\n      this.years = Array.from({ length: 12 }, (_, i) => this.yearRangeStart + i);\n    } else {\n      // For date mode year selector, show 16 years centered on current\n      this.years = Array.from({ length: 16 }, (_, i) => currentYear - 6 + i);\n    }\n  }\n\n  setView(view: 'date' | 'month' | 'year') {\n    this.currentView = view;\n    if (view === 'year') {\n      this.generateYears();\n    }\n  }\n\n  prev() {\n    if (this.mode === 'date') {\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    } else if (this.mode === 'month') {\n      this.viewDate = subYears(this.viewDate, 1);\n    } else if (this.mode === 'year') {\n      this.yearRangeStart -= 12;\n      this.generateYears();\n    } else if (this.mode === 'month-year') {\n      if (this.currentView === 'month') {\n        this.viewDate = subYears(this.viewDate, 1);\n      } else {\n        this.yearRangeStart -= 12;\n        this.generateYears();\n      }\n    }\n  }\n\n  next() {\n    if (this.mode === 'date') {\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    } else if (this.mode === 'month') {\n      this.viewDate = addYears(this.viewDate, 1);\n    } else if (this.mode === 'year') {\n      this.yearRangeStart += 12;\n      this.generateYears();\n    } else if (this.mode === 'month-year') {\n      if (this.currentView === 'month') {\n        this.viewDate = addYears(this.viewDate, 1);\n      } else {\n        this.yearRangeStart += 12;\n        this.generateYears();\n      }\n    }\n  }\n\n  prevYears() {\n    this.yearRangeStart -= 12;\n    this.generateYears();\n  }\n\n  nextYears() {\n    this.yearRangeStart += 12;\n    this.generateYears();\n  }\n\n  selectDate(date: Date) {\n    if (this.isDateDisabled(date)) return;\n\n    this.selectedDate = date;\n    this.onChange(date);\n    this.onTouched();\n    this.dateSelect.emit(date);\n  }\n\n  selectMonth(monthIndex: number) {\n    if (this.mode === 'date') {\n      this.viewDate = setMonth(this.viewDate, monthIndex);\n      this.currentView = 'date';\n      this.generateDays();\n    } else if (this.mode === 'month') {\n      this.viewDate = setMonth(this.viewDate, monthIndex);\n      this.selectedDate =  this.viewDate;\n      this.onChange( this.viewDate);\n      this.onTouched();\n      this.dateSelect.emit( this.viewDate);\n    }\n  }\n\n  selectYear(year: number) {\n    if (this.mode === 'date') {\n      this.viewDate = setYear(this.viewDate, year);\n      this.currentView = 'date';\n      this.generateDays();\n    } else if (this.mode === 'year' || this.mode === 'month') {\n      this.viewDate = setYear(this.viewDate, year);\n      this.selectedDate =  this.viewDate;\n      this.onChange( this.viewDate);\n      this.onTouched();\n      this.dateSelect.emit( this.viewDate);\n    }\n  }\n\n  getDayClass(date: Date) {\n    const isSelected = this.selectedDate && isSameDay(date, this.selectedDate);\n    const isTodayDate = isToday(date);\n    const isOutside = !isSameMonth(date, this.viewDate);\n    const isDisabled = this.isDateDisabled(date);\n\n    return cn(\n      'h-9 w-9 p-0 font-normal text-sm rounded-md transition-all flex items-center justify-center',\n      !isSelected && !isDisabled && 'hover:bg-accent hover:text-accent-foreground',\n      isSelected && 'bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground',\n      !isSelected && isTodayDate && 'bg-accent text-accent-foreground',\n      (isOutside || isDisabled) && 'text-muted-foreground opacity-50',\n      isDisabled && 'cursor-not-allowed'\n    );\n  }\n\n  getMonthClass(monthIndex: number) {\n    const isSelected = this.selectedDate &&\n      this.selectedDate.getMonth() === monthIndex &&\n      this.selectedDate.getFullYear() === this.viewDate.getFullYear();\n    const isCurrent = new Date().getMonth() === monthIndex &&\n      new Date().getFullYear() === this.viewDate.getFullYear();\n\n    return cn(\n      'text-sm py-2.5 rounded-md transition-colors',\n      isSelected ? 'bg-primary text-primary-foreground hover:bg-primary' :\n        isCurrent ? 'border border-primary/30 text-primary' :\n          'hover:bg-accent hover:text-accent-foreground'\n    );\n  }\n\n  getYearClass(year: number) {\n    const isSelected = this.selectedDate &&\n      this.selectedDate.getFullYear() === year;\n    const isCurrent = new Date().getFullYear() === year;\n\n    return cn(\n      'text-sm py-2 rounded-md transition-colors',\n      isSelected ? 'bg-primary text-primary-foreground hover:bg-primary' :\n        isCurrent ? 'border border-primary/30 text-primary' :\n          'hover:bg-accent hover:text-accent-foreground'\n    );\n  }\n\n  isDateDisabled(date: Date): boolean {\n    if (this.disablePastDates && isBefore(date, startOfDay(new Date()))) {\n      return true;\n    }\n    if (this.minDate && isBefore(date, this.minDate)) {\n      return true;\n    }\n    if (this.maxDate && isBefore(this.maxDate, date)) {\n      return true;\n    }\n    return false;\n  }\n\n  isTodayDisabled(): boolean {\n    return this.isDateDisabled(new Date());\n  }\n\n  selectToday() {\n    if (!this.isTodayDisabled()) {\n      this.selectDate(new Date());\n    }\n  }\n\n  clear() {\n    this.selectedDate = null;\n    this.onChange(null);\n    this.onTouched();\n    this.dateSelect.emit(null);\n  }\n\n  // CVA Implementation\n  writeValue(obj: any): void {\n    if (obj) {\n      const date = new Date(obj);\n      if (!isNaN(date.getTime())) {\n        this.selectedDate = date;\n        this.viewDate = date;\n        this.generateDays();\n        this.generateYears();\n      }\n    }\n  }\n  registerOnChange(fn: any): void { this.onChange = fn; }\n  registerOnTouched(fn: any): void { this.onTouched = fn; }\n}\n"]}
|