pdm-ui-kit 0.1.21 → 0.1.23

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.
@@ -1,71 +1,472 @@
1
- import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';
2
2
  import * as i0 from "@angular/core";
3
3
  import * as i1 from "@angular/common";
4
+ const DEFAULT_VIEW_MONTH = new Date();
5
+ const DAY_MS = 24 * 60 * 60 * 1000;
4
6
  export class PdmCalendarComponent {
5
- constructor() {
6
- this.mode = 'single';
7
- this.month = 6;
8
- this.year = 2025;
9
- this.selectedDay = 25;
10
- this.rangeStartDay = 25;
11
- this.rangeEndDay = 9;
7
+ constructor(cdr) {
8
+ this.cdr = cdr;
9
+ this._value = null;
10
+ this._rangeValue = null;
11
+ this._month = null;
12
+ this.variant = 'single';
12
13
  this.className = '';
14
+ this.disabledDates = [];
15
+ this.minDate = null;
16
+ this.maxDate = null;
17
+ this.isDateDisabled = null;
18
+ this.allowSameDayRange = true;
19
+ this.readonly = false;
20
+ this.valueChange = new EventEmitter();
21
+ this.rangeValueChange = new EventEmitter();
22
+ this.monthChange = new EventEmitter();
23
+ this.dateClick = new EventEmitter();
24
+ this.disabledDateClick = new EventEmitter();
13
25
  this.weekdays = ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'];
26
+ this.monthOptions = [
27
+ { value: 0, label: 'Jan' },
28
+ { value: 1, label: 'Feb' },
29
+ { value: 2, label: 'Mar' },
30
+ { value: 3, label: 'Apr' },
31
+ { value: 4, label: 'May' },
32
+ { value: 5, label: 'Jun' },
33
+ { value: 6, label: 'Jul' },
34
+ { value: 7, label: 'Aug' },
35
+ { value: 8, label: 'Sep' },
36
+ { value: 9, label: 'Oct' },
37
+ { value: 10, label: 'Nov' },
38
+ { value: 11, label: 'Dec' }
39
+ ];
40
+ this.trackByIndex = (index) => {
41
+ return index;
42
+ };
43
+ this.trackByDate = (_index, cell) => {
44
+ return this.dateKey(cell.date);
45
+ };
14
46
  }
15
- get leftMonthName() {
16
- return this.monthName(this.month - 1);
17
- }
18
- get rightMonthName() {
19
- const rightMonth = this.month % 12;
20
- return this.monthName(rightMonth);
21
- }
22
- get leftCells() {
23
- return this.buildMonthCells(this.year, this.month - 1, this.selectedDay, this.mode === 'range' ? this.rangeStartDay : undefined, this.mode === 'range' ? 31 : undefined);
24
- }
25
- get rightCells() {
26
- const rightMonth = this.month % 12;
27
- const rightYear = this.month === 12 ? this.year + 1 : this.year;
28
- return this.buildMonthCells(rightYear, rightMonth, this.mode === 'range' ? this.rangeEndDay : undefined, 1, this.mode === 'range' ? this.rangeEndDay : undefined);
29
- }
30
- monthName(monthIndex) {
31
- const safe = ((monthIndex % 12) + 12) % 12;
32
- return new Date(this.year, safe, 1).toLocaleString('en-US', { month: 'long' });
33
- }
34
- buildMonthCells(year, monthIndex, selectedDay, rangeStartDay, rangeEndDay) {
35
- const firstDay = new Date(year, monthIndex, 1);
36
- const startWeekday = firstDay.getDay();
37
- const daysInMonth = new Date(year, monthIndex + 1, 0).getDate();
38
- const cells = [];
39
- for (let i = 0; i < 42; i += 1) {
40
- const dayNumber = i - startWeekday + 1;
41
- const date = new Date(year, monthIndex, dayNumber);
42
- const outside = dayNumber < 1 || dayNumber > daysInMonth;
43
- const day = date.getDate();
44
- const selected = !outside && !!selectedDay && day === selectedDay;
45
- const inRange = !outside && !!rangeStartDay && !!rangeEndDay && day >= rangeStartDay && day <= rangeEndDay;
46
- cells.push({ date, day, outside, selected, inRange });
47
- }
48
- return cells;
47
+ set value(value) {
48
+ this._value = this.normalizeDate(value);
49
+ }
50
+ get value() {
51
+ return this._value;
52
+ }
53
+ set rangeValue(value) {
54
+ this._rangeValue = value
55
+ ? {
56
+ start: this.normalizeDate(value.start),
57
+ end: this.normalizeDate(value.end)
58
+ }
59
+ : null;
60
+ }
61
+ get rangeValue() {
62
+ return this._rangeValue;
63
+ }
64
+ set month(value) {
65
+ this._month = value ? this.startOfMonth(value) : null;
66
+ }
67
+ get month() {
68
+ return this._month;
69
+ }
70
+ get resolvedVariant() {
71
+ return this.variant === 'range' ? 'range' : 'single';
72
+ }
73
+ get visibleMonths() {
74
+ const baseMonth = this.getAnchorMonth();
75
+ if (this.resolvedVariant === 'single') {
76
+ return [
77
+ {
78
+ date: baseMonth,
79
+ title: this.formatMonthYear(baseMonth),
80
+ titleStyle: 'dropdowns',
81
+ dropdownMonth: this.formatMonthShort(baseMonth),
82
+ dropdownYear: String(baseMonth.getFullYear()),
83
+ showPrevButton: true,
84
+ showNextButton: true,
85
+ weeks: this.buildMonthWeeks(baseMonth, false)
86
+ }
87
+ ];
88
+ }
89
+ const nextMonth = this.addMonths(baseMonth, 1);
90
+ return [
91
+ {
92
+ date: baseMonth,
93
+ title: this.formatMonthYear(baseMonth),
94
+ titleStyle: 'plain',
95
+ showPrevButton: true,
96
+ showNextButton: false,
97
+ weeks: this.buildMonthWeeks(baseMonth, true)
98
+ },
99
+ {
100
+ date: nextMonth,
101
+ title: this.formatMonthYear(nextMonth),
102
+ titleStyle: 'plain',
103
+ showPrevButton: false,
104
+ showNextButton: true,
105
+ weeks: this.buildMonthWeeks(nextMonth, true)
106
+ }
107
+ ];
108
+ }
109
+ get singleHeaderMonth() {
110
+ return this.getAnchorMonth().getMonth();
111
+ }
112
+ get singleHeaderYear() {
113
+ return this.getAnchorMonth().getFullYear();
114
+ }
115
+ get yearOptions() {
116
+ const currentYear = this.singleHeaderYear;
117
+ const minYear = this.minDate ? this.minDate.getFullYear() : currentYear - 100;
118
+ const maxYear = this.maxDate ? this.maxDate.getFullYear() : currentYear + 100;
119
+ const years = [];
120
+ for (let year = minYear; year <= maxYear; year += 1) {
121
+ years.push(year);
122
+ }
123
+ return years;
124
+ }
125
+ get rootClasses() {
126
+ return [
127
+ 'border-border bg-background relative rounded-[10px] border p-3 shadow-sm',
128
+ this.resolvedVariant === 'range'
129
+ ? 'inline-flex items-start justify-center gap-4 shrink-0 grow-0 basis-auto'
130
+ : 'inline-flex flex-col gap-4 shrink-0 grow-0 basis-auto',
131
+ this.className
132
+ ];
133
+ }
134
+ get rootStyle() {
135
+ const width = this.resolvedVariant === 'range' ? '488px' : '250px';
136
+ return {
137
+ width,
138
+ minWidth: width,
139
+ maxWidth: width,
140
+ minHeight: '293px',
141
+ flex: '0 0 auto',
142
+ alignSelf: 'flex-start'
143
+ };
144
+ }
145
+ monthPanelClasses(_index) {
146
+ return ['flex flex-col items-start', this.resolvedVariant === 'range' ? 'w-[224px] gap-4' : 'w-full gap-0'];
147
+ }
148
+ headerClasses(month) {
149
+ return ['flex w-full items-center justify-between', month.titleStyle === 'dropdowns' ? '' : 'mb-4'];
150
+ }
151
+ navButtonClasses() {
152
+ return [
153
+ 'flex h-8 w-8 items-center justify-center rounded-md text-foreground',
154
+ 'disabled:pointer-events-none disabled:opacity-40'
155
+ ];
156
+ }
157
+ navPlaceholderClasses() {
158
+ return ['h-7 w-7 shrink-0'];
159
+ }
160
+ dropdownWrapClasses() {
161
+ return ['flex w-40 items-center justify-center gap-2'];
162
+ }
163
+ dropdownClasses(widthClass) {
164
+ return ['flex h-8 items-center justify-center gap-1 px-1', widthClass];
165
+ }
166
+ dropdownSelectClasses() {
167
+ return [
168
+ 'text-foreground h-full bg-transparent text-sm font-medium leading-5',
169
+ 'appearance-none border-0 outline-none ring-0 focus:outline-none focus:ring-0 text-center cursor-pointer'
170
+ ];
171
+ }
172
+ get dropdownSelectStyle() {
173
+ return {
174
+ appearance: 'none',
175
+ WebkitAppearance: 'none',
176
+ MozAppearance: 'none',
177
+ background: 'transparent',
178
+ border: '0',
179
+ boxShadow: 'none',
180
+ outline: '0',
181
+ padding: '0',
182
+ margin: '0'
183
+ };
184
+ }
185
+ calendarGridWrapClasses() {
186
+ return ['flex w-full flex-col items-start'];
187
+ }
188
+ weekdayRowClasses() {
189
+ return ['flex w-full items-center'];
190
+ }
191
+ weekdayCellClasses() {
192
+ return ['text-muted-foreground flex h-[21px] w-8 items-center justify-center rounded-md text-xs leading-4'];
193
+ }
194
+ weekRowClasses() {
195
+ return ['flex w-full items-start pt-2'];
196
+ }
197
+ dayCellClasses(cell) {
198
+ return [
199
+ 'relative flex h-8 w-8 shrink-0 items-center justify-center',
200
+ cell.rangeFill ? 'bg-accent' : '',
201
+ cell.rangeLeftCap ? 'rounded-l-md' : '',
202
+ cell.rangeRightCap ? 'rounded-r-md' : ''
203
+ ];
204
+ }
205
+ dayButtonClasses(cell) {
206
+ return [
207
+ 'relative z-10 flex h-8 w-8 items-center justify-center rounded-md text-sm leading-5',
208
+ cell.selected ? 'bg-primary text-primary-foreground' : cell.rangeFill ? 'text-accent-foreground' : 'text-foreground',
209
+ cell.muted && !cell.rangeFill ? 'opacity-50' : '',
210
+ cell.disabled ? 'cursor-not-allowed opacity-40' : '',
211
+ !cell.disabled && !this.readonly && !cell.selected ? 'hover:bg-accent/70' : ''
212
+ ];
213
+ }
214
+ dayLabelClasses(_cell) {
215
+ return ['font-normal'];
216
+ }
217
+ goToPreviousMonth() {
218
+ this.setAnchorMonth(this.addMonths(this.getAnchorMonth(), -1));
219
+ }
220
+ goToNextMonth() {
221
+ this.setAnchorMonth(this.addMonths(this.getAnchorMonth(), 1));
222
+ }
223
+ onSingleMonthChange(monthValue) {
224
+ const month = Number(monthValue);
225
+ if (Number.isNaN(month) || month < 0 || month > 11) {
226
+ return;
227
+ }
228
+ const anchor = this.getAnchorMonth();
229
+ this.setAnchorMonth(new Date(anchor.getFullYear(), month, 1));
230
+ }
231
+ onSingleYearChange(yearValue) {
232
+ const year = Number(yearValue);
233
+ if (Number.isNaN(year)) {
234
+ return;
235
+ }
236
+ const anchor = this.getAnchorMonth();
237
+ this.setAnchorMonth(new Date(year, anchor.getMonth(), 1));
238
+ }
239
+ onDatePressed(cell) {
240
+ const date = this.cloneDate(cell.date);
241
+ if (cell.disabled) {
242
+ this.disabledDateClick.emit(date);
243
+ return;
244
+ }
245
+ if (this.readonly) {
246
+ return;
247
+ }
248
+ this.dateClick.emit(date);
249
+ if (this.resolvedVariant === 'single') {
250
+ this._value = date;
251
+ this.valueChange.emit(this.cloneDate(date));
252
+ this.syncVisibleMonthToDate(date);
253
+ this.cdr.markForCheck();
254
+ return;
255
+ }
256
+ this.handleRangeSelection(date);
257
+ this.cdr.markForCheck();
258
+ }
259
+ handleRangeSelection(date) {
260
+ const current = this._rangeValue;
261
+ const currentStart = current?.start ? this.cloneDate(current.start) : null;
262
+ const currentEnd = current?.end ? this.cloneDate(current.end) : null;
263
+ if (!currentStart || (currentStart && currentEnd)) {
264
+ this._rangeValue = { start: date, end: null };
265
+ this.rangeValueChange.emit({ start: this.cloneDate(date), end: null });
266
+ this.syncVisibleMonthToDate(date);
267
+ return;
268
+ }
269
+ if (this.isSameDay(currentStart, date) && !this.allowSameDayRange) {
270
+ this._rangeValue = { start: date, end: null };
271
+ this.rangeValueChange.emit({ start: this.cloneDate(date), end: null });
272
+ return;
273
+ }
274
+ const start = this.compareDate(date, currentStart) < 0 ? date : currentStart;
275
+ const end = this.compareDate(date, currentStart) < 0 ? currentStart : date;
276
+ if (this.rangeContainsBlockedDate(start, end)) {
277
+ this.disabledDateClick.emit(this.cloneDate(date));
278
+ return;
279
+ }
280
+ this._rangeValue = { start, end };
281
+ this.rangeValueChange.emit({ start: this.cloneDate(start), end: this.cloneDate(end) });
282
+ this.syncVisibleMonthToDate(start);
283
+ }
284
+ buildMonthWeeks(month, includeRange) {
285
+ const firstOfMonth = this.startOfMonth(month);
286
+ const start = this.startOfWeek(firstOfMonth);
287
+ const end = this.endOfWeek(this.endOfMonth(firstOfMonth));
288
+ const weeks = [];
289
+ let cursor = this.cloneDate(start);
290
+ while (this.compareDate(cursor, end) <= 0) {
291
+ const row = [];
292
+ for (let col = 0; col < 7; col += 1) {
293
+ row.push(this.buildCell(cursor, month, includeRange));
294
+ cursor = this.addDays(cursor, 1);
295
+ }
296
+ weeks.push(row);
297
+ }
298
+ for (const row of weeks) {
299
+ for (let col = 0; col < row.length; col += 1) {
300
+ const cell = row[col];
301
+ if (!cell.inRange) {
302
+ continue;
303
+ }
304
+ const leftInRow = col > 0 ? row[col - 1].inRange : false;
305
+ const rightInRow = col < row.length - 1 ? row[col + 1].inRange : false;
306
+ cell.rangeLeftCap = !leftInRow;
307
+ cell.rangeRightCap = !rightInRow;
308
+ }
309
+ }
310
+ return weeks;
311
+ }
312
+ buildCell(date, visibleMonth, includeRange) {
313
+ const normalized = this.cloneDate(date);
314
+ const inCurrentMonth = normalized.getMonth() === visibleMonth.getMonth() && normalized.getFullYear() === visibleMonth.getFullYear();
315
+ const disabled = this.isBlocked(normalized);
316
+ const selectedSingle = this.resolvedVariant === 'single' && !!this._value && this.isSameDay(normalized, this._value);
317
+ const rangeStart = includeRange && !!this._rangeValue?.start && this.isSameDay(normalized, this._rangeValue.start);
318
+ const rangeEnd = includeRange && !!this._rangeValue?.end && this.isSameDay(normalized, this._rangeValue.end);
319
+ const inRange = includeRange &&
320
+ !!this._rangeValue?.start &&
321
+ !!this._rangeValue?.end &&
322
+ this.compareDate(normalized, this._rangeValue.start) >= 0 &&
323
+ this.compareDate(normalized, this._rangeValue.end) <= 0;
324
+ return {
325
+ date: normalized,
326
+ label: normalized.getDate(),
327
+ muted: !inCurrentMonth,
328
+ disabled,
329
+ selected: selectedSingle || rangeStart || rangeEnd,
330
+ inRange,
331
+ rangeFill: inRange,
332
+ rangeLeftCap: false,
333
+ rangeRightCap: false
334
+ };
335
+ }
336
+ getAnchorMonth() {
337
+ if (this._month) {
338
+ return this.cloneDate(this._month);
339
+ }
340
+ if (this.resolvedVariant === 'single' && this._value) {
341
+ return this.startOfMonth(this._value);
342
+ }
343
+ if (this.resolvedVariant === 'range' && this._rangeValue?.start) {
344
+ return this.startOfMonth(this._rangeValue.start);
345
+ }
346
+ return this.startOfMonth(DEFAULT_VIEW_MONTH);
347
+ }
348
+ setAnchorMonth(month) {
349
+ this._month = this.startOfMonth(month);
350
+ this.monthChange.emit(this.cloneDate(this._month));
351
+ }
352
+ syncVisibleMonthToDate(date) {
353
+ const nextMonth = this.startOfMonth(date);
354
+ if (!this._month || !this.isSameMonth(this._month, nextMonth)) {
355
+ this._month = nextMonth;
356
+ this.monthChange.emit(this.cloneDate(nextMonth));
357
+ }
358
+ }
359
+ isBlocked(date) {
360
+ if (this.minDate && this.compareDate(date, this.minDate) < 0) {
361
+ return true;
362
+ }
363
+ if (this.maxDate && this.compareDate(date, this.maxDate) > 0) {
364
+ return true;
365
+ }
366
+ const blockedSet = new Set((this.disabledDates || []).map((item) => this.dateKey(item)));
367
+ if (blockedSet.has(this.dateKey(date))) {
368
+ return true;
369
+ }
370
+ return !!this.isDateDisabled?.(this.cloneDate(date));
371
+ }
372
+ rangeContainsBlockedDate(start, end) {
373
+ let cursor = this.cloneDate(start);
374
+ while (this.compareDate(cursor, end) <= 0) {
375
+ if (this.isBlocked(cursor)) {
376
+ return true;
377
+ }
378
+ cursor = this.addDays(cursor, 1);
379
+ }
380
+ return false;
381
+ }
382
+ normalizeDate(value) {
383
+ if (!(value instanceof Date) || Number.isNaN(value.getTime())) {
384
+ return null;
385
+ }
386
+ return new Date(value.getFullYear(), value.getMonth(), value.getDate());
387
+ }
388
+ cloneDate(date) {
389
+ return new Date(date.getFullYear(), date.getMonth(), date.getDate());
390
+ }
391
+ startOfMonth(date) {
392
+ return new Date(date.getFullYear(), date.getMonth(), 1);
393
+ }
394
+ endOfMonth(date) {
395
+ return new Date(date.getFullYear(), date.getMonth() + 1, 0);
396
+ }
397
+ startOfWeek(date) {
398
+ return this.addDays(date, -date.getDay());
399
+ }
400
+ endOfWeek(date) {
401
+ return this.addDays(date, 6 - date.getDay());
402
+ }
403
+ addMonths(date, months) {
404
+ return new Date(date.getFullYear(), date.getMonth() + months, 1);
405
+ }
406
+ addDays(date, days) {
407
+ return new Date(date.getFullYear(), date.getMonth(), date.getDate() + days);
408
+ }
409
+ compareDate(a, b) {
410
+ return this.dateValue(a) - this.dateValue(b);
411
+ }
412
+ dateValue(date) {
413
+ return Date.UTC(date.getFullYear(), date.getMonth(), date.getDate()) / DAY_MS;
414
+ }
415
+ isSameDay(a, b) {
416
+ return this.compareDate(a, b) === 0;
417
+ }
418
+ isSameMonth(a, b) {
419
+ return a.getFullYear() === b.getFullYear() && a.getMonth() === b.getMonth();
420
+ }
421
+ dateKey(date) {
422
+ const y = date.getFullYear();
423
+ const m = String(date.getMonth() + 1).padStart(2, '0');
424
+ const d = String(date.getDate()).padStart(2, '0');
425
+ return `${y}-${m}-${d}`;
426
+ }
427
+ formatMonthShort(date) {
428
+ return date.toLocaleString('en-US', { month: 'short' });
429
+ }
430
+ formatMonthYear(date) {
431
+ return date.toLocaleString('en-US', { month: 'long', year: 'numeric' });
49
432
  }
50
433
  }
51
- PdmCalendarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PdmCalendarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
52
- PdmCalendarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: PdmCalendarComponent, selector: "pdm-calendar", inputs: { mode: "mode", month: "month", year: "year", selectedDay: "selectedDay", rangeStartDay: "rangeStartDay", rangeEndDay: "rangeEndDay", className: "className" }, ngImport: i0, template: "<section\n [ngClass]=\"[\n 'rounded-lg border border-border bg-background p-3 shadow-sm',\n mode === 'range' ? 'w-full max-w-2xl' : 'w-full max-w-sm',\n className\n ]\"\n>\n <div [ngClass]=\"['grid gap-4', mode === 'range' ? 'grid-cols-2' : 'grid-cols-1']\">\n <div class=\"space-y-4\">\n <header class=\"flex items-center justify-between\">\n <button type=\"button\" class=\"inline-flex h-8 w-8 items-center justify-center rounded-md text-foreground hover:bg-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\">\u2039</button>\n <div [ngClass]=\"['flex items-center justify-center gap-1', mode === 'range' ? 'w-full' : 'w-full']\">\n <ng-container *ngIf=\"mode === 'single'; else monthText\">\n <span class=\"inline-flex h-8 items-center rounded-md border border-border px-2 text-sm font-medium text-foreground shadow-sm\">Jun</span>\n <span class=\"inline-flex h-8 items-center rounded-md border border-border px-2 text-sm font-medium text-foreground shadow-sm\">2025</span>\n </ng-container>\n <ng-template #monthText>\n <span class=\"text-sm font-medium text-foreground\">{{ leftMonthName }} {{ year }}</span>\n </ng-template>\n </div>\n <button type=\"button\" class=\"inline-flex h-8 w-8 items-center justify-center rounded-md text-foreground hover:bg-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\">{{ mode === 'range' ? '' : '\u203A' }}</button>\n </header>\n\n <div>\n <div class=\"grid grid-cols-7\">\n <span *ngFor=\"let day of weekdays\" class=\"inline-flex h-5 items-center justify-center text-xs font-normal text-muted-foreground\">{{ day }}</span>\n </div>\n <div class=\"mt-2 grid grid-cols-7 gap-y-2\">\n <button\n *ngFor=\"let cell of leftCells\"\n type=\"button\"\n [ngClass]=\"[\n 'inline-flex h-8 w-8 items-center justify-center rounded-md text-sm font-normal transition-colors hover:bg-accent hover:text-accent-foreground',\n cell.outside ? 'text-muted-foreground opacity-50' : 'text-foreground',\n cell.inRange ? 'bg-accent text-accent-foreground' : '',\n cell.selected ? 'bg-primary text-primary-foreground' : ''\n ]\"\n >\n {{ cell.day }}\n </button>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"mode === 'range'\" class=\"space-y-4\">\n <header class=\"flex items-center justify-between\">\n <span class=\"text-sm font-medium text-foreground\">{{ rightMonthName }} {{ month === 12 ? year + 1 : year }}</span>\n <button type=\"button\" class=\"inline-flex h-8 w-8 items-center justify-center rounded-md text-foreground hover:bg-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\">\u203A</button>\n </header>\n\n <div>\n <div class=\"grid grid-cols-7\">\n <span *ngFor=\"let day of weekdays\" class=\"inline-flex h-5 items-center justify-center text-xs font-normal text-muted-foreground\">{{ day }}</span>\n </div>\n <div class=\"mt-2 grid grid-cols-7 gap-y-2\">\n <button\n *ngFor=\"let cell of rightCells\"\n type=\"button\"\n [ngClass]=\"[\n 'inline-flex h-8 w-8 items-center justify-center rounded-md text-sm font-normal transition-colors hover:bg-accent hover:text-accent-foreground',\n cell.outside ? 'text-muted-foreground opacity-50' : 'text-foreground',\n cell.inRange ? 'bg-accent text-accent-foreground' : '',\n cell.selected ? 'bg-primary text-primary-foreground' : ''\n ]\"\n >\n {{ cell.day }}\n </button>\n </div>\n </div>\n </div>\n </div>\n</section>\n", dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
434
+ PdmCalendarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PdmCalendarComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
435
+ PdmCalendarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: PdmCalendarComponent, selector: "pdm-calendar", inputs: { variant: "variant", className: "className", disabledDates: "disabledDates", minDate: "minDate", maxDate: "maxDate", isDateDisabled: "isDateDisabled", allowSameDayRange: "allowSameDayRange", readonly: "readonly", value: "value", rangeValue: "rangeValue", month: "month" }, outputs: { valueChange: "valueChange", rangeValueChange: "rangeValueChange", monthChange: "monthChange", dateClick: "dateClick", disabledDateClick: "disabledDateClick" }, ngImport: i0, template: "<div [ngClass]=\"rootClasses\" [ngStyle]=\"rootStyle\">\n <div *ngFor=\"let month of visibleMonths; let monthIndex = index; trackBy: trackByIndex\" [ngClass]=\"monthPanelClasses(monthIndex)\">\n <div [ngClass]=\"headerClasses(month)\">\n <button\n *ngIf=\"month.showPrevButton; else prevPlaceholder\"\n type=\"button\"\n [ngClass]=\"navButtonClasses()\"\n aria-label=\"Previous month\"\n (click)=\"goToPreviousMonth()\"\n [disabled]=\"readonly\"\n >\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n <path d=\"m15 18-6-6 6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n <ng-template #prevPlaceholder>\n <div [ngClass]=\"navPlaceholderClasses()\" aria-hidden=\"true\"></div>\n </ng-template>\n\n <ng-container *ngIf=\"month.titleStyle === 'dropdowns'; else plainTitle\">\n <div [ngClass]=\"dropdownWrapClasses()\">\n <div [ngClass]=\"dropdownClasses('w-[72px]')\">\n <select\n [ngClass]=\"dropdownSelectClasses()\"\n [ngStyle]=\"dropdownSelectStyle\"\n [value]=\"singleHeaderMonth\"\n aria-label=\"Month\"\n (change)=\"onSingleMonthChange($any($event.target).value)\"\n >\n <option *ngFor=\"let monthOption of monthOptions\" [value]=\"monthOption.value\">{{ monthOption.label }}</option>\n </select>\n <svg viewBox=\"0 0 24 24\" class=\"h-3 w-3 text-foreground\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n <path d=\"m6 9 6 6 6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </div>\n <div [ngClass]=\"dropdownClasses('w-[82px]')\">\n <select\n [ngClass]=\"dropdownSelectClasses()\"\n [ngStyle]=\"dropdownSelectStyle\"\n [value]=\"singleHeaderYear\"\n aria-label=\"Year\"\n (change)=\"onSingleYearChange($any($event.target).value)\"\n >\n <option *ngFor=\"let year of yearOptions\" [value]=\"year\">{{ year }}</option>\n </select>\n <svg viewBox=\"0 0 24 24\" class=\"h-3 w-3 text-foreground\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n <path d=\"m6 9 6 6 6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </div>\n </div>\n </ng-container>\n\n <ng-template #plainTitle>\n <div class=\"flex min-w-0 flex-1 items-center justify-center\">\n <p class=\"text-foreground text-center text-sm font-medium leading-5\">\n {{ month.title }}\n </p>\n </div>\n </ng-template>\n\n <button\n *ngIf=\"month.showNextButton; else nextPlaceholder\"\n type=\"button\"\n [ngClass]=\"navButtonClasses()\"\n aria-label=\"Next month\"\n (click)=\"goToNextMonth()\"\n [disabled]=\"readonly\"\n >\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n <path d=\"m9 18 6-6-6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n <ng-template #nextPlaceholder>\n <div [ngClass]=\"navPlaceholderClasses()\" aria-hidden=\"true\"></div>\n </ng-template>\n </div>\n\n <div [ngClass]=\"calendarGridWrapClasses()\">\n <div [ngClass]=\"weekdayRowClasses()\">\n <div *ngFor=\"let day of weekdays; trackBy: trackByIndex\" [ngClass]=\"weekdayCellClasses()\">\n <span>{{ day }}</span>\n </div>\n </div>\n\n <div *ngFor=\"let week of month.weeks; trackBy: trackByIndex\" [ngClass]=\"weekRowClasses()\">\n <div *ngFor=\"let cell of week; trackBy: trackByDate\" [ngClass]=\"dayCellClasses(cell)\">\n <button\n type=\"button\"\n [ngClass]=\"dayButtonClasses(cell)\"\n [disabled]=\"readonly\"\n [attr.aria-selected]=\"cell.selected\"\n [attr.aria-disabled]=\"cell.disabled || readonly\"\n [attr.title]=\"cell.date | date : 'yyyy-MM-dd'\"\n (click)=\"onDatePressed(cell)\"\n >\n <span [ngClass]=\"dayLabelClasses(cell)\">{{ cell.label }}</span>\n </button>\n </div>\n </div>\n </div>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "pipe", type: i1.DatePipe, name: "date" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
53
436
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PdmCalendarComponent, decorators: [{
54
437
  type: Component,
55
- args: [{ selector: 'pdm-calendar', changeDetection: ChangeDetectionStrategy.OnPush, template: "<section\n [ngClass]=\"[\n 'rounded-lg border border-border bg-background p-3 shadow-sm',\n mode === 'range' ? 'w-full max-w-2xl' : 'w-full max-w-sm',\n className\n ]\"\n>\n <div [ngClass]=\"['grid gap-4', mode === 'range' ? 'grid-cols-2' : 'grid-cols-1']\">\n <div class=\"space-y-4\">\n <header class=\"flex items-center justify-between\">\n <button type=\"button\" class=\"inline-flex h-8 w-8 items-center justify-center rounded-md text-foreground hover:bg-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\">\u2039</button>\n <div [ngClass]=\"['flex items-center justify-center gap-1', mode === 'range' ? 'w-full' : 'w-full']\">\n <ng-container *ngIf=\"mode === 'single'; else monthText\">\n <span class=\"inline-flex h-8 items-center rounded-md border border-border px-2 text-sm font-medium text-foreground shadow-sm\">Jun</span>\n <span class=\"inline-flex h-8 items-center rounded-md border border-border px-2 text-sm font-medium text-foreground shadow-sm\">2025</span>\n </ng-container>\n <ng-template #monthText>\n <span class=\"text-sm font-medium text-foreground\">{{ leftMonthName }} {{ year }}</span>\n </ng-template>\n </div>\n <button type=\"button\" class=\"inline-flex h-8 w-8 items-center justify-center rounded-md text-foreground hover:bg-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\">{{ mode === 'range' ? '' : '\u203A' }}</button>\n </header>\n\n <div>\n <div class=\"grid grid-cols-7\">\n <span *ngFor=\"let day of weekdays\" class=\"inline-flex h-5 items-center justify-center text-xs font-normal text-muted-foreground\">{{ day }}</span>\n </div>\n <div class=\"mt-2 grid grid-cols-7 gap-y-2\">\n <button\n *ngFor=\"let cell of leftCells\"\n type=\"button\"\n [ngClass]=\"[\n 'inline-flex h-8 w-8 items-center justify-center rounded-md text-sm font-normal transition-colors hover:bg-accent hover:text-accent-foreground',\n cell.outside ? 'text-muted-foreground opacity-50' : 'text-foreground',\n cell.inRange ? 'bg-accent text-accent-foreground' : '',\n cell.selected ? 'bg-primary text-primary-foreground' : ''\n ]\"\n >\n {{ cell.day }}\n </button>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"mode === 'range'\" class=\"space-y-4\">\n <header class=\"flex items-center justify-between\">\n <span class=\"text-sm font-medium text-foreground\">{{ rightMonthName }} {{ month === 12 ? year + 1 : year }}</span>\n <button type=\"button\" class=\"inline-flex h-8 w-8 items-center justify-center rounded-md text-foreground hover:bg-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\">\u203A</button>\n </header>\n\n <div>\n <div class=\"grid grid-cols-7\">\n <span *ngFor=\"let day of weekdays\" class=\"inline-flex h-5 items-center justify-center text-xs font-normal text-muted-foreground\">{{ day }}</span>\n </div>\n <div class=\"mt-2 grid grid-cols-7 gap-y-2\">\n <button\n *ngFor=\"let cell of rightCells\"\n type=\"button\"\n [ngClass]=\"[\n 'inline-flex h-8 w-8 items-center justify-center rounded-md text-sm font-normal transition-colors hover:bg-accent hover:text-accent-foreground',\n cell.outside ? 'text-muted-foreground opacity-50' : 'text-foreground',\n cell.inRange ? 'bg-accent text-accent-foreground' : '',\n cell.selected ? 'bg-primary text-primary-foreground' : ''\n ]\"\n >\n {{ cell.day }}\n </button>\n </div>\n </div>\n </div>\n </div>\n</section>\n" }]
56
- }], propDecorators: { mode: [{
438
+ args: [{ selector: 'pdm-calendar', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div [ngClass]=\"rootClasses\" [ngStyle]=\"rootStyle\">\n <div *ngFor=\"let month of visibleMonths; let monthIndex = index; trackBy: trackByIndex\" [ngClass]=\"monthPanelClasses(monthIndex)\">\n <div [ngClass]=\"headerClasses(month)\">\n <button\n *ngIf=\"month.showPrevButton; else prevPlaceholder\"\n type=\"button\"\n [ngClass]=\"navButtonClasses()\"\n aria-label=\"Previous month\"\n (click)=\"goToPreviousMonth()\"\n [disabled]=\"readonly\"\n >\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n <path d=\"m15 18-6-6 6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n <ng-template #prevPlaceholder>\n <div [ngClass]=\"navPlaceholderClasses()\" aria-hidden=\"true\"></div>\n </ng-template>\n\n <ng-container *ngIf=\"month.titleStyle === 'dropdowns'; else plainTitle\">\n <div [ngClass]=\"dropdownWrapClasses()\">\n <div [ngClass]=\"dropdownClasses('w-[72px]')\">\n <select\n [ngClass]=\"dropdownSelectClasses()\"\n [ngStyle]=\"dropdownSelectStyle\"\n [value]=\"singleHeaderMonth\"\n aria-label=\"Month\"\n (change)=\"onSingleMonthChange($any($event.target).value)\"\n >\n <option *ngFor=\"let monthOption of monthOptions\" [value]=\"monthOption.value\">{{ monthOption.label }}</option>\n </select>\n <svg viewBox=\"0 0 24 24\" class=\"h-3 w-3 text-foreground\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n <path d=\"m6 9 6 6 6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </div>\n <div [ngClass]=\"dropdownClasses('w-[82px]')\">\n <select\n [ngClass]=\"dropdownSelectClasses()\"\n [ngStyle]=\"dropdownSelectStyle\"\n [value]=\"singleHeaderYear\"\n aria-label=\"Year\"\n (change)=\"onSingleYearChange($any($event.target).value)\"\n >\n <option *ngFor=\"let year of yearOptions\" [value]=\"year\">{{ year }}</option>\n </select>\n <svg viewBox=\"0 0 24 24\" class=\"h-3 w-3 text-foreground\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n <path d=\"m6 9 6 6 6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </div>\n </div>\n </ng-container>\n\n <ng-template #plainTitle>\n <div class=\"flex min-w-0 flex-1 items-center justify-center\">\n <p class=\"text-foreground text-center text-sm font-medium leading-5\">\n {{ month.title }}\n </p>\n </div>\n </ng-template>\n\n <button\n *ngIf=\"month.showNextButton; else nextPlaceholder\"\n type=\"button\"\n [ngClass]=\"navButtonClasses()\"\n aria-label=\"Next month\"\n (click)=\"goToNextMonth()\"\n [disabled]=\"readonly\"\n >\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n <path d=\"m9 18 6-6-6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n <ng-template #nextPlaceholder>\n <div [ngClass]=\"navPlaceholderClasses()\" aria-hidden=\"true\"></div>\n </ng-template>\n </div>\n\n <div [ngClass]=\"calendarGridWrapClasses()\">\n <div [ngClass]=\"weekdayRowClasses()\">\n <div *ngFor=\"let day of weekdays; trackBy: trackByIndex\" [ngClass]=\"weekdayCellClasses()\">\n <span>{{ day }}</span>\n </div>\n </div>\n\n <div *ngFor=\"let week of month.weeks; trackBy: trackByIndex\" [ngClass]=\"weekRowClasses()\">\n <div *ngFor=\"let cell of week; trackBy: trackByDate\" [ngClass]=\"dayCellClasses(cell)\">\n <button\n type=\"button\"\n [ngClass]=\"dayButtonClasses(cell)\"\n [disabled]=\"readonly\"\n [attr.aria-selected]=\"cell.selected\"\n [attr.aria-disabled]=\"cell.disabled || readonly\"\n [attr.title]=\"cell.date | date : 'yyyy-MM-dd'\"\n (click)=\"onDatePressed(cell)\"\n >\n <span [ngClass]=\"dayLabelClasses(cell)\">{{ cell.label }}</span>\n </button>\n </div>\n </div>\n </div>\n </div>\n</div>\n" }]
439
+ }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { variant: [{
57
440
  type: Input
58
- }], month: [{
441
+ }], className: [{
59
442
  type: Input
60
- }], year: [{
443
+ }], disabledDates: [{
61
444
  type: Input
62
- }], selectedDay: [{
445
+ }], minDate: [{
63
446
  type: Input
64
- }], rangeStartDay: [{
447
+ }], maxDate: [{
65
448
  type: Input
66
- }], rangeEndDay: [{
449
+ }], isDateDisabled: [{
67
450
  type: Input
68
- }], className: [{
451
+ }], allowSameDayRange: [{
452
+ type: Input
453
+ }], readonly: [{
454
+ type: Input
455
+ }], valueChange: [{
456
+ type: Output
457
+ }], rangeValueChange: [{
458
+ type: Output
459
+ }], monthChange: [{
460
+ type: Output
461
+ }], dateClick: [{
462
+ type: Output
463
+ }], disabledDateClick: [{
464
+ type: Output
465
+ }], value: [{
466
+ type: Input
467
+ }], rangeValue: [{
468
+ type: Input
469
+ }], month: [{
69
470
  type: Input
70
471
  }] } });
71
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.component.js","sourceRoot":"","sources":["../../../../../../src/lib/components/calendar/calendar.component.ts","../../../../../../src/lib/components/calendar/calendar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input } from '@angular/core';\n\nexport type PdmCalendarMode = 'single' | 'range';\n\ninterface CalendarCell {\n  date: Date;\n  day: number;\n  outside: boolean;\n  selected: boolean;\n  inRange: boolean;\n}\n\n@Component({\n  selector: 'pdm-calendar',\n  templateUrl: './calendar.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PdmCalendarComponent {\n  @Input() mode: PdmCalendarMode = 'single';\n  @Input() month = 6;\n  @Input() year = 2025;\n  @Input() selectedDay = 25;\n  @Input() rangeStartDay = 25;\n  @Input() rangeEndDay = 9;\n  @Input() className = '';\n\n  readonly weekdays = ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'];\n\n  get leftMonthName(): string {\n    return this.monthName(this.month - 1);\n  }\n\n  get rightMonthName(): string {\n    const rightMonth = this.month % 12;\n    return this.monthName(rightMonth);\n  }\n\n  get leftCells(): CalendarCell[] {\n    return this.buildMonthCells(this.year, this.month - 1, this.selectedDay, this.mode === 'range' ? this.rangeStartDay : undefined, this.mode === 'range' ? 31 : undefined);\n  }\n\n  get rightCells(): CalendarCell[] {\n    const rightMonth = this.month % 12;\n    const rightYear = this.month === 12 ? this.year + 1 : this.year;\n    return this.buildMonthCells(rightYear, rightMonth, this.mode === 'range' ? this.rangeEndDay : undefined, 1, this.mode === 'range' ? this.rangeEndDay : undefined);\n  }\n\n  private monthName(monthIndex: number): string {\n    const safe = ((monthIndex % 12) + 12) % 12;\n    return new Date(this.year, safe, 1).toLocaleString('en-US', { month: 'long' });\n  }\n\n  private buildMonthCells(\n    year: number,\n    monthIndex: number,\n    selectedDay?: number,\n    rangeStartDay?: number,\n    rangeEndDay?: number\n  ): CalendarCell[] {\n    const firstDay = new Date(year, monthIndex, 1);\n    const startWeekday = firstDay.getDay();\n    const daysInMonth = new Date(year, monthIndex + 1, 0).getDate();\n    const cells: CalendarCell[] = [];\n\n    for (let i = 0; i < 42; i += 1) {\n      const dayNumber = i - startWeekday + 1;\n      const date = new Date(year, monthIndex, dayNumber);\n      const outside = dayNumber < 1 || dayNumber > daysInMonth;\n      const day = date.getDate();\n      const selected = !outside && !!selectedDay && day === selectedDay;\n      const inRange = !outside && !!rangeStartDay && !!rangeEndDay && day >= rangeStartDay && day <= rangeEndDay;\n      cells.push({ date, day, outside, selected, inRange });\n    }\n\n    return cells;\n  }\n}\n\n","<section\n  [ngClass]=\"[\n    'rounded-lg border border-border bg-background p-3 shadow-sm',\n    mode === 'range' ? 'w-full max-w-2xl' : 'w-full max-w-sm',\n    className\n  ]\"\n>\n  <div [ngClass]=\"['grid gap-4', mode === 'range' ? 'grid-cols-2' : 'grid-cols-1']\">\n    <div class=\"space-y-4\">\n      <header class=\"flex items-center justify-between\">\n        <button type=\"button\" class=\"inline-flex h-8 w-8 items-center justify-center rounded-md text-foreground hover:bg-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\">‹</button>\n        <div [ngClass]=\"['flex items-center justify-center gap-1', mode === 'range' ? 'w-full' : 'w-full']\">\n          <ng-container *ngIf=\"mode === 'single'; else monthText\">\n            <span class=\"inline-flex h-8 items-center rounded-md border border-border px-2 text-sm font-medium text-foreground shadow-sm\">Jun</span>\n            <span class=\"inline-flex h-8 items-center rounded-md border border-border px-2 text-sm font-medium text-foreground shadow-sm\">2025</span>\n          </ng-container>\n          <ng-template #monthText>\n            <span class=\"text-sm font-medium text-foreground\">{{ leftMonthName }} {{ year }}</span>\n          </ng-template>\n        </div>\n        <button type=\"button\" class=\"inline-flex h-8 w-8 items-center justify-center rounded-md text-foreground hover:bg-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\">{{ mode === 'range' ? '' : '›' }}</button>\n      </header>\n\n      <div>\n        <div class=\"grid grid-cols-7\">\n          <span *ngFor=\"let day of weekdays\" class=\"inline-flex h-5 items-center justify-center text-xs font-normal text-muted-foreground\">{{ day }}</span>\n        </div>\n        <div class=\"mt-2 grid grid-cols-7 gap-y-2\">\n          <button\n            *ngFor=\"let cell of leftCells\"\n            type=\"button\"\n            [ngClass]=\"[\n              'inline-flex h-8 w-8 items-center justify-center rounded-md text-sm font-normal transition-colors hover:bg-accent hover:text-accent-foreground',\n              cell.outside ? 'text-muted-foreground opacity-50' : 'text-foreground',\n              cell.inRange ? 'bg-accent text-accent-foreground' : '',\n              cell.selected ? 'bg-primary text-primary-foreground' : ''\n            ]\"\n          >\n            {{ cell.day }}\n          </button>\n        </div>\n      </div>\n    </div>\n\n    <div *ngIf=\"mode === 'range'\" class=\"space-y-4\">\n      <header class=\"flex items-center justify-between\">\n        <span class=\"text-sm font-medium text-foreground\">{{ rightMonthName }} {{ month === 12 ? year + 1 : year }}</span>\n        <button type=\"button\" class=\"inline-flex h-8 w-8 items-center justify-center rounded-md text-foreground hover:bg-accent focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\">›</button>\n      </header>\n\n      <div>\n        <div class=\"grid grid-cols-7\">\n          <span *ngFor=\"let day of weekdays\" class=\"inline-flex h-5 items-center justify-center text-xs font-normal text-muted-foreground\">{{ day }}</span>\n        </div>\n        <div class=\"mt-2 grid grid-cols-7 gap-y-2\">\n          <button\n            *ngFor=\"let cell of rightCells\"\n            type=\"button\"\n            [ngClass]=\"[\n              'inline-flex h-8 w-8 items-center justify-center rounded-md text-sm font-normal transition-colors hover:bg-accent hover:text-accent-foreground',\n              cell.outside ? 'text-muted-foreground opacity-50' : 'text-foreground',\n              cell.inRange ? 'bg-accent text-accent-foreground' : '',\n              cell.selected ? 'bg-primary text-primary-foreground' : ''\n            ]\"\n          >\n            {{ cell.day }}\n          </button>\n        </div>\n      </div>\n    </div>\n  </div>\n</section>\n"]}
472
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.component.js","sourceRoot":"","sources":["../../../../../../src/lib/components/calendar/calendar.component.ts","../../../../../../src/lib/components/calendar/calendar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, Input, Output } from '@angular/core';\n\nexport type PdmCalendarVariant = 'single' | 'range';\n\nexport interface PdmCalendarRange {\n  start: Date | null;\n  end: Date | null;\n}\n\ninterface PdmCalendarCell {\n  date: Date;\n  label: number;\n  muted: boolean;\n  disabled: boolean;\n  selected: boolean;\n  inRange: boolean;\n  rangeFill: boolean;\n  rangeLeftCap: boolean;\n  rangeRightCap: boolean;\n}\n\ninterface PdmCalendarMonthView {\n  date: Date;\n  title: string;\n  titleStyle: 'dropdowns' | 'plain';\n  dropdownMonth?: string;\n  dropdownYear?: string;\n  showPrevButton: boolean;\n  showNextButton: boolean;\n  weeks: readonly (readonly PdmCalendarCell[])[];\n}\n\nconst DEFAULT_VIEW_MONTH = new Date();\nconst DAY_MS = 24 * 60 * 60 * 1000;\n\n@Component({\n  selector: 'pdm-calendar',\n  templateUrl: './calendar.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PdmCalendarComponent {\n  private _value: Date | null = null;\n  private _rangeValue: PdmCalendarRange | null = null;\n  private _month: Date | null = null;\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  @Input() variant: PdmCalendarVariant | string = 'single';\n  @Input() className = '';\n  @Input() disabledDates: Date[] = [];\n  @Input() minDate: Date | null = null;\n  @Input() maxDate: Date | null = null;\n  @Input() isDateDisabled: ((date: Date) => boolean) | null = null;\n  @Input() allowSameDayRange = true;\n  @Input() readonly = false;\n\n  @Output() valueChange = new EventEmitter<Date | null>();\n  @Output() rangeValueChange = new EventEmitter<PdmCalendarRange | null>();\n  @Output() monthChange = new EventEmitter<Date>();\n  @Output() dateClick = new EventEmitter<Date>();\n  @Output() disabledDateClick = new EventEmitter<Date>();\n\n  @Input()\n  set value(value: Date | null) {\n    this._value = this.normalizeDate(value);\n  }\n  get value(): Date | null {\n    return this._value;\n  }\n\n  @Input()\n  set rangeValue(value: PdmCalendarRange | null) {\n    this._rangeValue = value\n      ? {\n          start: this.normalizeDate(value.start),\n          end: this.normalizeDate(value.end)\n        }\n      : null;\n  }\n  get rangeValue(): PdmCalendarRange | null {\n    return this._rangeValue;\n  }\n\n  @Input()\n  set month(value: Date | null) {\n    this._month = value ? this.startOfMonth(value) : null;\n  }\n  get month(): Date | null {\n    return this._month;\n  }\n\n  readonly weekdays = ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'] as const;\n  readonly monthOptions = [\n    { value: 0, label: 'Jan' },\n    { value: 1, label: 'Feb' },\n    { value: 2, label: 'Mar' },\n    { value: 3, label: 'Apr' },\n    { value: 4, label: 'May' },\n    { value: 5, label: 'Jun' },\n    { value: 6, label: 'Jul' },\n    { value: 7, label: 'Aug' },\n    { value: 8, label: 'Sep' },\n    { value: 9, label: 'Oct' },\n    { value: 10, label: 'Nov' },\n    { value: 11, label: 'Dec' }\n  ] as const;\n\n  get resolvedVariant(): PdmCalendarVariant {\n    return this.variant === 'range' ? 'range' : 'single';\n  }\n\n  get visibleMonths(): readonly PdmCalendarMonthView[] {\n    const baseMonth = this.getAnchorMonth();\n\n    if (this.resolvedVariant === 'single') {\n      return [\n        {\n          date: baseMonth,\n          title: this.formatMonthYear(baseMonth),\n          titleStyle: 'dropdowns',\n          dropdownMonth: this.formatMonthShort(baseMonth),\n          dropdownYear: String(baseMonth.getFullYear()),\n          showPrevButton: true,\n          showNextButton: true,\n          weeks: this.buildMonthWeeks(baseMonth, false)\n        }\n      ];\n    }\n\n    const nextMonth = this.addMonths(baseMonth, 1);\n    return [\n      {\n        date: baseMonth,\n        title: this.formatMonthYear(baseMonth),\n        titleStyle: 'plain',\n        showPrevButton: true,\n        showNextButton: false,\n        weeks: this.buildMonthWeeks(baseMonth, true)\n      },\n      {\n        date: nextMonth,\n        title: this.formatMonthYear(nextMonth),\n        titleStyle: 'plain',\n        showPrevButton: false,\n        showNextButton: true,\n        weeks: this.buildMonthWeeks(nextMonth, true)\n      }\n    ];\n  }\n\n  get singleHeaderMonth(): number {\n    return this.getAnchorMonth().getMonth();\n  }\n\n  get singleHeaderYear(): number {\n    return this.getAnchorMonth().getFullYear();\n  }\n\n  get yearOptions(): readonly number[] {\n    const currentYear = this.singleHeaderYear;\n    const minYear = this.minDate ? this.minDate.getFullYear() : currentYear - 100;\n    const maxYear = this.maxDate ? this.maxDate.getFullYear() : currentYear + 100;\n    const years: number[] = [];\n\n    for (let year = minYear; year <= maxYear; year += 1) {\n      years.push(year);\n    }\n\n    return years;\n  }\n\n  get rootClasses(): string[] {\n    return [\n      'border-border bg-background relative rounded-[10px] border p-3 shadow-sm',\n      this.resolvedVariant === 'range'\n        ? 'inline-flex items-start justify-center gap-4 shrink-0 grow-0 basis-auto'\n        : 'inline-flex flex-col gap-4 shrink-0 grow-0 basis-auto',\n      this.className\n    ];\n  }\n\n  get rootStyle(): Record<string, string> {\n    const width = this.resolvedVariant === 'range' ? '488px' : '250px';\n\n    return {\n      width,\n      minWidth: width,\n      maxWidth: width,\n      minHeight: '293px',\n      flex: '0 0 auto',\n      alignSelf: 'flex-start'\n    };\n  }\n\n  monthPanelClasses(_index: number): string[] {\n    return ['flex flex-col items-start', this.resolvedVariant === 'range' ? 'w-[224px] gap-4' : 'w-full gap-0'];\n  }\n\n  headerClasses(month: PdmCalendarMonthView): string[] {\n    return ['flex w-full items-center justify-between', month.titleStyle === 'dropdowns' ? '' : 'mb-4'];\n  }\n\n  navButtonClasses(): string[] {\n    return [\n      'flex h-8 w-8 items-center justify-center rounded-md text-foreground',\n      'disabled:pointer-events-none disabled:opacity-40'\n    ];\n  }\n\n  navPlaceholderClasses(): string[] {\n    return ['h-7 w-7 shrink-0'];\n  }\n\n  dropdownWrapClasses(): string[] {\n    return ['flex w-40 items-center justify-center gap-2'];\n  }\n\n  dropdownClasses(widthClass: string): string[] {\n    return ['flex h-8 items-center justify-center gap-1 px-1', widthClass];\n  }\n\n  dropdownSelectClasses(): string[] {\n    return [\n      'text-foreground h-full bg-transparent text-sm font-medium leading-5',\n      'appearance-none border-0 outline-none ring-0 focus:outline-none focus:ring-0 text-center cursor-pointer'\n    ];\n  }\n\n  get dropdownSelectStyle(): Record<string, string> {\n    return {\n      appearance: 'none',\n      WebkitAppearance: 'none',\n      MozAppearance: 'none',\n      background: 'transparent',\n      border: '0',\n      boxShadow: 'none',\n      outline: '0',\n      padding: '0',\n      margin: '0'\n    };\n  }\n\n  calendarGridWrapClasses(): string[] {\n    return ['flex w-full flex-col items-start'];\n  }\n\n  weekdayRowClasses(): string[] {\n    return ['flex w-full items-center'];\n  }\n\n  weekdayCellClasses(): string[] {\n    return ['text-muted-foreground flex h-[21px] w-8 items-center justify-center rounded-md text-xs leading-4'];\n  }\n\n  weekRowClasses(): string[] {\n    return ['flex w-full items-start pt-2'];\n  }\n\n  dayCellClasses(cell: PdmCalendarCell): string[] {\n    return [\n      'relative flex h-8 w-8 shrink-0 items-center justify-center',\n      cell.rangeFill ? 'bg-accent' : '',\n      cell.rangeLeftCap ? 'rounded-l-md' : '',\n      cell.rangeRightCap ? 'rounded-r-md' : ''\n    ];\n  }\n\n  dayButtonClasses(cell: PdmCalendarCell): string[] {\n    return [\n      'relative z-10 flex h-8 w-8 items-center justify-center rounded-md text-sm leading-5',\n      cell.selected ? 'bg-primary text-primary-foreground' : cell.rangeFill ? 'text-accent-foreground' : 'text-foreground',\n      cell.muted && !cell.rangeFill ? 'opacity-50' : '',\n      cell.disabled ? 'cursor-not-allowed opacity-40' : '',\n      !cell.disabled && !this.readonly && !cell.selected ? 'hover:bg-accent/70' : ''\n    ];\n  }\n\n  dayLabelClasses(_cell: PdmCalendarCell): string[] {\n    return ['font-normal'];\n  }\n\n  trackByIndex = (index: number): number => {\n    return index;\n  }\n\n  trackByDate = (_index: number, cell: PdmCalendarCell): string => {\n    return this.dateKey(cell.date);\n  }\n\n  goToPreviousMonth(): void {\n    this.setAnchorMonth(this.addMonths(this.getAnchorMonth(), -1));\n  }\n\n  goToNextMonth(): void {\n    this.setAnchorMonth(this.addMonths(this.getAnchorMonth(), 1));\n  }\n\n  onSingleMonthChange(monthValue: string): void {\n    const month = Number(monthValue);\n    if (Number.isNaN(month) || month < 0 || month > 11) {\n      return;\n    }\n\n    const anchor = this.getAnchorMonth();\n    this.setAnchorMonth(new Date(anchor.getFullYear(), month, 1));\n  }\n\n  onSingleYearChange(yearValue: string): void {\n    const year = Number(yearValue);\n    if (Number.isNaN(year)) {\n      return;\n    }\n\n    const anchor = this.getAnchorMonth();\n    this.setAnchorMonth(new Date(year, anchor.getMonth(), 1));\n  }\n\n  onDatePressed(cell: PdmCalendarCell): void {\n    const date = this.cloneDate(cell.date);\n\n    if (cell.disabled) {\n      this.disabledDateClick.emit(date);\n      return;\n    }\n\n    if (this.readonly) {\n      return;\n    }\n\n    this.dateClick.emit(date);\n\n    if (this.resolvedVariant === 'single') {\n      this._value = date;\n      this.valueChange.emit(this.cloneDate(date));\n      this.syncVisibleMonthToDate(date);\n      this.cdr.markForCheck();\n      return;\n    }\n\n    this.handleRangeSelection(date);\n    this.cdr.markForCheck();\n  }\n\n  private handleRangeSelection(date: Date): void {\n    const current = this._rangeValue;\n    const currentStart = current?.start ? this.cloneDate(current.start) : null;\n    const currentEnd = current?.end ? this.cloneDate(current.end) : null;\n\n    if (!currentStart || (currentStart && currentEnd)) {\n      this._rangeValue = { start: date, end: null };\n      this.rangeValueChange.emit({ start: this.cloneDate(date), end: null });\n      this.syncVisibleMonthToDate(date);\n      return;\n    }\n\n    if (this.isSameDay(currentStart, date) && !this.allowSameDayRange) {\n      this._rangeValue = { start: date, end: null };\n      this.rangeValueChange.emit({ start: this.cloneDate(date), end: null });\n      return;\n    }\n\n    const start = this.compareDate(date, currentStart) < 0 ? date : currentStart;\n    const end = this.compareDate(date, currentStart) < 0 ? currentStart : date;\n\n    if (this.rangeContainsBlockedDate(start, end)) {\n      this.disabledDateClick.emit(this.cloneDate(date));\n      return;\n    }\n\n    this._rangeValue = { start, end };\n    this.rangeValueChange.emit({ start: this.cloneDate(start), end: this.cloneDate(end) });\n    this.syncVisibleMonthToDate(start);\n  }\n\n  private buildMonthWeeks(month: Date, includeRange: boolean): readonly (readonly PdmCalendarCell[])[] {\n    const firstOfMonth = this.startOfMonth(month);\n    const start = this.startOfWeek(firstOfMonth);\n    const end = this.endOfWeek(this.endOfMonth(firstOfMonth));\n    const weeks: PdmCalendarCell[][] = [];\n\n    let cursor = this.cloneDate(start);\n    while (this.compareDate(cursor, end) <= 0) {\n      const row: PdmCalendarCell[] = [];\n\n      for (let col = 0; col < 7; col += 1) {\n        row.push(this.buildCell(cursor, month, includeRange));\n        cursor = this.addDays(cursor, 1);\n      }\n\n      weeks.push(row);\n    }\n\n    for (const row of weeks) {\n      for (let col = 0; col < row.length; col += 1) {\n        const cell = row[col];\n        if (!cell.inRange) {\n          continue;\n        }\n\n        const leftInRow = col > 0 ? row[col - 1].inRange : false;\n        const rightInRow = col < row.length - 1 ? row[col + 1].inRange : false;\n\n        cell.rangeLeftCap = !leftInRow;\n        cell.rangeRightCap = !rightInRow;\n      }\n    }\n\n    return weeks;\n  }\n\n  private buildCell(date: Date, visibleMonth: Date, includeRange: boolean): PdmCalendarCell {\n    const normalized = this.cloneDate(date);\n    const inCurrentMonth = normalized.getMonth() === visibleMonth.getMonth() && normalized.getFullYear() === visibleMonth.getFullYear();\n    const disabled = this.isBlocked(normalized);\n\n    const selectedSingle = this.resolvedVariant === 'single' && !!this._value && this.isSameDay(normalized, this._value);\n    const rangeStart = includeRange && !!this._rangeValue?.start && this.isSameDay(normalized, this._rangeValue.start);\n    const rangeEnd = includeRange && !!this._rangeValue?.end && this.isSameDay(normalized, this._rangeValue.end);\n    const inRange =\n      includeRange &&\n      !!this._rangeValue?.start &&\n      !!this._rangeValue?.end &&\n      this.compareDate(normalized, this._rangeValue.start) >= 0 &&\n      this.compareDate(normalized, this._rangeValue.end) <= 0;\n\n    return {\n      date: normalized,\n      label: normalized.getDate(),\n      muted: !inCurrentMonth,\n      disabled,\n      selected: selectedSingle || rangeStart || rangeEnd,\n      inRange,\n      rangeFill: inRange,\n      rangeLeftCap: false,\n      rangeRightCap: false\n    };\n  }\n\n  private getAnchorMonth(): Date {\n    if (this._month) {\n      return this.cloneDate(this._month);\n    }\n\n    if (this.resolvedVariant === 'single' && this._value) {\n      return this.startOfMonth(this._value);\n    }\n\n    if (this.resolvedVariant === 'range' && this._rangeValue?.start) {\n      return this.startOfMonth(this._rangeValue.start);\n    }\n\n    return this.startOfMonth(DEFAULT_VIEW_MONTH);\n  }\n\n  private setAnchorMonth(month: Date): void {\n    this._month = this.startOfMonth(month);\n    this.monthChange.emit(this.cloneDate(this._month));\n  }\n\n  private syncVisibleMonthToDate(date: Date): void {\n    const nextMonth = this.startOfMonth(date);\n    if (!this._month || !this.isSameMonth(this._month, nextMonth)) {\n      this._month = nextMonth;\n      this.monthChange.emit(this.cloneDate(nextMonth));\n    }\n  }\n\n  private isBlocked(date: Date): boolean {\n    if (this.minDate && this.compareDate(date, this.minDate) < 0) {\n      return true;\n    }\n\n    if (this.maxDate && this.compareDate(date, this.maxDate) > 0) {\n      return true;\n    }\n\n    const blockedSet = new Set((this.disabledDates || []).map((item) => this.dateKey(item)));\n    if (blockedSet.has(this.dateKey(date))) {\n      return true;\n    }\n\n    return !!this.isDateDisabled?.(this.cloneDate(date));\n  }\n\n  private rangeContainsBlockedDate(start: Date, end: Date): boolean {\n    let cursor = this.cloneDate(start);\n\n    while (this.compareDate(cursor, end) <= 0) {\n      if (this.isBlocked(cursor)) {\n        return true;\n      }\n      cursor = this.addDays(cursor, 1);\n    }\n\n    return false;\n  }\n\n  private normalizeDate(value: Date | null | undefined): Date | null {\n    if (!(value instanceof Date) || Number.isNaN(value.getTime())) {\n      return null;\n    }\n\n    return new Date(value.getFullYear(), value.getMonth(), value.getDate());\n  }\n\n  private cloneDate(date: Date): Date {\n    return new Date(date.getFullYear(), date.getMonth(), date.getDate());\n  }\n\n  private startOfMonth(date: Date): Date {\n    return new Date(date.getFullYear(), date.getMonth(), 1);\n  }\n\n  private endOfMonth(date: Date): Date {\n    return new Date(date.getFullYear(), date.getMonth() + 1, 0);\n  }\n\n  private startOfWeek(date: Date): Date {\n    return this.addDays(date, -date.getDay());\n  }\n\n  private endOfWeek(date: Date): Date {\n    return this.addDays(date, 6 - date.getDay());\n  }\n\n  private addMonths(date: Date, months: number): Date {\n    return new Date(date.getFullYear(), date.getMonth() + months, 1);\n  }\n\n  private addDays(date: Date, days: number): Date {\n    return new Date(date.getFullYear(), date.getMonth(), date.getDate() + days);\n  }\n\n  private compareDate(a: Date, b: Date): number {\n    return this.dateValue(a) - this.dateValue(b);\n  }\n\n  private dateValue(date: Date): number {\n    return Date.UTC(date.getFullYear(), date.getMonth(), date.getDate()) / DAY_MS;\n  }\n\n  private isSameDay(a: Date, b: Date): boolean {\n    return this.compareDate(a, b) === 0;\n  }\n\n  private isSameMonth(a: Date, b: Date): boolean {\n    return a.getFullYear() === b.getFullYear() && a.getMonth() === b.getMonth();\n  }\n\n  private dateKey(date: Date): string {\n    const y = date.getFullYear();\n    const m = String(date.getMonth() + 1).padStart(2, '0');\n    const d = String(date.getDate()).padStart(2, '0');\n    return `${y}-${m}-${d}`;\n  }\n\n  private formatMonthShort(date: Date): string {\n    return date.toLocaleString('en-US', { month: 'short' });\n  }\n\n  private formatMonthYear(date: Date): string {\n    return date.toLocaleString('en-US', { month: 'long', year: 'numeric' });\n  }\n}\n","<div [ngClass]=\"rootClasses\" [ngStyle]=\"rootStyle\">\n  <div *ngFor=\"let month of visibleMonths; let monthIndex = index; trackBy: trackByIndex\" [ngClass]=\"monthPanelClasses(monthIndex)\">\n    <div [ngClass]=\"headerClasses(month)\">\n      <button\n        *ngIf=\"month.showPrevButton; else prevPlaceholder\"\n        type=\"button\"\n        [ngClass]=\"navButtonClasses()\"\n        aria-label=\"Previous month\"\n        (click)=\"goToPreviousMonth()\"\n        [disabled]=\"readonly\"\n      >\n        <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n          <path d=\"m15 18-6-6 6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n        </svg>\n      </button>\n      <ng-template #prevPlaceholder>\n        <div [ngClass]=\"navPlaceholderClasses()\" aria-hidden=\"true\"></div>\n      </ng-template>\n\n      <ng-container *ngIf=\"month.titleStyle === 'dropdowns'; else plainTitle\">\n        <div [ngClass]=\"dropdownWrapClasses()\">\n          <div [ngClass]=\"dropdownClasses('w-[72px]')\">\n            <select\n              [ngClass]=\"dropdownSelectClasses()\"\n              [ngStyle]=\"dropdownSelectStyle\"\n              [value]=\"singleHeaderMonth\"\n              aria-label=\"Month\"\n              (change)=\"onSingleMonthChange($any($event.target).value)\"\n            >\n              <option *ngFor=\"let monthOption of monthOptions\" [value]=\"monthOption.value\">{{ monthOption.label }}</option>\n            </select>\n            <svg viewBox=\"0 0 24 24\" class=\"h-3 w-3 text-foreground\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n              <path d=\"m6 9 6 6 6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n            </svg>\n          </div>\n          <div [ngClass]=\"dropdownClasses('w-[82px]')\">\n            <select\n              [ngClass]=\"dropdownSelectClasses()\"\n              [ngStyle]=\"dropdownSelectStyle\"\n              [value]=\"singleHeaderYear\"\n              aria-label=\"Year\"\n              (change)=\"onSingleYearChange($any($event.target).value)\"\n            >\n              <option *ngFor=\"let year of yearOptions\" [value]=\"year\">{{ year }}</option>\n            </select>\n            <svg viewBox=\"0 0 24 24\" class=\"h-3 w-3 text-foreground\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n              <path d=\"m6 9 6 6 6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n            </svg>\n          </div>\n        </div>\n      </ng-container>\n\n      <ng-template #plainTitle>\n        <div class=\"flex min-w-0 flex-1 items-center justify-center\">\n          <p class=\"text-foreground text-center text-sm font-medium leading-5\">\n            {{ month.title }}\n          </p>\n        </div>\n      </ng-template>\n\n      <button\n        *ngIf=\"month.showNextButton; else nextPlaceholder\"\n        type=\"button\"\n        [ngClass]=\"navButtonClasses()\"\n        aria-label=\"Next month\"\n        (click)=\"goToNextMonth()\"\n        [disabled]=\"readonly\"\n      >\n        <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" aria-hidden=\"true\">\n          <path d=\"m9 18 6-6-6-6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n        </svg>\n      </button>\n      <ng-template #nextPlaceholder>\n        <div [ngClass]=\"navPlaceholderClasses()\" aria-hidden=\"true\"></div>\n      </ng-template>\n    </div>\n\n    <div [ngClass]=\"calendarGridWrapClasses()\">\n      <div [ngClass]=\"weekdayRowClasses()\">\n        <div *ngFor=\"let day of weekdays; trackBy: trackByIndex\" [ngClass]=\"weekdayCellClasses()\">\n          <span>{{ day }}</span>\n        </div>\n      </div>\n\n      <div *ngFor=\"let week of month.weeks; trackBy: trackByIndex\" [ngClass]=\"weekRowClasses()\">\n        <div *ngFor=\"let cell of week; trackBy: trackByDate\" [ngClass]=\"dayCellClasses(cell)\">\n          <button\n            type=\"button\"\n            [ngClass]=\"dayButtonClasses(cell)\"\n            [disabled]=\"readonly\"\n            [attr.aria-selected]=\"cell.selected\"\n            [attr.aria-disabled]=\"cell.disabled || readonly\"\n            [attr.title]=\"cell.date | date : 'yyyy-MM-dd'\"\n            (click)=\"onDatePressed(cell)\"\n          >\n            <span [ngClass]=\"dayLabelClasses(cell)\">{{ cell.label }}</span>\n          </button>\n        </div>\n      </div>\n    </div>\n  </div>\n</div>\n"]}