@energycap/components 0.42.0 → 0.42.2-ECAP-26676-readonly-attribute-suppresses-help-popovers-on-checkboxes.20250122-1358

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 (41) 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/checkbox/checkbox.component.mjs +2 -2
  6. package/esm2022/lib/controls/date-input/date-input-selection-strategies/date-input-selection-strategy-base.mjs +57 -0
  7. package/esm2022/lib/controls/date-input/date-input-selection-strategies/day-selection-strategy.mjs +62 -0
  8. package/esm2022/lib/controls/date-input/date-input-selection-strategies/last-28-days-selection-strategy.mjs +100 -0
  9. package/esm2022/lib/controls/date-input/date-input-selection-strategies/last-7-days-selection-strategy.mjs +101 -0
  10. package/esm2022/lib/controls/date-input/date-input-selection-strategies/month-selection-strategy.mjs +76 -0
  11. package/esm2022/lib/controls/date-input/date-input-selection-strategies/quarter-selection-strategy.mjs +79 -0
  12. package/esm2022/lib/controls/date-input/date-input-selection-strategies/range-selection-strategy.mjs +210 -0
  13. package/esm2022/lib/controls/date-input/date-input-selection-strategies/year-selection-strategy.mjs +81 -0
  14. package/esm2022/lib/controls/date-input/date-input.component.mjs +322 -113
  15. package/esm2022/lib/controls/date-input/date-input.types.mjs +44 -0
  16. package/esm2022/lib/controls/file-upload/file-upload.component.mjs +1 -1
  17. package/esm2022/lib/controls/form-control/form-control.component.mjs +6 -12
  18. package/esm2022/lib/core/date-time-helper.mjs +10 -2
  19. package/esm2022/lib/shared/directives/keyboard-nav-container/keyboard-nav-container.directive.mjs +100 -0
  20. package/esm2022/public-api.mjs +63 -61
  21. package/fesm2022/energycap-components.mjs +1668 -509
  22. package/fesm2022/energycap-components.mjs.map +1 -1
  23. package/lib/components.module.d.ts +9 -8
  24. package/lib/controls/calendar/calendar-item.component.d.ts +11 -6
  25. package/lib/controls/calendar/calendar.component.d.ts +21 -23
  26. package/lib/controls/calendar/calendar.types.d.ts +11 -7
  27. package/lib/controls/date-input/date-input-selection-strategies/date-input-selection-strategy-base.d.ts +42 -0
  28. package/lib/controls/date-input/date-input-selection-strategies/day-selection-strategy.d.ts +21 -0
  29. package/lib/controls/date-input/date-input-selection-strategies/last-28-days-selection-strategy.d.ts +21 -0
  30. package/lib/controls/date-input/date-input-selection-strategies/last-7-days-selection-strategy.d.ts +21 -0
  31. package/lib/controls/date-input/date-input-selection-strategies/month-selection-strategy.d.ts +18 -0
  32. package/lib/controls/date-input/date-input-selection-strategies/quarter-selection-strategy.d.ts +18 -0
  33. package/lib/controls/date-input/date-input-selection-strategies/range-selection-strategy.d.ts +21 -0
  34. package/lib/controls/date-input/date-input-selection-strategies/year-selection-strategy.d.ts +20 -0
  35. package/lib/controls/date-input/date-input.component.d.ts +63 -28
  36. package/lib/controls/date-input/date-input.types.d.ts +62 -0
  37. package/lib/controls/form-control/form-control.component.d.ts +4 -6
  38. package/lib/shared/directives/keyboard-nav-container/keyboard-nav-container.directive.d.ts +23 -0
  39. package/package.json +1 -1
  40. package/public-api.d.ts +62 -60
  41. package/src/assets/locales/en_US.json +9 -1
