@energycap/components 0.42.0 → 0.42.1

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 (40) hide show
  1. package/esm2022/lib/components.module.mjs +10 -5
  2. package/esm2022/lib/controls/calendar/calendar-item.component.mjs +46 -14
  3. package/esm2022/lib/controls/calendar/calendar.component.mjs +169 -121
  4. package/esm2022/lib/controls/calendar/calendar.types.mjs +2 -4
  5. package/esm2022/lib/controls/date-input/date-input-selection-strategies/date-input-selection-strategy-base.mjs +57 -0
  6. package/esm2022/lib/controls/date-input/date-input-selection-strategies/day-selection-strategy.mjs +62 -0
  7. package/esm2022/lib/controls/date-input/date-input-selection-strategies/last-28-days-selection-strategy.mjs +100 -0
  8. package/esm2022/lib/controls/date-input/date-input-selection-strategies/last-7-days-selection-strategy.mjs +101 -0
  9. package/esm2022/lib/controls/date-input/date-input-selection-strategies/month-selection-strategy.mjs +76 -0
  10. package/esm2022/lib/controls/date-input/date-input-selection-strategies/quarter-selection-strategy.mjs +79 -0
  11. package/esm2022/lib/controls/date-input/date-input-selection-strategies/range-selection-strategy.mjs +210 -0
  12. package/esm2022/lib/controls/date-input/date-input-selection-strategies/year-selection-strategy.mjs +81 -0
  13. package/esm2022/lib/controls/date-input/date-input.component.mjs +322 -113
  14. package/esm2022/lib/controls/date-input/date-input.types.mjs +44 -0
  15. package/esm2022/lib/controls/file-upload/file-upload.component.mjs +1 -1
  16. package/esm2022/lib/controls/form-control/form-control.component.mjs +6 -12
  17. package/esm2022/lib/core/date-time-helper.mjs +10 -2
  18. package/esm2022/lib/shared/directives/keyboard-nav-container/keyboard-nav-container.directive.mjs +100 -0
  19. package/esm2022/public-api.mjs +63 -61
  20. package/fesm2022/energycap-components.mjs +1666 -507
  21. package/fesm2022/energycap-components.mjs.map +1 -1
  22. package/lib/components.module.d.ts +9 -8
  23. package/lib/controls/calendar/calendar-item.component.d.ts +11 -6
  24. package/lib/controls/calendar/calendar.component.d.ts +21 -23
  25. package/lib/controls/calendar/calendar.types.d.ts +11 -7
  26. package/lib/controls/date-input/date-input-selection-strategies/date-input-selection-strategy-base.d.ts +42 -0
  27. package/lib/controls/date-input/date-input-selection-strategies/day-selection-strategy.d.ts +21 -0
  28. package/lib/controls/date-input/date-input-selection-strategies/last-28-days-selection-strategy.d.ts +21 -0
  29. package/lib/controls/date-input/date-input-selection-strategies/last-7-days-selection-strategy.d.ts +21 -0
  30. package/lib/controls/date-input/date-input-selection-strategies/month-selection-strategy.d.ts +18 -0
  31. package/lib/controls/date-input/date-input-selection-strategies/quarter-selection-strategy.d.ts +18 -0
  32. package/lib/controls/date-input/date-input-selection-strategies/range-selection-strategy.d.ts +21 -0
  33. package/lib/controls/date-input/date-input-selection-strategies/year-selection-strategy.d.ts +20 -0
  34. package/lib/controls/date-input/date-input.component.d.ts +63 -28
  35. package/lib/controls/date-input/date-input.types.d.ts +62 -0
  36. package/lib/controls/form-control/form-control.component.d.ts +4 -6
  37. package/lib/shared/directives/keyboard-nav-container/keyboard-nav-container.directive.d.ts +23 -0
  38. package/package.json +1 -1
  39. package/public-api.d.ts +62 -60
  40. package/src/assets/locales/en_US.json +9 -1
@@ -2,7 +2,6 @@ import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, Input, O
2
2
  import { range } from 'lodash';
3
3
  import moment from 'moment';
4
4
  import { DateTimeHelper } from '../../core/date-time-helper';
5
- import { isCalendarSelectionSingleDate } from './calendar.types';
6
5
  import * as i0 from "@angular/core";
7
6
  import * as i1 from "@angular/common";
8
7
  import * as i2 from "./calendar-item.component";
