pdm-ui-kit 0.1.21 → 0.1.22
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/esm2020/lib/components/calendar/calendar.component.mjs +454 -53
- package/esm2020/lib/components/date-picker/date-picker.component.mjs +252 -34
- package/fesm2015/pdm-ui-kit.mjs +727 -103
- package/fesm2015/pdm-ui-kit.mjs.map +1 -1
- package/fesm2020/pdm-ui-kit.mjs +721 -103
- package/fesm2020/pdm-ui-kit.mjs.map +1 -1
- package/lib/components/calendar/calendar.component.d.ts +137 -18
- package/lib/components/date-picker/date-picker.component.d.ts +61 -17
- package/package.json +2 -1
|
@@ -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(2025, 5, 1); // June 2025 (Figma default)
|
|
5
|
+
const DAY_MS = 24 * 60 * 60 * 1000;
|
|
4
6
|
export class PdmCalendarComponent {
|
|
5
|
-
constructor() {
|
|
6
|
-
this.
|
|
7
|
-
this.
|
|
8
|
-
this.
|
|
9
|
-
this.
|
|
10
|
-
this.
|
|
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
|
-
|
|
16
|
-
|
|
17
|
-
}
|
|
18
|
-
get
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
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: {
|
|
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: "<
|
|
56
|
-
}], propDecorators: {
|
|
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
|
-
}],
|
|
441
|
+
}], className: [{
|
|
59
442
|
type: Input
|
|
60
|
-
}],
|
|
443
|
+
}], disabledDates: [{
|
|
61
444
|
type: Input
|
|
62
|
-
}],
|
|
445
|
+
}], minDate: [{
|
|
63
446
|
type: Input
|
|
64
|
-
}],
|
|
447
|
+
}], maxDate: [{
|
|
65
448
|
type: Input
|
|
66
|
-
}],
|
|
449
|
+
}], isDateDisabled: [{
|
|
67
450
|
type: Input
|
|
68
|
-
}],
|
|
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":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;;;AAiB1E,MAAM,OAAO,oBAAoB;IALjC;QAMW,SAAI,GAAoB,QAAQ,CAAC;QACjC,UAAK,GAAG,CAAC,CAAC;QACV,SAAI,GAAG,IAAI,CAAC;QACZ,gBAAW,GAAG,EAAE,CAAC;QACjB,kBAAa,GAAG,EAAE,CAAC;QACnB,gBAAW,GAAG,CAAC,CAAC;QAChB,cAAS,GAAG,EAAE,CAAC;QAEf,aAAQ,GAAG,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;KAkDhE;IAhDC,IAAI,aAAa;QACf,OAAO,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;IACxC,CAAC;IAED,IAAI,cAAc;QAChB,MAAM,UAAU,GAAG,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC;QACnC,OAAO,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;IACpC,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,GAAG,CAAC,EAAE,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS,EAAE,IAAI,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;IAC3K,CAAC;IAED,IAAI,UAAU;QACZ,MAAM,UAAU,GAAG,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC;QACnC,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;QAChE,OAAO,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,UAAU,EAAE,IAAI,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,SAAS,EAAE,CAAC,EAAE,IAAI,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;IACpK,CAAC;IAEO,SAAS,CAAC,UAAkB;QAClC,MAAM,IAAI,GAAG,CAAC,CAAC,UAAU,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC;QAC3C,OAAO,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,cAAc,CAAC,OAAO,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;IACjF,CAAC;IAEO,eAAe,CACrB,IAAY,EACZ,UAAkB,EAClB,WAAoB,EACpB,aAAsB,EACtB,WAAoB;QAEpB,MAAM,QAAQ,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,UAAU,EAAE,CAAC,CAAC,CAAC;QAC/C,MAAM,YAAY,GAAG,QAAQ,CAAC,MAAM,EAAE,CAAC;QACvC,MAAM,WAAW,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,UAAU,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,EAAE,CAAC;QAChE,MAAM,KAAK,GAAmB,EAAE,CAAC;QAEjC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE;YAC9B,MAAM,SAAS,GAAG,CAAC,GAAG,YAAY,GAAG,CAAC,CAAC;YACvC,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,UAAU,EAAE,SAAS,CAAC,CAAC;YACnD,MAAM,OAAO,GAAG,SAAS,GAAG,CAAC,IAAI,SAAS,GAAG,WAAW,CAAC;YACzD,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;YAC3B,MAAM,QAAQ,GAAG,CAAC,OAAO,IAAI,CAAC,CAAC,WAAW,IAAI,GAAG,KAAK,WAAW,CAAC;YAClE,MAAM,OAAO,GAAG,CAAC,OAAO,IAAI,CAAC,CAAC,aAAa,IAAI,CAAC,CAAC,WAAW,IAAI,GAAG,IAAI,aAAa,IAAI,GAAG,IAAI,WAAW,CAAC;YAC3G,KAAK,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,GAAG,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAC;SACvD;QAED,OAAO,KAAK,CAAC;IACf,CAAC;;iHA1DU,oBAAoB;qGAApB,oBAAoB,4NCjBjC,kgIAwEA;2FDvDa,oBAAoB;kBALhC,SAAS;+BACE,cAAc,mBAEP,uBAAuB,CAAC,MAAM;8BAGtC,IAAI;sBAAZ,KAAK;gBACG,KAAK;sBAAb,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,aAAa;sBAArB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,SAAS;sBAAjB,KAAK","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(2025, 5, 1); // June 2025 (Figma default)\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"]}
|