@@ -0,0 +1,79 @@
1
+ import moment from "moment";
2
+ import { DateInput } from "../date-input.types";
3
+ import { DateInputSelectionStrategyBase } from "./date-input-selection-strategy-base";
4
+ /**
5
+ * Selection strategy for the 'month' selection mode.
6
+ */
7
+ export class QuarterSelectionStrategy extends DateInputSelectionStrategyBase {
8
+ constructor() {
9
+ super(...arguments);
10
+ this.selectionViewMode = 'quarter';
11
+ this.validViewModes = ['quarter', 'year'];
12
+ this.showSecondaryTextbox = false;
13
+ this.showSecondaryCalendar = false;
14
+ }
15
+ formatSelection(selection) {
16
+ // Format the the quarter range as "MMM–MMM, YYYY"
17
+ // We only have one textbox in this mode
18
+ if (DateInput.isSelectionRange(selection)) {
19
+ const start = moment(selection.start).format('MMM');
20
+ const end = moment(selection.end).format('MMM');
21
+ const year = moment(selection.end).format('YYYY');
22
+ return {
23
+ textbox: `${start}–${end}, ${year}`,
24
+ textbox2: null
25
+ };
26
+ }
27
+ // This shouldn't happen since the selection should always be a range in quarter mode
28
+ return { textbox: null, textbox2: null };
29
+ }
30
+ parseTextboxValues(value, parseFormats, opts) {
31
+ // Only one textbox is used in this mode
32
+ const date = this.parseString(value.textbox, parseFormats, opts);
33
+ if (!date) {
34
+ return null;
35
+ }
36
+ return this.getSelectionFromDate(date);
37
+ }
38
+ getSelectionFromDate(date) {
39
+ return {
40
+ start: moment(date).startOf('quarter').toDate(),
41
+ end: moment(date).endOf('quarter').toDate()
42
+ };
43
+ }
44
+ getNewSelectionFromExisting(previousSelection) {
45
+ if (DateInput.isSelectionRange(previousSelection)) {
46
+ // Base the quarter range off of the existing selection's end date, falling back to the start date
47
+ const date = previousSelection.end || previousSelection.start;
48
+ if (date) {
49
+ return this.getSelectionFromDate(date);
50
+ }
51
+ }
52
+ else if (DateInput.isSelectionSingleDate(previousSelection)) {
53
+ // If the previous selection was a single date, make the range that quarter
54
+ return this.getSelectionFromDate(previousSelection);
55
+ }
56
+ return null;
57
+ }
58
+ getNextSelection(selection) {
59
+ // Move the selection to the next quarter
60
+ if (DateInput.isSelectionRange(selection) && selection.start) {
61
+ const date = moment(selection.start).add(1, 'quarter');
62
+ return this.getSelectionFromDate(date.toDate());
63
+ }
64
+ // This shouldn't happen since the next stepper is disabled if the selection only has one date.
65
+ // If it does, just return the selection as-is.
66
+ return selection;
67
+ }
68
+ getPreviousSelection(selection) {
69
+ // Move the selection to the previous quarter
70
+ if (DateInput.isSelectionRange(selection) && selection.start) {
71
+ const date = moment(selection.start).subtract(1, 'quarter');
72
+ return this.getSelectionFromDate(date.toDate());
73
+ }
74
+ // This shouldn't happen since the previous stepper is disabled if the selection only has one date.
75
+ // If it does, just return the selection as-is.
76
+ return selection;
77
+ }
78
+ }
79
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"quarter-selection-strategy.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/controls/date-input/date-input-selection-strategies/quarter-selection-strategy.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,QAAQ,CAAC;AAE5B,OAAO,EAAE,SAAS,EAAE,MAAM,qBAAqB,CAAC;AAChD,OAAO,EAAE,8BAA8B,EAAE,MAAM,sCAAsC,CAAC;AAEtF;;GAEG;AACH,MAAM,OAAO,wBAAyB,SAAQ,8BAA8B;IAA5E;;QACkB,sBAAiB,GAAsB,SAAS,CAAC;QACjD,mBAAc,GAAwB,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;QAC1D,yBAAoB,GAAG,KAAK,CAAC;QAC7B,0BAAqB,GAAG,KAAK,CAAC;IA2EhD,CAAC;IAzEC,eAAe,CAAC,SAAsC;QACpD,kDAAkD;QAClD,wCAAwC;QACxC,IAAI,SAAS,CAAC,gBAAgB,CAAC,SAAS,CAAC,EAAE,CAAC;YAC1C,MAAM,KAAK,GAAG,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YACpD,MAAM,GAAG,GAAG,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAChD,MAAM,IAAI,GAAG,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YAClD,OAAO;gBACL,OAAO,EAAE,GAAG,KAAK,IAAI,GAAG,KAAK,IAAI,EAAE;gBACnC,QAAQ,EAAE,IAAI;aACf,CAAC;QACJ,CAAC;QAED,qFAAqF;QACrF,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC;IAC3C,CAAC;IAED,kBAAkB,CAAC,KAAsC,EAAE,YAAsB,EAAE,IAA+B;QAChH,wCAAwC;QACxC,MAAM,IAAI,GAAG,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,OAAO,EAAE,YAAY,EAAE,IAAI,CAAC,CAAC;QAEjE,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC;IACzC,CAAC;IAED,oBAAoB,CAAC,IAAU;QAC7B,OAAO;YACL,KAAK,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE;YAC/C,GAAG,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE;SAC5C,CAAC;IACJ,CAAC;IAED,2BAA2B,CAAC,iBAAuC;QACjE,IAAI,SAAS,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,EAAE,CAAC;YAClD,kGAAkG;YAClG,MAAM,IAAI,GAAG,iBAAiB,CAAC,GAAG,IAAI,iBAAiB,CAAC,KAAK,CAAC;YAC9D,IAAI,IAAI,EAAE,CAAC;gBACT,OAAO,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC;YACzC,CAAC;QACH,CAAC;aAAM,IAAI,SAAS,CAAC,qBAAqB,CAAC,iBAAiB,CAAC,EAAE,CAAC;YAC9D,2EAA2E;YAC3E,OAAO,IAAI,CAAC,oBAAoB,CAAC,iBAAiB,CAAC,CAAC;QACtD,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED,gBAAgB,CAAC,SAAqC;QACpD,yCAAyC;QACzC,IAAI,SAAS,CAAC,gBAAgB,CAAC,SAAS,CAAC,IAAI,SAAS,CAAC,KAAK,EAAE,CAAC;YAC7D,MAAM,IAAI,GAAG,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC;YACvD,OAAO,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;QAClD,CAAC;QAED,+FAA+F;QAC/F,+CAA+C;QAC/C,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,oBAAoB,CAAC,SAAqC;QACxD,6CAA6C;QAC7C,IAAI,SAAS,CAAC,gBAAgB,CAAC,SAAS,CAAC,IAAI,SAAS,CAAC,KAAK,EAAE,CAAC;YAC7D,MAAM,IAAI,GAAG,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC;YAC5D,OAAO,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;QAClD,CAAC;QAED,mGAAmG;QACnG,+CAA+C;QAC/C,OAAO,SAAS,CAAC;IACnB,CAAC;CACF","sourcesContent":["import moment from \"moment\";\r\nimport { Calendar } from \"../../calendar/calendar.types\";\r\nimport { DateInput } from \"../date-input.types\";\r\nimport { DateInputSelectionStrategyBase } from \"./date-input-selection-strategy-base\";\r\n\r\n/**\r\n * Selection strategy for the 'month' selection mode.\r\n */\r\nexport class QuarterSelectionStrategy extends DateInputSelectionStrategyBase {\r\n  public readonly selectionViewMode: Calendar.ViewMode = 'quarter';\r\n  public readonly validViewModes: Calendar.ViewMode[] = ['quarter', 'year'];\r\n  public readonly showSecondaryTextbox = false;\r\n  public readonly showSecondaryCalendar = false;\r\n\r\n  formatSelection(selection?: DateInput.Selection | null): DateInput.TextboxGroup['value'] {\r\n    // Format the the quarter range as \"MMM–MMM, YYYY\"\r\n    // We only have one textbox in this mode\r\n    if (DateInput.isSelectionRange(selection)) {\r\n      const start = moment(selection.start).format('MMM');\r\n      const end = moment(selection.end).format('MMM');\r\n      const year = moment(selection.end).format('YYYY');\r\n      return {\r\n        textbox: `${start}–${end}, ${year}`,\r\n        textbox2: null\r\n      };\r\n    }\r\n    \r\n    // This shouldn't happen since the selection should always be a range in quarter mode\r\n    return { textbox: null, textbox2: null };\r\n  }\r\n\r\n  parseTextboxValues(value: DateInput.TextboxGroup['value'], parseFormats: string[], opts?: DateInput.ParsingOptions): DateInput.Selection | null {\r\n    // Only one textbox is used in this mode\r\n    const date = this.parseString(value.textbox, parseFormats, opts);\r\n    \r\n    if (!date) {\r\n      return null;\r\n    }\r\n\r\n    return this.getSelectionFromDate(date);\r\n  }\r\n\r\n  getSelectionFromDate(date: Date): DateInput.Selection | null {\r\n    return {\r\n      start: moment(date).startOf('quarter').toDate(),\r\n      end: moment(date).endOf('quarter').toDate()\r\n    };\r\n  }\r\n\r\n  getNewSelectionFromExisting(previousSelection?: DateInput.Selection): DateInput.Selection | null {\r\n    if (DateInput.isSelectionRange(previousSelection)) {\r\n      // Base the quarter range off of the existing selection's end date, falling back to the start date\r\n      const date = previousSelection.end || previousSelection.start;\r\n      if (date) {\r\n        return this.getSelectionFromDate(date);\r\n      }\r\n    } else if (DateInput.isSelectionSingleDate(previousSelection)) {\r\n      // If the previous selection was a single date, make the range that quarter\r\n      return this.getSelectionFromDate(previousSelection);\r\n    }\r\n\r\n    return null;\r\n  }\r\n\r\n  getNextSelection(selection: DateInput.Selection | null): DateInput.Selection | null {\r\n    // Move the selection to the next quarter\r\n    if (DateInput.isSelectionRange(selection) && selection.start) {\r\n      const date = moment(selection.start).add(1, 'quarter');\r\n      return this.getSelectionFromDate(date.toDate());\r\n    }\r\n\r\n    // This shouldn't happen since the next stepper is disabled if the selection only has one date.\r\n    // If it does, just return the selection as-is.\r\n    return selection;\r\n  }\r\n\r\n  getPreviousSelection(selection: DateInput.Selection | null): DateInput.Selection | null {\r\n    // Move the selection to the previous quarter\r\n    if (DateInput.isSelectionRange(selection) && selection.start) {\r\n      const date = moment(selection.start).subtract(1, 'quarter');\r\n      return this.getSelectionFromDate(date.toDate());\r\n    }\r\n\r\n    // This shouldn't happen since the previous stepper is disabled if the selection only has one date.\r\n    // If it does, just return the selection as-is.\r\n    return selection;\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,210 @@
1
+ import moment from "moment";
2
+ import { DateInput } from "../date-input.types";
3
+ import { DateInputSelectionStrategyBase } from "./date-input-selection-strategy-base";
4
+ /**
5
+ * Selection strategy for the 'range' selection mode.
6
+ */
7
+ export class RangeSelectionStrategy extends DateInputSelectionStrategyBase {
8
+ constructor() {
9
+ super(...arguments);
10
+ this.selectionViewMode = 'day';
11
+ this.validViewModes = ['day', 'month', 'year'];
12
+ this.showSecondaryTextbox = true;
13
+ this.showSecondaryCalendar = true;
14
+ }
15
+ formatSelection(selection) {
16
+ if (!selection) {
17
+ return { textbox: null, textbox2: null };
18
+ }
19
+ // We use the 'll' moment medium format here to make sure the formatted date is formatted according to the user's locale
20
+ if (DateInput.isSelectionSingleDate(selection)) {
21
+ return { textbox: moment(selection).format('ll'), textbox2: null };
22
+ }
23
+ else {
24
+ const start = selection.start ? moment(selection.start).format('ll') : null;
25
+ const end = selection.end ? moment(selection.end).format('ll') : null;
26
+ return { textbox: start, textbox2: end };
27
+ }
28
+ }
29
+ parseTextboxValues(value, parseFormats, opts) {
30
+ let start = this.parseString(value.textbox, parseFormats, opts);
31
+ let end = this.parseString(value.textbox2, parseFormats, opts);
32
+ // If the start date is after the end date, we need to clear out the invalid date. This is different from the last 7/28 days strategies
33
+ // because we don't have a fixed range to automatically adjust the dates to. If the user is changing the begin date or end date and ends
34
+ // up crossing the other date, we should clear out the date that's not being changed so that the user can enter a valid range using the
35
+ // begin/end date they just entered.
36
+ if (start && end && moment(start).isAfter(end)) {
37
+ if (opts?.parseFromEnd) {
38
+ start = null;
39
+ }
40
+ else {
41
+ end = null;
42
+ }
43
+ }
44
+ if (!start && !end) {
45
+ return null;
46
+ }
47
+ else {
48
+ return { start, end };
49
+ }
50
+ }
51
+ getSelectionFromDate(date, existingSelection) {
52
+ if (DateInput.isSelectionRange(existingSelection)) {
53
+ const { start, end } = existingSelection;
54
+ if (start && !end) {
55
+ // If the selected date is before the existing start date, set it as the new start date and use the existing start date as the end date
56
+ return moment(date).isBefore(start) ? { start: date, end: start } : { start, end: date };
57
+ }
58
+ if (end && !start) {
59
+ // If the selected date is after the existing end date, set it as the new end date and use the existing end date as the start date
60
+ return moment(date).isBefore(end) ? { start: date, end } : { start: end, end: date };
61
+ }
62
+ }
63
+ else if (DateInput.isSelectionSingleDate(existingSelection)) {
64
+ // If the selected date is before the existing date, set it as the new start date and use the existing date as the end date
65
+ return moment(date).isBefore(existingSelection) ? { start: date, end: existingSelection } : { start: existingSelection, end: date };
66
+ }
67
+ // If no selection exists or we already have a complete selection, start a new one with the selected date as the start date.
68
+ return { start: date, end: null };
69
+ }
70
+ getNewSelectionFromExisting(previousSelection) {
71
+ if (DateInput.isSelectionRange(previousSelection)) {
72
+ // If we have a range, just return it as is. We don't have a fixed range to adjust the dates to.
73
+ // Even if it's incomplete, the user will still be able to select the remaining date.
74
+ return previousSelection;
75
+ }
76
+ else if (DateInput.isSelectionSingleDate(previousSelection)) {
77
+ // If we have a single date selection, use it as the start date for the new one.
78
+ return { start: previousSelection, end: null };
79
+ }
80
+ return null;
81
+ }
82
+ getSelectionForQuickSelectDate(date, existingSelection) {
83
+ // If we have a valid range, we'll shift the range to the date provided, using it as the end date
84
+ // The new selection will have the same length as the previous.
85
+ if (DateInput.isSelectionRange(existingSelection) && existingSelection.start && existingSelection.end) {
86
+ const length = moment(existingSelection.end).diff(existingSelection.start, 'days');
87
+ return {
88
+ start: moment(date).subtract(length, 'days').toDate(),
89
+ end: date
90
+ };
91
+ }
92
+ // If we only have a single date, we'll use the date provided as the end date.
93
+ // The user will be able to select the start date.
94
+ return { start: null, end: date };
95
+ }
96
+ getNextSelection(selection) {
97
+ // We can only determine the next selection if we have a valid range
98
+ if (DateInput.isSelectionRange(selection) && selection.start && selection.end) {
99
+ // Shift the range forward by the number of days between the start and end dates
100
+ const length = moment(selection.end).diff(selection.start, 'days');
101
+ return {
102
+ start: moment(selection.end).add(1, 'days').toDate(),
103
+ end: moment(selection.end).add(1 + length, 'days').toDate()
104
+ };
105
+ }
106
+ // This shouldn't happen since the next stepper is disabled if the selection only has one date.
107
+ // If it does, just return the selection as-is.
108
+ return selection;
109
+ }
110
+ getPreviousSelection(selection) {
111
+ // We can only determine the previous selection if we have a valid range
112
+ if (DateInput.isSelectionRange(selection) && selection.start && selection.end) {
113
+ // Shift the range back by the number of days between the start and end dates
114
+ const length = moment(selection.end).diff(selection.start, 'days');
115
+ return {
116
+ start: moment(selection.start).subtract(1 + length, 'days').toDate(),
117
+ end: moment(selection.start).subtract(1, 'days').toDate()
118
+ };
119
+ }
120
+ // This shouldn't happen since the previous stepper is disabled if the selection only has one date.
121
+ // If it does, just return the selection as-is.
122
+ return selection;
123
+ }
124
+ getPrimaryCalendarView(selection, minDate, maxDate, currentView) {
125
+ let viewDate;
126
+ if (DateInput.isSelectionRange(selection)) {
127
+ let start = selection.start ? moment(selection.start) : null;
128
+ let end = selection.end ? moment(selection.end) : null;
129
+ // Since we're in range mode, the primary calendar is mostly concerned with the start date
130
+ if (start) {
131
+ // If we have a start date in the selection and it's before the min date, shift the view to the min date.
132
+ if (start.isBefore(minDate, 'month')) {
133
+ viewDate = minDate;
134
+ // If the start is after the max date, shift the view to the max date.
135
+ }
136
+ else if (start.isSameOrAfter(maxDate, 'month')) {
137
+ // the secondary calendar will show the max date month.
138
+ // The primary calendar will show the month before the max date.
139
+ viewDate = moment(maxDate).subtract(1, 'month').toDate();
140
+ }
141
+ else {
142
+ viewDate = start.toDate();
143
+ }
144
+ }
145
+ else if (end && end.isSameOrBefore(minDate, 'month')) {
146
+ // If we don't have a start date, and the end date is the same month or before the min date, shift the view to the min date.
147
+ viewDate = minDate;
148
+ }
149
+ else if (end && end.isSameOrBefore(new Date(), 'month')) {
150
+ // If we don't have a start date, and the end date is the same month or before the current month, shift the view to the month before the end
151
+ viewDate = moment(end).subtract(1, 'month').toDate();
152
+ }
153
+ else {
154
+ // Otherwise, show the current month.
155
+ viewDate = new Date();
156
+ }
157
+ }
158
+ else {
159
+ viewDate = selection || new Date();
160
+ }
161
+ // If the current calendar view mode is valid for the selection mode, use it, otherwise use the selection view mode
162
+ // This prevents the calendar from changing view modes as the user types/blurs the textboxes as long as the view mode is valid
163
+ const viewMode = this.validViewModes.includes(currentView.mode) ? currentView.mode : this.selectionViewMode;
164
+ return { mode: viewMode, date: viewDate };
165
+ }
166
+ getSecondaryCalendarView(selection, minDate, maxDate, currentView) {
167
+ let viewDate;
168
+ if (DateInput.isSelectionRange(selection)) {
169
+ let start = selection.start ? moment(selection.start) : null;
170
+ let end = selection.end ? moment(selection.end) : null;
171
+ // The secondary calendar is primarily concerned with the end date in the selection.
172
+ if (end) {
173
+ if (end.isAfter(maxDate, 'month')) {
174
+ // If the end date is after the max date, shift the view to the max date.
175
+ viewDate = maxDate;
176
+ }
177
+ else if (end.isSameOrBefore(minDate, 'month')) {
178
+ // If the end date is the same month or before the min date, shift the view to show
179
+ // the month after the min date. The primary calendar will show the min date month.
180
+ viewDate = moment(minDate).add(1, 'month').toDate();
181
+ }
182
+ else {
183
+ // Otherwise, show the end date month.
184
+ viewDate = end.toDate();
185
+ }
186
+ }
187
+ else if (start && start.isSameOrAfter(maxDate, 'month')) {
188
+ // If we don't have an end date, and the start date is after the max date, show the max date month
189
+ viewDate = maxDate;
190
+ }
191
+ else if (start && start.isSameOrAfter(new Date(), 'month')) {
192
+ // If we don't have an end date, and the start date is on or after the current month,
193
+ // show the month after the start date.
194
+ viewDate = moment(start).add(1, 'month').toDate();
195
+ }
196
+ else {
197
+ // Otherwise, show the current month.
198
+ viewDate = new Date();
199
+ }
200
+ }
201
+ else {
202
+ viewDate = selection || new Date();
203
+ }
204
+ // If the current calendar view mode is valid for the selection mode, use it, otherwise use the selection view mode
205
+ // This prevents the calendar from changing view modes as the user types/blurs the textboxes as long as the view mode is valid
206
+ const viewMode = this.validViewModes.includes(currentView.mode) ? currentView.mode : this.selectionViewMode;
207
+ return { mode: viewMode, date: viewDate };
208
+ }
209
+ }
210
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"range-selection-strategy.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/controls/date-input/date-input-selection-strategies/range-selection-strategy.ts"],"names":[],"mappings":"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","sourcesContent":["import moment from \"moment\";\r\nimport { Calendar } from \"../../calendar/calendar.types\";\r\nimport { DateInput } from \"../date-input.types\";\r\nimport { DateInputSelectionStrategyBase } from \"./date-input-selection-strategy-base\";\r\n\r\n/**\r\n * Selection strategy for the 'range' selection mode.\r\n */\r\nexport class RangeSelectionStrategy extends DateInputSelectionStrategyBase {\r\n  public readonly selectionViewMode: Calendar.ViewMode = 'day';\r\n  public readonly validViewModes: Calendar.ViewMode[] = ['day', 'month', 'year'];\r\n  public readonly showSecondaryTextbox = true;\r\n  public readonly showSecondaryCalendar = true;\r\n\r\n  formatSelection(selection?: DateInput.Selection | null): DateInput.TextboxGroup['value'] {\r\n    if (!selection) {\r\n      return { textbox: null, textbox2: null };\r\n    }\r\n\r\n    // We use the 'll' moment medium format here to make sure the formatted date is formatted according to the user's locale\r\n    if (DateInput.isSelectionSingleDate(selection)) {\r\n      return { textbox: moment(selection).format('ll'), textbox2: null };\r\n    } else {\r\n      const start = selection.start ? moment(selection.start).format('ll') : null;\r\n      const end = selection.end ? moment(selection.end).format('ll') : null;\r\n\r\n      return { textbox: start, textbox2: end };\r\n    }\r\n  }\r\n\r\n  parseTextboxValues(value: DateInput.TextboxGroup['value'], parseFormats: string[], opts?: DateInput.ParsingOptions): DateInput.Selection | null {\r\n    let start = this.parseString(value.textbox, parseFormats, opts);\r\n    let end = this.parseString(value.textbox2, parseFormats, opts);\r\n\r\n    // If the start date is after the end date, we need to clear out the invalid date. This is different from the last 7/28 days strategies\r\n    // because we don't have a fixed range to automatically adjust the dates to. If the user is changing the begin date or end date and ends\r\n    // up crossing the other date, we should clear out the date that's not being changed so that the user can enter a valid range using the\r\n    // begin/end date they just entered.\r\n    if (start && end && moment(start).isAfter(end)) {\r\n      if (opts?.parseFromEnd) {\r\n        start = null;\r\n      } else {\r\n        end = null;\r\n      }\r\n    }\r\n\r\n    if (!start && !end) {\r\n      return null;\r\n    } else {\r\n      return { start, end };\r\n    }\r\n  }\r\n\r\n  getSelectionFromDate(date: Date, existingSelection: DateInput.Selection | null): DateInput.Selection | null {\r\n    if (DateInput.isSelectionRange(existingSelection)) {\r\n      const { start, end } = existingSelection;\r\n      if (start && !end) {\r\n        // If the selected date is before the existing start date, set it as the new start date and use the existing start date as the end date\r\n        return moment(date).isBefore(start) ? { start: date, end: start } : { start, end: date };\r\n      }\r\n      if (end && !start) {\r\n        // If the selected date is after the existing end date, set it as the new end date and use the existing end date as the start date\r\n        return moment(date).isBefore(end) ? { start: date, end } : { start: end, end: date };\r\n      }\r\n    } else if (DateInput.isSelectionSingleDate(existingSelection)) {\r\n      // If the selected date is before the existing date, set it as the new start date and use the existing date as the end date\r\n      return moment(date).isBefore(existingSelection) ? { start: date, end: existingSelection } : { start: existingSelection, end: date };\r\n    }\r\n\r\n    // If no selection exists or we already have a complete selection, start a new one with the selected date as the start date.\r\n    return { start: date, end: null };\r\n  }\r\n\r\n  getNewSelectionFromExisting(previousSelection?: DateInput.Selection): DateInput.Selection | null {\r\n    if (DateInput.isSelectionRange(previousSelection)) {\r\n      // If we have a range, just return it as is. We don't have a fixed range to adjust the dates to.\r\n      // Even if it's incomplete, the user will still be able to select the remaining date.\r\n      return previousSelection;\r\n    } else if (DateInput.isSelectionSingleDate(previousSelection)) {\r\n      // If we have a single date selection, use it as the start date for the new one.\r\n      return { start: previousSelection, end: null };\r\n    }\r\n\r\n    return null;\r\n  }\r\n\r\n  getSelectionForQuickSelectDate(date: Date, existingSelection?: DateInput.Selection | null): DateInput.Selection | null {\r\n    // If we have a valid range, we'll shift the range to the date provided, using it as the end date\r\n    // The new selection will have the same length as the previous.\r\n    if (DateInput.isSelectionRange(existingSelection) && existingSelection.start && existingSelection.end) {\r\n      const length = moment(existingSelection.end).diff(existingSelection.start, 'days');\r\n      return {\r\n        start: moment(date).subtract(length, 'days').toDate(),\r\n        end: date\r\n      };\r\n    }\r\n\r\n    // If we only have a single date, we'll use the date provided as the end date.\r\n    // The user will be able to select the start date.\r\n    return { start: null, end: date };\r\n  }\r\n\r\n  getNextSelection(selection: DateInput.Selection | null): DateInput.Selection | null {\r\n    // We can only determine the next selection if we have a valid range\r\n    if (DateInput.isSelectionRange(selection) && selection.start && selection.end) {\r\n      // Shift the range forward by the number of days between the start and end dates\r\n      const length = moment(selection.end).diff(selection.start, 'days');\r\n      return {\r\n        start: moment(selection.end).add(1, 'days').toDate(),\r\n        end: moment(selection.end).add(1 + length, 'days').toDate()\r\n      };\r\n    }\r\n\r\n    // This shouldn't happen since the next stepper is disabled if the selection only has one date.\r\n    // If it does, just return the selection as-is.\r\n    return selection;\r\n  }\r\n\r\n  getPreviousSelection(selection: DateInput.Selection | null): DateInput.Selection | null {\r\n    // We can only determine the previous selection if we have a valid range\r\n    if (DateInput.isSelectionRange(selection) && selection.start && selection.end) {\r\n      // Shift the range back by the number of days between the start and end dates\r\n      const length = moment(selection.end).diff(selection.start, 'days');\r\n      return {\r\n        start: moment(selection.start).subtract(1 + length, 'days').toDate(),\r\n        end: moment(selection.start).subtract(1, 'days').toDate()\r\n      };\r\n    }\r\n\r\n    // This shouldn't happen since the previous stepper is disabled if the selection only has one date.\r\n    // If it does, just return the selection as-is.\r\n    return selection;\r\n  }\r\n\r\n  getPrimaryCalendarView(selection: DateInput.Selection | null, minDate: Date, maxDate: Date, currentView: Calendar.View): Calendar.View {\r\n    let viewDate: Date;\r\n    if (DateInput.isSelectionRange(selection)) {\r\n      let start = selection.start ? moment(selection.start) : null;\r\n      let end = selection.end ? moment(selection.end) : null;\r\n\r\n      // Since we're in range mode, the primary calendar is mostly concerned with the start date\r\n      if (start) {\r\n        // If we have a start date in the selection and it's before the min date, shift the view to the min date.\r\n        if (start.isBefore(minDate, 'month')) {\r\n          viewDate = minDate;\r\n          // If the start is after the max date, shift the view to the max date.\r\n        } else if (start.isSameOrAfter(maxDate, 'month')) {\r\n          // the secondary calendar will show the max date month.\r\n          // The primary calendar will show the month before the max date.\r\n          viewDate = moment(maxDate).subtract(1, 'month').toDate();\r\n        } else {\r\n          viewDate = start.toDate();\r\n        }\r\n      } else if (end && end.isSameOrBefore(minDate, 'month')) {\r\n        // If we don't have a start date, and the end date is the same month or before the min date, shift the view to the min date.\r\n        viewDate = minDate;\r\n      } else if (end && end.isSameOrBefore(new Date(), 'month')) {\r\n        // If we don't have a start date, and the end date is the same month or before the current month, shift the view to the month before the end\r\n        viewDate = moment(end).subtract(1, 'month').toDate();\r\n      } else {\r\n        // Otherwise, show the current month.\r\n        viewDate = new Date();\r\n      }\r\n    } else {\r\n      viewDate = selection || new Date();\r\n    }\r\n\r\n    // If the current calendar view mode is valid for the selection mode, use it, otherwise use the selection view mode\r\n    // This prevents the calendar from changing view modes as the user types/blurs the textboxes as long as the view mode is valid\r\n    const viewMode = this.validViewModes.includes(currentView.mode) ? currentView.mode : this.selectionViewMode;\r\n\r\n    return { mode: viewMode, date: viewDate };\r\n  }\r\n\r\n  getSecondaryCalendarView(selection: DateInput.Selection | null, minDate: Date, maxDate: Date, currentView: Calendar.View): Calendar.View {\r\n    let viewDate: Date;\r\n\r\n    if (DateInput.isSelectionRange(selection)) {\r\n      let start = selection.start ? moment(selection.start) : null;\r\n      let end = selection.end ? moment(selection.end) : null;\r\n\r\n      // The secondary calendar is primarily concerned with the end date in the selection.\r\n      if (end) {\r\n        if (end.isAfter(maxDate, 'month')) {\r\n          // If the end date is after the max date, shift the view to the max date.\r\n          viewDate = maxDate;\r\n        } else if (end.isSameOrBefore(minDate, 'month')) {\r\n          // If the end date is the same month or before the min date, shift the view to show\r\n          // the month after the min date. The primary calendar will show the min date month.\r\n          viewDate = moment(minDate).add(1, 'month').toDate();\r\n        } else {\r\n          // Otherwise, show the end date month.\r\n          viewDate = end.toDate();\r\n        }\r\n      } else if (start && start.isSameOrAfter(maxDate, 'month')) {\r\n        // If we don't have an end date, and the start date is after the max date, show the max date month\r\n        viewDate = maxDate;\r\n      } else if (start && start.isSameOrAfter(new Date(), 'month')) {\r\n        // If we don't have an end date, and the start date is on or after the current month,\r\n        // show the month after the start date.\r\n        viewDate = moment(start).add(1, 'month').toDate();\r\n      } else {\r\n        // Otherwise, show the current month.\r\n        viewDate = new Date();\r\n      }\r\n    } else {\r\n      viewDate = selection || new Date();\r\n    }\r\n\r\n    // If the current calendar view mode is valid for the selection mode, use it, otherwise use the selection view mode\r\n    // This prevents the calendar from changing view modes as the user types/blurs the textboxes as long as the view mode is valid\r\n    const viewMode = this.validViewModes.includes(currentView.mode) ? currentView.mode : this.selectionViewMode;\r\n\r\n    return { mode: viewMode, date: viewDate };\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,81 @@
1
+ import moment from "moment";
2
+ import { DateInput } from "../date-input.types";
3
+ import { DateInputSelectionStrategyBase } from "./date-input-selection-strategy-base";
4
+ /**
5
+ * Selection strategy for the 'year' selection mode.
6
+ */
7
+ export class YearSelectionStrategy extends DateInputSelectionStrategyBase {
8
+ constructor() {
9
+ super(...arguments);
10
+ this.selectionViewMode = 'year';
11
+ this.validViewModes = ['year'];
12
+ this.showSecondaryTextbox = false;
13
+ this.showSecondaryCalendar = false;
14
+ }
15
+ formatSelection(selection) {
16
+ if (!selection) {
17
+ return { textbox: null, textbox2: null };
18
+ }
19
+ // Only one date is needed for formatting since the ranges span a single year.
20
+ // There shouldn't be any case where there's only a single date but it's covered anyway.
21
+ const date = DateInput.isSelectionRange(selection) ? selection.start : selection;
22
+ return { textbox: moment(date).format('YYYY'), textbox2: null };
23
+ }
24
+ parseString(value, parseFormats, opts) {
25
+ // Including 'YY' and 'YYYY' as valid formats for year input. These aren't included in the default parse
26
+ // formats because they cause issues with the default date parsing.
27
+ return super.parseString(value, ['YY', 'YYYY', ...parseFormats], opts);
28
+ }
29
+ parseTextboxValues(value, parseFormats, opts) {
30
+ // Only a single textbox is used in year mode
31
+ const date = this.parseString(value.textbox, parseFormats, opts);
32
+ if (!date) {
33
+ return null;
34
+ }
35
+ return this.getSelectionFromDate(date);
36
+ }
37
+ getSelectionFromDate(date) {
38
+ return {
39
+ start: moment(date).startOf('year').toDate(),
40
+ end: moment(date).endOf('year').toDate()
41
+ };
42
+ }
43
+ getNewSelectionFromExisting(previousSelection) {
44
+ if (DateInput.isSelectionRange(previousSelection)) {
45
+ // Base the year range off of the existing selection's end date, falling back to the start date
46
+ const date = previousSelection.end || previousSelection.start;
47
+ if (date) {
48
+ return this.getSelectionFromDate(date);
49
+ }
50
+ }
51
+ else if (DateInput.isSelectionSingleDate(previousSelection)) {
52
+ // If the previous selection was a single date, make the range that year
53
+ return this.getSelectionFromDate(previousSelection);
54
+ }
55
+ return null;
56
+ }
57
+ getSelectionForQuickSelectDate(date) {
58
+ return this.getSelectionFromDate(date);
59
+ }
60
+ getNextSelection(selection) {
61
+ // Move the selection to the next year
62
+ if (DateInput.isSelectionRange(selection) && selection.start) {
63
+ const date = moment(selection.start).add(1, 'year');
64
+ return this.getSelectionFromDate(date.toDate());
65
+ }
66
+ // This shouldn't happen since the next stepper is disabled if the selection only has one date.
67
+ // If it does, just return the selection as-is.
68
+ return selection;
69
+ }
70
+ getPreviousSelection(selection) {
71
+ // Move the selection to the previous year
72
+ if (DateInput.isSelectionRange(selection) && selection.start) {
73
+ const date = moment(selection.start).subtract(1, 'year');
74
+ return this.getSelectionFromDate(date.toDate());
75
+ }
76
+ // This shouldn't happen since the previous stepper is disabled if the selection only has one date.
77
+ // If it does, just return the selection as-is.
78
+ return selection;
79
+ }
80
+ }
81
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"year-selection-strategy.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/controls/date-input/date-input-selection-strategies/year-selection-strategy.ts"],"names":[],"mappings":"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","sourcesContent":["import moment from \"moment\";\r\nimport { Calendar } from \"../../calendar/calendar.types\";\r\nimport { DateInput } from \"../date-input.types\";\r\nimport { DateInputSelectionStrategyBase } from \"./date-input-selection-strategy-base\";\r\n\r\n/**\r\n * Selection strategy for the 'year' selection mode.\r\n */\r\nexport class YearSelectionStrategy extends DateInputSelectionStrategyBase {\r\n  public readonly selectionViewMode: Calendar.ViewMode = 'year';\r\n  public readonly validViewModes: Calendar.ViewMode[] = ['year'];\r\n  public readonly showSecondaryTextbox = false;\r\n  public readonly showSecondaryCalendar = false;\r\n\r\n  formatSelection(selection?: DateInput.Selection | null): DateInput.TextboxGroup['value'] {\r\n    if (!selection) {\r\n      return { textbox: null, textbox2: null };\r\n    }\r\n\r\n    // Only one date is needed for formatting since the ranges span a single year.\r\n    // There shouldn't be any case where there's only a single date but it's covered anyway.\r\n    const date = DateInput.isSelectionRange(selection) ? selection.start : selection;\r\n    return { textbox: moment(date).format('YYYY'), textbox2: null };\r\n  }\r\n\r\n  parseString(value: string | null | undefined, parseFormats: string[], opts?: DateInput.ParsingOptions): Date | null {\r\n    // Including 'YY' and 'YYYY' as valid formats for year input. These aren't included in the default parse\r\n    // formats because they cause issues with the default date parsing.\r\n    return super.parseString(value, ['YY', 'YYYY', ...parseFormats], opts);\r\n  }\r\n\r\n  parseTextboxValues(value: DateInput.TextboxGroup['value'], parseFormats: string[], opts?: DateInput.ParsingOptions): DateInput.Selection | null {\r\n    // Only a single textbox is used in year mode\r\n    const date = this.parseString(value.textbox, parseFormats, opts);\r\n\r\n    if (!date) {\r\n      return null;\r\n    }\r\n\r\n    return this.getSelectionFromDate(date);\r\n  }\r\n\r\n  getSelectionFromDate(date: Date): DateInput.Selection | null {\r\n    return {\r\n      start: moment(date).startOf('year').toDate(),\r\n      end: moment(date).endOf('year').toDate()\r\n    };\r\n  }\r\n\r\n  getNewSelectionFromExisting(previousSelection?: DateInput.Selection): DateInput.Selection | null {\r\n    if (DateInput.isSelectionRange(previousSelection)) {\r\n      // Base the year range off of the existing selection's end date, falling back to the start date\r\n      const date = previousSelection.end || previousSelection.start;\r\n      if (date) {\r\n        return this.getSelectionFromDate(date);\r\n      }\r\n    } else if (DateInput.isSelectionSingleDate(previousSelection)) {\r\n      // If the previous selection was a single date, make the range that year\r\n      return this.getSelectionFromDate(previousSelection);\r\n    }\r\n\r\n    return null;\r\n  }\r\n\r\n  getSelectionForQuickSelectDate(date: Date): DateInput.Selection | null {\r\n    return this.getSelectionFromDate(date);\r\n  }\r\n\r\n  getNextSelection(selection: DateInput.Selection): DateInput.Selection | null {\r\n    // Move the selection to the next year\r\n    if (DateInput.isSelectionRange(selection) && selection.start) {\r\n      const date = moment(selection.start).add(1, 'year');\r\n      return this.getSelectionFromDate(date.toDate());\r\n    }\r\n\r\n    // This shouldn't happen since the next stepper is disabled if the selection only has one date.\r\n    // If it does, just return the selection as-is.\r\n    return selection;\r\n  }\r\n\r\n  getPreviousSelection(selection: DateInput.Selection): DateInput.Selection | null {\r\n    // Move the selection to the previous year\r\n    if (DateInput.isSelectionRange(selection) && selection.start) {\r\n      const date = moment(selection.start).subtract(1, 'year');\r\n      return this.getSelectionFromDate(date.toDate());\r\n    }\r\n    \r\n    // This shouldn't happen since the previous stepper is disabled if the selection only has one date.\r\n    // If it does, just return the selection as-is.\r\n    return selection;\r\n  }\r\n}\r\n"]}