@updevs/components 1.0.0-alpha.48 → 1.0.0-alpha.49

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (90) hide show
  1. package/README.md +1 -13
  2. package/calendar/calendar.component.d.ts +58 -122
  3. package/calendar/models/calendar-date.model.d.ts +10 -0
  4. package/calendar/models/calendar-month.model.d.ts +7 -0
  5. package/calendar/models/calendar-selected.model.d.ts +5 -0
  6. package/calendar/models/calendar-year.model.d.ts +6 -0
  7. package/calendar/models/date-range.model.d.ts +0 -4
  8. package/calendar/models/enums/calendar-view.enum.d.ts +5 -0
  9. package/calendar/models/public-api.d.ts +5 -0
  10. package/calendar/upd-calendar.module.d.ts +6 -11
  11. package/card/public-api.d.ts +1 -0
  12. package/card/types/public-api.d.ts +5 -0
  13. package/dropdown/dropdown.component.d.ts +3 -0
  14. package/esm2022/calendar/calendar.component.mjs +288 -217
  15. package/esm2022/calendar/models/calendar-date.model.mjs +2 -0
  16. package/esm2022/calendar/models/calendar-month.model.mjs +2 -0
  17. package/esm2022/calendar/models/calendar-selected.model.mjs +2 -0
  18. package/esm2022/calendar/models/calendar-year.model.mjs +2 -0
  19. package/esm2022/calendar/models/date-range.model.mjs +1 -1
  20. package/esm2022/calendar/models/enums/calendar-view.enum.mjs +7 -0
  21. package/esm2022/calendar/models/public-api.mjs +6 -1
  22. package/esm2022/calendar/upd-calendar.module.mjs +2 -20
  23. package/esm2022/card/public-api.mjs +2 -1
  24. package/esm2022/card/types/public-api.mjs +6 -0
  25. package/esm2022/dropdown/dropdown.component.mjs +23 -7
  26. package/esm2022/form-controls/date-picker/date-picker.component.mjs +45 -60
  27. package/esm2022/form-controls/input/input.component.mjs +3 -3
  28. package/esm2022/popover/popover.directive.mjs +10 -26
  29. package/esm2022/pricing-cards/index.mjs +2 -0
  30. package/esm2022/pricing-cards/models/price-item-feature.model.mjs +2 -0
  31. package/esm2022/pricing-cards/models/price-item-features-group.model.mjs +2 -0
  32. package/esm2022/pricing-cards/models/price-item.model.mjs +2 -0
  33. package/esm2022/pricing-cards/models/public-api.mjs +4 -0
  34. package/esm2022/pricing-cards/pricing-cards.component.mjs +24 -0
  35. package/esm2022/pricing-cards/public-api.mjs +4 -0
  36. package/esm2022/pricing-cards/upd-pricing-cards.module.mjs +36 -0
  37. package/esm2022/pricing-cards/updevs-components-pricing-cards.mjs +5 -0
  38. package/esm2022/table/abstractions/base.column.mjs +30 -44
  39. package/esm2022/table/table.component.mjs +9 -12
  40. package/fesm2022/updevs-components-calendar.mjs +319 -1449
  41. package/fesm2022/updevs-components-calendar.mjs.map +1 -1
  42. package/fesm2022/updevs-components-dropdown.mjs +22 -6
  43. package/fesm2022/updevs-components-dropdown.mjs.map +1 -1
  44. package/fesm2022/updevs-components-form-controls-date-picker.mjs +44 -59
  45. package/fesm2022/updevs-components-form-controls-date-picker.mjs.map +1 -1
  46. package/fesm2022/updevs-components-form-controls-input.mjs +2 -2
  47. package/fesm2022/updevs-components-form-controls-input.mjs.map +1 -1
  48. package/fesm2022/updevs-components-popover.mjs +9 -25
  49. package/fesm2022/updevs-components-popover.mjs.map +1 -1
  50. package/fesm2022/updevs-components-pricing-cards.mjs +64 -0
  51. package/fesm2022/updevs-components-pricing-cards.mjs.map +1 -0
  52. package/fesm2022/updevs-components-table.mjs +38 -55
  53. package/fesm2022/updevs-components-table.mjs.map +1 -1
  54. package/form-controls/date-picker/date-picker.component.d.ts +10 -31
  55. package/package.json +37 -31
  56. package/pricing-cards/index.d.ts +1 -0
  57. package/pricing-cards/models/price-item-feature.model.d.ts +5 -0
  58. package/pricing-cards/models/price-item-features-group.model.d.ts +5 -0
  59. package/pricing-cards/models/price-item.model.d.ts +19 -0
  60. package/pricing-cards/models/public-api.d.ts +3 -0
  61. package/pricing-cards/pricing-cards.component.d.ts +11 -0
  62. package/pricing-cards/public-api.d.ts +3 -0
  63. package/pricing-cards/upd-pricing-cards.module.d.ts +11 -0
  64. package/table/abstractions/base.column.d.ts +4 -0
  65. package/calendar/components/calendar-body/calendar-body.component.d.ts +0 -263
  66. package/calendar/components/calendar-header/calendar-header.component.d.ts +0 -57
  67. package/calendar/components/calendar-month-view/calendar-month-view.component.d.ts +0 -151
  68. package/calendar/components/calendar-multi-year-view/calendar-multi-year-view.component.d.ts +0 -10
  69. package/calendar/components/calendar-year-view/calendar-year-view.component.d.ts +0 -9
  70. package/calendar/models/calendar-cell.model.d.ts +0 -16
  71. package/calendar/services/calendar.service.d.ts +0 -15
  72. package/calendar/tools/events-options.d.ts +0 -12
  73. package/calendar/tools/keyboard-tools.d.ts +0 -9
  74. package/calendar/tools/passive-listeners.d.ts +0 -16
  75. package/calendar/types/calendar-cell-class-function.type.d.ts +0 -3
  76. package/calendar/types/calendar-cell-css-classes.type.d.ts +0 -3
  77. package/calendar/types/modifier-key.type.d.ts +0 -1
  78. package/esm2022/calendar/components/calendar-body/calendar-body.component.mjs +0 -515
  79. package/esm2022/calendar/components/calendar-header/calendar-header.component.mjs +0 -144
  80. package/esm2022/calendar/components/calendar-month-view/calendar-month-view.component.mjs +0 -423
  81. package/esm2022/calendar/components/calendar-multi-year-view/calendar-multi-year-view.component.mjs +0 -18
  82. package/esm2022/calendar/components/calendar-year-view/calendar-year-view.component.mjs +0 -17
  83. package/esm2022/calendar/models/calendar-cell.model.mjs +0 -18
  84. package/esm2022/calendar/services/calendar.service.mjs +0 -40
  85. package/esm2022/calendar/tools/events-options.mjs +0 -14
  86. package/esm2022/calendar/tools/keyboard-tools.mjs +0 -14
  87. package/esm2022/calendar/tools/passive-listeners.mjs +0 -29
  88. package/esm2022/calendar/types/calendar-cell-class-function.type.mjs +0 -2
  89. package/esm2022/calendar/types/calendar-cell-css-classes.type.mjs +0 -2
  90. package/esm2022/calendar/types/modifier-key.type.mjs +0 -2
@@ -1,247 +1,318 @@
1
- import { ChangeDetectorRef, Component, EventEmitter, Input, Output, ViewChild, inject } from '@angular/core';
1
+ import { Component, inject, signal, computed, input, output } from '@angular/core';
2
2
  import { DateService } from '@updevs/sdk/core';
3
- import { DateRangeModel } from './models/date-range.model';
4
- import { CalendarMonthViewComponent } from './components/calendar-month-view/calendar-month-view.component';
5
- import { CalendarYearViewComponent } from './components/calendar-year-view/calendar-year-view.component';
6
- import { CalendarMultiYearViewComponent } from './components/calendar-multi-year-view/calendar-multi-year-view.component';
3
+ import { CalendarViewEnum } from './models/enums/calendar-view.enum';
7
4
  import * as i0 from "@angular/core";