@@ -10,191 +9,240 @@ export class CalendarComponent {
10
9
  constructor() {
11
10
  this.id = 'calendar';
12
11
  this.selection = null;
13
- this.selectionChange = new EventEmitter();
12
+ this.selectionMode = 'day';
14
13
  this.minDate = DateTimeHelper.minDatePickerDate;
15
14
  this.maxDate = DateTimeHelper.maxDatePickerDate;
16
- this.focusOutStart = new EventEmitter();
17
- this.focusOutEnd = new EventEmitter();
18
- /** Array of calendar items to display in the calendar grid. */
15
+ this.dateSelected = new EventEmitter();
16
+ /** Determines the current view of the calendar. */
17
+ this.view = { mode: 'day', date: new Date() };
18
+ this.viewChange = new EventEmitter();
19
+ /** Array of calendar item rows to display in the calendar. */
19
20
  this.calendarItems = [];
20
- /** Determines the current view mode of the calendar. */
21
- this.view = 'day';
22
- /** TrackBy function for the calendar items. This avoids re-drawing calendar items if a date is shared between view updates */
23
- this.calendarItemTrackByDate = (index, item) => item.date.toISOString();
21
+ /** Weekday labels to show at the top of the calendar in day view. */
22
+ this.weekDays = moment.weekdaysShort().map(d => d.slice(0, 1));
24
23
  this.disablePreviousButton = false;
25
24
  this.disableNextButton = false;
26
- // By default, the active date is today.
27
- // This is replaced with the date of the selection if one is provided on init.
28
- this.activeDate = new Date();
29
- // Using moment here to get the weekday labels so they are localized.
30
- this.weekDays = moment.weekdaysShort().map(d => d.slice(0, 1));
25
+ // Trackby functions for ngFor loops. These reduce the number of DOM elements that need to be updated when the calendar is re-drawn.
26
+ // Without these there are some change-detection issues on the first-time rendering of the calendar.
27
+ this.trackByDateRow = (index, row) => `${row[0].date.toISOString()}-${row[row.length - 1].date.toISOString()}`;
28
+ this.trackByDate = (index, item) => item.date.toISOString();
31
29
  }
32
30
  ngOnChanges(changes) {
33
- if (changes.selection && this.selection) {
34
- // Only need to update the view if the selection is out of view.
35
- if (isCalendarSelectionSingleDate(this.selection) &&
36
- !this.isDateInView(this.selection) &&
37
- moment(this.selection).isBetween(this.minDate, this.maxDate, 'day', '[]')) {
38
- this.drawCalendar('day', this.selection);
39
- }
40
- else {
41
- // TODO ECAP-26841: determine the active date for a range selection and draw the calendar
42
- }
31
+ if (changes.view) {
32
+ this.drawCalendar(this.view);
33
+ }
34
+ if (changes.minDate || changes.maxDate) {
35
+ this.updateNextPreviousStates();
43
36
  }
44
37
  }
45
38
  ngOnInit() {
46
39
  if (!this.calendarItems.length) {
47
- this.drawCalendar('day');
48
- }
49
- }
50
- onItemSelected(item) {
51
- if (this.view === 'day') {
52
- this.selection = item.date;
53
- this.selectionChange.emit(this.selection);
54
- }
55
- if (this.view === 'month') {
56
- this.drawCalendar('day', item.date);
57
- }
58
- if (this.view === 'year') {
59
- this.drawCalendar('month', item.date);
40
+ this.drawCalendar(this.view);
60
41
  }
61
42
  }
62
43
  onNextClick() {
63
44
  // If we're in day view, we're incrementing by month
64
- // In both month and year views we're incrementing by year
65
- // In year view, we're moving by 16 years at a time
66
- const unit = this.view === 'day' ? 'month' : 'year';
45
+ // In month, quarter, and year views we're incrementing by year\
46
+ const unit = this.view.mode === 'day' ? 'month' : 'year';
67
47
  // In year view, we're moving by 16 years at a time. Otherwise we're just moving by 1 month/year.
68
- const count = this.view === 'year' ? 16 : 1;
69
- const goToDate = moment(this.activeDate).add(count, unit).toDate();
70
- this.drawCalendar(this.view, goToDate);
48
+ const count = this.view.mode === 'year' ? 16 : 1;
49
+ const goToDate = moment(this.view.date).add(count, unit).toDate();
50
+ this.drawCalendar({ mode: this.view.mode, date: goToDate });
71
51
  }
72
52
  onPreviousClick() {
73
53
  // If we're in day view, we're incrementing by month
74
- // In both month and year views we're incrementing years
75
- const unit = this.view === 'day' ? 'month' : 'year';
54
+ // In month, quarter, and year views we're incrementing by year
55
+ const unit = this.view.mode === 'day' ? 'month' : 'year';
76
56
  // In year view, we're moving by 16 years at a time. Otherwise we're just moving by 1 month/year.
77
- const count = this.view === 'year' ? 16 : 1;
78
- const goToDate = moment(this.activeDate).subtract(count, unit).toDate();
79
- this.drawCalendar(this.view, goToDate);
57
+ const count = this.view.mode === 'year' ? 16 : 1;
58
+ const goToDate = moment(this.view.date).subtract(count, unit).toDate();
59
+ this.drawCalendar({ mode: this.view.mode, date: goToDate });
80
60
  }
81
61
  /** Switches the calendar to month view. */
82
62
  onMonthClick() {
83
- this.drawCalendar('month');
63
+ this.drawCalendar({ mode: 'month', date: this.view.date });
84
64
  }
85
65
  /** Switches the calendar to year view. */
86
66
  onYearClick() {
87
- this.drawCalendar('year');
88
- }
89
- onFirstItemKeydown(event) {
90
- // If the user is tabbing backwards from the first item, emit the focusOutStart event.
91
- if (event.key === 'Tab' && event.shiftKey) {
92
- this.focusOutStart.emit(event);
93
- }
67
+ this.drawCalendar({ mode: 'year', date: this.view.date });
94
68
  }
95
- onLastItemKeydown(event) {
96
- // If the user is tabbing forwards from the last item, emit the focusOutEnd event.
97
- if (event.key === 'Tab' && !event.shiftKey) {
98
- this.focusOutEnd.emit(event);
69
+ onItemSelected(item) {
70
+ switch (this.view.mode) {
71
+ case 'day':
72
+ this.onDaySelected(item);
73
+ break;
74
+ case 'month':
75
+ this.onMonthSelected(item);
76
+ break;
77
+ case 'quarter':
78
+ this.onQuarterSelected(item);
79
+ break;
80
+ case 'year':
81
+ this.onYearSelected(item);
82
+ break;
99
83
  }
100
84
  }
101
- drawCalendar(view, goTo) {
102
- // If a goTo date is provided, update the active date so we know what to increment on next/previous clicks.
103
- if (goTo) {
104
- this.activeDate = goTo;
85
+ onYearSelected(item) {
86
+ // If in year selection mode, emit the calendar item so the date picker can update the selection.
87
+ if (this.selectionMode === 'year') {
88
+ this.dateSelected.emit(item.date);
89
+ // If we're in quarter selection mode, draw the quarter view.
105
90
  }
106
- this.month = moment(this.activeDate).format('MMM');
107
- this.year = moment(this.activeDate).format('YYYY');
108
- this.view = view;
109
- this.calendarItems = this.getCalendarItems(view, this.activeDate);
110
- const startYear = this.calendarItems[0].date < this.minDate ? this.minDate : this.calendarItems[0].date;
111
- const endYear = this.calendarItems[this.calendarItems.length - 1].date > this.maxDate ? this.maxDate : this.calendarItems[this.calendarItems.length - 1].date;
112
- this.yearRange = `${moment(startYear).format('YYYY')}&ndash;${moment(endYear).format('YYYY')}`;
113
- // If the active date is less than the min date or the min date is in view, disable the previous button.
114
- if (this.activeDate < this.minDate || this.isDateInView(this.minDate)) {
115
- this.disablePreviousButton = true;
91
+ else if (this.selectionMode === 'quarter') {
92
+ this.drawCalendar({ mode: 'quarter', date: item.date });
93
+ // All other selection modes should zoom in to the month view when a year is selected.
116
94
  }
117
95
  else {
118
- this.disablePreviousButton = false;
96
+ this.drawCalendar({ mode: 'month', date: item.date });
119
97
  }
120
- // If the active date is greater than the max date or the max date is in view, disable the next button.
121
- if (this.activeDate > this.maxDate || this.isDateInView(this.maxDate)) {
122
- this.disableNextButton = true;
98
+ }
99
+ onQuarterSelected(item) {
100
+ // Quarter view is only accessible when in quarter selection mode,
101
+ // so there's no other views to draw from here. Just emit the selected item.
102
+ this.dateSelected.emit(item.date);
103
+ }
104
+ onMonthSelected(item) {
105
+ // When in month selection mode, emit the selected month item and don't change the view.
106
+ if (this.selectionMode === 'month') {
107
+ this.dateSelected.emit(item.date);
108
+ // Month view is only accessible from day, last7days, last28days, month, and range selection modes.
109
+ // If we're not in month selection mode, we need to zoom in to the day view.
123
110
  }
124
111
  else {
125
- this.disableNextButton = false;
112
+ this.drawCalendar({ mode: 'day', date: item.date });
126
113
  }
127
114
  }
128
- getCalendarItems(view, activeDate) {
129
- switch (view) {
130
- case 'day': return this.getDayViewItems(activeDate);
131
- case 'month': return this.getMonthViewItems(activeDate);
132
- case 'year': return this.getYearViewItems(activeDate);
115
+ onDaySelected(item) {
116
+ // No further views other than day mode. Just emit the selected item.
117
+ this.dateSelected.emit(item.date);
118
+ }
119
+ drawCalendar(view) {
120
+ // If the view has changed, emit the new view.
121
+ if (view.date !== this.view.date || view.mode !== this.view.mode) {
122
+ this.view = view;
123
+ this.viewChange.emit(view);
124
+ }
125
+ this.month = moment(this.view.date).format('MMM');
126
+ this.year = moment(this.view.date).format('YYYY');
127
+ this.calendarItems = this.getCalendarItems(this.view);
128
+ const startYear = this.calendarItems[0][0].date < this.minDate ? this.minDate : this.calendarItems[0][0].date;
129
+ const lastItem = this.getLastItem();
130
+ const endYear = lastItem.date > this.maxDate ? this.maxDate : lastItem.date;
131
+ this.yearRange = `${moment(startYear).format('YYYY')}&ndash;${moment(endYear).format('YYYY')}`;
132
+ this.updateNextPreviousStates();
133
+ }
134
+ getCalendarItems(view) {
135
+ switch (view.mode) {
136
+ case 'day': return this.getDayViewItems(view.date);
137
+ case 'month': return this.getMonthViewItems(view.date);
138
+ case 'quarter': return this.getQuarterViewItems(view.date);
139
+ case 'year': return this.getYearViewItems(view.date);
133
140
  }
134
141
  }
135
142
  getDayViewItems(activeDate) {
136
143
  // 6 rows of 7 days = 42 days
137
- return range(0, 42).map(i => {
138
- // Use the start of the week of the first day of the month.
139
- // This pads out the first week with any days from the previous month.
140
- const date = moment(activeDate).startOf('month').startOf('week').add(i, 'day');
141
- return {
142
- date: date.toDate(),
143
- label: date.format('D')
144
- };
144
+ return range(0, 6).map(r => {
145
+ return range(0, 7).map(d => {
146
+ // Use the start of the week of the first day of the month.
147
+ // This pads out the first week with any days from the previous month.
148
+ const date = moment(activeDate).startOf('month').startOf('week').add(r * 7 + d, 'day');
149
+ return {
150
+ date: date.toDate(),
151
+ label: date.format('D')
152
+ };
153
+ });
145
154
  });
146
155
  }
147
156
  getMonthViewItems(activeDate) {
148
- return range(0, 12).map(i => {
149
- const date = moment(activeDate).startOf('year').add(i, 'month');
150
- return {
151
- date: date.toDate(),
152
- label: date.format('MMM')
153
- };
157
+ // 4 rows of 3 months = 12 months
158
+ return range(0, 4).map(r => {
159
+ return range(0, 3).map(m => {
160
+ const date = moment(activeDate).startOf('year').add(r * 3 + m, 'month');
161
+ return {
162
+ date: date.toDate(),
163
+ label: date.format('MMM')
164
+ };
165
+ });
166
+ });
167
+ }
168
+ getQuarterViewItems(activeDate) {
169
+ // 2 rows of 2 quarters = 4 quarters
170
+ return range(0, 2).map(r => {
171
+ return range(0, 2).map(q => {
172
+ const date = moment(activeDate).startOf('year').add(r * 2 + q, 'quarter');
173
+ const endDate = moment(date).endOf('quarter');
174
+ return {
175
+ date: date.toDate(),
176
+ label: `${date.format('MMM')}&ndash;${endDate.format('MMM')}`
177
+ };
178
+ });
154
179
  });
155
180
  }
156
181
  getYearViewItems(activeDate) {
157
- // 4x4 grid of years = 16 years
158
- return range(0, 16).map(i => {
159
- // Put the current active year at the beginning of the 3rd row.
160
- const date = moment(activeDate).startOf('year').subtract(8, 'year').add(i, 'year');
161
- return {
162
- date: date.toDate(),
163
- label: date.format('YYYY')
164
- };
182
+ // 4 rows of 4 years = 16 years
183
+ return range(0, 4).map(r => {
184
+ return range(0, 4).map(y => {
185
+ // Subtracting 8 years to put the current active year at the beginning of the 3rd row.
186
+ const date = moment(activeDate).startOf('year').subtract(8, 'years').add(r * 4 + y, 'year');
187
+ return {
188
+ date: date.toDate(),
189
+ label: date.format('YYYY')
190
+ };
191
+ });
165
192
  });
166
193
  }
167
- /** Returns true if the date is within the current calendar view */
168
- isDateInView(date) {
169
- if (!this.calendarItems.length) {
170
- return false;
194
+ updateNextPreviousStates() {
195
+ // We change the granularity of the min and max date checks based on the current view mode.
196
+ // We don't want to disable the next/previous buttons if the min or max date is within the
197
+ // next or previous view
198
+ const unit = this.view.mode === 'day' ? 'month' : 'year';
199
+ // When in year view, we need to subtract 9 years to determine if we're past the min date.
200
+ // The year view is 16 years long, and the current date is the 9th year in the view.
201
+ const subCount = this.view.mode === 'year' ? 9 : 1;
202
+ if (moment(this.view.date).subtract(subCount, unit).isBefore(this.minDate, unit)) {
203
+ this.disablePreviousButton = true;
204
+ }
205
+ else {
206
+ this.disablePreviousButton = false;
207
+ }
208
+ // When in year view, we need to add 8 years to determine if we're past the max date.
209
+ // The year view is 16 years long, and the current date is the 9th year in the view.
210
+ const addCount = this.view.mode === 'year' ? 8 : 1;
211
+ if (moment(this.view.date).add(addCount, unit).isAfter(this.maxDate, unit)) {
212
+ this.disableNextButton = true;
213
+ }
214
+ else {
215
+ this.disableNextButton = false;
171
216
  }
172
- const viewStart = this.calendarItems[0].date;
173
- const viewEnd = this.calendarItems[this.calendarItems.length - 1].date;
174
- return date >= viewStart && date <= viewEnd;
217
+ }
218
+ getLastItem() {
219
+ const lastRow = this.calendarItems[this.calendarItems.length - 1];
220
+ return lastRow[lastRow.length - 1];
175
221
  }
176
222
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: CalendarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
177
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: CalendarComponent, selector: "ec-calendar", inputs: { id: "id", selection: "selection", minDate: "minDate", maxDate: "maxDate" }, outputs: { selectionChange: "selectionChange", focusOutStart: "focusOutStart", focusOutEnd: "focusOutEnd" }, host: { properties: { "attr.id": "this.id" } }, usesOnChanges: true, ngImport: i0, template: "<header class=\"d-flex align-items-center mb-2\">\r\n <button id=\"{{id}}_prev_button\"\r\n class=\"mr-auto\"\r\n (click)=\"onPreviousClick()\"\r\n (keydown)=\"!disablePreviousButton ? onFirstItemKeydown($event) : undefined\"\r\n [disabled]=\"disablePreviousButton\">\r\n <i class=\"ec-icon icon-angle-down rotate-90\"></i>\r\n </button>\r\n\r\n <button *ngIf=\"view === 'day'\"\r\n id=\"{{id}}_month_button\"\r\n class=\"text-body-1 font-weight-bold\"\r\n (click)=\"onMonthClick()\"\r\n (keydown)=\"disablePreviousButton ? onFirstItemKeydown($event) : undefined\">\r\n {{month}}\r\n </button>\r\n\r\n <button *ngIf=\"view !== 'year'\"\r\n id=\"{{id}}_year_button\"\r\n class=\"text-body-1 font-weight-bold\"\r\n (click)=\"onYearClick()\"\r\n (keydown)=\"disablePreviousButton && view !== 'day' ? onFirstItemKeydown($event) : undefined\">\r\n {{year}}\r\n </button>\r\n\r\n <div id=\"{{id}}_year_range\"\r\n *ngIf=\"view === 'year'\"\r\n class=\"text-body-1 font-weight-bold\"\r\n [innerHTML]=\"yearRange\">\r\n </div>\r\n\r\n <button id=\"{{id}}_next_button\"\r\n class=\"ml-auto\"\r\n (click)=\"onNextClick()\"\r\n [disabled]=\"disableNextButton\"\r\n (keydown)=\"disablePreviousButton && view === 'year' ? onFirstItemKeydown($event) : undefined\">\r\n <i class=\"ec-icon icon-angle-down rotate-270\"></i>\r\n </button>\r\n</header>\r\n\r\n<ul class=\"px-1 {{view}}-view\">\r\n <ng-container *ngIf=\"view === 'day'\">\r\n <li *ngFor=\"let day of weekDays\"\r\n class=\"d-flex align-items-center justify-content-center text-heading-3\">\r\n {{day}}\r\n </li>\r\n </ng-container>\r\n\r\n <li *ngFor=\"let item of calendarItems; last as isLast; trackBy: calendarItemTrackByDate;\">\r\n <button id=\"{{id}}_item_{{item.date | date:'MM_dd_yyyy'}}\"\r\n ec-calendar-item\r\n [item]=\"item\"\r\n [activeDate]=\"activeDate\"\r\n [selection]=\"selection\"\r\n [view]=\"view\"\r\n (click)=\"onItemSelected(item)\"\r\n (keydown)=\"isLast ? onLastItemKeydown($event) : undefined\"\r\n [hidden]=\"item.date < minDate || item.date > maxDate\">\r\n </button>\r\n </li>\r\n</ul>", styles: [":host{display:inline-block}button{font-size:var(--ec-font-size-action);height:2rem;line-height:1.25rem;padding:.3125rem .5rem;border:0;border-radius:var(--ec-border-radius);display:flex;align-items:center;justify-content:center;cursor:pointer;background-color:transparent}button .label{display:flex;align-items:center;justify-content:center;white-space:nowrap;flex:auto}button .ec-icon{flex:none}button .ec-icon+.label{flex:none;margin-left:.25rem}button.has-badge{padding-right:.0625rem}button:focus{outline:none;position:relative;z-index:1}button:disabled{background-color:var(--ec-background-color-disabled);border:1px solid var(--ec-form-control-border-color-disabled);color:var(--ec-color-disabled-dark);opacity:var(--ec-form-control-opacity-disabled);cursor:default}button:disabled{background-color:transparent;border-color:transparent;color:var(--ec-color-hint-dark);opacity:1;--ec-color-icon: var(--ec-color-hint-dark)}button:hover:not(:disabled){background-color:var(--ec-background-color-hover)}button:active:not(:disabled){background-color:var(--ec-background-color-selected);font-weight:700}button:focus:not(:disabled){box-shadow:var(--ec-button-box-shadow-active, 0 0 0 2px var(--ec-border-color-focus))}ul{list-style:none;padding:0;margin:0;display:grid;align-items:stretch;justify-items:stretch}button[ec-calendar-item]{height:100%;width:100%}ul.day-view{grid-template-columns:repeat(7,2rem);grid-auto-rows:1.75rem;row-gap:.25rem}ul.month-view{grid-template-columns:repeat(3,4.67rem);grid-auto-rows:3.25rem;row-gap:.25rem}ul.year-view{grid-template-columns:repeat(4,3.5rem);grid-auto-rows:3.25rem;row-gap:.25rem}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.CalendarItemComponent, selector: "button[ec-calendar-item]", inputs: ["item", "activeDate", "selection", "view"] }, { kind: "pipe", type: i1.DatePipe, name: "date" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
223
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: CalendarComponent, selector: "ec-calendar", inputs: { id: "id", selection: "selection", selectionMode: "selectionMode", minDate: "minDate", maxDate: "maxDate", view: "view" }, outputs: { dateSelected: "dateSelected", viewChange: "viewChange" }, host: { properties: { "attr.id": "this.id" } }, usesOnChanges: true, ngImport: i0, template: "<header class=\"d-flex align-items-center mt-1 mb-2 px-1\">\r\n <button id=\"{{id}}_prev_button\"\r\n class=\"mr-auto\"\r\n (click)=\"onPreviousClick()\"\r\n [disabled]=\"disablePreviousButton\">\r\n <i class=\"ec-icon icon-angle-down rotate-90\"></i>\r\n </button>\r\n\r\n <button *ngIf=\"view.mode === 'day'\"\r\n id=\"{{id}}_month_button\"\r\n class=\"text-body-1 font-weight-bold\"\r\n (click)=\"onMonthClick()\">\r\n {{month}}\r\n </button>\r\n\r\n <button *ngIf=\"view.mode !== 'year'\"\r\n id=\"{{id}}_year_button\"\r\n class=\"text-body-1 font-weight-bold\"\r\n (click)=\"onYearClick()\">\r\n {{year}}\r\n </button>\r\n\r\n <div id=\"{{id}}_year_range\"\r\n *ngIf=\"view.mode === 'year'\"\r\n class=\"text-body-1 font-weight-bold\"\r\n [innerHTML]=\"yearRange\">\r\n </div>\r\n\r\n <button id=\"{{id}}_next_button\"\r\n class=\"ml-auto\"\r\n (click)=\"onNextClick()\"\r\n [disabled]=\"disableNextButton\">\r\n <i class=\"ec-icon icon-angle-down rotate-270\"></i>\r\n </button>\r\n</header>\r\n\r\n<div *ngIf=\"view.mode === 'day'\"\r\n class=\"mb-1 px-2 d-flex\">\r\n <h3 *ngFor=\"let day of weekDays\"\r\n class=\"text-heading-3 d-flex justify-content-center align-items-center\"\r\n style=\"width: 2rem; height: 1.75rem;\">{{day}}</h3>\r\n</div>\r\n\r\n<div *ngFor=\"let row of calendarItems; last as isLast; trackBy: trackByDateRow\"\r\n class=\"px-2 d-flex {{view.mode}}-view\"\r\n [class.mb-1]=\"!isLast\"\r\n [class.mb-2]=\"isLast\">\r\n <button *ngFor=\"let item of row; trackBy: trackByDate\"\r\n id=\"{{id}}_item_{{item.date | date:'MM_dd_yyyy'}}\"\r\n ec-calendar-item\r\n [item]=\"item\"\r\n [selection]=\"selection\"\r\n [view]=\"view\"\r\n (click)=\"onItemSelected(item)\"\r\n [innerHTML]=\"item.label\"\r\n [minDate]=\"minDate\"\r\n [maxDate]=\"maxDate\">\r\n </button>\r\n</div>\r\n", styles: [":host{display:inline-block}button{font-size:var(--ec-font-size-action);height:2rem;line-height:1.25rem;padding:.3125rem .5rem;border:0;border-radius:var(--ec-border-radius);display:flex;align-items:center;justify-content:center;cursor:pointer;background-color:transparent}button .label{display:flex;align-items:center;justify-content:center;white-space:nowrap;flex:auto}button .ec-icon{flex:none}button .ec-icon+.label{flex:none;margin-left:.25rem}button.has-badge{padding-right:.0625rem}button:focus{outline:none;position:relative;z-index:1}button:disabled{background-color:var(--ec-background-color-disabled);border:1px solid var(--ec-form-control-border-color-disabled);color:var(--ec-color-disabled-dark);opacity:var(--ec-form-control-opacity-disabled);cursor:default}button:disabled{background-color:transparent;border-color:transparent;color:var(--ec-color-hint-dark);--ec-color-icon: var(--ec-color-hint-dark)}button:hover:not(:disabled){background-color:var(--ec-background-color-hover)}button:active:not(:disabled){background-color:var(--ec-background-color-selected);font-weight:700}button:focus:not(:disabled){box-shadow:var(--ec-button-box-shadow-active, 0 0 0 2px var(--ec-border-color-focus))}button.is-selected{background-color:var(--ec-background-color-selected);font-weight:700}.day-view button{width:2rem;height:1.75rem}.month-view button{width:4.67rem;height:3.25rem}.quarter-view button{width:7rem;height:6.75rem}.year-view button{width:3.5rem;height:3.25rem}.is-selected:has(+.is-selected:not(:disabled)){border-top-right-radius:0;border-bottom-right-radius:0}.is-selected:not(:disabled)+.is-selected{border-top-left-radius:0;border-bottom-left-radius:0}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.CalendarItemComponent, selector: "button[ec-calendar-item]", inputs: ["item", "selection", "view", "minDate", "maxDate"] }, { kind: "pipe", type: i1.DatePipe, name: "date" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
178
224
  }
179
225
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: CalendarComponent, decorators: [{
180
226
  type: Component,
181
- args: [{ selector: 'ec-calendar', changeDetection: ChangeDetectionStrategy.OnPush, template: "<header class=\"d-flex align-items-center mb-2\">\r\n <button id=\"{{id}}_prev_button\"\r\n class=\"mr-auto\"\r\n (click)=\"onPreviousClick()\"\r\n (keydown)=\"!disablePreviousButton ? onFirstItemKeydown($event) : undefined\"\r\n [disabled]=\"disablePreviousButton\">\r\n <i class=\"ec-icon icon-angle-down rotate-90\"></i>\r\n </button>\r\n\r\n <button *ngIf=\"view === 'day'\"\r\n id=\"{{id}}_month_button\"\r\n class=\"text-body-1 font-weight-bold\"\r\n (click)=\"onMonthClick()\"\r\n (keydown)=\"disablePreviousButton ? onFirstItemKeydown($event) : undefined\">\r\n {{month}}\r\n </button>\r\n\r\n <button *ngIf=\"view !== 'year'\"\r\n id=\"{{id}}_year_button\"\r\n class=\"text-body-1 font-weight-bold\"\r\n (click)=\"onYearClick()\"\r\n (keydown)=\"disablePreviousButton && view !== 'day' ? onFirstItemKeydown($event) : undefined\">\r\n {{year}}\r\n </button>\r\n\r\n <div id=\"{{id}}_year_range\"\r\n *ngIf=\"view === 'year'\"\r\n class=\"text-body-1 font-weight-bold\"\r\n [innerHTML]=\"yearRange\">\r\n </div>\r\n\r\n <button id=\"{{id}}_next_button\"\r\n class=\"ml-auto\"\r\n (click)=\"onNextClick()\"\r\n [disabled]=\"disableNextButton\"\r\n (keydown)=\"disablePreviousButton && view === 'year' ? onFirstItemKeydown($event) : undefined\">\r\n <i class=\"ec-icon icon-angle-down rotate-270\"></i>\r\n </button>\r\n</header>\r\n\r\n<ul class=\"px-1 {{view}}-view\">\r\n <ng-container *ngIf=\"view === 'day'\">\r\n <li *ngFor=\"let day of weekDays\"\r\n class=\"d-flex align-items-center justify-content-center text-heading-3\">\r\n {{day}}\r\n </li>\r\n </ng-container>\r\n\r\n <li *ngFor=\"let item of calendarItems; last as isLast; trackBy: calendarItemTrackByDate;\">\r\n <button id=\"{{id}}_item_{{item.date | date:'MM_dd_yyyy'}}\"\r\n ec-calendar-item\r\n [item]=\"item\"\r\n [activeDate]=\"activeDate\"\r\n [selection]=\"selection\"\r\n [view]=\"view\"\r\n (click)=\"onItemSelected(item)\"\r\n (keydown)=\"isLast ? onLastItemKeydown($event) : undefined\"\r\n [hidden]=\"item.date < minDate || item.date > maxDate\">\r\n </button>\r\n </li>\r\n</ul>", styles: [":host{display:inline-block}button{font-size:var(--ec-font-size-action);height:2rem;line-height:1.25rem;padding:.3125rem .5rem;border:0;border-radius:var(--ec-border-radius);display:flex;align-items:center;justify-content:center;cursor:pointer;background-color:transparent}button .label{display:flex;align-items:center;justify-content:center;white-space:nowrap;flex:auto}button .ec-icon{flex:none}button .ec-icon+.label{flex:none;margin-left:.25rem}button.has-badge{padding-right:.0625rem}button:focus{outline:none;position:relative;z-index:1}button:disabled{background-color:var(--ec-background-color-disabled);border:1px solid var(--ec-form-control-border-color-disabled);color:var(--ec-color-disabled-dark);opacity:var(--ec-form-control-opacity-disabled);cursor:default}button:disabled{background-color:transparent;border-color:transparent;color:var(--ec-color-hint-dark);opacity:1;--ec-color-icon: var(--ec-color-hint-dark)}button:hover:not(:disabled){background-color:var(--ec-background-color-hover)}button:active:not(:disabled){background-color:var(--ec-background-color-selected);font-weight:700}button:focus:not(:disabled){box-shadow:var(--ec-button-box-shadow-active, 0 0 0 2px var(--ec-border-color-focus))}ul{list-style:none;padding:0;margin:0;display:grid;align-items:stretch;justify-items:stretch}button[ec-calendar-item]{height:100%;width:100%}ul.day-view{grid-template-columns:repeat(7,2rem);grid-auto-rows:1.75rem;row-gap:.25rem}ul.month-view{grid-template-columns:repeat(3,4.67rem);grid-auto-rows:3.25rem;row-gap:.25rem}ul.year-view{grid-template-columns:repeat(4,3.5rem);grid-auto-rows:3.25rem;row-gap:.25rem}\n"] }]
182
- }], ctorParameters: () => [], propDecorators: { id: [{
227
+ args: [{ selector: 'ec-calendar', changeDetection: ChangeDetectionStrategy.OnPush, template: "<header class=\"d-flex align-items-center mt-1 mb-2 px-1\">\r\n <button id=\"{{id}}_prev_button\"\r\n class=\"mr-auto\"\r\n (click)=\"onPreviousClick()\"\r\n [disabled]=\"disablePreviousButton\">\r\n <i class=\"ec-icon icon-angle-down rotate-90\"></i>\r\n </button>\r\n\r\n <button *ngIf=\"view.mode === 'day'\"\r\n id=\"{{id}}_month_button\"\r\n class=\"text-body-1 font-weight-bold\"\r\n (click)=\"onMonthClick()\">\r\n {{month}}\r\n </button>\r\n\r\n <button *ngIf=\"view.mode !== 'year'\"\r\n id=\"{{id}}_year_button\"\r\n class=\"text-body-1 font-weight-bold\"\r\n (click)=\"onYearClick()\">\r\n {{year}}\r\n </button>\r\n\r\n <div id=\"{{id}}_year_range\"\r\n *ngIf=\"view.mode === 'year'\"\r\n class=\"text-body-1 font-weight-bold\"\r\n [innerHTML]=\"yearRange\">\r\n </div>\r\n\r\n <button id=\"{{id}}_next_button\"\r\n class=\"ml-auto\"\r\n (click)=\"onNextClick()\"\r\n [disabled]=\"disableNextButton\">\r\n <i class=\"ec-icon icon-angle-down rotate-270\"></i>\r\n </button>\r\n</header>\r\n\r\n<div *ngIf=\"view.mode === 'day'\"\r\n class=\"mb-1 px-2 d-flex\">\r\n <h3 *ngFor=\"let day of weekDays\"\r\n class=\"text-heading-3 d-flex justify-content-center align-items-center\"\r\n style=\"width: 2rem; height: 1.75rem;\">{{day}}</h3>\r\n</div>\r\n\r\n<div *ngFor=\"let row of calendarItems; last as isLast; trackBy: trackByDateRow\"\r\n class=\"px-2 d-flex {{view.mode}}-view\"\r\n [class.mb-1]=\"!isLast\"\r\n [class.mb-2]=\"isLast\">\r\n <button *ngFor=\"let item of row; trackBy: trackByDate\"\r\n id=\"{{id}}_item_{{item.date | date:'MM_dd_yyyy'}}\"\r\n ec-calendar-item\r\n [item]=\"item\"\r\n [selection]=\"selection\"\r\n [view]=\"view\"\r\n (click)=\"onItemSelected(item)\"\r\n [innerHTML]=\"item.label\"\r\n [minDate]=\"minDate\"\r\n [maxDate]=\"maxDate\">\r\n </button>\r\n</div>\r\n", styles: [":host{display:inline-block}button{font-size:var(--ec-font-size-action);height:2rem;line-height:1.25rem;padding:.3125rem .5rem;border:0;border-radius:var(--ec-border-radius);display:flex;align-items:center;justify-content:center;cursor:pointer;background-color:transparent}button .label{display:flex;align-items:center;justify-content:center;white-space:nowrap;flex:auto}button .ec-icon{flex:none}button .ec-icon+.label{flex:none;margin-left:.25rem}button.has-badge{padding-right:.0625rem}button:focus{outline:none;position:relative;z-index:1}button:disabled{background-color:var(--ec-background-color-disabled);border:1px solid var(--ec-form-control-border-color-disabled);color:var(--ec-color-disabled-dark);opacity:var(--ec-form-control-opacity-disabled);cursor:default}button:disabled{background-color:transparent;border-color:transparent;color:var(--ec-color-hint-dark);--ec-color-icon: var(--ec-color-hint-dark)}button:hover:not(:disabled){background-color:var(--ec-background-color-hover)}button:active:not(:disabled){background-color:var(--ec-background-color-selected);font-weight:700}button:focus:not(:disabled){box-shadow:var(--ec-button-box-shadow-active, 0 0 0 2px var(--ec-border-color-focus))}button.is-selected{background-color:var(--ec-background-color-selected);font-weight:700}.day-view button{width:2rem;height:1.75rem}.month-view button{width:4.67rem;height:3.25rem}.quarter-view button{width:7rem;height:6.75rem}.year-view button{width:3.5rem;height:3.25rem}.is-selected:has(+.is-selected:not(:disabled)){border-top-right-radius:0;border-bottom-right-radius:0}.is-selected:not(:disabled)+.is-selected{border-top-left-radius:0;border-bottom-left-radius:0}\n"] }]
228
+ }], propDecorators: { id: [{
183
229
  type: HostBinding,
184
230
  args: ['attr.id']
185
231
  }, {
186
232
  type: Input
187
233
  }], selection: [{
188
234
  type: Input
189
- }], selectionChange: [{
190
- type: Output
235
+ }], selectionMode: [{
236
+ type: Input
191
237
  }], minDate: [{
192
238
  type: Input
193
239
  }], maxDate: [{
194
240
  type: Input
195
- }], focusOutStart: [{
241
+ }], dateSelected: [{
196
242
  type: Output
197
- }], focusOutEnd: [{
243
+ }], view: [{
244
+ type: Input
245
+ }], viewChange: [{
198
246
  type: Output
199
247
  }] } });
200
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/controls/calendar/calendar.component.ts","../../../../../../projects/components/src/lib/controls/calendar/calendar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, Input, OnChanges, OnInit, Output, SimpleChanges } from '@angular/core';\r\nimport { range } from 'lodash';\r\nimport moment from 'moment';\r\nimport { DateTimeHelper } from '../../core/date-time-helper';\r\nimport { CalendarItem, CalendarSelection, CalendarView, isCalendarSelectionSingleDate } from './calendar.types';\r\n\r\n@Component({\r\n  selector: 'ec-calendar',\r\n  templateUrl: './calendar.component.html',\r\n  styleUrls: ['./calendar.component.scss'],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n})\r\nexport class CalendarComponent implements OnChanges, OnInit {\r\n\r\n  @HostBinding('attr.id')\r\n  @Input() public id: string = 'calendar';\r\n\r\n  @Input() selection: CalendarSelection | null = null;\r\n  @Output() selectionChange = new EventEmitter<CalendarSelection | null>();\r\n\r\n  @Input() minDate: Date = DateTimeHelper.minDatePickerDate;\r\n  @Input() maxDate: Date = DateTimeHelper.maxDatePickerDate;\r\n\r\n  @Output() focusOutStart = new EventEmitter<KeyboardEvent>();\r\n  @Output() focusOutEnd = new EventEmitter<KeyboardEvent>();\r\n\r\n  /** Month of the active date displayed in the header buttons. */\r\n  public month?: string;\r\n\r\n  /** Year of the active date displayed in the header buttons. */\r\n  public year?: string;\r\n\r\n  /** Range of years to display in the year view. */\r\n  public yearRange?: string;\r\n\r\n  /** Array of calendar items to display in the calendar grid. */\r\n  public calendarItems: CalendarItem[] = [];\r\n\r\n  /** Weekday labels to show at the top of the calendar in day view. */\r\n  public weekDays: string[];\r\n\r\n  /** Determines the current view mode of the calendar. */\r\n  public view: CalendarView = 'day';\r\n\r\n  /** \r\n   * Tracks the month and year being viewed in the calendar.\r\n   * This is separate from the selection, which may be out of view.\r\n   */\r\n  public activeDate: Date;\r\n\r\n  /** TrackBy function for the calendar items. This avoids re-drawing calendar items if a date is shared between view updates */\r\n  calendarItemTrackByDate = (index: number, item: CalendarItem) => item.date.toISOString();\r\n\r\n  public disablePreviousButton: boolean = false;\r\n  public disableNextButton: boolean = false;\r\n\r\n  constructor() {\r\n    // By default, the active date is today.\r\n    // This is replaced with the date of the selection if one is provided on init.\r\n    this.activeDate = new Date();\r\n    // Using moment here to get the weekday labels so they are localized.\r\n    this.weekDays = moment.weekdaysShort().map(d => d.slice(0, 1));\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes.selection && this.selection) {\r\n      // Only need to update the view if the selection is out of view.\r\n      if (\r\n        isCalendarSelectionSingleDate(this.selection) &&\r\n        !this.isDateInView(this.selection) &&\r\n        moment(this.selection).isBetween(this.minDate, this.maxDate, 'day', '[]')\r\n      ) {\r\n        this.drawCalendar('day', this.selection);\r\n      } else {\r\n        // TODO ECAP-26841: determine the active date for a range selection and draw the calendar\r\n      }\r\n    }\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (!this.calendarItems.length) {\r\n      this.drawCalendar('day');\r\n    }\r\n  }\r\n\r\n  public onItemSelected(item: CalendarItem): void {\r\n    if (this.view === 'day') {\r\n      this.selection = item.date;\r\n      this.selectionChange.emit(this.selection);\r\n    }\r\n\r\n    if (this.view === 'month') {\r\n      this.drawCalendar('day', item.date);\r\n    }\r\n\r\n    if (this.view === 'year') {\r\n      this.drawCalendar('month', item.date);\r\n    }\r\n  }\r\n\r\n  public onNextClick(): void {\r\n    // If we're in day view, we're incrementing by month\r\n    // In both month and year views we're incrementing by year\r\n    // In year view, we're moving by 16 years at a time\r\n    const unit = this.view === 'day' ? 'month' : 'year';\r\n    // In year view, we're moving by 16 years at a time. Otherwise we're just moving by 1 month/year.\r\n    const count = this.view === 'year' ? 16 : 1;\r\n    const goToDate = moment(this.activeDate).add(count, unit).toDate();\r\n    this.drawCalendar(this.view, goToDate);\r\n  }\r\n\r\n  public onPreviousClick(): void {\r\n    // If we're in day view, we're incrementing by month\r\n    // In both month and year views we're incrementing years\r\n    const unit = this.view === 'day' ? 'month' : 'year';\r\n    // In year view, we're moving by 16 years at a time. Otherwise we're just moving by 1 month/year.\r\n    const count = this.view === 'year' ? 16 : 1;\r\n    const goToDate = moment(this.activeDate).subtract(count, unit).toDate();\r\n    this.drawCalendar(this.view, goToDate);\r\n  }\r\n\r\n  /** Switches the calendar to month view. */\r\n  public onMonthClick(): void {\r\n    this.drawCalendar('month');\r\n  }\r\n\r\n  /** Switches the calendar to year view. */\r\n  public onYearClick(): void {\r\n    this.drawCalendar('year');\r\n  }\r\n\r\n  public onFirstItemKeydown(event: KeyboardEvent): void {\r\n    // If the user is tabbing backwards from the first item, emit the focusOutStart event.\r\n    if (event.key === 'Tab' && event.shiftKey) {\r\n      this.focusOutStart.emit(event);\r\n    }\r\n  }\r\n\r\n  public onLastItemKeydown(event: KeyboardEvent): void {\r\n    // If the user is tabbing forwards from the last item, emit the focusOutEnd event.\r\n    if (event.key === 'Tab' && !event.shiftKey) {\r\n      this.focusOutEnd.emit(event);\r\n    }\r\n  }\r\n\r\n  private drawCalendar(view: CalendarView, goTo?: Date): void {\r\n    // If a goTo date is provided, update the active date so we know what to increment on next/previous clicks.\r\n    if (goTo) {\r\n      this.activeDate = goTo;\r\n    }\r\n    \r\n    this.month = moment(this.activeDate).format('MMM');\r\n    this.year = moment(this.activeDate).format('YYYY');\r\n    this.view = view;\r\n    this.calendarItems = this.getCalendarItems(view, this.activeDate);\r\n\r\n    const startYear = this.calendarItems[0].date < this.minDate ? this.minDate : this.calendarItems[0].date;\r\n    const endYear = this.calendarItems[this.calendarItems.length - 1].date > this.maxDate ? this.maxDate : this.calendarItems[this.calendarItems.length - 1].date;\r\n    this.yearRange = `${moment(startYear).format('YYYY')}&ndash;${moment(endYear).format('YYYY')}`;\r\n\r\n    // If the active date is less than the min date or the min date is in view, disable the previous button.\r\n    if (this.activeDate < this.minDate || this.isDateInView(this.minDate)) {\r\n      this.disablePreviousButton = true;\r\n    } else {\r\n      this.disablePreviousButton = false;\r\n    }\r\n\r\n    // If the active date is greater than the max date or the max date is in view, disable the next button.\r\n    if (this.activeDate > this.maxDate || this.isDateInView(this.maxDate)) {\r\n      this.disableNextButton = true;\r\n    } else {\r\n      this.disableNextButton = false;\r\n    }\r\n  }\r\n\r\n  private getCalendarItems(view: CalendarView, activeDate: Date): CalendarItem[] {\r\n    switch (view) {\r\n      case 'day': return this.getDayViewItems(activeDate);\r\n      case 'month': return this.getMonthViewItems(activeDate);\r\n      case 'year': return this.getYearViewItems(activeDate);\r\n    }\r\n  }\r\n\r\n  private getDayViewItems(activeDate: Date): CalendarItem[] {\r\n    // 6 rows of 7 days = 42 days\r\n    return range(0, 42).map(i => {\r\n      // Use the start of the week of the first day of the month.\r\n      // This pads out the first week with any days from the previous month.\r\n      const date = moment(activeDate).startOf('month').startOf('week').add(i, 'day');\r\n      return {\r\n        date: date.toDate(),\r\n        label: date.format('D')\r\n      };\r\n    });\r\n  }\r\n\r\n  private getMonthViewItems(activeDate: Date): CalendarItem[] {\r\n    return range(0, 12).map(i => {\r\n      const date = moment(activeDate).startOf('year').add(i, 'month');\r\n      return {\r\n        date: date.toDate(),\r\n        label: date.format('MMM')\r\n      };\r\n    });\r\n  }\r\n\r\n  private getYearViewItems(activeDate: Date): CalendarItem[] {\r\n    // 4x4 grid of years = 16 years\r\n    return range(0, 16).map(i => {\r\n      // Put the current active year at the beginning of the 3rd row.\r\n      const date = moment(activeDate).startOf('year').subtract(8, 'year').add(i, 'year');\r\n      return {\r\n        date: date.toDate(),\r\n        label: date.format('YYYY')\r\n      };\r\n    });\r\n  }\r\n\r\n  /** Returns true if the date is within the current calendar view */\r\n  private isDateInView(date: Date): boolean {\r\n    if (!this.calendarItems.length) {\r\n      return false;\r\n    }\r\n\r\n    const viewStart = this.calendarItems[0].date;\r\n    const viewEnd = this.calendarItems[this.calendarItems.length - 1].date;\r\n    return date >= viewStart && date <= viewEnd;\r\n  }\r\n}\r\n","<header class=\"d-flex align-items-center mb-2\">\r\n  <button id=\"{{id}}_prev_button\"\r\n          class=\"mr-auto\"\r\n          (click)=\"onPreviousClick()\"\r\n          (keydown)=\"!disablePreviousButton ? onFirstItemKeydown($event) : undefined\"\r\n          [disabled]=\"disablePreviousButton\">\r\n    <i class=\"ec-icon icon-angle-down rotate-90\"></i>\r\n  </button>\r\n\r\n  <button *ngIf=\"view === 'day'\"\r\n          id=\"{{id}}_month_button\"\r\n          class=\"text-body-1 font-weight-bold\"\r\n          (click)=\"onMonthClick()\"\r\n          (keydown)=\"disablePreviousButton ? onFirstItemKeydown($event) : undefined\">\r\n    {{month}}\r\n  </button>\r\n\r\n  <button *ngIf=\"view !== 'year'\"\r\n          id=\"{{id}}_year_button\"\r\n          class=\"text-body-1 font-weight-bold\"\r\n          (click)=\"onYearClick()\"\r\n          (keydown)=\"disablePreviousButton && view !== 'day' ? onFirstItemKeydown($event) : undefined\">\r\n    {{year}}\r\n  </button>\r\n\r\n  <div id=\"{{id}}_year_range\"\r\n       *ngIf=\"view === 'year'\"\r\n       class=\"text-body-1 font-weight-bold\"\r\n       [innerHTML]=\"yearRange\">\r\n  </div>\r\n\r\n  <button id=\"{{id}}_next_button\"\r\n          class=\"ml-auto\"\r\n          (click)=\"onNextClick()\"\r\n          [disabled]=\"disableNextButton\"\r\n          (keydown)=\"disablePreviousButton && view === 'year' ? onFirstItemKeydown($event) : undefined\">\r\n    <i class=\"ec-icon icon-angle-down rotate-270\"></i>\r\n  </button>\r\n</header>\r\n\r\n<ul class=\"px-1 {{view}}-view\">\r\n  <ng-container *ngIf=\"view === 'day'\">\r\n    <li *ngFor=\"let day of weekDays\"\r\n        class=\"d-flex align-items-center justify-content-center text-heading-3\">\r\n      {{day}}\r\n    </li>\r\n  </ng-container>\r\n\r\n  <li *ngFor=\"let item of calendarItems; last as isLast; trackBy: calendarItemTrackByDate;\">\r\n    <button id=\"{{id}}_item_{{item.date | date:'MM_dd_yyyy'}}\"\r\n            ec-calendar-item\r\n            [item]=\"item\"\r\n            [activeDate]=\"activeDate\"\r\n            [selection]=\"selection\"\r\n            [view]=\"view\"\r\n            (click)=\"onItemSelected(item)\"\r\n            (keydown)=\"isLast ? onLastItemKeydown($event) : undefined\"\r\n            [hidden]=\"item.date < minDate || item.date > maxDate\">\r\n    </button>\r\n  </li>\r\n</ul>"]}
248
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/controls/calendar/calendar.component.ts","../../../../../../projects/components/src/lib/controls/calendar/calendar.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,WAAW,EAAE,KAAK,EAAqB,MAAM,EAAiB,MAAM,eAAe,CAAC;AAC/I,OAAO,EAAE,KAAK,EAAE,MAAM,QAAQ,CAAC;AAC/B,OAAO,MAAM,MAAM,QAAQ,CAAC;AAC5B,OAAO,EAAE,cAAc,EAAE,MAAM,6BAA6B,CAAC;;;;AAU7D,MAAM,OAAO,iBAAiB;IAN9B;QASkB,OAAE,GAAW,UAAU,CAAC;QAE/B,cAAS,GAA+B,IAAI,CAAC;QAC7C,kBAAa,GAA4B,KAAK,CAAC;QAE/C,YAAO,GAAS,cAAc,CAAC,iBAAiB,CAAC;QACjD,YAAO,GAAS,cAAc,CAAC,iBAAiB,CAAC;QAEhD,iBAAY,GAAG,IAAI,YAAY,EAAQ,CAAC;QAElD,mDAAmD;QACnC,SAAI,GAAkB,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,IAAI,IAAI,EAAE,EAAE,CAAC;QACvD,eAAU,GAAG,IAAI,YAAY,EAAiB,CAAC;QAWhE,8DAA8D;QACvD,kBAAa,GAAsB,EAAE,CAAC;QAE7C,qEAAqE;QAC9D,aAAQ,GAAG,MAAM,CAAC,aAAa,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QAE1D,0BAAqB,GAAY,KAAK,CAAC;QACvC,sBAAiB,GAAY,KAAK,CAAC;QAE1C,oIAAoI;QACpI,oGAAoG;QAC7F,mBAAc,GAAG,CAAC,KAAa,EAAE,GAAoB,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,GAAG,CAAC,GAAG,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC;QACnI,gBAAW,GAAG,CAAC,KAAa,EAAE,IAAmB,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;KAqNtF;IAnNC,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC/B,CAAC;QAED,IAAI,OAAO,CAAC,OAAO,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACvC,IAAI,CAAC,wBAAwB,EAAE,CAAC;QAClC,CAAC;IACH,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC;YAC/B,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC/B,CAAC;IACH,CAAC;IAEM,WAAW;QAChB,oDAAoD;QACpD,gEAAgE;QAChE,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC;QACzD,iGAAiG;QACjG,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QACjD,MAAM,QAAQ,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC,MAAM,EAAE,CAAC;QAClE,IAAI,CAAC,YAAY,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC,CAAC;IAC9D,CAAC;IAEM,eAAe;QACpB,oDAAoD;QACpD,+DAA+D;QAC/D,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC;QACzD,iGAAiG;QACjG,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QACjD,MAAM,QAAQ,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC,MAAM,EAAE,CAAC;QACvE,IAAI,CAAC,YAAY,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC,CAAC;IAC9D,CAAC;IAED,2CAA2C;IACpC,YAAY;QACjB,IAAI,CAAC,YAAY,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;IAC7D,CAAC;IAED,0CAA0C;IACnC,WAAW;QAChB,IAAI,CAAC,YAAY,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;IAC5D,CAAC;IAEM,cAAc,CAAC,IAAmB;QACvC,QAAO,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;YACtB,KAAK,KAAK;gBACR,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;gBACzB,MAAM;YACR,KAAK,OAAO;gBACV,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;gBAC3B,MAAM;YACR,KAAK,SAAS;gBACZ,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;gBAC7B,MAAM;YACR,KAAK,MAAM;gBACT,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;gBAC1B,MAAM;QACV,CAAC;IACH,CAAC;IAEO,cAAc,CAAC,IAAmB;QACxC,iGAAiG;QACjG,IAAI,IAAI,CAAC,aAAa,KAAK,MAAM,EAAE,CAAC;YAClC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACpC,6DAA6D;QAC7D,CAAC;aAAM,IAAI,IAAI,CAAC,aAAa,KAAK,SAAS,EAAE,CAAC;YAC5C,IAAI,CAAC,YAAY,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;YAC1D,sFAAsF;QACtF,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,YAAY,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QACxD,CAAC;IACH,CAAC;IAEO,iBAAiB,CAAC,IAAmB;QAC3C,kEAAkE;QAClE,4EAA4E;QAC5E,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACpC,CAAC;IAEO,eAAe,CAAC,IAAmB;QACzC,wFAAwF;QACxF,IAAI,IAAI,CAAC,aAAa,KAAK,OAAO,EAAE,CAAC;YACnC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACpC,mGAAmG;YACnG,4EAA4E;QAC5E,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,YAAY,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QACtD,CAAC;IACH,CAAC;IAEO,aAAa,CAAC,IAAmB;QACvC,qEAAqE;QACrE,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACpC,CAAC;IAEO,YAAY,CAAC,IAAmB;QACtC,8CAA8C;QAC9C,IAAI,IAAI,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;YACjE,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;YACjB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC7B,CAAC;QAED,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAClD,IAAI,CAAC,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAClD,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAEtD,MAAM,SAAS,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAC9G,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QACpC,MAAM,OAAO,GAAG,QAAQ,CAAC,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC;QAC5E,IAAI,CAAC,SAAS,GAAG,GAAG,MAAM,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,UAAU,MAAM,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC;QAE/F,IAAI,CAAC,wBAAwB,EAAE,CAAC;IAClC,CAAC;IAEO,gBAAgB,CAAC,IAAmB;QAC1C,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;YAClB,KAAK,KAAK,CAAC,CAAC,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACnD,KAAK,OAAO,CAAC,CAAC,OAAO,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACvD,KAAK,SAAS,CAAC,CAAC,OAAO,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC3D,KAAK,MAAM,CAAC,CAAC,OAAO,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACvD,CAAC;IACH,CAAC;IAEO,eAAe,CAAC,UAAgB;QACtC,6BAA6B;QAC7B,OAAO,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;YACzB,OAAO,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;gBACzB,2DAA2D;gBAC3D,sEAAsE;gBACtE,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;gBACvF,OAAO;oBACL,IAAI,EAAE,IAAI,CAAC,MAAM,EAAE;oBACnB,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC;iBACxB,CAAC;YACJ,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,iBAAiB,CAAC,UAAgB;QACxC,iCAAiC;QACjC,OAAO,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;YACzB,OAAO,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;gBACzB,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,OAAO,CAAC,CAAC;gBACxE,OAAO;oBACL,IAAI,EAAE,IAAI,CAAC,MAAM,EAAE;oBACnB,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC;iBAC1B,CAAC;YACJ,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,mBAAmB,CAAC,UAAgB;QAC1C,oCAAoC;QACpC,OAAO,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;YACzB,OAAO,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;gBACzB,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,SAAS,CAAC,CAAC;gBAC1E,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;gBAC9C,OAAO;oBACL,IAAI,EAAE,IAAI,CAAC,MAAM,EAAE;oBACnB,KAAK,EAAE,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,UAAU,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE;iBAC9D,CAAC;YACJ,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,gBAAgB,CAAC,UAAgB;QACvC,+BAA+B;QAC/B,OAAO,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;YACzB,OAAO,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;gBACzB,sFAAsF;gBACtF,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,MAAM,CAAC,CAAC;gBAC5F,OAAO;oBACL,IAAI,EAAE,IAAI,CAAC,MAAM,EAAE;oBACnB,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC;iBAC3B,CAAC;YACJ,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,wBAAwB;QAC9B,2FAA2F;QAC3F,0FAA0F;QAC1F,wBAAwB;QACxB,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC;QAEzD,0FAA0F;QAC1F,oFAAoF;QACpF,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACnD,IAAI,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,EAAE,CAAC;YACjF,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC;QACpC,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,qBAAqB,GAAG,KAAK,CAAC;QACrC,CAAC;QAED,qFAAqF;QACrF,oFAAoF;QACpF,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACnD,IAAI,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,EAAE,CAAC;YAC3E,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;QAChC,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC;QACjC,CAAC;IACH,CAAC;IAEO,WAAW;QACjB,MAAM,OAAO,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAClE,OAAO,OAAO,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;IACrC,CAAC;+GA1PU,iBAAiB;mGAAjB,iBAAiB,iUCb9B,ihEA2DA;;4FD9Ca,iBAAiB;kBAN7B,SAAS;+BACE,aAAa,mBAGN,uBAAuB,CAAC,MAAM;8BAK/B,EAAE;sBADjB,WAAW;uBAAC,SAAS;;sBACrB,KAAK;gBAEG,SAAS;sBAAjB,KAAK;gBACG,aAAa;sBAArB,KAAK;gBAEG,OAAO;sBAAf,KAAK;gBACG,OAAO;sBAAf,KAAK;gBAEI,YAAY;sBAArB,MAAM;gBAGS,IAAI;sBAAnB,KAAK;gBACW,UAAU;sBAA1B,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, Input, OnChanges, OnInit, Output, SimpleChanges } from '@angular/core';\r\nimport { range } from 'lodash';\r\nimport moment from 'moment';\r\nimport { DateTimeHelper } from '../../core/date-time-helper';\r\nimport { DateInput } from '../date-input/date-input.types';\r\nimport { Calendar } from './calendar.types';\r\n\r\n@Component({\r\n  selector: 'ec-calendar',\r\n  templateUrl: './calendar.component.html',\r\n  styleUrls: ['./calendar.component.scss'],\r\n  changeDetection: ChangeDetectionStrategy.OnPush\r\n})\r\nexport class CalendarComponent implements OnChanges, OnInit {\r\n\r\n  @HostBinding('attr.id')\r\n  @Input() public id: string = 'calendar';\r\n\r\n  @Input() selection: DateInput.Selection | null = null;\r\n  @Input() selectionMode: DateInput.SelectionMode = 'day';\r\n\r\n  @Input() minDate: Date = DateTimeHelper.minDatePickerDate;\r\n  @Input() maxDate: Date = DateTimeHelper.maxDatePickerDate;\r\n\r\n  @Output() dateSelected = new EventEmitter<Date>();\r\n\r\n  /** Determines the current view of the calendar. */\r\n  @Input() public view: Calendar.View = { mode: 'day', date: new Date() };\r\n  @Output() public viewChange = new EventEmitter<Calendar.View>();\r\n\r\n  /** Month of the active date displayed in the header buttons. */\r\n  public month?: string;\r\n\r\n  /** Year of the active date displayed in the header buttons. */\r\n  public year?: string;\r\n\r\n  /** Range of years to display in the year view. */\r\n  public yearRange?: string;\r\n\r\n  /** Array of calendar item rows to display in the calendar. */\r\n  public calendarItems: Calendar.Item[][] = [];\r\n\r\n  /** Weekday labels to show at the top of the calendar in day view. */\r\n  public weekDays = moment.weekdaysShort().map(d => d.slice(0, 1));\r\n\r\n  public disablePreviousButton: boolean = false;\r\n  public disableNextButton: boolean = false;\r\n\r\n  // Trackby functions for ngFor loops. These reduce the number of DOM elements that need to be updated when the calendar is re-drawn.\r\n  // Without these there are some change-detection issues on the first-time rendering of the calendar.\r\n  public trackByDateRow = (index: number, row: Calendar.Item[]) => `${row[0].date.toISOString()}-${row[row.length - 1].date.toISOString()}`;\r\n  public trackByDate = (index: number, item: Calendar.Item) => item.date.toISOString();\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes.view) {\r\n      this.drawCalendar(this.view);\r\n    }\r\n\r\n    if (changes.minDate || changes.maxDate) {\r\n      this.updateNextPreviousStates();\r\n    }\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (!this.calendarItems.length) {\r\n      this.drawCalendar(this.view);\r\n    }\r\n  }\r\n\r\n  public onNextClick(): void {\r\n    // If we're in day view, we're incrementing by month\r\n    // In month, quarter, and year views we're incrementing by year\\\r\n    const unit = this.view.mode === 'day' ? 'month' : 'year';\r\n    // In year view, we're moving by 16 years at a time. Otherwise we're just moving by 1 month/year.\r\n    const count = this.view.mode === 'year' ? 16 : 1;\r\n    const goToDate = moment(this.view.date).add(count, unit).toDate();\r\n    this.drawCalendar({ mode: this.view.mode, date: goToDate });\r\n  }\r\n\r\n  public onPreviousClick(): void {\r\n    // If we're in day view, we're incrementing by month\r\n    // In month, quarter, and year views we're incrementing by year\r\n    const unit = this.view.mode === 'day' ? 'month' : 'year';\r\n    // In year view, we're moving by 16 years at a time. Otherwise we're just moving by 1 month/year.\r\n    const count = this.view.mode === 'year' ? 16 : 1;\r\n    const goToDate = moment(this.view.date).subtract(count, unit).toDate();\r\n    this.drawCalendar({ mode: this.view.mode, date: goToDate });\r\n  }\r\n\r\n  /** Switches the calendar to month view. */\r\n  public onMonthClick(): void {\r\n    this.drawCalendar({ mode: 'month', date: this.view.date });\r\n  }\r\n\r\n  /** Switches the calendar to year view. */\r\n  public onYearClick(): void {\r\n    this.drawCalendar({ mode: 'year', date: this.view.date });\r\n  }\r\n\r\n  public onItemSelected(item: Calendar.Item): void {\r\n    switch(this.view.mode) {\r\n      case 'day':\r\n        this.onDaySelected(item);\r\n        break;\r\n      case 'month':\r\n        this.onMonthSelected(item);\r\n        break;\r\n      case 'quarter':\r\n        this.onQuarterSelected(item);\r\n        break;\r\n      case 'year':\r\n        this.onYearSelected(item);\r\n        break;\r\n    }\r\n  }\r\n\r\n  private onYearSelected(item: Calendar.Item): void {\r\n    // If in year selection mode, emit the calendar item so the date picker can update the selection.\r\n    if (this.selectionMode === 'year') {\r\n      this.dateSelected.emit(item.date);\r\n    // If we're in quarter selection mode, draw the quarter view.\r\n    } else if (this.selectionMode === 'quarter') {\r\n      this.drawCalendar({ mode: 'quarter', date: item.date });\r\n    // All other selection modes should zoom in to the month view when a year is selected.\r\n    } else {\r\n      this.drawCalendar({ mode: 'month', date: item.date });\r\n    }\r\n  }\r\n\r\n  private onQuarterSelected(item: Calendar.Item): void {\r\n    // Quarter view is only accessible when in quarter selection mode,\r\n    // so there's no other views to draw from here. Just emit the selected item.\r\n    this.dateSelected.emit(item.date);\r\n  }\r\n\r\n  private onMonthSelected(item: Calendar.Item): void {\r\n    // When in month selection mode, emit the selected month item and don't change the view.\r\n    if (this.selectionMode === 'month') {\r\n      this.dateSelected.emit(item.date);\r\n    // Month view is only accessible from day, last7days, last28days, month, and range selection modes.\r\n    // If we're not in month selection mode, we need to zoom in to the day view.\r\n    } else {\r\n      this.drawCalendar({ mode: 'day', date: item.date });\r\n    }\r\n  }\r\n\r\n  private onDaySelected(item: Calendar.Item): void {\r\n    // No further views other than day mode. Just emit the selected item.\r\n    this.dateSelected.emit(item.date);\r\n  }\r\n\r\n  private drawCalendar(view: Calendar.View): void {\r\n    // If the view has changed, emit the new view.\r\n    if (view.date !== this.view.date || view.mode !== this.view.mode) {\r\n      this.view = view;\r\n      this.viewChange.emit(view);\r\n    }\r\n    \r\n    this.month = moment(this.view.date).format('MMM');\r\n    this.year = moment(this.view.date).format('YYYY');\r\n    this.calendarItems = this.getCalendarItems(this.view);\r\n\r\n    const startYear = this.calendarItems[0][0].date < this.minDate ? this.minDate : this.calendarItems[0][0].date;\r\n    const lastItem = this.getLastItem();\r\n    const endYear = lastItem.date > this.maxDate ? this.maxDate : lastItem.date;\r\n    this.yearRange = `${moment(startYear).format('YYYY')}&ndash;${moment(endYear).format('YYYY')}`;\r\n\r\n    this.updateNextPreviousStates();\r\n  }\r\n\r\n  private getCalendarItems(view: Calendar.View): Calendar.Item[][] {\r\n    switch (view.mode) {\r\n      case 'day': return this.getDayViewItems(view.date);\r\n      case 'month': return this.getMonthViewItems(view.date);\r\n      case 'quarter': return this.getQuarterViewItems(view.date);\r\n      case 'year': return this.getYearViewItems(view.date);\r\n    }\r\n  }\r\n\r\n  private getDayViewItems(activeDate: Date): Calendar.Item[][] {\r\n    // 6 rows of 7 days = 42 days\r\n    return range(0, 6).map(r => {\r\n      return range(0, 7).map(d => {\r\n        // Use the start of the week of the first day of the month.\r\n        // This pads out the first week with any days from the previous month.\r\n        const date = moment(activeDate).startOf('month').startOf('week').add(r * 7 + d, 'day');\r\n        return {\r\n          date: date.toDate(),\r\n          label: date.format('D')\r\n        };\r\n      });\r\n    });\r\n  }\r\n\r\n  private getMonthViewItems(activeDate: Date): Calendar.Item[][] {\r\n    // 4 rows of 3 months = 12 months\r\n    return range(0, 4).map(r => {\r\n      return range(0, 3).map(m => {\r\n        const date = moment(activeDate).startOf('year').add(r * 3 + m, 'month');\r\n        return {\r\n          date: date.toDate(),\r\n          label: date.format('MMM')\r\n        };\r\n      });\r\n    });\r\n  }\r\n\r\n  private getQuarterViewItems(activeDate: Date): Calendar.Item[][] {\r\n    // 2 rows of 2 quarters = 4 quarters\r\n    return range(0, 2).map(r => {\r\n      return range(0, 2).map(q => {\r\n        const date = moment(activeDate).startOf('year').add(r * 2 + q, 'quarter');\r\n        const endDate = moment(date).endOf('quarter');\r\n        return {\r\n          date: date.toDate(),\r\n          label: `${date.format('MMM')}&ndash;${endDate.format('MMM')}`\r\n        };\r\n      });\r\n    });\r\n  }\r\n\r\n  private getYearViewItems(activeDate: Date): Calendar.Item[][] {\r\n    // 4 rows of 4 years = 16 years\r\n    return range(0, 4).map(r => {\r\n      return range(0, 4).map(y => {\r\n        // Subtracting 8 years to put the current active year at the beginning of the 3rd row.\r\n        const date = moment(activeDate).startOf('year').subtract(8, 'years').add(r * 4 + y, 'year');\r\n        return {\r\n          date: date.toDate(),\r\n          label: date.format('YYYY')\r\n        };\r\n      });\r\n    });\r\n  }\r\n\r\n  private updateNextPreviousStates(): void {\r\n    // We change the granularity of the min and max date checks based on the current view mode.\r\n    // We don't want to disable the next/previous buttons if the min or max date is within the\r\n    // next or previous view\r\n    const unit = this.view.mode === 'day' ? 'month' : 'year';\r\n\r\n    // When in year view, we need to subtract 9 years to determine if we're past the min date.\r\n    // The year view is 16 years long, and the current date is the 9th year in the view.\r\n    const subCount = this.view.mode === 'year' ? 9 : 1;\r\n    if (moment(this.view.date).subtract(subCount, unit).isBefore(this.minDate, unit)) {\r\n      this.disablePreviousButton = true;\r\n    } else {\r\n      this.disablePreviousButton = false;\r\n    }\r\n\r\n    // When in year view, we need to add 8 years to determine if we're past the max date.\r\n    // The year view is 16 years long, and the current date is the 9th year in the view.\r\n    const addCount = this.view.mode === 'year' ? 8 : 1;\r\n    if (moment(this.view.date).add(addCount, unit).isAfter(this.maxDate, unit)) {\r\n      this.disableNextButton = true;\r\n    } else {\r\n      this.disableNextButton = false;\r\n    }\r\n  }\r\n\r\n  private getLastItem(): Calendar.Item {\r\n    const lastRow = this.calendarItems[this.calendarItems.length - 1];\r\n    return lastRow[lastRow.length - 1];\r\n  }\r\n}\r\n","<header class=\"d-flex align-items-center mt-1 mb-2 px-1\">\r\n  <button id=\"{{id}}_prev_button\"\r\n          class=\"mr-auto\"\r\n          (click)=\"onPreviousClick()\"\r\n          [disabled]=\"disablePreviousButton\">\r\n    <i class=\"ec-icon icon-angle-down rotate-90\"></i>\r\n  </button>\r\n\r\n  <button *ngIf=\"view.mode === 'day'\"\r\n          id=\"{{id}}_month_button\"\r\n          class=\"text-body-1 font-weight-bold\"\r\n          (click)=\"onMonthClick()\">\r\n    {{month}}\r\n  </button>\r\n\r\n  <button *ngIf=\"view.mode !== 'year'\"\r\n          id=\"{{id}}_year_button\"\r\n          class=\"text-body-1 font-weight-bold\"\r\n          (click)=\"onYearClick()\">\r\n    {{year}}\r\n  </button>\r\n\r\n  <div id=\"{{id}}_year_range\"\r\n       *ngIf=\"view.mode === 'year'\"\r\n       class=\"text-body-1 font-weight-bold\"\r\n       [innerHTML]=\"yearRange\">\r\n  </div>\r\n\r\n  <button id=\"{{id}}_next_button\"\r\n          class=\"ml-auto\"\r\n          (click)=\"onNextClick()\"\r\n          [disabled]=\"disableNextButton\">\r\n    <i class=\"ec-icon icon-angle-down rotate-270\"></i>\r\n  </button>\r\n</header>\r\n\r\n<div *ngIf=\"view.mode === 'day'\"\r\n     class=\"mb-1 px-2 d-flex\">\r\n  <h3 *ngFor=\"let day of weekDays\"\r\n      class=\"text-heading-3 d-flex justify-content-center align-items-center\"\r\n      style=\"width: 2rem; height: 1.75rem;\">{{day}}</h3>\r\n</div>\r\n\r\n<div *ngFor=\"let row of calendarItems; last as isLast; trackBy: trackByDateRow\"\r\n     class=\"px-2 d-flex {{view.mode}}-view\"\r\n     [class.mb-1]=\"!isLast\"\r\n     [class.mb-2]=\"isLast\">\r\n  <button *ngFor=\"let item of row; trackBy: trackByDate\"\r\n          id=\"{{id}}_item_{{item.date | date:'MM_dd_yyyy'}}\"\r\n          ec-calendar-item\r\n          [item]=\"item\"\r\n          [selection]=\"selection\"\r\n          [view]=\"view\"\r\n          (click)=\"onItemSelected(item)\"\r\n          [innerHTML]=\"item.label\"\r\n          [minDate]=\"minDate\"\r\n          [maxDate]=\"maxDate\">\r\n  </button>\r\n</div>\r\n"]}
@@ -1,4 +1,2 @@
1
- export function isCalendarSelectionSingleDate(selection) {
2
- return selection instanceof Date;
3
- }
4
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXIudHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb21wb25lbnRzL3NyYy9saWIvY29udHJvbHMvY2FsZW5kYXIvY2FsZW5kYXIudHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBU0EsTUFBTSxVQUFVLDZCQUE2QixDQUFDLFNBQW9DO0lBQ2hGLE9BQU8sU0FBUyxZQUFZLElBQUksQ0FBQztBQUNuQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgQ2FsZW5kYXJJdGVtID0ge1xyXG4gIGxhYmVsOiBzdHJpbmc7XHJcbiAgZGF0ZTogRGF0ZTtcclxufTtcclxuXHJcbmV4cG9ydCB0eXBlIENhbGVuZGFyU2VsZWN0aW9uID0gRGF0ZSB8IFtEYXRlLCBEYXRlXTtcclxuXHJcbmV4cG9ydCB0eXBlIENhbGVuZGFyVmlldyA9ICdkYXknIHwgJ21vbnRoJyB8ICd5ZWFyJztcclxuXHJcbmV4cG9ydCBmdW5jdGlvbiBpc0NhbGVuZGFyU2VsZWN0aW9uU2luZ2xlRGF0ZShzZWxlY3Rpb24/OiBDYWxlbmRhclNlbGVjdGlvbiB8IG51bGwpOiBzZWxlY3Rpb24gaXMgRGF0ZSB7XHJcbiAgcmV0dXJuIHNlbGVjdGlvbiBpbnN0YW5jZW9mIERhdGU7XHJcbn1cclxuIl19
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXIudHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb21wb25lbnRzL3NyYy9saWIvY29udHJvbHMvY2FsZW5kYXIvY2FsZW5kYXIudHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBuYW1lc3BhY2UgQ2FsZW5kYXIge1xyXG4gIGV4cG9ydCB0eXBlIFZpZXdNb2RlID0gJ2RheScgfCAnbW9udGgnIHwgJ3F1YXJ0ZXInIHwgJ3llYXInO1xyXG5cclxuICBleHBvcnQgdHlwZSBWaWV3ID0ge1xyXG4gICAgbW9kZTogVmlld01vZGU7XHJcbiAgICBkYXRlOiBEYXRlO1xyXG4gIH07XHJcblxyXG4gIGV4cG9ydCB0eXBlIEl0ZW0gPSB7XHJcbiAgICBsYWJlbDogc3RyaW5nO1xyXG4gICAgZGF0ZTogRGF0ZTtcclxuICB9O1xyXG59XHJcbiJdfQ==