8
- import * as i1 from "./components/calendar-month-view/calendar-month-view.component";
9
5
  export class CalendarComponent {
10
6
  constructor() {
11
- /**
12
- * Whether the calendar should be started in month or year view.
13
- */
14
- this.startView = 'month';
15
- /** Whether the user is currently selecting a range of dates. */
16
- this.isRange = false;
17
- /** Emits when the currently selected date changes. */
18
- this.selectedChange = new EventEmitter();
19
- /**
20
- * Emits the year chosen in multiyear view.
21
- * This doesn't imply a change on the selected date.
22
- */
23
- this.yearSelected = new EventEmitter();
24
- /**
25
- * Emits the month chosen in year view.
26
- * This doesn't imply a change on the selected date.
27
- */
28
- this.monthSelected = new EventEmitter();
29
- /**
30
- * Emits when the current view changes.
31
- */
32
- this.viewChanged = new EventEmitter(true);
33
- /** Emits when any date is selected. */
34
- this.userSelection = new EventEmitter();
35
- /** Emits a new date range value when the user completes a drag drop operation. */
36
- this.userDragDrop = new EventEmitter();
37
- /**
38
- * Used for scheduling that focus should be moved to the active cell on the next tick.
39
- * We need to schedule it, rather than do it immediately, because we have to wait
40
- * for Angular to re-evaluate the view children.
41
- */
42
- this._moveFocusOnNextTick = false;
7
+ this.selectedDate = input();
8
+ this.selectedStartDate = input();
9
+ this.selectedEndDate = input();
10
+ this.minDate = input();
11
+ this.maxDate = input();
12
+ this.disabledDatesFilter = input();
13
+ this.isRange = input(false);
14
+ this.shouldRemoveExternalBorder = input(false);
15
+ this.selected = output();
16
+ this.monthChanged = output();
17
+ this.CalendarViewEnum = CalendarViewEnum;
18
+ this.currentView = signal(CalendarViewEnum.Days);
19
+ this.currentDate = signal(new Date());
20
+ this.yearRange = signal({ start: 2020, end: 2030 });
21
+ this.currentYear = computed(() => this.currentDate().getFullYear());
22
+ this.currentMonth = computed(() => this.currentDate().getMonth());
23
+ this.currentMonthName = computed(() => this.dateService.format(this.currentDate(), { month: 'short', year: 'numeric' }).toUpperCase());
24
+ this.calendarDays = computed(() => this.generateCalendarDays());
25
+ this.calendarMonths = computed(() => this.generateCalendarMonths());
26
+ this.calendarYears = computed(() => this.generateCalendarYears());
27
+ this.weekDayNames = computed(() => this.dateService.getDayOfWeekNames('narrow'));
28
+ this.internalSelectedDate = signal(undefined);
29
+ this.internalSelectedStartDate = signal(undefined);
30
+ this.internalSelectedEndDate = signal(undefined);
43
31
  this.dateService = inject(DateService);
44
- this._changeDetectorRef = inject(ChangeDetectorRef);
32
+ this.rangeSelectionState = 'start';
45
33
  }
46
- /** A date representing the period (month or year) to start the calendar in. */
47
- get startAt() {
48
- return this._startAt;
34
+ ngOnInit() {
35
+ this.syncInternalState();
36
+ if (this.selectedDate()) {
37
+ this.currentDate.set(new Date(this.selectedDate()));
38
+ }
39
+ else if (this.selectedStartDate()) {
40
+ this.currentDate.set(new Date(this.selectedStartDate()));
41
+ }
42
+ this.updateYearRange();
49
43
  }
50
- set startAt(value) {
51
- this._startAt = this.dateService.getValidOrUndefined(this.dateService.deserialize(value));
44
+ onHeaderClick() {
45
+ switch (this.currentView()) {
46
+ case CalendarViewEnum.Days:
47
+ this.currentView.set(CalendarViewEnum.Months);
48
+ break;
49
+ case CalendarViewEnum.Months:
50
+ this.currentView.set(CalendarViewEnum.Years);
51
+ break;
52
+ }
53
+ }
54
+ onPreviousClick() {
55
+ const current = this.currentDate();
56
+ switch (this.currentView()) {
57
+ case CalendarViewEnum.Days:
58
+ this.currentDate.set(this.dateService.addMonths(current, -1));
59
+ this.monthChanged.emit(this.currentDate());
60
+ break;
61
+ case CalendarViewEnum.Months:
62
+ this.currentDate.set(this.dateService.addYears(current, -1));
63
+ break;
64
+ case CalendarViewEnum.Years:
65
+ this.moveYearRange(-12);
66
+ break;
67
+ }
52
68
  }
53
- /**
54
- * The currently selected date.
55
- */
56
- get selected() {
57
- return this._selected;
69
+ onNextClick() {
70
+ const current = this.currentDate();
71
+ switch (this.currentView()) {
72
+ case CalendarViewEnum.Days:
73
+ this.currentDate.set(this.dateService.addMonths(current, 1));
74
+ this.monthChanged.emit(this.currentDate());
75
+ break;
76
+ case CalendarViewEnum.Months:
77
+ this.currentDate.set(this.dateService.addYears(current, 1));
78
+ break;
79
+ case CalendarViewEnum.Years:
80
+ this.moveYearRange(12);
81
+ break;
82
+ }
58
83
  }
59
- set selected(value) {
60
- console.log('setting selected', value);
61
- if (value instanceof DateRangeModel) {
62
- this._selected = value;
84
+ onDateClick(dateModel) {
85
+ if (!dateModel.isCurrentMonth) {
86
+ return;
87
+ }
88
+ if (dateModel.isDisabled) {
89
+ return;
90
+ }
91
+ if (this.isRange()) {
92
+ this.handleRangeSelection(dateModel.date);
63
93
  }
64
94
  else {
65
- this._selected = this.dateService.getValidOrUndefined(this.dateService.deserialize(value));
66
- }
67
- }
68
- /**
69
- * The minimum selectable date.
70
- */
71
- get minDate() {
72
- return this._minDate;
73
- }
74
- set minDate(value) {
75
- this._minDate = this.dateService.getValidOrUndefined(this.dateService.deserialize(value));
76
- }
77
- /**
78
- * The maximum selectable date.
79
- */
80
- get maxDate() {
81
- return this._maxDate;
82
- }
83
- set maxDate(value) {
84
- this._maxDate = this.dateService.getValidOrUndefined(this.dateService.deserialize(value));
85
- }
86
- /**
87
- * The current active date. This determines which time period is shown and which date is
88
- * highlighted when using keyboard navigation.
89
- */
90
- get activeDate() {
91
- return this._clampedActiveDate;
92
- }
93
- set activeDate(value) {
94
- this._clampedActiveDate = this.dateService.clamp(value, this.minDate, this.maxDate);
95
- this._changeDetectorRef.markForCheck();
96
- }
97
- /** Whether the calendar is in month view. */
98
- get currentView() {
99
- return this._currentView;
100
- }
101
- set currentView(value) {
102
- const viewChangedResult = this._currentView !== value ? value : undefined;
103
- this._currentView = value;
104
- this._moveFocusOnNextTick = true;
105
- this._changeDetectorRef.markForCheck();
106
- if (viewChangedResult) {
107
- this.viewChanged.emit(viewChangedResult);
108
- }
109
- }
110
- ngAfterContentInit() {
111
- this.activeDate = this.startAt || new Date();
112
- // Assign to the private property since we don't want to move focus on init.
113
- this._currentView = this.startView;
114
- }
115
- ngAfterViewChecked() {
116
- if (this._moveFocusOnNextTick) {
117
- this._moveFocusOnNextTick = false;
118
- this.focusActiveCell();
119
- }
120
- }
121
- ngOnChanges(changes) {
122
- // Ignore date changes that are at a different time on the same day. This fixes issues where
123
- // the calendar re-renders when there is no meaningful change to [minDate] or [maxDate]
124
- // (#24435).
125
- const minDateChange = changes['minDate'] && !this.dateService.areEqual(changes['minDate'].previousValue, changes['minDate'].currentValue)
126
- ? changes['minDate']
127
- : undefined;
128
- const maxDateChange = changes['maxDate'] && !this.dateService.areEqual(changes['maxDate'].previousValue, changes['maxDate'].currentValue)
129
- ? changes['maxDate']
130
- : undefined;
131
- const change = minDateChange || maxDateChange || changes['dateFilter'];
132
- if (change && !change.firstChange) {
133
- const view = this._getCurrentViewComponent();
134
- if (view) {
135
- // We need to `detectChanges` manually here, because the `minDate`, `maxDate` etc. are
136
- // passed down to the view via data bindings which won't be up-to-date when we call `_init`.
137
- this._changeDetectorRef.detectChanges();
138
- view.init();
95
+ this.internalSelectedDate.set(dateModel.date);
96
+ this.selected.emit({ selectedDate: dateModel.date });
97
+ }
98
+ }
99
+ onMonthClick(monthModel) {
100
+ if (monthModel.isDisabled) {
101
+ return;
102
+ }
103
+ const newDate = new Date(this.currentYear(), monthModel.monthIndex, 1);
104
+ this.currentDate.set(newDate);
105
+ this.currentView.set(CalendarViewEnum.Days);
106
+ this.monthChanged.emit(newDate);
107
+ }
108
+ onYearClick(yearModel) {
109
+ if (yearModel.isDisabled) {
110
+ return;
111
+ }
112
+ const newDate = new Date(yearModel.year, this.currentMonth(), 1);
113
+ this.currentDate.set(newDate);
114
+ this.currentView.set(CalendarViewEnum.Months);
115
+ }
116
+ clearSelection() {
117
+ this.internalSelectedDate.set(undefined);
118
+ this.internalSelectedStartDate.set(undefined);
119
+ this.internalSelectedEndDate.set(undefined);
120
+ this.rangeSelectionState = 'start';
121
+ this.selected.emit(undefined);
122
+ }
123
+ syncInternalState() {
124
+ this.internalSelectedDate.set(this.selectedDate());
125
+ this.internalSelectedStartDate.set(this.selectedStartDate());
126
+ this.internalSelectedEndDate.set(this.selectedEndDate());
127
+ }
128
+ handleRangeSelection(selectedDate) {
129
+ if (this.rangeSelectionState === 'start' || !this.internalSelectedStartDate()) {
130
+ this.internalSelectedStartDate.set(selectedDate);
131
+ this.internalSelectedEndDate.set(undefined);
132
+ this.rangeSelectionState = 'end';
133
+ this.selected.emit({ startDate: selectedDate });
134
+ }
135
+ else {
136
+ const startDate = this.internalSelectedStartDate();
137
+ const endDate = selectedDate;
138
+ if (this.dateService.compare(selectedDate, startDate) < 0) {
139
+ this.internalSelectedStartDate.set(selectedDate);
140
+ this.internalSelectedEndDate.set(startDate);
141
+ this.selected.emit({
142
+ startDate: selectedDate,
143
+ endDate: startDate
144
+ });
145
+ }
146
+ else {
147
+ this.internalSelectedEndDate.set(endDate);
148
+ this.selected.emit({
149
+ startDate: startDate,
150
+ endDate: endDate
151
+ });
139
152
  }
153
+ this.rangeSelectionState = 'start';
140
154
  }
141
155
  }
142
- /** Focuses the active date. */
143
- focusActiveCell() {
144
- this._getCurrentViewComponent().focusActiveCell(false);
156
+ generateCalendarDays() {
157
+ const current = this.currentDate();
158
+ const year = current.getFullYear();
159
+ const month = current.getMonth();
160
+ const firstDayOfMonth = new Date(year, month, 1);
161
+ const lastDayOfMonth = new Date(year, month + 1, 0);
162
+ const firstDayOfWeek = firstDayOfMonth.getDay();
163
+ const days = [];
164
+ const today = new Date();
165
+ if (firstDayOfWeek > 0) {
166
+ const prevMonthLastDay = new Date(year, month, 0);
167
+ const startDate = prevMonthLastDay.getDate() - firstDayOfWeek + 1;
168
+ for (let day = startDate; day <= prevMonthLastDay.getDate(); day++) {
169
+ const date = new Date(year, month - 1, day);
170
+ days.push({
171
+ date,
172
+ isCurrentMonth: false,
173
+ isToday: this.dateService.areEqual(date, today),
174
+ isSelected: false,
175
+ isDisabled: true,
176
+ isInRange: false,
177
+ isRangeStart: false,
178
+ isRangeEnd: false
179
+ });
180
+ }
181
+ }
182
+ for (let day = 1; day <= lastDayOfMonth.getDate(); day++) {
183
+ const date = new Date(year, month, day);
184
+ days.push({
185
+ date,
186
+ isCurrentMonth: true,
187
+ isToday: this.dateService.areEqual(date, today),
188
+ isSelected: this.isDateSelected(date),
189
+ isDisabled: this.isDateDisabled(date),
190
+ isInRange: this.isDateInRange(date),
191
+ isRangeStart: this.isDateRangeStart(date),
192
+ isRangeEnd: this.isDateRangeEnd(date)
193
+ });
194
+ }
195
+ const totalCells = Math.ceil(days.length / 7) * 7;
196
+ const nextMonthDays = totalCells - days.length;
197
+ for (let day = 1; day <= nextMonthDays; day++) {
198
+ const date = new Date(year, month + 1, day);
199
+ days.push({
200
+ date,
201
+ isCurrentMonth: false,
202
+ isToday: this.dateService.areEqual(date, today),
203
+ isSelected: false,
204
+ isDisabled: true,
205
+ isInRange: false,
206
+ isRangeStart: false,
207
+ isRangeEnd: false
208
+ });
209
+ }
210
+ return days;
145
211
  }
146
- /** Updates today's date after an update of the active date */
147
- updateTodaysDate() {
148
- this._getCurrentViewComponent().init();
212
+ generateCalendarMonths() {
213
+ const currentYear = this.currentYear();
214
+ const currentMonth = this.currentMonth();
215
+ const today = new Date();
216
+ const monthNames = this.dateService.getMonthNames('short');
217
+ return monthNames.map((name, index) => ({
218
+ monthIndex: index,
219
+ name: name.toUpperCase(),
220
+ isCurrentMonth: today.getFullYear() === currentYear && today.getMonth() === index,
221
+ isSelected: currentMonth === index,
222
+ isDisabled: this.isMonthDisabled(currentYear, index)
223
+ }));
149
224
  }
150
- /** Handles date selection in the month view. */
151
- _dateSelected(event) {
152
- const date = event.value;
153
- if (this.selected instanceof DateRangeModel || (date && !this.dateService.areEqual(date, this.selected))) {
154
- this.selectedChange.emit(date);
225
+ generateCalendarYears() {
226
+ const range = this.yearRange();
227
+ const currentYear = this.currentYear();
228
+ const today = new Date();
229
+ const years = [];
230
+ for (let year = range.start; year <= range.end; year++) {
231
+ years.push({
232
+ year,
233
+ isCurrentYear: today.getFullYear() === year,
234
+ isSelected: currentYear === year,
235
+ isDisabled: this.isYearDisabled(year)
236
+ });
155
237
  }
156
- this.userSelection.emit(event);
238
+ return years;
157
239
  }
158
- /** Handles year selection in the multiyear view. */
159
- _yearSelectedInMultiYearView(normalizedYear) {
160
- this.yearSelected.emit(normalizedYear);
240
+ isDateSelected(date) {
241
+ if (this.isRange()) {
242
+ if (this.internalSelectedStartDate() && this.dateService.areEqual(date, this.internalSelectedStartDate())) {
243
+ return true;
244
+ }
245
+ return !!(this.internalSelectedEndDate() && this.dateService.areEqual(date, this.internalSelectedEndDate()));
246
+ }
247
+ else {
248
+ return this.internalSelectedDate() ? this.dateService.areEqual(date, this.internalSelectedDate()) : false;
249
+ }
161
250
  }
162
- /** Handles month selection in the year view. */
163
- _monthSelectedInYearView(normalizedMonth) {
164
- this.monthSelected.emit(normalizedMonth);
251
+ isDateInRange(date) {
252
+ if (!this.isRange() || !this.internalSelectedStartDate() || !this.internalSelectedEndDate()) {
253
+ return false;
254
+ }
255
+ return this.dateService.compare(date, this.internalSelectedStartDate()) > 0
256
+ && this.dateService.compare(date, this.internalSelectedEndDate()) < 0;
165
257
  }
166
- /** Handles year/month selection in the multi-year/year views. */
167
- _goToDateInView(date, view) {
168
- this.activeDate = date;
169
- this.currentView = view;
258
+ isDateRangeStart(date) {
259
+ return this.isRange() && this.internalSelectedStartDate()
260
+ ? this.dateService.areEqual(date, this.internalSelectedStartDate())
261
+ : false;
170
262
  }
171
- /** Called when the user starts dragging to change a date range. */
172
- _dragStarted(event) {
173
- this._activeDrag = event;
263
+ isDateRangeEnd(date) {
264
+ return this.isRange() && this.internalSelectedEndDate()
265
+ ? this.dateService.areEqual(date, this.internalSelectedEndDate())
266
+ : false;
174
267
  }
175
- /**
176
- * Called when a drag completes. It may end in cancelation or in the selection
177
- * of a new range.
178
- */
179
- _dragEnded(event) {
180
- if (!this._activeDrag) {
181
- return;
268
+ isDateDisabled(date) {
269
+ if (this.minDate() && this.dateService.compare(date, this.minDate()) < 0) {
270
+ return true;
182
271
  }
183
- if (event.value) {
184
- this.userDragDrop.emit(event);
272
+ if (this.maxDate() && this.dateService.compare(date, this.maxDate()) > 0) {
273
+ return true;
185
274
  }
186
- this._activeDrag = undefined;
275
+ const filterFn = this.disabledDatesFilter();
276
+ if (!!filterFn) {
277
+ return !filterFn(date);
278
+ }
279
+ return false;
280
+ }
281
+ isMonthDisabled(year, month) {
282
+ const firstDayOfMonth = new Date(year, month, 1);
283
+ const lastDayOfMonth = new Date(year, month + 1, 0);
284
+ if (this.minDate() && this.dateService.compare(lastDayOfMonth, this.minDate()) < 0) {
285
+ return true;
286
+ }
287
+ return !!(this.maxDate() && this.dateService.compare(firstDayOfMonth, this.maxDate()) > 0);
288
+ }
289
+ isYearDisabled(year) {
290
+ const firstDayOfYear = new Date(year, 0, 1);
291
+ const lastDayOfYear = new Date(year, 11, 31);
292
+ if (this.minDate() && this.dateService.compare(lastDayOfYear, this.minDate()) < 0) {
293
+ return true;
294
+ }
295
+ return !!(this.maxDate() && this.dateService.compare(firstDayOfYear, this.maxDate()) > 0);
296
+ }
297
+ updateYearRange() {
298
+ const currentYear = this.currentYear();
299
+ this.yearRange.set({
300
+ start: currentYear - 6,
301
+ end: currentYear + 5
302
+ });
187
303
  }
188
- /** Returns the component instance that corresponds to the current calendar view. */
189
- _getCurrentViewComponent() {
190
- // The return type is explicitly written as a union to ensure that the Closure compiler does
191
- // not optimize calls to _init(). Without the explicit return type, TypeScript narrows it to
192
- // only the first component type. See https://github.com/angular/components/issues/22996.
193
- return (this.monthView || this.yearView || this.multiYearView);
304
+ moveYearRange(offset) {
305
+ const current = this.yearRange();
306
+ this.yearRange.set({
307
+ start: current.start + offset,
308
+ end: current.end + offset
309
+ });
194
310
  }
195
311
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: CalendarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
196
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.1.0", type: CalendarComponent, selector: "upd-calendar", inputs: { startAt: "startAt", startView: "startView", selected: "selected", minDate: "minDate", maxDate: "maxDate", dateFilter: "dateFilter", dateClass: "dateClass", comparisonStart: "comparisonStart", comparisonEnd: "comparisonEnd", startDateAccessibleName: "startDateAccessibleName", endDateAccessibleName: "endDateAccessibleName", isRange: "isRange" }, outputs: { selectedChange: "selectedChange", yearSelected: "yearSelected", monthSelected: "monthSelected", viewChanged: "viewChanged", userSelection: "userSelection", userDragDrop: "userDragDrop" }, viewQueries: [{ propertyName: "monthView", first: true, predicate: CalendarMonthViewComponent, descendants: true }, { propertyName: "yearView", first: true, predicate: CalendarYearViewComponent, descendants: true }, { propertyName: "multiYearView", first: true, predicate: CalendarMultiYearViewComponent, descendants: true }], usesOnChanges: true, ngImport: i0, template: "@switch (currentView) {\n @case ('month') {\n <upd-calendar-month-view [(activeDate)]=\"activeDate\" [isRange]=\"isRange\" [selected]=\"selected\" [dateFilter]=\"dateFilter\"\n [maxDate]=\"maxDate\" [minDate]=\"minDate\" [dateClass]=\"dateClass\" [comparisonStart]=\"comparisonStart\"\n [comparisonEnd]=\"comparisonEnd\" [startDateAccessibleName]=\"startDateAccessibleName\"\n [endDateAccessibleName]=\"endDateAccessibleName\" (userSelection)=\"_dateSelected($event)\" (dragStarted)=\"_dragStarted($event)\"\n (dragEnded)=\"_dragEnded($event)\" [activeDrag]=\"_activeDrag\">\n </upd-calendar-month-view>\n }\n @case ('year') {\n YEAR!\n <!-- <upd-calendar-year-view-->\n <!-- [(activeDate)]=\"activeDate\"-->\n <!-- [selected]=\"selected\"-->\n <!-- [dateFilter]=\"dateFilter\"-->\n <!-- [maxDate]=\"maxDate\"-->\n <!-- [minDate]=\"minDate\"-->\n <!-- [dateClass]=\"dateClass\"-->\n <!-- (monthSelected)=\"_monthSelectedInYearView($event)\"-->\n <!-- (selectedChange)=\"_goToDateInView($event, 'month')\"></upd-calendar-year-view>-->\n }\n @case ('multi-year') {\n MUUMUMUULTI-YEAR!\n <!-- <upd-calendar-multi-year-view-->\n <!-- [(activeDate)]=\"activeDate\"-->\n <!-- [selected]=\"selected\"-->\n <!-- [dateFilter]=\"dateFilter\"-->\n <!-- [maxDate]=\"maxDate\"-->\n <!-- [minDate]=\"minDate\"-->\n <!-- [dateClass]=\"dateClass\"-->\n <!-- (yearSelected)=\"_yearSelectedInMultiYearView($event)\"-->\n <!-- (selectedChange)=\"_goToDateInView($event, 'year')\"></upd-calendar-multi-year-view>-->\n }\n}\n", styles: [""], dependencies: [{ kind: "component", type: i1.CalendarMonthViewComponent, selector: "upd-calendar-month-view", inputs: ["activeDate", "selected", "minDate", "maxDate", "dateFilter", "dateClass", "comparisonStart", "comparisonEnd", "startDateAccessibleName", "endDateAccessibleName", "firstDayOfWeek", "activeDrag", "isRange"], outputs: ["selectedChange", "userSelection", "dragStarted", "dragEnded", "activeDateChange"] }] }); }
312
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.1.0", type: CalendarComponent, selector: "upd-calendar", inputs: { selectedDate: { classPropertyName: "selectedDate", publicName: "selectedDate", isSignal: true, isRequired: false, transformFunction: null }, selectedStartDate: { classPropertyName: "selectedStartDate", publicName: "selectedStartDate", isSignal: true, isRequired: false, transformFunction: null }, selectedEndDate: { classPropertyName: "selectedEndDate", publicName: "selectedEndDate", isSignal: true, isRequired: false, transformFunction: null }, minDate: { classPropertyName: "minDate", publicName: "minDate", isSignal: true, isRequired: false, transformFunction: null }, maxDate: { classPropertyName: "maxDate", publicName: "maxDate", isSignal: true, isRequired: false, transformFunction: null }, disabledDatesFilter: { classPropertyName: "disabledDatesFilter", publicName: "disabledDatesFilter", isSignal: true, isRequired: false, transformFunction: null }, isRange: { classPropertyName: "isRange", publicName: "isRange", isSignal: true, isRequired: false, transformFunction: null }, shouldRemoveExternalBorder: { classPropertyName: "shouldRemoveExternalBorder", publicName: "shouldRemoveExternalBorder", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { selected: "selected", monthChanged: "monthChanged" }, ngImport: i0, template: "<div class=\"calendar\" [class.calendar-border]=\"!shouldRemoveExternalBorder()\">\n <div class=\"calendar-header\">\n <button type=\"button\" class=\"btn btn-sm btn-ghost-secondary\" (click)=\"onPreviousClick()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" class=\"icon\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" stroke-width=\"2\"\n stroke=\"currentColor\" fill=\"none\">\n <path stroke=\"none\" d=\"m0 0h24v24H0z\" fill=\"none\" />\n <path d=\"m15 6-6 6 6 6\" />\n </svg>\n </button>\n\n <button type=\"button\" class=\"btn btn-sm btn-ghost-secondary calendar-title\" (click)=\"onHeaderClick()\">\n @switch (currentView()) {\n @case (CalendarViewEnum.Days) {\n {{ currentMonthName() }}\n }\n @case (CalendarViewEnum.Months) {\n {{ currentYear() }}\n }\n @case (CalendarViewEnum.Years) {\n {{ yearRange().start }} - {{ yearRange().end }}\n }\n }\n <svg xmlns=\"http://www.w3.org/2000/svg\" class=\"icon ms-1\" width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" stroke-width=\"2\"\n stroke=\"currentColor\" fill=\"none\">\n <path stroke=\"none\" d=\"m0 0h24v24H0z\" fill=\"none\" />\n <path d=\"m6 9 6 6 6-6\" />\n </svg>\n </button>\n\n <button type=\"button\" class=\"btn btn-sm btn-ghost-secondary\" (click)=\"onNextClick()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" class=\"icon\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" stroke-width=\"2\"\n stroke=\"currentColor\" fill=\"none\">\n <path stroke=\"none\" d=\"m0 0h24v24H0z\" fill=\"none\" />\n <path d=\"m9 6 6 6-6 6\" />\n </svg>\n </button>\n </div>\n\n @if (currentView() === CalendarViewEnum.Days) {\n <div class=\"calendar-weekdays\">\n @for (dayName of weekDayNames(); track dayName) {\n <div class=\"calendar-weekday\">{{ dayName }}</div>\n }\n </div>\n\n <div class=\"calendar-days\">\n @for (day of calendarDays(); track day.date.getTime()) {\n <button type=\"button\" class=\"calendar-day\" [class.calendar-day-other-month]=\"!day.isCurrentMonth\"\n [class.calendar-day-today]=\"day.isToday\" [class.calendar-day-selected]=\"day.isSelected\"\n [class.calendar-day-in-range]=\"day.isInRange\" [class.calendar-day-range-start]=\"day.isRangeStart\"\n [class.calendar-day-range-end]=\"day.isRangeEnd\" [disabled]=\"day.isDisabled\" (click)=\"onDateClick(day)\">\n {{ day.date.getDate() }}\n </button>\n }\n </div>\n }\n\n @if (currentView() === CalendarViewEnum.Months) {\n <div class=\"calendar-months\">\n @for (month of calendarMonths(); track month.monthIndex) {\n <button type=\"button\" class=\"calendar-month\" [class.calendar-month-current]=\"month.isCurrentMonth\"\n [class.calendar-month-selected]=\"month.isSelected\" [disabled]=\"month.isDisabled\" (click)=\"onMonthClick(month)\">\n {{ month.name }}\n </button>\n }\n </div>\n }\n\n @if (currentView() === CalendarViewEnum.Years) {\n <div class=\"calendar-years\">\n @for (year of calendarYears(); track year.year) {\n <button type=\"button\" class=\"calendar-year\" [class.calendar-year-current]=\"year.isCurrentYear\"\n [class.calendar-year-selected]=\"year.isSelected\" [disabled]=\"year.isDisabled\" (click)=\"onYearClick(year)\">\n {{ year.year }}\n </button>\n }\n </div>\n }\n</div>\n", styles: [".calendar{background:var(--bs-body-bg);padding:1rem;width:100%;height:100%;border:0!important}.calendar-border{border:1px solid var(--bs-border-color)!important;border-radius:var(--bs-border-radius, .375rem);box-shadow:var(--bs-box-shadow-sm, 0 .125rem .25rem rgba(0, 0, 0, .075))}.calendar-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:1rem}.calendar-header .calendar-title{font-weight:600;flex:1;text-align:center;display:flex;align-items:center;justify-content:center;cursor:pointer}.calendar-header .calendar-title:hover{color:var(--bs-primary)}.calendar-header button{background:transparent;border:none;padding:.25rem;border-radius:var(--bs-border-radius-sm, .25rem);cursor:pointer;color:var(--bs-secondary)}.calendar-header button:hover{background-color:var(--bs-gray-100);color:var(--bs-gray-700)}.calendar-weekdays{display:grid;grid-template-columns:repeat(7,1fr);gap:.25rem;margin-bottom:.5rem}.calendar-weekday{text-align:center;font-size:.75rem;font-weight:600;color:var(--bs-secondary);padding:.25rem;text-transform:uppercase}.calendar-days{display:grid;grid-template-columns:repeat(7,1fr);gap:.125rem}.calendar-day{aspect-ratio:1;border:none;background:transparent;border-radius:var(--bs-border-radius-sm, .25rem);font-size:.875rem;cursor:pointer;transition:all .15s ease-in-out;display:flex;align-items:center;justify-content:center;min-height:32px;font-weight:400;position:relative}.calendar-day:hover:not(:disabled):not(.calendar-day-other-month){background-color:var(--bs-gray-100)}.calendar-day:disabled{cursor:not-allowed}.calendar-day.calendar-day-other-month{color:var(--bs-gray-400, #adb5bd);cursor:default;pointer-events:none}.calendar-day.calendar-day-other-month:hover{background-color:transparent}.calendar-day.calendar-day-today{background-color:var(--bs-primary-bg-subtle, rgba(13, 110, 253, .1));color:var(--bs-primary, #0d6efd);font-weight:600}.calendar-day.calendar-day-today:hover:not(.calendar-day-other-month){background-color:var(--bs-primary-border-subtle, rgba(13, 110, 253, .2))}.calendar-day.calendar-day-selected{background-color:var(--bs-primary, #0d6efd);color:var(--bs-white, white);font-weight:600}.calendar-day.calendar-day-selected:hover:not(.calendar-day-other-month){background-color:var(--bs-primary-dark, #0a58ca)}.calendar-day.calendar-day-in-range{border-radius:0;background-color:var(--bs-secondary-bg-subtle, rgba(108, 117, 125, .1));color:var(--bs-secondary-emphasis, #495057)}.calendar-day.calendar-day-in-range:hover:not(.calendar-day-other-month){background-color:var(--bs-secondary-border-subtle, rgba(108, 117, 125, .2))}.calendar-day.calendar-day-range-start{background-color:var(--bs-primary, #0d6efd);color:var(--bs-white, white);font-weight:600;border-top-right-radius:0;border-bottom-right-radius:0}.calendar-day.calendar-day-range-start:hover:not(.calendar-day-other-month){background-color:var(--bs-primary-dark, #0a58ca)}.calendar-day.calendar-day-range-end{background-color:var(--bs-primary, #0d6efd);color:var(--bs-white, white);font-weight:600;border-top-left-radius:0;border-bottom-left-radius:0}.calendar-day.calendar-day-range-end:hover:not(.calendar-day-other-month){background-color:var(--bs-primary-dark, #0a58ca)}.calendar-day.calendar-day-range-start.calendar-day-range-end{border-radius:var(--bs-border-radius-sm, .25rem)}.calendar-day.calendar-day-other-month.calendar-day-selected{background-color:transparent;color:var(--bs-gray-400, #adb5bd);font-weight:400}.calendar-months{display:grid;grid-template-columns:repeat(3,1fr);gap:.5rem}.calendar-month{padding:.75rem .5rem;border:none;background:transparent;border-radius:var(--bs-border-radius-sm, .25rem);font-size:.875rem;font-weight:500;cursor:pointer;transition:all .15s ease-in-out}.calendar-month:hover:not(:disabled){background-color:var(--bs-gray-100, #f8f9fa)}.calendar-month:disabled{color:var(--bs-gray-400, #adb5bd);cursor:not-allowed}.calendar-month.calendar-month-current{background-color:var(--bs-primary-bg-subtle, rgba(13, 110, 253, .1));color:var(--bs-primary, #0d6efd);font-weight:600}.calendar-month.calendar-month-selected{background-color:var(--bs-primary, #0d6efd);color:var(--bs-white, white);font-weight:600}.calendar-month.calendar-month-selected:hover{background-color:var(--bs-primary-dark, #0a58ca)}.calendar-years{display:grid;grid-template-columns:repeat(3,1fr);gap:.5rem}.calendar-year{padding:.75rem .5rem;border:none;background:transparent;border-radius:var(--bs-border-radius-sm, .25rem);font-size:.875rem;font-weight:500;cursor:pointer;transition:all .15s ease-in-out}.calendar-year:hover:not(:disabled){background-color:var(--bs-gray-100, #f8f9fa)}.calendar-year:disabled{color:var(--bs-gray-400, #adb5bd);cursor:not-allowed}.calendar-year.calendar-year-current{background-color:var(--bs-primary-bg-subtle, rgba(13, 110, 253, .1));color:var(--bs-primary, #0d6efd);font-weight:600}.calendar-year.calendar-year-selected{background-color:var(--bs-primary, #0d6efd);color:var(--bs-white, white);font-weight:600}.calendar-year.calendar-year-selected:hover{background-color:var(--bs-primary-dark, #0a58ca)}.calendar-day-not-selectable{pointer-events:none;opacity:.5}\n"] }); }
197
313
  }
198
314
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: CalendarComponent, decorators: [{
199
315
  type: Component,
200
- args: [{ selector: 'upd-calendar', template: "@switch (currentView) {\n @case ('month') {\n <upd-calendar-month-view [(activeDate)]=\"activeDate\" [isRange]=\"isRange\" [selected]=\"selected\" [dateFilter]=\"dateFilter\"\n [maxDate]=\"maxDate\" [minDate]=\"minDate\" [dateClass]=\"dateClass\" [comparisonStart]=\"comparisonStart\"\n [comparisonEnd]=\"comparisonEnd\" [startDateAccessibleName]=\"startDateAccessibleName\"\n [endDateAccessibleName]=\"endDateAccessibleName\" (userSelection)=\"_dateSelected($event)\" (dragStarted)=\"_dragStarted($event)\"\n (dragEnded)=\"_dragEnded($event)\" [activeDrag]=\"_activeDrag\">\n </upd-calendar-month-view>\n }\n @case ('year') {\n YEAR!\n <!-- <upd-calendar-year-view-->\n <!-- [(activeDate)]=\"activeDate\"-->\n <!-- [selected]=\"selected\"-->\n <!-- [dateFilter]=\"dateFilter\"-->\n <!-- [maxDate]=\"maxDate\"-->\n <!-- [minDate]=\"minDate\"-->\n <!-- [dateClass]=\"dateClass\"-->\n <!-- (monthSelected)=\"_monthSelectedInYearView($event)\"-->\n <!-- (selectedChange)=\"_goToDateInView($event, 'month')\"></upd-calendar-year-view>-->\n }\n @case ('multi-year') {\n MUUMUMUULTI-YEAR!\n <!-- <upd-calendar-multi-year-view-->\n <!-- [(activeDate)]=\"activeDate\"-->\n <!-- [selected]=\"selected\"-->\n <!-- [dateFilter]=\"dateFilter\"-->\n <!-- [maxDate]=\"maxDate\"-->\n <!-- [minDate]=\"minDate\"-->\n <!-- [dateClass]=\"dateClass\"-->\n <!-- (yearSelected)=\"_yearSelectedInMultiYearView($event)\"-->\n <!-- (selectedChange)=\"_goToDateInView($event, 'year')\"></upd-calendar-multi-year-view>-->\n }\n}\n" }]
201
- }], propDecorators: { monthView: [{
202
- type: ViewChild,
203
- args: [CalendarMonthViewComponent]
204
- }], yearView: [{
205
- type: ViewChild,
206
- args: [CalendarYearViewComponent]
207
- }], multiYearView: [{
208
- type: ViewChild,
209
- args: [CalendarMultiYearViewComponent]
210
- }], startAt: [{
211
- type: Input
212
- }], startView: [{
213
- type: Input
214
- }], selected: [{
215
- type: Input
216
- }], minDate: [{
217
- type: Input
218
- }], maxDate: [{
219
- type: Input
220
- }], dateFilter: [{
221
- type: Input
222
- }], dateClass: [{
223
- type: Input
224
- }], comparisonStart: [{
225
- type: Input
226
- }], comparisonEnd: [{
227
- type: Input
228
- }], startDateAccessibleName: [{
229
- type: Input
230
- }], endDateAccessibleName: [{
231
- type: Input
232
- }], isRange: [{
233
- type: Input
234
- }], selectedChange: [{
235
- type: Output
236
- }], yearSelected: [{
237
- type: Output
238
- }], monthSelected: [{
239
- type: Output
240
- }], viewChanged: [{
241
- type: Output
242
- }], userSelection: [{
243
- type: Output
244
- }], userDragDrop: [{
245
- type: Output
246
- }] } });
247
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.component.js","sourceRoot":"","sources":["../../../../../libs/components/calendar/src/calendar.component.ts","../../../../../libs/components/calendar/src/calendar.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterContentInit,\n    AfterViewChecked,\n    ChangeDetectorRef,\n    Component,\n    EventEmitter,\n    Input,\n    OnChanges,\n    Output,\n    SimpleChange,\n    SimpleChanges,\n    ViewChild,\n    inject\n} from '@angular/core';\nimport { OptionalType } from '@updevs/sdk/types';\nimport { DateService } from '@updevs/sdk/core';\n\nimport { DateRangeModel } from './models/date-range.model';\nimport { CalendarViewType } from './types/calendar-view.type';\nimport { CalendarUserEventModel } from './models/calendar-user-event.model';\nimport { CalendarCellClassFunctionType } from './types/calendar-cell-class-function.type';\nimport { CalendarMonthViewComponent } from './components/calendar-month-view/calendar-month-view.component';\nimport { CalendarYearViewComponent } from './components/calendar-year-view/calendar-year-view.component';\nimport { CalendarMultiYearViewComponent } from './components/calendar-multi-year-view/calendar-multi-year-view.component';\n\n@Component({\n    selector: 'upd-calendar',\n    templateUrl: './calendar.component.html',\n    styleUrl: './calendar.component.scss'\n})\nexport class CalendarComponent implements AfterContentInit, AfterViewChecked, OnChanges {\n    /** Reference to the current month view component. */\n    @ViewChild(CalendarMonthViewComponent) monthView?: CalendarMonthViewComponent;\n\n    /** Reference to the current year view component. */\n    @ViewChild(CalendarYearViewComponent) yearView?: CalendarYearViewComponent;\n\n    /** Reference to the current multi-year view component. */\n    @ViewChild(CalendarMultiYearViewComponent) multiYearView?: CalendarMultiYearViewComponent;\n\n    /** A date representing the period (month or year) to start the calendar in. */\n    @Input()\n    get startAt(): OptionalType<Date> {\n        return this._startAt;\n    }\n    set startAt(value: OptionalType<Date>) {\n        this._startAt = this.dateService.getValidOrUndefined(this.dateService.deserialize(value));\n    }\n\n    /**\n     * Whether the calendar should be started in month or year view.\n     */\n    @Input() startView: CalendarViewType = 'month';\n    /**\n     * The currently selected date.\n     */\n    @Input()\n    get selected(): DateRangeModel | OptionalType<Date> {\n        return this._selected;\n    }\n    set selected(value: DateRangeModel | OptionalType<Date>) {\n        console.log('setting selected', value);\n        if (value instanceof DateRangeModel) {\n            this._selected = value;\n        } else {\n            this._selected = this.dateService.getValidOrUndefined(this.dateService.deserialize(value));\n        }\n    }\n    /**\n     * The minimum selectable date.\n     */\n    @Input()\n    get minDate(): OptionalType<Date> {\n        return this._minDate;\n    }\n    set minDate(value: OptionalType<Date>) {\n        this._minDate = this.dateService.getValidOrUndefined(this.dateService.deserialize(value));\n    }\n    /**\n     * The maximum selectable date.\n     */\n    @Input()\n    get maxDate(): OptionalType<Date> {\n        return this._maxDate;\n    }\n    set maxDate(value: OptionalType<Date>) {\n        this._maxDate = this.dateService.getValidOrUndefined(this.dateService.deserialize(value));\n    }\n    /**\n     * Function used to filter which dates are selectable.\n     */\n    @Input() dateFilter?: (date: Date) => boolean;\n\n    /** Function that can be used to add custom CSS classes to dates. */\n    @Input() dateClass?: CalendarCellClassFunctionType;\n\n    /** Start of the comparison range. */\n    @Input() comparisonStart?: Date;\n\n    /** End of the comparison range. */\n    @Input() comparisonEnd?: Date;\n\n    /** ARIA Accessible name of the `<input matStartDate/>` */\n    @Input() startDateAccessibleName?: string;\n\n    /** ARIA Accessible name of the `<input matEndDate/>` */\n    @Input() endDateAccessibleName?: string;\n    /** Whether the user is currently selecting a range of dates. */\n    @Input() isRange = false;\n\n    /** Emits when the currently selected date changes. */\n    @Output() readonly selectedChange = new EventEmitter<OptionalType<Date>>();\n\n    /**\n     * Emits the year chosen in multiyear view.\n     * This doesn't imply a change on the selected date.\n     */\n    @Output() readonly yearSelected = new EventEmitter<Date>();\n\n    /**\n     * Emits the month chosen in year view.\n     * This doesn't imply a change on the selected date.\n     */\n    @Output() readonly monthSelected = new EventEmitter<Date>();\n\n    /**\n     * Emits when the current view changes.\n     */\n    @Output() readonly viewChanged = new EventEmitter<CalendarViewType>(true);\n\n    /** Emits when any date is selected. */\n    @Output() readonly userSelection = new EventEmitter<CalendarUserEventModel<OptionalType<Date>>>();\n\n    /** Emits a new date range value when the user completes a drag drop operation. */\n    @Output() readonly userDragDrop = new EventEmitter<CalendarUserEventModel<DateRangeModel>>();\n\n    /**\n     * The current active date. This determines which time period is shown and which date is\n     * highlighted when using keyboard navigation.\n     */\n    get activeDate(): Date {\n        return this._clampedActiveDate;\n    }\n    set activeDate(value: Date) {\n        this._clampedActiveDate = this.dateService.clamp(value, this.minDate, this.maxDate);\n        this._changeDetectorRef.markForCheck();\n    }\n\n    /** Whether the calendar is in month view. */\n    get currentView(): CalendarViewType {\n        return this._currentView;\n    }\n    set currentView(value: CalendarViewType) {\n        const viewChangedResult = this._currentView !== value ? value : undefined;\n        this._currentView = value;\n        this._moveFocusOnNextTick = true;\n\n        this._changeDetectorRef.markForCheck();\n\n        if (viewChangedResult) {\n            this.viewChanged.emit(viewChangedResult);\n        }\n    }\n\n    /** Origin of active drag, or null when dragging is not active. */\n    protected _activeDrag?: CalendarUserEventModel<Date>;\n\n    private _clampedActiveDate!: Date;\n    private _startAt?: Date;\n    private _selected?: DateRangeModel | Date;\n    private _minDate?: Date;\n    private _maxDate?: Date;\n\n    /**\n     * Used for scheduling that focus should be moved to the active cell on the next tick.\n     * We need to schedule it, rather than do it immediately, because we have to wait\n     * for Angular to re-evaluate the view children.\n     */\n    private _moveFocusOnNextTick = false;\n    private _currentView!: CalendarViewType;\n\n    private readonly dateService = inject(DateService);\n    private readonly _changeDetectorRef = inject(ChangeDetectorRef);\n\n    ngAfterContentInit(): void {\n        this.activeDate = this.startAt || new Date();\n        // Assign to the private property since we don't want to move focus on init.\n        this._currentView = this.startView;\n    }\n\n    ngAfterViewChecked(): void {\n        if (this._moveFocusOnNextTick) {\n            this._moveFocusOnNextTick = false;\n            this.focusActiveCell();\n        }\n    }\n\n    ngOnChanges(changes: SimpleChanges): void {\n        // Ignore date changes that are at a different time on the same day. This fixes issues where\n        // the calendar re-renders when there is no meaningful change to [minDate] or [maxDate]\n        // (#24435).\n        const minDateChange: OptionalType<SimpleChange> =\n            changes['minDate'] && !this.dateService.areEqual(changes['minDate'].previousValue, changes['minDate'].currentValue)\n                ? changes['minDate']\n                : undefined;\n        const maxDateChange: OptionalType<SimpleChange> =\n            changes['maxDate'] && !this.dateService.areEqual(changes['maxDate'].previousValue, changes['maxDate'].currentValue)\n                ? changes['maxDate']\n                : undefined;\n\n        const change = minDateChange || maxDateChange || changes['dateFilter'];\n\n        if (change && !change.firstChange) {\n            const view = this._getCurrentViewComponent();\n\n            if (view) {\n                // We need to `detectChanges` manually here, because the `minDate`, `maxDate` etc. are\n                // passed down to the view via data bindings which won't be up-to-date when we call `_init`.\n                this._changeDetectorRef.detectChanges();\n                view.init();\n            }\n        }\n    }\n\n    /** Focuses the active date. */\n    focusActiveCell() {\n        this._getCurrentViewComponent().focusActiveCell(false);\n    }\n\n    /** Updates today's date after an update of the active date */\n    updateTodaysDate() {\n        this._getCurrentViewComponent().init();\n    }\n\n    /** Handles date selection in the month view. */\n    _dateSelected(event: CalendarUserEventModel<OptionalType<Date>>): void {\n        const date = event.value;\n\n        if (this.selected instanceof DateRangeModel || (date && !this.dateService.areEqual(date, this.selected))) {\n            this.selectedChange.emit(date);\n        }\n\n        this.userSelection.emit(event);\n    }\n\n    /** Handles year selection in the multiyear view. */\n    _yearSelectedInMultiYearView(normalizedYear: Date) {\n        this.yearSelected.emit(normalizedYear);\n    }\n\n    /** Handles month selection in the year view. */\n    _monthSelectedInYearView(normalizedMonth: Date) {\n        this.monthSelected.emit(normalizedMonth);\n    }\n\n    /** Handles year/month selection in the multi-year/year views. */\n    _goToDateInView(date: Date, view: CalendarViewType): void {\n        this.activeDate = date;\n        this.currentView = view;\n    }\n\n    /** Called when the user starts dragging to change a date range. */\n    _dragStarted(event: CalendarUserEventModel<Date>) {\n        this._activeDrag = event;\n    }\n\n    /**\n     * Called when a drag completes. It may end in cancelation or in the selection\n     * of a new range.\n     */\n    _dragEnded(event: CalendarUserEventModel<OptionalType<DateRangeModel>>) {\n        if (!this._activeDrag) {\n            return;\n        }\n\n        if (event.value) {\n            this.userDragDrop.emit(event as CalendarUserEventModel<DateRangeModel>);\n        }\n\n        this._activeDrag = undefined;\n    }\n\n    /** Returns the component instance that corresponds to the current calendar view. */\n    private _getCurrentViewComponent(): CalendarMonthViewComponent | CalendarYearViewComponent | CalendarMultiYearViewComponent {\n        // The return type is explicitly written as a union to ensure that the Closure compiler does\n        // not optimize calls to _init(). Without the explicit return type, TypeScript narrows it to\n        // only the first component type. See https://github.com/angular/components/issues/22996.\n        return (this.monthView || this.yearView || this.multiYearView)!;\n    }\n}\n","@switch (currentView) {\n    @case ('month') {\n        <upd-calendar-month-view [(activeDate)]=\"activeDate\" [isRange]=\"isRange\" [selected]=\"selected\" [dateFilter]=\"dateFilter\"\n            [maxDate]=\"maxDate\" [minDate]=\"minDate\" [dateClass]=\"dateClass\" [comparisonStart]=\"comparisonStart\"\n            [comparisonEnd]=\"comparisonEnd\" [startDateAccessibleName]=\"startDateAccessibleName\"\n            [endDateAccessibleName]=\"endDateAccessibleName\" (userSelection)=\"_dateSelected($event)\" (dragStarted)=\"_dragStarted($event)\"\n            (dragEnded)=\"_dragEnded($event)\" [activeDrag]=\"_activeDrag\">\n        </upd-calendar-month-view>\n    }\n    @case ('year') {\n        YEAR!\n            <!--        <upd-calendar-year-view-->\n            <!--            [(activeDate)]=\"activeDate\"-->\n            <!--            [selected]=\"selected\"-->\n            <!--            [dateFilter]=\"dateFilter\"-->\n            <!--            [maxDate]=\"maxDate\"-->\n            <!--            [minDate]=\"minDate\"-->\n            <!--            [dateClass]=\"dateClass\"-->\n            <!--            (monthSelected)=\"_monthSelectedInYearView($event)\"-->\n            <!--            (selectedChange)=\"_goToDateInView($event, 'month')\"></upd-calendar-year-view>-->\n    }\n    @case ('multi-year') {\n        MUUMUMUULTI-YEAR!\n            <!--        <upd-calendar-multi-year-view-->\n            <!--            [(activeDate)]=\"activeDate\"-->\n            <!--            [selected]=\"selected\"-->\n            <!--            [dateFilter]=\"dateFilter\"-->\n            <!--            [maxDate]=\"maxDate\"-->\n            <!--            [minDate]=\"minDate\"-->\n            <!--            [dateClass]=\"dateClass\"-->\n            <!--            (yearSelected)=\"_yearSelectedInMultiYearView($event)\"-->\n            <!--            (selectedChange)=\"_goToDateInView($event, 'year')\"></upd-calendar-multi-year-view>-->\n    }\n}\n"]}
316
+ args: [{ selector: 'upd-calendar', template: "<div class=\"calendar\" [class.calendar-border]=\"!shouldRemoveExternalBorder()\">\n <div class=\"calendar-header\">\n <button type=\"button\" class=\"btn btn-sm btn-ghost-secondary\" (click)=\"onPreviousClick()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" class=\"icon\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" stroke-width=\"2\"\n stroke=\"currentColor\" fill=\"none\">\n <path stroke=\"none\" d=\"m0 0h24v24H0z\" fill=\"none\" />\n <path d=\"m15 6-6 6 6 6\" />\n </svg>\n </button>\n\n <button type=\"button\" class=\"btn btn-sm btn-ghost-secondary calendar-title\" (click)=\"onHeaderClick()\">\n @switch (currentView()) {\n @case (CalendarViewEnum.Days) {\n {{ currentMonthName() }}\n }\n @case (CalendarViewEnum.Months) {\n {{ currentYear() }}\n }\n @case (CalendarViewEnum.Years) {\n {{ yearRange().start }} - {{ yearRange().end }}\n }\n }\n <svg xmlns=\"http://www.w3.org/2000/svg\" class=\"icon ms-1\" width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" stroke-width=\"2\"\n stroke=\"currentColor\" fill=\"none\">\n <path stroke=\"none\" d=\"m0 0h24v24H0z\" fill=\"none\" />\n <path d=\"m6 9 6 6 6-6\" />\n </svg>\n </button>\n\n <button type=\"button\" class=\"btn btn-sm btn-ghost-secondary\" (click)=\"onNextClick()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" class=\"icon\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" stroke-width=\"2\"\n stroke=\"currentColor\" fill=\"none\">\n <path stroke=\"none\" d=\"m0 0h24v24H0z\" fill=\"none\" />\n <path d=\"m9 6 6 6-6 6\" />\n </svg>\n </button>\n </div>\n\n @if (currentView() === CalendarViewEnum.Days) {\n <div class=\"calendar-weekdays\">\n @for (dayName of weekDayNames(); track dayName) {\n <div class=\"calendar-weekday\">{{ dayName }}</div>\n }\n </div>\n\n <div class=\"calendar-days\">\n @for (day of calendarDays(); track day.date.getTime()) {\n <button type=\"button\" class=\"calendar-day\" [class.calendar-day-other-month]=\"!day.isCurrentMonth\"\n [class.calendar-day-today]=\"day.isToday\" [class.calendar-day-selected]=\"day.isSelected\"\n [class.calendar-day-in-range]=\"day.isInRange\" [class.calendar-day-range-start]=\"day.isRangeStart\"\n [class.calendar-day-range-end]=\"day.isRangeEnd\" [disabled]=\"day.isDisabled\" (click)=\"onDateClick(day)\">\n {{ day.date.getDate() }}\n </button>\n }\n </div>\n }\n\n @if (currentView() === CalendarViewEnum.Months) {\n <div class=\"calendar-months\">\n @for (month of calendarMonths(); track month.monthIndex) {\n <button type=\"button\" class=\"calendar-month\" [class.calendar-month-current]=\"month.isCurrentMonth\"\n [class.calendar-month-selected]=\"month.isSelected\" [disabled]=\"month.isDisabled\" (click)=\"onMonthClick(month)\">\n {{ month.name }}\n </button>\n }\n </div>\n }\n\n @if (currentView() === CalendarViewEnum.Years) {\n <div class=\"calendar-years\">\n @for (year of calendarYears(); track year.year) {\n <button type=\"button\" class=\"calendar-year\" [class.calendar-year-current]=\"year.isCurrentYear\"\n [class.calendar-year-selected]=\"year.isSelected\" [disabled]=\"year.isDisabled\" (click)=\"onYearClick(year)\">\n {{ year.year }}\n </button>\n }\n </div>\n }\n</div>\n", styles: [".calendar{background:var(--bs-body-bg);padding:1rem;width:100%;height:100%;border:0!important}.calendar-border{border:1px solid var(--bs-border-color)!important;border-radius:var(--bs-border-radius, .375rem);box-shadow:var(--bs-box-shadow-sm, 0 .125rem .25rem rgba(0, 0, 0, .075))}.calendar-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:1rem}.calendar-header .calendar-title{font-weight:600;flex:1;text-align:center;display:flex;align-items:center;justify-content:center;cursor:pointer}.calendar-header .calendar-title:hover{color:var(--bs-primary)}.calendar-header button{background:transparent;border:none;padding:.25rem;border-radius:var(--bs-border-radius-sm, .25rem);cursor:pointer;color:var(--bs-secondary)}.calendar-header button:hover{background-color:var(--bs-gray-100);color:var(--bs-gray-700)}.calendar-weekdays{display:grid;grid-template-columns:repeat(7,1fr);gap:.25rem;margin-bottom:.5rem}.calendar-weekday{text-align:center;font-size:.75rem;font-weight:600;color:var(--bs-secondary);padding:.25rem;text-transform:uppercase}.calendar-days{display:grid;grid-template-columns:repeat(7,1fr);gap:.125rem}.calendar-day{aspect-ratio:1;border:none;background:transparent;border-radius:var(--bs-border-radius-sm, .25rem);font-size:.875rem;cursor:pointer;transition:all .15s ease-in-out;display:flex;align-items:center;justify-content:center;min-height:32px;font-weight:400;position:relative}.calendar-day:hover:not(:disabled):not(.calendar-day-other-month){background-color:var(--bs-gray-100)}.calendar-day:disabled{cursor:not-allowed}.calendar-day.calendar-day-other-month{color:var(--bs-gray-400, #adb5bd);cursor:default;pointer-events:none}.calendar-day.calendar-day-other-month:hover{background-color:transparent}.calendar-day.calendar-day-today{background-color:var(--bs-primary-bg-subtle, rgba(13, 110, 253, .1));color:var(--bs-primary, #0d6efd);font-weight:600}.calendar-day.calendar-day-today:hover:not(.calendar-day-other-month){background-color:var(--bs-primary-border-subtle, rgba(13, 110, 253, .2))}.calendar-day.calendar-day-selected{background-color:var(--bs-primary, #0d6efd);color:var(--bs-white, white);font-weight:600}.calendar-day.calendar-day-selected:hover:not(.calendar-day-other-month){background-color:var(--bs-primary-dark, #0a58ca)}.calendar-day.calendar-day-in-range{border-radius:0;background-color:var(--bs-secondary-bg-subtle, rgba(108, 117, 125, .1));color:var(--bs-secondary-emphasis, #495057)}.calendar-day.calendar-day-in-range:hover:not(.calendar-day-other-month){background-color:var(--bs-secondary-border-subtle, rgba(108, 117, 125, .2))}.calendar-day.calendar-day-range-start{background-color:var(--bs-primary, #0d6efd);color:var(--bs-white, white);font-weight:600;border-top-right-radius:0;border-bottom-right-radius:0}.calendar-day.calendar-day-range-start:hover:not(.calendar-day-other-month){background-color:var(--bs-primary-dark, #0a58ca)}.calendar-day.calendar-day-range-end{background-color:var(--bs-primary, #0d6efd);color:var(--bs-white, white);font-weight:600;border-top-left-radius:0;border-bottom-left-radius:0}.calendar-day.calendar-day-range-end:hover:not(.calendar-day-other-month){background-color:var(--bs-primary-dark, #0a58ca)}.calendar-day.calendar-day-range-start.calendar-day-range-end{border-radius:var(--bs-border-radius-sm, .25rem)}.calendar-day.calendar-day-other-month.calendar-day-selected{background-color:transparent;color:var(--bs-gray-400, #adb5bd);font-weight:400}.calendar-months{display:grid;grid-template-columns:repeat(3,1fr);gap:.5rem}.calendar-month{padding:.75rem .5rem;border:none;background:transparent;border-radius:var(--bs-border-radius-sm, .25rem);font-size:.875rem;font-weight:500;cursor:pointer;transition:all .15s ease-in-out}.calendar-month:hover:not(:disabled){background-color:var(--bs-gray-100, #f8f9fa)}.calendar-month:disabled{color:var(--bs-gray-400, #adb5bd);cursor:not-allowed}.calendar-month.calendar-month-current{background-color:var(--bs-primary-bg-subtle, rgba(13, 110, 253, .1));color:var(--bs-primary, #0d6efd);font-weight:600}.calendar-month.calendar-month-selected{background-color:var(--bs-primary, #0d6efd);color:var(--bs-white, white);font-weight:600}.calendar-month.calendar-month-selected:hover{background-color:var(--bs-primary-dark, #0a58ca)}.calendar-years{display:grid;grid-template-columns:repeat(3,1fr);gap:.5rem}.calendar-year{padding:.75rem .5rem;border:none;background:transparent;border-radius:var(--bs-border-radius-sm, .25rem);font-size:.875rem;font-weight:500;cursor:pointer;transition:all .15s ease-in-out}.calendar-year:hover:not(:disabled){background-color:var(--bs-gray-100, #f8f9fa)}.calendar-year:disabled{color:var(--bs-gray-400, #adb5bd);cursor:not-allowed}.calendar-year.calendar-year-current{background-color:var(--bs-primary-bg-subtle, rgba(13, 110, 253, .1));color:var(--bs-primary, #0d6efd);font-weight:600}.calendar-year.calendar-year-selected{background-color:var(--bs-primary, #0d6efd);color:var(--bs-white, white);font-weight:600}.calendar-year.calendar-year-selected:hover{background-color:var(--bs-primary-dark, #0a58ca)}.calendar-day-not-selectable{pointer-events:none;opacity:.5}\n"] }]
317
+ }] });
318
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.component.js","sourceRoot":"","sources":["../../../../../libs/components/calendar/src/calendar.component.ts","../../../../../libs/components/calendar/src/calendar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, inject, signal, computed, input, output } from '@angular/core';\nimport { DateService } from '@updevs/sdk/core';\nimport { OptionalType } from '@updevs/sdk/types';\n\nimport { CalendarViewEnum } from './models/enums/calendar-view.enum';\nimport { CalendarDateModel } from './models/calendar-date.model';\nimport { CalendarMonthModel } from './models/calendar-month.model';\nimport { CalendarYearModel } from './models/calendar-year.model';\nimport { CalendarSelectedModel } from './models/calendar-selected.model';\n\n@Component({\n    selector: 'upd-calendar',\n    templateUrl: './calendar.component.html',\n    styleUrls: ['./calendar.component.scss']\n})\nexport class CalendarComponent implements OnInit {\n    readonly selectedDate = input<OptionalType<Date>>();\n    readonly selectedStartDate = input<OptionalType<Date>>();\n    readonly selectedEndDate = input<OptionalType<Date>>();\n    readonly minDate = input<OptionalType<Date>>();\n    readonly maxDate = input<OptionalType<Date>>();\n    readonly disabledDatesFilter = input<OptionalType<(date: Date) => boolean>>();\n    readonly isRange = input<boolean>(false);\n    readonly shouldRemoveExternalBorder = input<boolean>(false);\n\n    readonly selected = output<OptionalType<CalendarSelectedModel>>();\n    readonly monthChanged = output<Date>();\n\n    CalendarViewEnum = CalendarViewEnum;\n\n    readonly currentView = signal<CalendarViewEnum>(CalendarViewEnum.Days);\n    readonly currentDate = signal<Date>(new Date());\n    readonly yearRange = signal<{ start: number; end: number }>({ start: 2020, end: 2030 });\n    readonly currentYear = computed(() => this.currentDate().getFullYear());\n    readonly currentMonth = computed(() => this.currentDate().getMonth());\n    readonly currentMonthName = computed(() =>\n        this.dateService.format(this.currentDate(), { month: 'short', year: 'numeric' }).toUpperCase()\n    );\n    readonly calendarDays = computed(() => this.generateCalendarDays());\n    readonly calendarMonths = computed(() => this.generateCalendarMonths());\n    readonly calendarYears = computed(() => this.generateCalendarYears());\n    readonly weekDayNames = computed(() => this.dateService.getDayOfWeekNames('narrow'));\n\n    private readonly internalSelectedDate = signal<OptionalType<Date>>(undefined);\n    private readonly internalSelectedStartDate = signal<OptionalType<Date>>(undefined);\n    private readonly internalSelectedEndDate = signal<OptionalType<Date>>(undefined);\n    private readonly dateService = inject(DateService);\n    private rangeSelectionState: 'start' | 'end' = 'start';\n\n    ngOnInit(): void {\n        this.syncInternalState();\n\n        if (this.selectedDate()) {\n            this.currentDate.set(new Date(this.selectedDate()!));\n        } else if (this.selectedStartDate()) {\n            this.currentDate.set(new Date(this.selectedStartDate()!));\n        }\n\n        this.updateYearRange();\n    }\n\n    onHeaderClick(): void {\n        switch (this.currentView()) {\n            case CalendarViewEnum.Days:\n                this.currentView.set(CalendarViewEnum.Months);\n                break;\n            case CalendarViewEnum.Months:\n                this.currentView.set(CalendarViewEnum.Years);\n                break;\n        }\n    }\n\n    onPreviousClick(): void {\n        const current = this.currentDate();\n\n        switch (this.currentView()) {\n            case CalendarViewEnum.Days:\n                this.currentDate.set(this.dateService.addMonths(current, -1));\n                this.monthChanged.emit(this.currentDate());\n                break;\n            case CalendarViewEnum.Months:\n                this.currentDate.set(this.dateService.addYears(current, -1));\n                break;\n            case CalendarViewEnum.Years:\n                this.moveYearRange(-12);\n                break;\n        }\n    }\n\n    onNextClick(): void {\n        const current = this.currentDate();\n\n        switch (this.currentView()) {\n            case CalendarViewEnum.Days:\n                this.currentDate.set(this.dateService.addMonths(current, 1));\n                this.monthChanged.emit(this.currentDate());\n                break;\n            case CalendarViewEnum.Months:\n                this.currentDate.set(this.dateService.addYears(current, 1));\n                break;\n            case CalendarViewEnum.Years:\n                this.moveYearRange(12);\n                break;\n        }\n    }\n\n    onDateClick(dateModel: CalendarDateModel): void {\n        if (!dateModel.isCurrentMonth) {\n            return;\n        }\n\n        if (dateModel.isDisabled) {\n            return;\n        }\n\n        if (this.isRange()) {\n            this.handleRangeSelection(dateModel.date);\n        } else {\n            this.internalSelectedDate.set(dateModel.date);\n            this.selected.emit({ selectedDate: dateModel.date });\n        }\n    }\n\n    onMonthClick(monthModel: CalendarMonthModel): void {\n        if (monthModel.isDisabled) {\n            return;\n        }\n\n        const newDate = new Date(this.currentYear(), monthModel.monthIndex, 1);\n\n        this.currentDate.set(newDate);\n        this.currentView.set(CalendarViewEnum.Days);\n        this.monthChanged.emit(newDate);\n    }\n\n    onYearClick(yearModel: CalendarYearModel): void {\n        if (yearModel.isDisabled) {\n            return;\n        }\n\n        const newDate = new Date(yearModel.year, this.currentMonth(), 1);\n\n        this.currentDate.set(newDate);\n        this.currentView.set(CalendarViewEnum.Months);\n    }\n\n    clearSelection(): void {\n        this.internalSelectedDate.set(undefined);\n        this.internalSelectedStartDate.set(undefined);\n        this.internalSelectedEndDate.set(undefined);\n\n        this.rangeSelectionState = 'start';\n\n        this.selected.emit(undefined);\n    }\n\n    private syncInternalState(): void {\n        this.internalSelectedDate.set(this.selectedDate());\n        this.internalSelectedStartDate.set(this.selectedStartDate());\n        this.internalSelectedEndDate.set(this.selectedEndDate());\n    }\n\n    private handleRangeSelection(selectedDate: Date): void {\n        if (this.rangeSelectionState === 'start' || !this.internalSelectedStartDate()) {\n            this.internalSelectedStartDate.set(selectedDate);\n            this.internalSelectedEndDate.set(undefined);\n            this.rangeSelectionState = 'end';\n\n            this.selected.emit({ startDate: selectedDate });\n        } else {\n            const startDate = this.internalSelectedStartDate()!;\n            const endDate = selectedDate;\n\n            if (this.dateService.compare(selectedDate, startDate) < 0) {\n                this.internalSelectedStartDate.set(selectedDate);\n                this.internalSelectedEndDate.set(startDate);\n                this.selected.emit({\n                    startDate: selectedDate,\n                    endDate: startDate\n                });\n            } else {\n                this.internalSelectedEndDate.set(endDate);\n                this.selected.emit({\n                    startDate: startDate,\n                    endDate: endDate\n                });\n            }\n\n            this.rangeSelectionState = 'start';\n        }\n    }\n\n    private generateCalendarDays(): CalendarDateModel[] {\n        const current = this.currentDate();\n        const year = current.getFullYear();\n        const month = current.getMonth();\n        const firstDayOfMonth = new Date(year, month, 1);\n        const lastDayOfMonth = new Date(year, month + 1, 0);\n        const firstDayOfWeek = firstDayOfMonth.getDay();\n        const days: CalendarDateModel[] = [];\n        const today = new Date();\n\n        if (firstDayOfWeek > 0) {\n            const prevMonthLastDay = new Date(year, month, 0);\n            const startDate = prevMonthLastDay.getDate() - firstDayOfWeek + 1;\n\n            for (let day = startDate; day <= prevMonthLastDay.getDate(); day++) {\n                const date = new Date(year, month - 1, day);\n                days.push({\n                    date,\n                    isCurrentMonth: false,\n                    isToday: this.dateService.areEqual(date, today),\n                    isSelected: false,\n                    isDisabled: true,\n                    isInRange: false,\n                    isRangeStart: false,\n                    isRangeEnd: false\n                });\n            }\n        }\n\n        for (let day = 1; day <= lastDayOfMonth.getDate(); day++) {\n            const date = new Date(year, month, day);\n            days.push({\n                date,\n                isCurrentMonth: true,\n                isToday: this.dateService.areEqual(date, today),\n                isSelected: this.isDateSelected(date),\n                isDisabled: this.isDateDisabled(date),\n                isInRange: this.isDateInRange(date),\n                isRangeStart: this.isDateRangeStart(date),\n                isRangeEnd: this.isDateRangeEnd(date)\n            });\n        }\n\n        const totalCells = Math.ceil(days.length / 7) * 7;\n        const nextMonthDays = totalCells - days.length;\n\n        for (let day = 1; day <= nextMonthDays; day++) {\n            const date = new Date(year, month + 1, day);\n            days.push({\n                date,\n                isCurrentMonth: false,\n                isToday: this.dateService.areEqual(date, today),\n                isSelected: false,\n                isDisabled: true,\n                isInRange: false,\n                isRangeStart: false,\n                isRangeEnd: false\n            });\n        }\n\n        return days;\n    }\n\n    private generateCalendarMonths(): CalendarMonthModel[] {\n        const currentYear = this.currentYear();\n        const currentMonth = this.currentMonth();\n        const today = new Date();\n        const monthNames = this.dateService.getMonthNames('short');\n\n        return monthNames.map((name, index) => ({\n            monthIndex: index,\n            name: name.toUpperCase(),\n            isCurrentMonth: today.getFullYear() === currentYear && today.getMonth() === index,\n            isSelected: currentMonth === index,\n            isDisabled: this.isMonthDisabled(currentYear, index)\n        }));\n    }\n\n    private generateCalendarYears(): CalendarYearModel[] {\n        const range = this.yearRange();\n        const currentYear = this.currentYear();\n        const today = new Date();\n        const years: CalendarYearModel[] = [];\n\n        for (let year = range.start; year <= range.end; year++) {\n            years.push({\n                year,\n                isCurrentYear: today.getFullYear() === year,\n                isSelected: currentYear === year,\n                isDisabled: this.isYearDisabled(year)\n            });\n        }\n\n        return years;\n    }\n\n    private isDateSelected(date: Date): boolean {\n        if (this.isRange()) {\n            if (this.internalSelectedStartDate() && this.dateService.areEqual(date, this.internalSelectedStartDate()!)) {\n                return true;\n            }\n\n            return !!(this.internalSelectedEndDate() && this.dateService.areEqual(date, this.internalSelectedEndDate()!));\n        } else {\n            return this.internalSelectedDate() ? this.dateService.areEqual(date, this.internalSelectedDate()!) : false;\n        }\n    }\n\n    private isDateInRange(date: Date): boolean {\n        if (!this.isRange() || !this.internalSelectedStartDate() || !this.internalSelectedEndDate()) {\n            return false;\n        }\n\n        return this.dateService.compare(date, this.internalSelectedStartDate()!) > 0\n            && this.dateService.compare(date, this.internalSelectedEndDate()!) < 0;\n    }\n\n    private isDateRangeStart(date: Date): boolean {\n        return this.isRange() && this.internalSelectedStartDate()\n            ? this.dateService.areEqual(date, this.internalSelectedStartDate()!)\n            : false;\n    }\n\n    private isDateRangeEnd(date: Date): boolean {\n        return this.isRange() && this.internalSelectedEndDate()\n            ? this.dateService.areEqual(date, this.internalSelectedEndDate()!)\n            : false;\n    }\n\n    private isDateDisabled(date: Date): boolean {\n        if (this.minDate() && this.dateService.compare(date, this.minDate()!) < 0) {\n            return true;\n        }\n\n        if (this.maxDate() && this.dateService.compare(date, this.maxDate()!) > 0) {\n            return true;\n        }\n\n        const filterFn = this.disabledDatesFilter();\n\n        if (!!filterFn) {\n            return !filterFn(date);\n        }\n\n        return false;\n    }\n\n    private isMonthDisabled(year: number, month: number): boolean {\n        const firstDayOfMonth = new Date(year, month, 1);\n        const lastDayOfMonth = new Date(year, month + 1, 0);\n\n        if (this.minDate() && this.dateService.compare(lastDayOfMonth, this.minDate()!) < 0) {\n            return true;\n        }\n\n        return !!(this.maxDate() && this.dateService.compare(firstDayOfMonth, this.maxDate()!) > 0);\n    }\n\n    private isYearDisabled(year: number): boolean {\n        const firstDayOfYear = new Date(year, 0, 1);\n        const lastDayOfYear = new Date(year, 11, 31);\n\n        if (this.minDate() && this.dateService.compare(lastDayOfYear, this.minDate()!) < 0) {\n            return true;\n        }\n\n        return !!(this.maxDate() && this.dateService.compare(firstDayOfYear, this.maxDate()!) > 0);\n    }\n\n    private updateYearRange(): void {\n        const currentYear = this.currentYear();\n        this.yearRange.set({\n            start: currentYear - 6,\n            end: currentYear + 5\n        });\n    }\n\n    private moveYearRange(offset: number): void {\n        const current = this.yearRange();\n        this.yearRange.set({\n            start: current.start + offset,\n            end: current.end + offset\n        });\n    }\n}\n","<div class=\"calendar\" [class.calendar-border]=\"!shouldRemoveExternalBorder()\">\n    <div class=\"calendar-header\">\n        <button type=\"button\" class=\"btn btn-sm btn-ghost-secondary\" (click)=\"onPreviousClick()\">\n            <svg xmlns=\"http://www.w3.org/2000/svg\" class=\"icon\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" stroke-width=\"2\"\n                stroke=\"currentColor\" fill=\"none\">\n                <path stroke=\"none\" d=\"m0 0h24v24H0z\" fill=\"none\" />\n                <path d=\"m15 6-6 6 6 6\" />\n            </svg>\n        </button>\n\n        <button type=\"button\" class=\"btn btn-sm btn-ghost-secondary calendar-title\" (click)=\"onHeaderClick()\">\n            @switch (currentView()) {\n                @case (CalendarViewEnum.Days) {\n                    {{ currentMonthName() }}\n                }\n                @case (CalendarViewEnum.Months) {\n                    {{ currentYear() }}\n                }\n                @case (CalendarViewEnum.Years) {\n                    {{ yearRange().start }} - {{ yearRange().end }}\n                }\n            }\n            <svg xmlns=\"http://www.w3.org/2000/svg\" class=\"icon ms-1\" width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" stroke-width=\"2\"\n                stroke=\"currentColor\" fill=\"none\">\n                <path stroke=\"none\" d=\"m0 0h24v24H0z\" fill=\"none\" />\n                <path d=\"m6 9 6 6 6-6\" />\n            </svg>\n        </button>\n\n        <button type=\"button\" class=\"btn btn-sm btn-ghost-secondary\" (click)=\"onNextClick()\">\n            <svg xmlns=\"http://www.w3.org/2000/svg\" class=\"icon\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" stroke-width=\"2\"\n                stroke=\"currentColor\" fill=\"none\">\n                <path stroke=\"none\" d=\"m0 0h24v24H0z\" fill=\"none\" />\n                <path d=\"m9 6 6 6-6 6\" />\n            </svg>\n        </button>\n    </div>\n\n    @if (currentView() === CalendarViewEnum.Days) {\n        <div class=\"calendar-weekdays\">\n            @for (dayName of weekDayNames(); track dayName) {\n                <div class=\"calendar-weekday\">{{ dayName }}</div>\n            }\n        </div>\n\n        <div class=\"calendar-days\">\n            @for (day of calendarDays(); track day.date.getTime()) {\n                <button type=\"button\" class=\"calendar-day\" [class.calendar-day-other-month]=\"!day.isCurrentMonth\"\n                    [class.calendar-day-today]=\"day.isToday\" [class.calendar-day-selected]=\"day.isSelected\"\n                    [class.calendar-day-in-range]=\"day.isInRange\" [class.calendar-day-range-start]=\"day.isRangeStart\"\n                    [class.calendar-day-range-end]=\"day.isRangeEnd\" [disabled]=\"day.isDisabled\" (click)=\"onDateClick(day)\">\n                    {{ day.date.getDate() }}\n                </button>\n            }\n        </div>\n    }\n\n    @if (currentView() === CalendarViewEnum.Months) {\n        <div class=\"calendar-months\">\n            @for (month of calendarMonths(); track month.monthIndex) {\n                <button type=\"button\" class=\"calendar-month\" [class.calendar-month-current]=\"month.isCurrentMonth\"\n                    [class.calendar-month-selected]=\"month.isSelected\" [disabled]=\"month.isDisabled\" (click)=\"onMonthClick(month)\">\n                    {{ month.name }}\n                </button>\n            }\n        </div>\n    }\n\n    @if (currentView() === CalendarViewEnum.Years) {\n        <div class=\"calendar-years\">\n            @for (year of calendarYears(); track year.year) {\n                <button type=\"button\" class=\"calendar-year\" [class.calendar-year-current]=\"year.isCurrentYear\"\n                    [class.calendar-year-selected]=\"year.isSelected\" [disabled]=\"year.isDisabled\" (click)=\"onYearClick(year)\">\n                    {{ year.year }}\n                </button>\n            }\n        </div>\n    }\n</div>\n"]}