@skyux/datetime 10.18.0 → 10.20.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,404 +1,355 @@
1
- import { ChangeDetectionStrategy, Component, HostBinding, Input, Optional, booleanAttribute, forwardRef, inject, } from '@angular/core';
2
- import { NG_VALIDATORS, NG_VALUE_ACCESSOR, UntypedFormControl, } from '@angular/forms';
3
- import { SkyAppFormat } from '@skyux/core';
4
- import { SkyFormFieldLabelTextRequiredService } from '@skyux/forms';
5
- import { SkyLibResourcesService } from '@skyux/i18n';
6
- import { Subject, combineLatest } from 'rxjs';
7
- import { distinctUntilChanged, first, takeUntil } from 'rxjs/operators';
8
- import { SkyDateFormatter } from '../datepicker/date-formatter';
9
- import { SkyDateRangeCalculatorId } from './types/date-range-calculator-id';
1
+ import { CommonModule } from '@angular/common';
2
+ import { ChangeDetectionStrategy, ChangeDetectorRef, Component, HostBinding, Injector, Input, booleanAttribute, inject, } from '@angular/core';
3
+ import { FormBuilder, FormControl, FormsModule, NG_VALIDATORS, NG_VALUE_ACCESSOR, NgControl, ReactiveFormsModule, Validators, } from '@angular/forms';
4
+ import { SkyFormFieldLabelTextRequiredService, SkyInputBoxModule, } from '@skyux/forms';
5
+ import { Subject, distinctUntilChanged, merge, takeUntil } from 'rxjs';
6
+ import { SkyDatepickerModule } from '../datepicker/datepicker.module';
7
+ import { SkyDatetimeResourcesModule } from '../shared/sky-datetime-resources.module';
8
+ import { SkyDateRangePickerEndDateResourceKeyPipe } from './date-range-picker-end-date-resource-key.pipe';
9
+ import { SkyDateRangePickerStartDateResourceKeyPipe } from './date-range-picker-start-date-resource-key.pipe';
10
+ import { SkyDateRangeService } from './date-range.service';
10
11
  import { SkyDateRangeCalculatorType } from './types/date-range-calculator-type';
12
+ import { SKY_DEFAULT_CALCULATOR_IDS } from './types/date-range-default-calculator-configs';
11
13
  import * as i0 from "@angular/core";
12
- import * as i1 from "./date-range.service";
14
+ import * as i1 from "@angular/common";
13
15
  import * as i2 from "@angular/forms";
14
- import * as i3 from "@skyux/i18n";
15
- import * as i4 from "@skyux/theme";
16
- import * as i5 from "@angular/common";
17
- import * as i6 from "../datepicker/datepicker.component";
18
- import * as i7 from "../datepicker/datepicker-input.directive";
19
- import * as i8 from "@skyux/forms";
20
- import * as i9 from "./date-range-picker-end-date-resource-key.pipe";
21
- import * as i10 from "./date-range-picker-start-date-resource-key.pipe";
22
- const SKY_DATE_RANGE_PICKER_VALUE_ACCESSOR = {
23
- provide: NG_VALUE_ACCESSOR,
24
- useExisting: forwardRef(() => SkyDateRangePickerComponent),
25
- multi: true,
26
- };
27
- const SKY_DATE_RANGE_PICKER_VALIDATOR = {
28
- provide: NG_VALIDATORS,
29
- useExisting: forwardRef(() => SkyDateRangePickerComponent),
30
- multi: true,
31
- };
32
- let uniqueId = 0;
33
- /**
34
- * Acts as a form control with a form model of type `SkyDateRangeCalculation`.
35
- * @example
36
- * ```
37
- * <sky-date-range-picker
38
- * formControlName="myPicker"
39
- * >
40
- * </sky-date-range-picker>
41
- * ```
42
- */
16
+ import * as i3 from "../datepicker/datepicker.component";
17
+ import * as i4 from "../datepicker/datepicker-input.directive";
18
+ import * as i5 from "@skyux/i18n";
19
+ import * as i6 from "@skyux/forms";
20
+ function areDateRangesEqual(rangeA, rangeB) {
21
+ return (!!rangeA &&
22
+ !!rangeB &&
23
+ rangeA.calculatorId === rangeB.calculatorId &&
24
+ areDatesEqual(rangeA.startDate, rangeB.startDate) &&
25
+ areDatesEqual(rangeA.endDate, rangeB.endDate));
26
+ }
27
+ function areDatesEqual(a, b) {
28
+ if (typeof a !== typeof b) {
29
+ return false;
30
+ }
31
+ if (!a && !b) {
32
+ return true;
33
+ }
34
+ if (typeof a === 'string' && a === b) {
35
+ return true;
36
+ }
37
+ return a instanceof Date && b instanceof Date && a.getTime() === b.getTime();
38
+ }
39
+ function isNullOrUndefined(value) {
40
+ return value === undefined || value === null;
41
+ }
42
+ function isPartialValue(value) {
43
+ return (isNullOrUndefined(value) ||
44
+ isNullOrUndefined(value.calculatorId) ||
45
+ !('endDate' in value) ||
46
+ !('startDate' in value));
47
+ }
43
48
  export class SkyDateRangePickerComponent {
44
49
  /**
45
50
  * IDs for the date range options to include in the picker's dropdown.
46
51
  * The options specify calculator objects that return two `Date` objects to represent date ranges.
47
52
  * By default, this property includes all `SkyDateRangeCalculatorId` values.
48
53
  */
49
- set calculatorIds(value) {
50
- this.#_calculatorIds = value || [
51
- SkyDateRangeCalculatorId.AnyTime,
52
- SkyDateRangeCalculatorId.Before,
53
- SkyDateRangeCalculatorId.After,
54
- SkyDateRangeCalculatorId.SpecificRange,
55
- SkyDateRangeCalculatorId.Yesterday,
56
- SkyDateRangeCalculatorId.Today,
57
- SkyDateRangeCalculatorId.Tomorrow,
58
- SkyDateRangeCalculatorId.LastWeek,
59
- SkyDateRangeCalculatorId.ThisWeek,
60
- SkyDateRangeCalculatorId.NextWeek,
61
- SkyDateRangeCalculatorId.LastMonth,
62
- SkyDateRangeCalculatorId.ThisMonth,
63
- SkyDateRangeCalculatorId.NextMonth,
64
- SkyDateRangeCalculatorId.LastQuarter,
65
- SkyDateRangeCalculatorId.ThisQuarter,
66
- SkyDateRangeCalculatorId.NextQuarter,
67
- SkyDateRangeCalculatorId.LastCalendarYear,
68
- SkyDateRangeCalculatorId.ThisCalendarYear,
69
- SkyDateRangeCalculatorId.NextCalendarYear,
70
- SkyDateRangeCalculatorId.LastFiscalYear,
71
- SkyDateRangeCalculatorId.ThisFiscalYear,
72
- SkyDateRangeCalculatorId.NextFiscalYear,
73
- ];
54
+ set calculatorIds(calculatorIds) {
55
+ const currentCalculatorId = this.#getValue().calculatorId;
56
+ this.#_calculatorIds = calculatorIds ?? SKY_DEFAULT_CALCULATOR_IDS;
57
+ this.calculators = this.#dateRangeSvc.filterCalculators(this.#_calculatorIds);
58
+ // If the currently selected calculator isn't available anymore,
59
+ // select the first calculator in the new array.
60
+ if (!this.#_calculatorIds.includes(currentCalculatorId)) {
61
+ this.#setValue({ calculatorId: this.calculatorIds[0] }, { emitEvent: true });
62
+ }
74
63
  }
75
64
  get calculatorIds() {
76
65
  return this.#_calculatorIds;
77
66
  }
78
67
  /**
79
- * The date format for
80
- * [the `sky-datepicker` components](https://developer.blackbaud.com/skyux/components/datepicker)
81
- * that make up the date range picker. The text input is a composite component of
82
- * up to two `sky-datepicker` components.
83
- * @default "MM/DD/YYYY"
84
- */
85
- set dateFormat(value) {
86
- this.#_dateFormat = value;
87
- this.dateFormatOrDefault = value || this.#preferredShortDateFormat;
88
- }
89
- get dateFormat() {
90
- return this.#_dateFormat;
91
- }
92
- /**
93
- * Whether to disable the date range picker on template-driven forms. Don't use this input on reactive forms because they may overwrite the input or leave the control out of sync.
94
- * To set the disabled state on reactive forms, use the `FormControl` instead.
68
+ * Whether to disable the date range picker on template-driven forms. Don't use
69
+ * this input on reactive forms because they may overwrite the input or leave
70
+ * the control out of sync. To set the disabled state on reactive forms,
71
+ * use the `FormControl` instead.
95
72
  * @default false
96
73
  */
97
74
  set disabled(value) {
98
- this.#_disabled = value;
99
- if (this.formGroup) {
100
- if (this.#_disabled) {
101
- this.formGroup.disable();
102
- }
103
- else {
104
- this.formGroup.enable();
105
- }
75
+ if (value) {
76
+ this.formGroup.disable();
77
+ }
78
+ else {
79
+ this.formGroup.enable();
106
80
  }
107
- this.#changeDetector.markForCheck();
108
- }
109
- get disabled() {
110
- return this.#_disabled;
111
81
  }
112
82
  get #calculatorIdControl() {
113
- return this.formGroup?.get('calculatorId');
114
- }
115
- get #defaultCalculator() {
116
- return this.calculators[0];
117
- }
118
- get #defaultValue() {
119
- return this.#defaultCalculator?.getValue();
83
+ return this.formGroup.get('calculatorId');
120
84
  }
121
85
  get #endDateControl() {
122
- return this.formGroup?.get('endDate');
86
+ return this.formGroup.get('endDate');
123
87
  }
124
88
  get #startDateControl() {
125
- return this.formGroup?.get('startDate');
126
- }
127
- #value;
128
- set #valueOrDefault(value) {
129
- this.#_valueOrDefault = value;
130
- this.#updateSelectedCalculator();
131
- }
132
- get #valueOrDefault() {
133
- return this.#_valueOrDefault;
89
+ return this.formGroup.get('startDate');
134
90
  }
135
- #labelTextRequired;
136
- #appFormatter;
137
- #control;
138
- #preferredShortDateFormat;
139
- #ngUnsubscribe;
140
- #resourceSvc;
141
91
  #_calculatorIds;
142
- #_dateFormat;
143
- #_disabled;
144
- #_valueOrDefault;
92
+ #_value;
93
+ #hostControl;
94
+ #ngUnsubscribe;
95
+ #notifyChange;
96
+ #notifyTouched;
145
97
  #changeDetector;
146
- #dateRangeService;
147
- #formBuilder;
148
- #localeProvider;
149
- #ngZone;
150
- constructor(changeDetector, dateRangeService, formBuilder, localeProvider, ngZone, themeSvc) {
151
- /**
152
- * Whether to require users to specify a start date.
153
- * @default false
154
- */
155
- this.startDateRequired = false;
98
+ #dateRangeSvc;
99
+ #injector;
100
+ #labelTextRequiredSvc;
101
+ constructor() {
156
102
  /**
157
103
  * Whether to require users to specify a end date.
158
- * @default false
104
+ * @deprecated Use the `required` directive or Angular's `Validators.required`
105
+ * on the form control to mark the date range picker as required.
159
106
  */
160
107
  this.endDateRequired = false;
108
+ /**
109
+ * Whether the date range picker requires a value.
110
+ */
111
+ this.required = false;
161
112
  /**
162
113
  * Whether the date range picker is stacked on another form component. When specified, the appropriate
163
114
  * vertical spacing is automatically added to the date range picker.
164
115
  * @preview
165
116
  */
166
117
  this.stacked = false;
167
- this.dateRangePickerId = `sky-date-range-picker-${uniqueId++}`;
118
+ /**
119
+ * Whether to require users to specify a start date.
120
+ * @deprecated Use the `required` directive or Angular's `Validators.required`
121
+ * on the form control to mark the date range picker as required.
122
+ */
123
+ this.startDateRequired = false;
168
124
  this.calculators = [];
169
- this.isReady = false;
125
+ this.hasErrors = false;
170
126
  this.showEndDatePicker = false;
171
127
  this.showStartDatePicker = false;
172
- this.#labelTextRequired = inject(SkyFormFieldLabelTextRequiredService, {
173
- optional: true,
174
- });
175
- this.#appFormatter = inject(SkyAppFormat);
128
+ this.#_calculatorIds = SKY_DEFAULT_CALCULATOR_IDS;
176
129
  this.#ngUnsubscribe = new Subject();
177
- this.#resourceSvc = inject(SkyLibResourcesService);
178
- this.#_calculatorIds = [
179
- SkyDateRangeCalculatorId.AnyTime,
180
- SkyDateRangeCalculatorId.Before,
181
- SkyDateRangeCalculatorId.After,
182
- SkyDateRangeCalculatorId.SpecificRange,
183
- SkyDateRangeCalculatorId.Yesterday,
184
- SkyDateRangeCalculatorId.Today,
185
- SkyDateRangeCalculatorId.Tomorrow,
186
- SkyDateRangeCalculatorId.LastWeek,
187
- SkyDateRangeCalculatorId.ThisWeek,
188
- SkyDateRangeCalculatorId.NextWeek,
189
- SkyDateRangeCalculatorId.LastMonth,
190
- SkyDateRangeCalculatorId.ThisMonth,
191
- SkyDateRangeCalculatorId.NextMonth,
192
- SkyDateRangeCalculatorId.LastQuarter,
193
- SkyDateRangeCalculatorId.ThisQuarter,
194
- SkyDateRangeCalculatorId.NextQuarter,
195
- SkyDateRangeCalculatorId.LastCalendarYear,
196
- SkyDateRangeCalculatorId.ThisCalendarYear,
197
- SkyDateRangeCalculatorId.NextCalendarYear,
198
- SkyDateRangeCalculatorId.LastFiscalYear,
199
- SkyDateRangeCalculatorId.ThisFiscalYear,
200
- SkyDateRangeCalculatorId.NextFiscalYear,
201
- ];
202
- this.#_disabled = false;
203
- /* istanbul ignore next */
204
- // eslint-disable-next-line @typescript-eslint/no-empty-function, @typescript-eslint/no-unused-vars
205
- this.#onChange = (_) => { };
206
- /* istanbul ignore next */
207
- // eslint-disable-next-line @typescript-eslint/no-empty-function
208
- this.#onTouched = () => { };
209
- this.#changeDetector = changeDetector;
210
- this.#dateRangeService = dateRangeService;
211
- this.#formBuilder = formBuilder;
212
- this.#localeProvider = localeProvider;
213
- this.#ngZone = ngZone;
214
- this.dateFormatOrDefault = this.dateFormat;
215
- this.#localeProvider
216
- .getLocaleInfo()
217
- .pipe(takeUntil(this.#ngUnsubscribe))
218
- .subscribe((localeInfo) => {
219
- SkyDateFormatter.setLocale(localeInfo.locale);
220
- this.#preferredShortDateFormat =
221
- SkyDateFormatter.getPreferredShortDateFormat();
222
- this.dateFormatOrDefault =
223
- this.dateFormat || this.#preferredShortDateFormat;
130
+ this.#changeDetector = inject(ChangeDetectorRef);
131
+ this.#dateRangeSvc = inject(SkyDateRangeService);
132
+ this.#injector = inject(Injector);
133
+ this.#labelTextRequiredSvc = inject(SkyFormFieldLabelTextRequiredService, {
134
+ optional: true,
224
135
  });
225
- // Update icons when theme changes.
226
- /* istanbul ignore next */
227
- themeSvc?.settingsChange
228
- .pipe(takeUntil(this.#ngUnsubscribe))
229
- .subscribe(() => {
230
- this.#changeDetector.markForCheck();
136
+ this.calculators = this.#dateRangeSvc.calculators;
137
+ this.selectedCalculator = this.calculators[0];
138
+ const initialValue = this.#getDefaultValue(this.selectedCalculator);
139
+ this.#_value = initialValue;
140
+ this.formGroup = inject(FormBuilder).group({
141
+ calculatorId: new FormControl(initialValue.calculatorId),
142
+ startDate: new FormControl(initialValue.startDate),
143
+ endDate: new FormControl(initialValue.endDate),
231
144
  });
232
145
  }
233
146
  ngOnInit() {
234
- this.#createForm();
235
- this.#updateCalculators().then(() => {
236
- if (!this.#value || !this.#value.calculatorId) {
237
- this.#valueOrDefault = this.#defaultValue;
147
+ if (this.#labelTextRequiredSvc) {
148
+ this.#labelTextRequiredSvc.validateLabelText(this.label);
149
+ if (!this.label) {
150
+ this.display = 'none';
238
151
  }
239
- this.#addEventListeners();
240
- this.isReady = true;
241
- this.#showRelevantFormFields();
242
- // We need to let Angular be stable and have rendered the components prior to setting the values and form controls. This ensures all initial validation will be ran correctly.
243
- this.#ngZone.onStable.pipe(first()).subscribe(() => {
244
- // Fill in any unprovided values after the calculators have been initialized.
245
- // For example, if the control is initialized with only the `calculatorId`,
246
- // allow the calculator to fill in the missing start and end dates.
247
- const defaultValue = this.selectedCalculator?.getValue(this.#valueOrDefault?.startDate, this.#valueOrDefault?.endDate);
248
- const newValue = Object.assign({}, defaultValue, this.#valueOrDefault);
249
- this.#setValue(newValue, false);
250
- this.#resetFormGroupValue();
251
- // This is needed to address a bug in Angular 4.
252
- // When a control value is set initially, its value is not represented on the view.
253
- // See: https://github.com/angular/angular/issues/13792
254
- /* istanbul ignore else */
255
- if (this.#control) {
256
- this.#control.setValue(this.#valueOrDefault, {
257
- emitEvent: false,
258
- });
259
- }
260
- });
261
- });
262
- this.#resourceSvc
263
- .getString('skyux_datepicker_format_hint_text')
264
- .pipe(takeUntil(this.#ngUnsubscribe))
265
- .subscribe((templateString) => {
266
- this.hostHintText = this.#appFormatter.formatText(templateString, this.dateFormatOrDefault);
267
- });
268
- if (this.#labelTextRequired && !this.label) {
269
- this.display = 'none';
270
152
  }
271
- this.#labelTextRequired?.validateLabelText(this.label);
272
153
  }
273
- ngOnChanges(changes) {
274
- if (changes['calculatorIds'] &&
275
- changes['calculatorIds'].firstChange === false) {
276
- this.#updateCalculators().then(() => {
277
- const id = this.#calculatorIdControl?.value;
278
- // Maintain the currently selected values if the calculators change after
279
- // a value has been chosen.
280
- const found = this.calculators.find((calculator) => {
281
- return calculator.calculatorId === id;
154
+ ngAfterViewInit() {
155
+ this.#hostControl = this.#injector.get(NgControl, null, {
156
+ optional: true,
157
+ self: true,
158
+ })?.control;
159
+ // Set a default value on the control if it's undefined on init.
160
+ // We need to use setTimeout to avoid interfering with the first
161
+ // validation cycle.
162
+ if (isPartialValue(this.#hostControl?.value)) {
163
+ setTimeout(() => {
164
+ this.#hostControl?.setValue(this.#getValue(), {
165
+ emitEvent: false,
166
+ onlySelf: true,
282
167
  });
283
- /* istanbul ignore else */
284
- if (!found) {
285
- const newValue = this.#defaultCalculator?.getValue();
286
- this.#setValue(newValue);
287
- this.#resetFormGroupValue(newValue);
288
- this.#showRelevantFormFields();
168
+ });
169
+ }
170
+ // Update the view when "required" or "disabled" states are changed on the
171
+ // host control.
172
+ this.#hostControl?.statusChanges
173
+ .pipe(distinctUntilChanged(), takeUntil(this.#ngUnsubscribe))
174
+ .subscribe(() => {
175
+ this.#changeDetector.markForCheck();
176
+ });
177
+ // Start listening for changes after the first change detection cycle.
178
+ setTimeout(() => {
179
+ this.formGroup.valueChanges
180
+ .pipe(distinctUntilChanged(areDateRangesEqual), takeUntil(this.#ngUnsubscribe))
181
+ .subscribe((value) => {
182
+ if (!isNullOrUndefined(value?.calculatorId)) {
183
+ // The select element sets the calculator ID to a string, but we
184
+ // need it to be a number.
185
+ value.calculatorId = +value.calculatorId;
186
+ // If the calculator ID is changed, we need to reset the start and
187
+ // end date values and wait until the next valueChanges event to
188
+ // notify the host control.
189
+ if (value.calculatorId !== this.#getValue().calculatorId) {
190
+ this.#setValue({ calculatorId: value.calculatorId }, { emitEvent: true });
191
+ return;
192
+ }
193
+ }
194
+ this.#setValue(value, { emitEvent: false });
195
+ const newValue = this.#getValue();
196
+ // Update the host control if the value is different.
197
+ if (!areDateRangesEqual(this.#hostControl?.value, newValue)) {
198
+ this.#notifyChange?.(newValue);
289
199
  }
290
200
  });
201
+ });
202
+ // If the datepickers' statuses change, we want to retrigger the host
203
+ // control's validation so that their errors are reflected back to the host.
204
+ merge(this.#startDateControl.statusChanges, this.#endDateControl.statusChanges)
205
+ .pipe(distinctUntilChanged(), takeUntil(this.#ngUnsubscribe))
206
+ .subscribe(() => {
207
+ // Use a setTimeout to avoid an ExpressionChangedAfterChecked error,
208
+ // since multiple calls to updateValueAndValidity in the same
209
+ // cycle may collide with one another.
210
+ setTimeout(() => {
211
+ this.#hostControl?.updateValueAndValidity({
212
+ emitEvent: false,
213
+ onlySelf: true,
214
+ });
215
+ });
216
+ });
217
+ this.#updatePickerVisibility(this.selectedCalculator);
218
+ }
219
+ /**
220
+ * Check for touched status in ngDoCheck since Angular does not (currently)
221
+ * have an API to respond to touched status changes from the host control.
222
+ * @see https://github.com/angular/angular/issues/17736#issuecomment-310812368
223
+ * TODO: Angular 18 introduces a new API to respond to these statuses.
224
+ * @see https://github.com/angular/angular/issues/10887#issuecomment-2035267400
225
+ */
226
+ ngDoCheck() {
227
+ const control = this.#hostControl;
228
+ const touched = this.formGroup.touched;
229
+ if (control) {
230
+ if (control.touched && !touched) {
231
+ this.formGroup.markAllAsTouched();
232
+ this.#changeDetector.markForCheck();
233
+ }
234
+ else if (control.untouched && touched) {
235
+ this.formGroup.markAsUntouched();
236
+ this.#changeDetector.markForCheck();
237
+ }
238
+ this.hasErrors = !!control.errors && (control.touched || control.dirty);
291
239
  }
292
240
  }
293
241
  ngOnDestroy() {
294
242
  this.#ngUnsubscribe.next();
295
243
  this.#ngUnsubscribe.complete();
296
244
  }
297
- onFieldBlur() {
298
- this.#onTouched();
245
+ // Implemented as part of ControlValueAccessor.
246
+ registerOnChange(fn) {
247
+ this.#notifyChange = fn;
299
248
  }
300
- writeValue(value) {
301
- // Only update the underlying controls when the calculators are ready.
302
- const notifyChange = false;
303
- // (We still need to save the initial value set by the consumer's form, however.)
304
- this.#setValue(value, notifyChange);
305
- if (this.isReady) {
306
- // When the control's value is set to `null`,
307
- // set it to the default value.
308
- if (!value) {
309
- this.#onChange(this.#defaultValue);
310
- }
311
- this.#resetFormGroupValue();
312
- this.#showRelevantFormFields();
313
- }
249
+ // Implemented as part of ControlValueAccessor.
250
+ registerOnTouched(fn) {
251
+ this.#notifyTouched = fn;
252
+ }
253
+ // Implemented as part of ControlValueAccessor.
254
+ setDisabledState(isDisabled) {
255
+ this.disabled = isDisabled;
314
256
  }
257
+ // Implemented as part of Validator.
315
258
  validate(control) {
316
- if (!this.#control) {
317
- this.#control = control;
318
- }
319
- if (!this.isReady) {
320
- return null;
321
- }
322
- const value = control.value;
323
- const idControl = this.#calculatorIdControl;
324
- const result = this.selectedCalculator?.validate(value);
325
259
  let errors = null;
326
- if (result) {
260
+ const calculatorErrors = this.selectedCalculator.validate(control.value);
261
+ const startDateErrors = this.#startDateControl.errors;
262
+ const endDateErrors = this.#endDateControl.errors;
263
+ if (calculatorErrors) {
327
264
  errors = {
328
265
  skyDateRange: {
329
- calculatorId: idControl?.value,
330
- errors: result,
266
+ calculatorId: this.#getValue().calculatorId,
267
+ errors: calculatorErrors,
331
268
  },
332
269
  };
333
270
  }
334
- else {
335
- let startErrors = null;
336
- let endErrors = null;
337
- if (this.#startDateControl) {
338
- startErrors = this.#startDateControl.errors;
339
- }
340
- if (this.#endDateControl) {
341
- endErrors = this.#endDateControl.errors;
342
- }
343
- errors = startErrors || endErrors;
271
+ if (this.showStartDatePicker && startDateErrors) {
272
+ errors ||= {};
273
+ errors = { ...errors, ...startDateErrors };
344
274
  }
345
- if (!errors) {
346
- // Clear any errors on the calculator select.
347
- idControl?.setErrors(null);
348
- return null;
275
+ if (this.showEndDatePicker && endDateErrors) {
276
+ errors ||= {};
277
+ errors = { ...errors, ...endDateErrors };
349
278
  }
350
- idControl?.setErrors(errors);
351
- idControl?.markAsTouched();
352
- idControl?.markAsDirty();
353
- // Need to mark the control as touched for the error messages to appear.
354
- this.#control.markAsTouched();
355
- // Notify the view to display any errors.
279
+ // Set errors on the calculator select so that they appear beneath it.
280
+ this.#calculatorIdControl.setErrors(errors);
356
281
  this.#changeDetector.markForCheck();
357
282
  return errors;
358
283
  }
359
- registerOnChange(fn) {
360
- this.#onChange = fn;
284
+ // Implemented as part of ControlValueAccessor.
285
+ // The date range picker always has a value, so if the consumer passes in a
286
+ // partial value (via `patchValue`) or null, we need to update the host control's
287
+ // value with the complete value after it's been modified.
288
+ writeValue(value) {
289
+ this.#patchValue(value);
290
+ // Update the host control if it is set to a partial or null value.
291
+ if (isPartialValue(value)) {
292
+ this.#hostControl?.setValue(this.#getValue(), {
293
+ emitEvent: false,
294
+ onlySelf: true,
295
+ });
296
+ }
361
297
  }
362
- registerOnTouched(fn) {
363
- this.#onTouched = fn;
298
+ isRequired() {
299
+ return !!(this.required || this.#hostControl?.hasValidator(Validators.required));
364
300
  }
365
- setDisabledState(disabled) {
366
- this.disabled = disabled;
301
+ onBlur() {
302
+ this.#notifyTouched?.();
367
303
  }
368
- #setValue(value, notifyChange = true) {
369
- const isNewValue = !this.#dateRangesEqual(this.#value, value);
370
- if (isNewValue) {
371
- this.#value = value;
372
- if (!value || value.calculatorId === undefined) {
373
- this.#valueOrDefault = this.#defaultValue;
374
- }
375
- else {
376
- this.#valueOrDefault = value;
377
- }
378
- if (notifyChange) {
379
- this.#onChange(this.#valueOrDefault);
380
- }
304
+ #getCalculator(calculatorId) {
305
+ const found = this.calculators.find((c) => c.calculatorId === calculatorId);
306
+ /*safety check: should not happen*/
307
+ /*istanbul ignore if*/
308
+ if (!found) {
309
+ throw new Error(`A date range calculator with ID (${calculatorId}) could not be found.`);
381
310
  }
311
+ return found;
382
312
  }
383
- #patchValue(value) {
384
- const newValue = Object.assign({}, this.#valueOrDefault, value);
385
- this.#setValue(newValue);
313
+ #getDefaultValue(calculator) {
314
+ return calculator.getValue();
386
315
  }
387
- #createForm() {
388
- this.formGroup = this.#formBuilder.group({
389
- calculatorId: new UntypedFormControl(),
390
- startDate: new UntypedFormControl(),
391
- endDate: new UntypedFormControl(),
392
- });
393
- if (this.disabled) {
394
- this.formGroup.disable();
316
+ #getValue() {
317
+ return this.#_value;
318
+ }
319
+ #patchValue(partialValue) {
320
+ this.#setValue(isNullOrUndefined(partialValue)
321
+ ? null
322
+ : { ...this.#getValue(), ...partialValue }, { emitEvent: true });
323
+ }
324
+ /**
325
+ * Sets the value to be used by the date range picker form control.
326
+ */
327
+ #setValue(value, options) {
328
+ const oldValue = this.#getValue();
329
+ const valueOrDefault = !value || isNullOrUndefined(value.calculatorId)
330
+ ? this.#getDefaultValue(this.calculators[0])
331
+ : {
332
+ ...this.#getDefaultValue(this.#getCalculator(value.calculatorId)),
333
+ ...value,
334
+ };
335
+ // Ensure falsy values are set to null.
336
+ valueOrDefault.endDate = valueOrDefault.endDate || null;
337
+ valueOrDefault.startDate = valueOrDefault.startDate || null;
338
+ if (!areDateRangesEqual(oldValue, valueOrDefault)) {
339
+ this.#_value = valueOrDefault;
340
+ this.selectedCalculator = this.#getCalculator(valueOrDefault.calculatorId);
341
+ if (oldValue.calculatorId !== valueOrDefault.calculatorId) {
342
+ this.#updatePickerVisibility(this.selectedCalculator);
343
+ }
344
+ if (options?.emitEvent) {
345
+ this.formGroup.setValue(valueOrDefault);
346
+ }
395
347
  }
396
348
  }
397
- #showRelevantFormFields() {
398
- const calculator = this.selectedCalculator;
349
+ #updatePickerVisibility(calculator) {
399
350
  let showEndDatePicker = false;
400
351
  let showStartDatePicker = false;
401
- switch (calculator?.type) {
352
+ switch (calculator.type) {
402
353
  case SkyDateRangeCalculatorType.Before:
403
354
  showEndDatePicker = true;
404
355
  break;
@@ -416,146 +367,75 @@ export class SkyDateRangePickerComponent {
416
367
  this.showStartDatePicker = showStartDatePicker;
417
368
  this.#changeDetector.markForCheck();
418
369
  }
419
- #resetFormGroupValue(value) {
420
- this.formGroup?.reset(value || this.#valueOrDefault);
421
- }
422
- #addEventListeners() {
423
- // Watch for selected calculator change.
424
- this.#calculatorIdControl?.valueChanges
425
- .pipe(distinctUntilChanged(), takeUntil(this.#ngUnsubscribe))
426
- .subscribe((value) => {
427
- if (value !== this.#valueOrDefault?.calculatorId) {
428
- const id = parseInt(value, 10);
429
- // if the component is disabled during form creation, null is passed
430
- // as the value of the calculator id control
431
- // only handle the value changes if the calculator id is a number
432
- /* istanbul ignore else */
433
- if (!isNaN(id)) {
434
- const calculator = this.#getCalculatorById(id);
435
- const newValue = calculator?.getValue();
436
- this.#setValue(newValue);
437
- this.#resetFormGroupValue(newValue);
438
- this.#showRelevantFormFields();
439
- }
440
- }
441
- this.#control?.updateValueAndValidity({ emitEvent: false });
442
- });
443
- // Watch for start date value changes.
444
- this.#startDateControl?.valueChanges
445
- .pipe(distinctUntilChanged(), takeUntil(this.#ngUnsubscribe))
446
- .subscribe((startDate) => {
447
- this.#patchValue({ startDate });
448
- });
449
- // Watch for end date value changes.
450
- this.#endDateControl?.valueChanges
451
- .pipe(distinctUntilChanged(), takeUntil(this.#ngUnsubscribe))
452
- .subscribe((endDate) => {
453
- this.#patchValue({ endDate });
454
- });
455
- // Safety check
456
- /* istanbul ignore else */
457
- if (this.#startDateControl && this.#endDateControl) {
458
- // Detect errors from the date inputs and update ng- classes on picker.
459
- combineLatest([
460
- this.#startDateControl.statusChanges,
461
- this.#endDateControl.statusChanges,
462
- ])
463
- .pipe(takeUntil(this.#ngUnsubscribe))
464
- .subscribe(() => {
465
- this.#updateBasedOnControls();
466
- });
467
- }
468
- else if (this.#startDateControl) {
469
- this.#startDateControl.statusChanges
470
- .pipe(takeUntil(this.#ngUnsubscribe))
471
- .subscribe(() => {
472
- this.#updateBasedOnControls();
473
- });
474
- }
475
- else if (this.#endDateControl) {
476
- this.#endDateControl.statusChanges
477
- .pipe(takeUntil(this.#ngUnsubscribe))
478
- .subscribe(() => {
479
- this.#updateBasedOnControls();
480
- });
481
- }
482
- }
483
- #updateBasedOnControls() {
484
- this.#changeDetector.markForCheck();
485
- // Wait for initial validation to complete.
486
- this.#ngZone.onStable.pipe(first()).subscribe(() => {
487
- this.#control?.updateValueAndValidity({ emitEvent: false });
488
- });
489
- }
490
- #updateCalculators() {
491
- return this.#dateRangeService
492
- .getCalculators(this.calculatorIds)
493
- .then((calculators) => {
494
- this.calculators = calculators;
495
- // Ensure that any previously set value is used to determine the selected calculator
496
- this.#updateSelectedCalculator();
497
- this.#changeDetector.markForCheck();
498
- });
499
- }
500
- #getCalculatorById(id) {
501
- return this.calculators.find((calculator) => {
502
- return calculator.calculatorId === id;
503
- });
504
- }
505
- #dateRangesEqual(rangeA, rangeB) {
506
- return (!!rangeA && !!rangeB && JSON.stringify(rangeA) === JSON.stringify(rangeB));
507
- }
508
- #updateSelectedCalculator() {
509
- this.selectedCalculator = this.#valueOrDefault
510
- ? this.#getCalculatorById(this.#valueOrDefault.calculatorId)
511
- : undefined;
512
- }
513
- /* istanbul ignore next */
514
- // eslint-disable-next-line @typescript-eslint/no-empty-function, @typescript-eslint/no-unused-vars
515
- #onChange;
516
- /* istanbul ignore next */
517
- // eslint-disable-next-line @typescript-eslint/no-empty-function
518
- #onTouched;
519
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: SkyDateRangePickerComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.SkyDateRangeService }, { token: i2.UntypedFormBuilder }, { token: i3.SkyAppLocaleProvider }, { token: i0.NgZone }, { token: i4.SkyThemeService, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
520
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "16.1.0", version: "17.3.4", type: SkyDateRangePickerComponent, selector: "sky-date-range-picker", inputs: { calculatorIds: "calculatorIds", dateFormat: "dateFormat", disabled: "disabled", label: "label", hintText: "hintText", startDateRequired: "startDateRequired", endDateRequired: "endDateRequired", helpPopoverContent: "helpPopoverContent", helpPopoverTitle: "helpPopoverTitle", stacked: ["stacked", "stacked", booleanAttribute] }, host: { properties: { "class.sky-margin-stacked-lg": "this.stacked", "style.display": "this.display" } }, providers: [
521
- SKY_DATE_RANGE_PICKER_VALUE_ACCESSOR,
522
- SKY_DATE_RANGE_PICKER_VALIDATOR,
523
- ], usesOnChanges: true, ngImport: i0, template: "<div\n *ngIf=\"isReady && formGroup\"\n class=\"sky-date-range-picker\"\n [formGroup]=\"formGroup\"\n>\n <div\n class=\"sky-date-range-picker-form-group\"\n [ngClass]=\"{\n 'sky-date-range-picker-last-input':\n !showStartDatePicker && !showEndDatePicker\n }\"\n >\n <sky-input-box\n [disabled]=\"disabled\"\n [helpPopoverContent]=\"helpPopoverContent\"\n [helpPopoverTitle]=\"helpPopoverTitle\"\n [labelText]=\"\n label || ('skyux_date_range_picker_default_label' | skyLibResources)\n \"\n [hintText]=\"hintText\"\n >\n <select\n class=\"sky-form-control\"\n formControlName=\"calculatorId\"\n [attr.id]=\"dateRangePickerId + '-select-calculator'\"\n (blur)=\"onFieldBlur()\"\n >\n <option\n *ngFor=\"let calculator of calculators\"\n [value]=\"calculator.calculatorId\"\n >\n {{ calculator.shortDescription }}\n </option>\n </select>\n </sky-input-box>\n </div>\n <div\n class=\"sky-date-range-picker-form-group\"\n [ngClass]=\"{\n 'sky-date-range-picker-last-input':\n showStartDatePicker && !showEndDatePicker\n }\"\n [hidden]=\"!showStartDatePicker\"\n >\n <sky-input-box [disabled]=\"disabled\">\n <label\n class=\"sky-control-label\"\n [attr.for]=\"dateRangePickerId + '-start-date'\"\n [ngClass]=\"{ 'sky-control-label-required': startDateRequired }\"\n >\n {{\n selectedCalculator?.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources\n }}\n </label>\n\n <sky-datepicker>\n <input\n class=\"sky-form-control\"\n formControlName=\"startDate\"\n [attr.aria-label]=\"\n label\n ? ('skyux_date_range_picker_default_aria_label'\n | skyLibResources\n : (selectedCalculator?.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources)\n : label)\n : (selectedCalculator?.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources)\n \"\n [attr.id]=\"dateRangePickerId + '-start-date'\"\n [required]=\"!!startDateRequired\"\n [dateFormat]=\"dateFormatOrDefault\"\n (blur)=\"onFieldBlur()\"\n skyDatepickerInput\n />\n </sky-datepicker>\n </sky-input-box>\n </div>\n\n <div\n class=\"sky-date-range-picker-form-group\"\n [ngClass]=\"{ 'sky-date-range-picker-last-input': showEndDatePicker }\"\n [hidden]=\"!showEndDatePicker\"\n >\n <sky-input-box [disabled]=\"disabled\">\n <label\n class=\"sky-control-label\"\n [attr.for]=\"dateRangePickerId + '-end-date'\"\n [ngClass]=\"{ 'sky-control-label-required': endDateRequired }\"\n >\n {{\n selectedCalculator?.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources\n }}\n </label>\n\n <sky-datepicker>\n <input\n class=\"sky-form-control\"\n formControlName=\"endDate\"\n skyDatepickerInput\n [attr.aria-label]=\"\n label\n ? ('skyux_date_range_picker_default_aria_label'\n | skyLibResources\n : (selectedCalculator?.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources)\n : label)\n : (selectedCalculator?.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources)\n \"\n [attr.id]=\"dateRangePickerId + '-end-date'\"\n [dateFormat]=\"dateFormatOrDefault\"\n [required]=\"!!endDateRequired\"\n (blur)=\"onFieldBlur()\"\n />\n </sky-datepicker>\n </sky-input-box>\n </div>\n</div>\n", styles: [":host{display:block}.sky-date-range-picker{display:flex}@media (max-width: 768px){.sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-bottom:var(--sky-margin-stacked-lg)}}:host .sky-date-range-picker{flex-direction:column}:host .sky-date-range-picker-form-group{flex-basis:100%}:host .sky-date-range-picker-form-group:not(:first-of-type){padding-left:initial}:host .sky-date-range-picker-form-group:not(:last-of-type){padding-right:initial}:host-context(.sky-responsive-container-xs) .sky-date-range-picker,:host-context(.sky-responsive-container-sm) .sky-date-range-picker,:host-context(.sky-responsive-container-md) .sky-date-range-picker,:host-context(.sky-responsive-container-lg) .sky-date-range-picker{flex-direction:column}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group{flex-basis:100%}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:first-of-type){padding-left:initial}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:last-of-type){padding-right:initial}@media (min-width: 768px){:host .sky-date-range-picker{flex-direction:initial}:host .sky-date-range-picker-form-group{flex-basis:33.3333333333%}:host .sky-date-range-picker-form-group:not(:first-of-type){padding-left:5px}:host .sky-date-range-picker-form-group:not(:last-of-type){padding-right:5px}}:host-context(.sky-responsive-container-sm) .sky-date-range-picker,:host-context(.sky-responsive-container-md) .sky-date-range-picker,:host-context(.sky-responsive-container-lg) .sky-date-range-picker{flex-direction:initial}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group{flex-basis:33.3333333333%}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:first-of-type){padding-left:5px}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:last-of-type){padding-right:5px}@media (min-width: 768px){:host-context(sky-filter-inline-item) .sky-date-range-picker-form-group{flex-basis:100%}}.sky-responsive-container-sm :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group,.sky-responsive-container-md :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group,.sky-responsive-container-lg :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group{flex-basis:100%}:host-context(.sky-theme-modern) .sky-date-range-picker-form-group{padding:0}@media (min-width: 768px){:host-context(.sky-theme-modern) .sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-right:20px}}.sky-theme-modern .sky-date-range-picker-form-group{padding:0}@media (min-width: 768px){.sky-theme-modern .sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-right:20px}}\n"], dependencies: [{ kind: "directive", type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i2.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: i6.SkyDatepickerComponent, selector: "sky-datepicker", inputs: ["pickerClass"], outputs: ["calendarDateRangeChange", "dateFormatChange", "openChange"] }, { kind: "directive", type: i7.SkyDatepickerInputDirective, selector: "[skyDatepickerInput]", inputs: ["dateFormat", "disabled", "maxDate", "minDate", "startAtDate", "skyDatepickerInput", "skyDatepickerNoValidate", "startingDay", "strict"] }, { kind: "component", type: i8.λ10, selector: "sky-input-box", inputs: ["hasErrors", "disabled", "labelText", "characterLimit", "stacked", "helpPopoverTitle", "helpPopoverContent", "helpKey", "hintText"] }, { kind: "pipe", type: i3.SkyLibResourcesPipe, name: "skyLibResources" }, { kind: "pipe", type: i9.SkyDateRangePickerEndDateResourceKeyPipe, name: "skyDateRangePickerEndDateResourceKey" }, { kind: "pipe", type: i10.SkyDateRangePickerStartDateResourceKeyPipe, name: "skyDateRangePickerStartDateResourceKey" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
370
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: SkyDateRangePickerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
371
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "16.1.0", version: "17.3.4", type: SkyDateRangePickerComponent, isStandalone: true, selector: "sky-date-range-picker", inputs: { calculatorIds: "calculatorIds", dateFormat: "dateFormat", disabled: ["disabled", "disabled", booleanAttribute], endDateRequired: ["endDateRequired", "endDateRequired", booleanAttribute], helpPopoverContent: "helpPopoverContent", helpPopoverTitle: "helpPopoverTitle", hintText: "hintText", label: "label", required: ["required", "required", booleanAttribute], stacked: ["stacked", "stacked", booleanAttribute], startDateRequired: ["startDateRequired", "startDateRequired", booleanAttribute] }, host: { properties: { "class.sky-margin-stacked-lg": "this.stacked", "style.display": "this.display" } }, providers: [
372
+ {
373
+ provide: NG_VALIDATORS,
374
+ useExisting: SkyDateRangePickerComponent,
375
+ multi: true,
376
+ },
377
+ {
378
+ provide: NG_VALUE_ACCESSOR,
379
+ useExisting: SkyDateRangePickerComponent,
380
+ multi: true,
381
+ },
382
+ ], ngImport: i0, template: "<div\n class=\"sky-date-range-picker\"\n [formGroup]=\"formGroup\"\n (focusout)=\"onBlur()\"\n>\n <div\n class=\"sky-date-range-picker-form-group\"\n [ngClass]=\"{\n 'sky-date-range-picker-last-input':\n !showStartDatePicker && !showEndDatePicker\n }\"\n >\n <sky-input-box\n [hasErrors]=\"hasErrors\"\n [helpPopoverContent]=\"helpPopoverContent\"\n [helpPopoverTitle]=\"helpPopoverTitle\"\n [hintText]=\"hintText\"\n [labelText]=\"\n label || ('skyux_date_range_picker_default_label' | skyLibResources)\n \"\n >\n <select\n formControlName=\"calculatorId\"\n [required]=\"isRequired()\"\n (blur)=\"onBlur()\"\n >\n <option\n *ngFor=\"let calculator of calculators\"\n [value]=\"calculator.calculatorId\"\n >\n {{\n calculator._shortDescriptionResourceKey\n ? (calculator._shortDescriptionResourceKey | skyLibResources)\n : calculator.shortDescription\n }}\n </option>\n </select>\n </sky-input-box>\n </div>\n <div\n class=\"sky-date-range-picker-form-group sky-date-range-datepicker-wrapper\"\n [hidden]=\"!showStartDatePicker\"\n [ngClass]=\"{\n 'sky-date-range-picker-last-input':\n showStartDatePicker && !showEndDatePicker\n }\"\n >\n <sky-input-box\n [errorsScreenReaderOnly]=\"true\"\n [hasErrors]=\"hasErrors\"\n [labelText]=\"\n selectedCalculator.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources\n \"\n >\n <sky-datepicker>\n <input\n formControlName=\"startDate\"\n name=\"startDate\"\n skyDatepickerInput\n type=\"text\"\n [attr.aria-label]=\"\n label\n ? ('skyux_date_range_picker_default_aria_label'\n | skyLibResources\n : (selectedCalculator.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources)\n : label)\n : (selectedCalculator.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources)\n \"\n [dateFormat]=\"dateFormat\"\n [required]=\"\n showStartDatePicker && (startDateRequired || isRequired())\n \"\n />\n </sky-datepicker>\n </sky-input-box>\n </div>\n <div\n class=\"sky-date-range-picker-form-group sky-date-range-datepicker-wrapper\"\n [hidden]=\"!showEndDatePicker\"\n [ngClass]=\"{ 'sky-date-range-picker-last-input': showEndDatePicker }\"\n >\n <sky-input-box\n [errorsScreenReaderOnly]=\"true\"\n [hasErrors]=\"hasErrors\"\n [labelText]=\"\n selectedCalculator.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources\n \"\n >\n <sky-datepicker>\n <input\n formControlName=\"endDate\"\n name=\"endDate\"\n skyDatepickerInput\n type=\"text\"\n [attr.aria-label]=\"\n label\n ? ('skyux_date_range_picker_default_aria_label'\n | skyLibResources\n : (selectedCalculator.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources)\n : label)\n : (selectedCalculator.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources)\n \"\n [dateFormat]=\"dateFormat\"\n [required]=\"showEndDatePicker && (endDateRequired || isRequired())\"\n />\n </sky-datepicker>\n </sky-input-box>\n </div>\n</div>\n", styles: [":host{display:block}.sky-date-range-picker{display:flex}@media (max-width: 768px){.sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-bottom:var(--sky-margin-stacked-lg)}}:host .sky-date-range-picker{flex-direction:column}:host .sky-date-range-picker-form-group{flex-basis:100%}:host .sky-date-range-picker-form-group:not(:first-of-type){padding-left:initial}:host .sky-date-range-picker-form-group:not(:last-of-type){padding-right:initial}:host-context(.sky-responsive-container-xs) .sky-date-range-picker,:host-context(.sky-responsive-container-sm) .sky-date-range-picker,:host-context(.sky-responsive-container-md) .sky-date-range-picker,:host-context(.sky-responsive-container-lg) .sky-date-range-picker{flex-direction:column}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group{flex-basis:100%}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:first-of-type){padding-left:initial}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:last-of-type){padding-right:initial}@media (min-width: 768px){:host .sky-date-range-picker{flex-direction:initial}:host .sky-date-range-picker-form-group{flex-basis:33.3333333333%}:host .sky-date-range-picker-form-group:not(:first-of-type){padding-left:5px}:host .sky-date-range-picker-form-group:not(:last-of-type){padding-right:5px}}:host-context(.sky-responsive-container-sm) .sky-date-range-picker,:host-context(.sky-responsive-container-md) .sky-date-range-picker,:host-context(.sky-responsive-container-lg) .sky-date-range-picker{flex-direction:initial}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group{flex-basis:33.3333333333%}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:first-of-type){padding-left:5px}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:last-of-type){padding-right:5px}@media (min-width: 768px){:host-context(sky-filter-inline-item) .sky-date-range-picker-form-group{flex-basis:100%}}.sky-responsive-container-sm :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group,.sky-responsive-container-md :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group,.sky-responsive-container-lg :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group{flex-basis:100%}:host-context(.sky-theme-modern) .sky-date-range-picker-form-group{padding:0}@media (min-width: 768px){:host-context(.sky-theme-modern) .sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-right:20px}}.sky-theme-modern .sky-date-range-picker-form-group{padding:0}@media (min-width: 768px){.sky-theme-modern .sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-right:20px}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i2.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "ngmodule", type: SkyDatepickerModule }, { kind: "component", type: i3.SkyDatepickerComponent, selector: "sky-datepicker", inputs: ["pickerClass"], outputs: ["calendarDateRangeChange", "dateFormatChange", "openChange"] }, { kind: "directive", type: i4.SkyDatepickerInputDirective, selector: "[skyDatepickerInput]", inputs: ["dateFormat", "disabled", "maxDate", "minDate", "startAtDate", "skyDatepickerInput", "skyDatepickerNoValidate", "startingDay", "strict"] }, { kind: "pipe", type: SkyDateRangePickerEndDateResourceKeyPipe, name: "skyDateRangePickerEndDateResourceKey" }, { kind: "pipe", type: SkyDateRangePickerStartDateResourceKeyPipe, name: "skyDateRangePickerStartDateResourceKey" }, { kind: "ngmodule", type: SkyDatetimeResourcesModule }, { kind: "pipe", type: i5.SkyLibResourcesPipe, name: "skyLibResources" }, { kind: "ngmodule", type: SkyInputBoxModule }, { kind: "component", type: i6.λ10, selector: "sky-input-box", inputs: ["hasErrors", "disabled", "labelText", "characterLimit", "stacked", "helpPopoverTitle", "helpPopoverContent", "helpKey", "hintText", "errorsScreenReaderOnly"] }, { kind: "directive", type: i6.λ20, selector: "input:not([skyId]):not(.sky-form-control),select:not([skyId]):not(.sky-form-control),textarea:not([skyId]):not(.sky-form-control)", inputs: ["autocomplete"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
524
383
  }
525
384
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: SkyDateRangePickerComponent, decorators: [{
526
385
  type: Component,
527
- args: [{ selector: 'sky-date-range-picker', providers: [
528
- SKY_DATE_RANGE_PICKER_VALUE_ACCESSOR,
529
- SKY_DATE_RANGE_PICKER_VALIDATOR,
530
- ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n *ngIf=\"isReady && formGroup\"\n class=\"sky-date-range-picker\"\n [formGroup]=\"formGroup\"\n>\n <div\n class=\"sky-date-range-picker-form-group\"\n [ngClass]=\"{\n 'sky-date-range-picker-last-input':\n !showStartDatePicker && !showEndDatePicker\n }\"\n >\n <sky-input-box\n [disabled]=\"disabled\"\n [helpPopoverContent]=\"helpPopoverContent\"\n [helpPopoverTitle]=\"helpPopoverTitle\"\n [labelText]=\"\n label || ('skyux_date_range_picker_default_label' | skyLibResources)\n \"\n [hintText]=\"hintText\"\n >\n <select\n class=\"sky-form-control\"\n formControlName=\"calculatorId\"\n [attr.id]=\"dateRangePickerId + '-select-calculator'\"\n (blur)=\"onFieldBlur()\"\n >\n <option\n *ngFor=\"let calculator of calculators\"\n [value]=\"calculator.calculatorId\"\n >\n {{ calculator.shortDescription }}\n </option>\n </select>\n </sky-input-box>\n </div>\n <div\n class=\"sky-date-range-picker-form-group\"\n [ngClass]=\"{\n 'sky-date-range-picker-last-input':\n showStartDatePicker && !showEndDatePicker\n }\"\n [hidden]=\"!showStartDatePicker\"\n >\n <sky-input-box [disabled]=\"disabled\">\n <label\n class=\"sky-control-label\"\n [attr.for]=\"dateRangePickerId + '-start-date'\"\n [ngClass]=\"{ 'sky-control-label-required': startDateRequired }\"\n >\n {{\n selectedCalculator?.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources\n }}\n </label>\n\n <sky-datepicker>\n <input\n class=\"sky-form-control\"\n formControlName=\"startDate\"\n [attr.aria-label]=\"\n label\n ? ('skyux_date_range_picker_default_aria_label'\n | skyLibResources\n : (selectedCalculator?.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources)\n : label)\n : (selectedCalculator?.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources)\n \"\n [attr.id]=\"dateRangePickerId + '-start-date'\"\n [required]=\"!!startDateRequired\"\n [dateFormat]=\"dateFormatOrDefault\"\n (blur)=\"onFieldBlur()\"\n skyDatepickerInput\n />\n </sky-datepicker>\n </sky-input-box>\n </div>\n\n <div\n class=\"sky-date-range-picker-form-group\"\n [ngClass]=\"{ 'sky-date-range-picker-last-input': showEndDatePicker }\"\n [hidden]=\"!showEndDatePicker\"\n >\n <sky-input-box [disabled]=\"disabled\">\n <label\n class=\"sky-control-label\"\n [attr.for]=\"dateRangePickerId + '-end-date'\"\n [ngClass]=\"{ 'sky-control-label-required': endDateRequired }\"\n >\n {{\n selectedCalculator?.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources\n }}\n </label>\n\n <sky-datepicker>\n <input\n class=\"sky-form-control\"\n formControlName=\"endDate\"\n skyDatepickerInput\n [attr.aria-label]=\"\n label\n ? ('skyux_date_range_picker_default_aria_label'\n | skyLibResources\n : (selectedCalculator?.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources)\n : label)\n : (selectedCalculator?.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources)\n \"\n [attr.id]=\"dateRangePickerId + '-end-date'\"\n [dateFormat]=\"dateFormatOrDefault\"\n [required]=\"!!endDateRequired\"\n (blur)=\"onFieldBlur()\"\n />\n </sky-datepicker>\n </sky-input-box>\n </div>\n</div>\n", styles: [":host{display:block}.sky-date-range-picker{display:flex}@media (max-width: 768px){.sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-bottom:var(--sky-margin-stacked-lg)}}:host .sky-date-range-picker{flex-direction:column}:host .sky-date-range-picker-form-group{flex-basis:100%}:host .sky-date-range-picker-form-group:not(:first-of-type){padding-left:initial}:host .sky-date-range-picker-form-group:not(:last-of-type){padding-right:initial}:host-context(.sky-responsive-container-xs) .sky-date-range-picker,:host-context(.sky-responsive-container-sm) .sky-date-range-picker,:host-context(.sky-responsive-container-md) .sky-date-range-picker,:host-context(.sky-responsive-container-lg) .sky-date-range-picker{flex-direction:column}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group{flex-basis:100%}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:first-of-type){padding-left:initial}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:last-of-type){padding-right:initial}@media (min-width: 768px){:host .sky-date-range-picker{flex-direction:initial}:host .sky-date-range-picker-form-group{flex-basis:33.3333333333%}:host .sky-date-range-picker-form-group:not(:first-of-type){padding-left:5px}:host .sky-date-range-picker-form-group:not(:last-of-type){padding-right:5px}}:host-context(.sky-responsive-container-sm) .sky-date-range-picker,:host-context(.sky-responsive-container-md) .sky-date-range-picker,:host-context(.sky-responsive-container-lg) .sky-date-range-picker{flex-direction:initial}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group{flex-basis:33.3333333333%}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:first-of-type){padding-left:5px}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:last-of-type){padding-right:5px}@media (min-width: 768px){:host-context(sky-filter-inline-item) .sky-date-range-picker-form-group{flex-basis:100%}}.sky-responsive-container-sm :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group,.sky-responsive-container-md :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group,.sky-responsive-container-lg :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group{flex-basis:100%}:host-context(.sky-theme-modern) .sky-date-range-picker-form-group{padding:0}@media (min-width: 768px){:host-context(.sky-theme-modern) .sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-right:20px}}.sky-theme-modern .sky-date-range-picker-form-group{padding:0}@media (min-width: 768px){.sky-theme-modern .sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-right:20px}}\n"] }]
531
- }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }, { type: i1.SkyDateRangeService }, { type: i2.UntypedFormBuilder }, { type: i3.SkyAppLocaleProvider }, { type: i0.NgZone }, { type: i4.SkyThemeService, decorators: [{
532
- type: Optional
533
- }] }], propDecorators: { calculatorIds: [{
386
+ args: [{ changeDetection: ChangeDetectionStrategy.OnPush, imports: [
387
+ CommonModule,
388
+ FormsModule,
389
+ ReactiveFormsModule,
390
+ SkyDatepickerModule,
391
+ SkyDateRangePickerEndDateResourceKeyPipe,
392
+ SkyDateRangePickerStartDateResourceKeyPipe,
393
+ SkyDatetimeResourcesModule,
394
+ SkyInputBoxModule,
395
+ ], providers: [
396
+ {
397
+ provide: NG_VALIDATORS,
398
+ useExisting: SkyDateRangePickerComponent,
399
+ multi: true,
400
+ },
401
+ {
402
+ provide: NG_VALUE_ACCESSOR,
403
+ useExisting: SkyDateRangePickerComponent,
404
+ multi: true,
405
+ },
406
+ ], selector: 'sky-date-range-picker', standalone: true, template: "<div\n class=\"sky-date-range-picker\"\n [formGroup]=\"formGroup\"\n (focusout)=\"onBlur()\"\n>\n <div\n class=\"sky-date-range-picker-form-group\"\n [ngClass]=\"{\n 'sky-date-range-picker-last-input':\n !showStartDatePicker && !showEndDatePicker\n }\"\n >\n <sky-input-box\n [hasErrors]=\"hasErrors\"\n [helpPopoverContent]=\"helpPopoverContent\"\n [helpPopoverTitle]=\"helpPopoverTitle\"\n [hintText]=\"hintText\"\n [labelText]=\"\n label || ('skyux_date_range_picker_default_label' | skyLibResources)\n \"\n >\n <select\n formControlName=\"calculatorId\"\n [required]=\"isRequired()\"\n (blur)=\"onBlur()\"\n >\n <option\n *ngFor=\"let calculator of calculators\"\n [value]=\"calculator.calculatorId\"\n >\n {{\n calculator._shortDescriptionResourceKey\n ? (calculator._shortDescriptionResourceKey | skyLibResources)\n : calculator.shortDescription\n }}\n </option>\n </select>\n </sky-input-box>\n </div>\n <div\n class=\"sky-date-range-picker-form-group sky-date-range-datepicker-wrapper\"\n [hidden]=\"!showStartDatePicker\"\n [ngClass]=\"{\n 'sky-date-range-picker-last-input':\n showStartDatePicker && !showEndDatePicker\n }\"\n >\n <sky-input-box\n [errorsScreenReaderOnly]=\"true\"\n [hasErrors]=\"hasErrors\"\n [labelText]=\"\n selectedCalculator.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources\n \"\n >\n <sky-datepicker>\n <input\n formControlName=\"startDate\"\n name=\"startDate\"\n skyDatepickerInput\n type=\"text\"\n [attr.aria-label]=\"\n label\n ? ('skyux_date_range_picker_default_aria_label'\n | skyLibResources\n : (selectedCalculator.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources)\n : label)\n : (selectedCalculator.type\n | skyDateRangePickerStartDateResourceKey\n | skyLibResources)\n \"\n [dateFormat]=\"dateFormat\"\n [required]=\"\n showStartDatePicker && (startDateRequired || isRequired())\n \"\n />\n </sky-datepicker>\n </sky-input-box>\n </div>\n <div\n class=\"sky-date-range-picker-form-group sky-date-range-datepicker-wrapper\"\n [hidden]=\"!showEndDatePicker\"\n [ngClass]=\"{ 'sky-date-range-picker-last-input': showEndDatePicker }\"\n >\n <sky-input-box\n [errorsScreenReaderOnly]=\"true\"\n [hasErrors]=\"hasErrors\"\n [labelText]=\"\n selectedCalculator.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources\n \"\n >\n <sky-datepicker>\n <input\n formControlName=\"endDate\"\n name=\"endDate\"\n skyDatepickerInput\n type=\"text\"\n [attr.aria-label]=\"\n label\n ? ('skyux_date_range_picker_default_aria_label'\n | skyLibResources\n : (selectedCalculator.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources)\n : label)\n : (selectedCalculator.type\n | skyDateRangePickerEndDateResourceKey\n | skyLibResources)\n \"\n [dateFormat]=\"dateFormat\"\n [required]=\"showEndDatePicker && (endDateRequired || isRequired())\"\n />\n </sky-datepicker>\n </sky-input-box>\n </div>\n</div>\n", styles: [":host{display:block}.sky-date-range-picker{display:flex}@media (max-width: 768px){.sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-bottom:var(--sky-margin-stacked-lg)}}:host .sky-date-range-picker{flex-direction:column}:host .sky-date-range-picker-form-group{flex-basis:100%}:host .sky-date-range-picker-form-group:not(:first-of-type){padding-left:initial}:host .sky-date-range-picker-form-group:not(:last-of-type){padding-right:initial}:host-context(.sky-responsive-container-xs) .sky-date-range-picker,:host-context(.sky-responsive-container-sm) .sky-date-range-picker,:host-context(.sky-responsive-container-md) .sky-date-range-picker,:host-context(.sky-responsive-container-lg) .sky-date-range-picker{flex-direction:column}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group{flex-basis:100%}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:first-of-type){padding-left:initial}:host-context(.sky-responsive-container-xs) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:last-of-type){padding-right:initial}@media (min-width: 768px){:host .sky-date-range-picker{flex-direction:initial}:host .sky-date-range-picker-form-group{flex-basis:33.3333333333%}:host .sky-date-range-picker-form-group:not(:first-of-type){padding-left:5px}:host .sky-date-range-picker-form-group:not(:last-of-type){padding-right:5px}}:host-context(.sky-responsive-container-sm) .sky-date-range-picker,:host-context(.sky-responsive-container-md) .sky-date-range-picker,:host-context(.sky-responsive-container-lg) .sky-date-range-picker{flex-direction:initial}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group,:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group{flex-basis:33.3333333333%}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:first-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:first-of-type){padding-left:5px}:host-context(.sky-responsive-container-sm) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-md) .sky-date-range-picker-form-group:not(:last-of-type),:host-context(.sky-responsive-container-lg) .sky-date-range-picker-form-group:not(:last-of-type){padding-right:5px}@media (min-width: 768px){:host-context(sky-filter-inline-item) .sky-date-range-picker-form-group{flex-basis:100%}}.sky-responsive-container-sm :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group,.sky-responsive-container-md :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group,.sky-responsive-container-lg :host-context(sky-filter-inline-item) .sky-date-range-picker-form-group{flex-basis:100%}:host-context(.sky-theme-modern) .sky-date-range-picker-form-group{padding:0}@media (min-width: 768px){:host-context(.sky-theme-modern) .sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-right:20px}}.sky-theme-modern .sky-date-range-picker-form-group{padding:0}@media (min-width: 768px){.sky-theme-modern .sky-date-range-picker-form-group:not(.sky-date-range-picker-last-input){margin-right:20px}}\n"] }]
407
+ }], ctorParameters: () => [], propDecorators: { calculatorIds: [{
534
408
  type: Input
535
409
  }], dateFormat: [{
536
410
  type: Input
537
411
  }], disabled: [{
538
- type: Input
539
- }], label: [{
540
- type: Input
541
- }], hintText: [{
542
- type: Input
543
- }], startDateRequired: [{
544
- type: Input
412
+ type: Input,
413
+ args: [{ transform: booleanAttribute }]
545
414
  }], endDateRequired: [{
546
- type: Input
415
+ type: Input,
416
+ args: [{ transform: booleanAttribute }]
547
417
  }], helpPopoverContent: [{
548
418
  type: Input
549
419
  }], helpPopoverTitle: [{
550
420
  type: Input
421
+ }], hintText: [{
422
+ type: Input
423
+ }], label: [{
424
+ type: Input
425
+ }], required: [{
426
+ type: Input,
427
+ args: [{ transform: booleanAttribute }]
551
428
  }], stacked: [{
552
429
  type: Input,
553
430
  args: [{ transform: booleanAttribute }]
554
431
  }, {
555
432
  type: HostBinding,
556
433
  args: ['class.sky-margin-stacked-lg']
434
+ }], startDateRequired: [{
435
+ type: Input,
436
+ args: [{ transform: booleanAttribute }]
557
437
  }], display: [{
558
438
  type: HostBinding,
559
439
  args: ['style.display']
560
440
  }] } });
561
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-range-picker.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/datetime/src/lib/modules/date-range-picker/date-range-picker.component.ts","../../../../../../../../libs/components/datetime/src/lib/modules/date-range-picker/date-range-picker.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  HostBinding,\n  Input,\n  NgZone,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Optional,\n  SimpleChanges,\n  TemplateRef,\n  booleanAttribute,\n  forwardRef,\n  inject,\n} from '@angular/core';\nimport {\n  AbstractControl,\n  ControlValueAccessor,\n  NG_VALIDATORS,\n  NG_VALUE_ACCESSOR,\n  UntypedFormBuilder,\n  UntypedFormControl,\n  UntypedFormGroup,\n  ValidationErrors,\n  Validator,\n} from '@angular/forms';\nimport { SkyAppFormat } from '@skyux/core';\nimport { SkyFormFieldLabelTextRequiredService } from '@skyux/forms';\nimport { SkyAppLocaleProvider, SkyLibResourcesService } from '@skyux/i18n';\nimport { SkyThemeService } from '@skyux/theme';\n\nimport { Subject, combineLatest } from 'rxjs';\nimport { distinctUntilChanged, first, takeUntil } from 'rxjs/operators';\n\nimport { SkyDateFormatter } from '../datepicker/date-formatter';\n\nimport { SkyDateRangeService } from './date-range.service';\nimport { SkyDateRangeCalculation } from './types/date-range-calculation';\nimport { SkyDateRangeCalculator } from './types/date-range-calculator';\nimport { SkyDateRangeCalculatorId } from './types/date-range-calculator-id';\nimport { SkyDateRangeCalculatorType } from './types/date-range-calculator-type';\n\nconst SKY_DATE_RANGE_PICKER_VALUE_ACCESSOR = {\n  provide: NG_VALUE_ACCESSOR,\n  useExisting: forwardRef(() => SkyDateRangePickerComponent),\n  multi: true,\n};\n\nconst SKY_DATE_RANGE_PICKER_VALIDATOR = {\n  provide: NG_VALIDATORS,\n  useExisting: forwardRef(() => SkyDateRangePickerComponent),\n  multi: true,\n};\n\nlet uniqueId = 0;\n\n/**\n * Acts as a form control with a form model of type `SkyDateRangeCalculation`.\n * @example\n * ```\n * <sky-date-range-picker\n *   formControlName=\"myPicker\"\n * >\n * </sky-date-range-picker>\n * ```\n */\n@Component({\n  selector: 'sky-date-range-picker',\n  templateUrl: './date-range-picker.component.html',\n  styleUrls: ['./date-range-picker.component.scss'],\n  providers: [\n    SKY_DATE_RANGE_PICKER_VALUE_ACCESSOR,\n    SKY_DATE_RANGE_PICKER_VALIDATOR,\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SkyDateRangePickerComponent\n  implements OnInit, OnChanges, OnDestroy, ControlValueAccessor, Validator\n{\n  /**\n   * IDs for the date range options to include in the picker's dropdown.\n   * The options specify calculator objects that return two `Date` objects to represent date ranges.\n   * By default, this property includes all `SkyDateRangeCalculatorId` values.\n   */\n  @Input()\n  public set calculatorIds(value: SkyDateRangeCalculatorId[] | undefined) {\n    this.#_calculatorIds = value || [\n      SkyDateRangeCalculatorId.AnyTime,\n      SkyDateRangeCalculatorId.Before,\n      SkyDateRangeCalculatorId.After,\n      SkyDateRangeCalculatorId.SpecificRange,\n      SkyDateRangeCalculatorId.Yesterday,\n      SkyDateRangeCalculatorId.Today,\n      SkyDateRangeCalculatorId.Tomorrow,\n      SkyDateRangeCalculatorId.LastWeek,\n      SkyDateRangeCalculatorId.ThisWeek,\n      SkyDateRangeCalculatorId.NextWeek,\n      SkyDateRangeCalculatorId.LastMonth,\n      SkyDateRangeCalculatorId.ThisMonth,\n      SkyDateRangeCalculatorId.NextMonth,\n      SkyDateRangeCalculatorId.LastQuarter,\n      SkyDateRangeCalculatorId.ThisQuarter,\n      SkyDateRangeCalculatorId.NextQuarter,\n      SkyDateRangeCalculatorId.LastCalendarYear,\n      SkyDateRangeCalculatorId.ThisCalendarYear,\n      SkyDateRangeCalculatorId.NextCalendarYear,\n      SkyDateRangeCalculatorId.LastFiscalYear,\n      SkyDateRangeCalculatorId.ThisFiscalYear,\n      SkyDateRangeCalculatorId.NextFiscalYear,\n    ];\n  }\n\n  public get calculatorIds(): SkyDateRangeCalculatorId[] {\n    return this.#_calculatorIds;\n  }\n\n  /**\n   * The date format for\n   * [the `sky-datepicker` components](https://developer.blackbaud.com/skyux/components/datepicker)\n   * that make up the date range picker. The text input is a composite component of\n   * up to two `sky-datepicker` components.\n   * @default \"MM/DD/YYYY\"\n   */\n  @Input()\n  public set dateFormat(value: string | undefined) {\n    this.#_dateFormat = value;\n    this.dateFormatOrDefault = value || this.#preferredShortDateFormat;\n  }\n\n  public get dateFormat(): string | undefined {\n    return this.#_dateFormat;\n  }\n\n  /**\n   * Whether to disable the date range picker on template-driven forms. Don't use this input on reactive forms because they may overwrite the input or leave the control out of sync.\n   * To set the disabled state on reactive forms, use the `FormControl` instead.\n   * @default false\n   */\n  @Input()\n  public set disabled(value: boolean | undefined) {\n    this.#_disabled = value;\n\n    if (this.formGroup) {\n      if (this.#_disabled) {\n        this.formGroup.disable();\n      } else {\n        this.formGroup.enable();\n      }\n    }\n\n    this.#changeDetector.markForCheck();\n  }\n\n  public get disabled(): boolean | undefined {\n    return this.#_disabled;\n  }\n\n  /**\n   * The label for the date range picker.\n   * @required\n   */\n  @Input()\n  public label: string | undefined;\n\n  /**\n   * [Persistent inline help text](https://developer.blackbaud.com/skyux/design/guidelines/user-assistance#inline-help) that provides\n   * additional context to the user.\n   * @preview\n   */\n  @Input()\n  public hintText: string | undefined;\n\n  /**\n   * Whether to require users to specify a start date.\n   * @default false\n   */\n  @Input()\n  public startDateRequired: boolean | undefined = false;\n\n  /**\n   * Whether to require users to specify a end date.\n   * @default false\n   */\n  @Input()\n  public endDateRequired: boolean | undefined = false;\n\n  /**\n   * The content of the help popover. When specified along with `labelText`, a [help inline](https://developer.blackbaud.com/skyux/components/help-inline)\n   * button is added to date range picker. The help inline button displays a [popover](https://developer.blackbaud.com/skyux/components/popover)\n   * when clicked using the specified content and optional title.\n   * @preview\n   */\n  @Input()\n  public helpPopoverContent: string | TemplateRef<unknown> | undefined;\n\n  /**\n   * The title of the help popover. This property only applies when `helpPopoverContent` is\n   * also specified.\n   * @preview\n   */\n  @Input()\n  public helpPopoverTitle: string | undefined;\n\n  /**\n   * Whether the date range picker is stacked on another form component. When specified, the appropriate\n   * vertical spacing is automatically added to the date range picker.\n   * @preview\n   */\n  @Input({ transform: booleanAttribute })\n  @HostBinding('class.sky-margin-stacked-lg')\n  public stacked = false;\n\n  @HostBinding('style.display')\n  public display: string | undefined;\n\n  public selectedCalculator: SkyDateRangeCalculator | undefined;\n\n  public readonly dateRangePickerId = `sky-date-range-picker-${uniqueId++}`;\n\n  public calculators: SkyDateRangeCalculator[] = [];\n  public dateFormatOrDefault: string | undefined;\n  public formGroup: UntypedFormGroup | undefined;\n  public isReady = false;\n  public showEndDatePicker = false;\n  public showStartDatePicker = false;\n\n  protected hostHintText: string | undefined;\n\n  get #calculatorIdControl(): AbstractControl | undefined | null {\n    return this.formGroup?.get('calculatorId');\n  }\n\n  get #defaultCalculator(): SkyDateRangeCalculator | undefined {\n    return this.calculators[0];\n  }\n\n  get #defaultValue(): SkyDateRangeCalculation | undefined {\n    return this.#defaultCalculator?.getValue();\n  }\n\n  get #endDateControl(): AbstractControl | undefined | null {\n    return this.formGroup?.get('endDate');\n  }\n\n  get #startDateControl(): AbstractControl | undefined | null {\n    return this.formGroup?.get('startDate');\n  }\n\n  #value: SkyDateRangeCalculation | undefined;\n\n  set #valueOrDefault(value: SkyDateRangeCalculation | undefined) {\n    this.#_valueOrDefault = value;\n    this.#updateSelectedCalculator();\n  }\n\n  get #valueOrDefault(): SkyDateRangeCalculation | undefined {\n    return this.#_valueOrDefault;\n  }\n\n  readonly #labelTextRequired = inject(SkyFormFieldLabelTextRequiredService, {\n    optional: true,\n  });\n\n  readonly #appFormatter = inject(SkyAppFormat);\n  #control: AbstractControl | undefined;\n  #preferredShortDateFormat: string | undefined;\n  #ngUnsubscribe = new Subject<void>();\n  readonly #resourceSvc = inject(SkyLibResourcesService);\n\n  #_calculatorIds: SkyDateRangeCalculatorId[] = [\n    SkyDateRangeCalculatorId.AnyTime,\n    SkyDateRangeCalculatorId.Before,\n    SkyDateRangeCalculatorId.After,\n    SkyDateRangeCalculatorId.SpecificRange,\n    SkyDateRangeCalculatorId.Yesterday,\n    SkyDateRangeCalculatorId.Today,\n    SkyDateRangeCalculatorId.Tomorrow,\n    SkyDateRangeCalculatorId.LastWeek,\n    SkyDateRangeCalculatorId.ThisWeek,\n    SkyDateRangeCalculatorId.NextWeek,\n    SkyDateRangeCalculatorId.LastMonth,\n    SkyDateRangeCalculatorId.ThisMonth,\n    SkyDateRangeCalculatorId.NextMonth,\n    SkyDateRangeCalculatorId.LastQuarter,\n    SkyDateRangeCalculatorId.ThisQuarter,\n    SkyDateRangeCalculatorId.NextQuarter,\n    SkyDateRangeCalculatorId.LastCalendarYear,\n    SkyDateRangeCalculatorId.ThisCalendarYear,\n    SkyDateRangeCalculatorId.NextCalendarYear,\n    SkyDateRangeCalculatorId.LastFiscalYear,\n    SkyDateRangeCalculatorId.ThisFiscalYear,\n    SkyDateRangeCalculatorId.NextFiscalYear,\n  ];\n  #_dateFormat: string | undefined;\n  #_disabled: boolean | undefined = false;\n  #_valueOrDefault: SkyDateRangeCalculation | undefined;\n\n  #changeDetector: ChangeDetectorRef;\n  #dateRangeService: SkyDateRangeService;\n  #formBuilder: UntypedFormBuilder;\n  #localeProvider: SkyAppLocaleProvider;\n  #ngZone: NgZone;\n\n  constructor(\n    changeDetector: ChangeDetectorRef,\n    dateRangeService: SkyDateRangeService,\n    formBuilder: UntypedFormBuilder,\n    localeProvider: SkyAppLocaleProvider,\n    ngZone: NgZone,\n    @Optional() themeSvc?: SkyThemeService,\n  ) {\n    this.#changeDetector = changeDetector;\n    this.#dateRangeService = dateRangeService;\n    this.#formBuilder = formBuilder;\n    this.#localeProvider = localeProvider;\n    this.#ngZone = ngZone;\n\n    this.dateFormatOrDefault = this.dateFormat;\n\n    this.#localeProvider\n      .getLocaleInfo()\n      .pipe(takeUntil(this.#ngUnsubscribe))\n      .subscribe((localeInfo) => {\n        SkyDateFormatter.setLocale(localeInfo.locale);\n        this.#preferredShortDateFormat =\n          SkyDateFormatter.getPreferredShortDateFormat();\n        this.dateFormatOrDefault =\n          this.dateFormat || this.#preferredShortDateFormat;\n      });\n\n    // Update icons when theme changes.\n    /* istanbul ignore next */\n    themeSvc?.settingsChange\n      .pipe(takeUntil(this.#ngUnsubscribe))\n      .subscribe(() => {\n        this.#changeDetector.markForCheck();\n      });\n  }\n\n  public ngOnInit(): void {\n    this.#createForm();\n\n    this.#updateCalculators().then(() => {\n      if (!this.#value || !this.#value.calculatorId) {\n        this.#valueOrDefault = this.#defaultValue;\n      }\n\n      this.#addEventListeners();\n\n      this.isReady = true;\n\n      this.#showRelevantFormFields();\n\n      // We need to let Angular be stable and have rendered the components prior to setting the values and form controls. This ensures all initial validation will be ran correctly.\n      this.#ngZone.onStable.pipe(first()).subscribe(() => {\n        // Fill in any unprovided values after the calculators have been initialized.\n        // For example, if the control is initialized with only the `calculatorId`,\n        // allow the calculator to fill in the missing start and end dates.\n        const defaultValue = this.selectedCalculator?.getValue(\n          this.#valueOrDefault?.startDate,\n          this.#valueOrDefault?.endDate,\n        );\n        const newValue = Object.assign({}, defaultValue, this.#valueOrDefault);\n\n        this.#setValue(newValue, false);\n\n        this.#resetFormGroupValue();\n\n        // This is needed to address a bug in Angular 4.\n        // When a control value is set initially, its value is not represented on the view.\n        // See: https://github.com/angular/angular/issues/13792\n        /* istanbul ignore else */\n        if (this.#control) {\n          this.#control.setValue(this.#valueOrDefault, {\n            emitEvent: false,\n          });\n        }\n      });\n    });\n\n    this.#resourceSvc\n      .getString('skyux_datepicker_format_hint_text')\n      .pipe(takeUntil(this.#ngUnsubscribe))\n      .subscribe((templateString) => {\n        this.hostHintText = this.#appFormatter.formatText(\n          templateString,\n          this.dateFormatOrDefault,\n        );\n      });\n\n    if (this.#labelTextRequired && !this.label) {\n      this.display = 'none';\n    }\n    this.#labelTextRequired?.validateLabelText(this.label);\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    if (\n      changes['calculatorIds'] &&\n      changes['calculatorIds'].firstChange === false\n    ) {\n      this.#updateCalculators().then(() => {\n        const id = this.#calculatorIdControl?.value;\n\n        // Maintain the currently selected values if the calculators change after\n        // a value has been chosen.\n        const found = this.calculators.find((calculator) => {\n          return calculator.calculatorId === id;\n        });\n\n        /* istanbul ignore else */\n        if (!found) {\n          const newValue = this.#defaultCalculator?.getValue();\n          this.#setValue(newValue);\n          this.#resetFormGroupValue(newValue);\n          this.#showRelevantFormFields();\n        }\n      });\n    }\n  }\n\n  public ngOnDestroy(): void {\n    this.#ngUnsubscribe.next();\n    this.#ngUnsubscribe.complete();\n  }\n\n  public onFieldBlur(): void {\n    this.#onTouched();\n  }\n\n  public writeValue(value: SkyDateRangeCalculation): void {\n    // Only update the underlying controls when the calculators are ready.\n    const notifyChange = false;\n\n    // (We still need to save the initial value set by the consumer's form, however.)\n    this.#setValue(value, notifyChange);\n\n    if (this.isReady) {\n      // When the control's value is set to `null`,\n      // set it to the default value.\n      if (!value) {\n        this.#onChange(this.#defaultValue);\n      }\n\n      this.#resetFormGroupValue();\n      this.#showRelevantFormFields();\n    }\n  }\n\n  public validate(control: AbstractControl): ValidationErrors | null {\n    if (!this.#control) {\n      this.#control = control;\n    }\n\n    if (!this.isReady) {\n      return null;\n    }\n\n    const value = control.value;\n    const idControl = this.#calculatorIdControl;\n    const result = this.selectedCalculator?.validate(value);\n\n    let errors: ValidationErrors | null = null;\n\n    if (result) {\n      errors = {\n        skyDateRange: {\n          calculatorId: idControl?.value,\n          errors: result,\n        },\n      };\n    } else {\n      let startErrors: ValidationErrors | null = null;\n      let endErrors: ValidationErrors | null = null;\n      if (this.#startDateControl) {\n        startErrors = this.#startDateControl.errors;\n      }\n      if (this.#endDateControl) {\n        endErrors = this.#endDateControl.errors;\n      }\n\n      errors = startErrors || endErrors;\n    }\n\n    if (!errors) {\n      // Clear any errors on the calculator select.\n      idControl?.setErrors(null);\n      return null;\n    }\n\n    idControl?.setErrors(errors);\n    idControl?.markAsTouched();\n    idControl?.markAsDirty();\n\n    // Need to mark the control as touched for the error messages to appear.\n    this.#control.markAsTouched();\n\n    // Notify the view to display any errors.\n    this.#changeDetector.markForCheck();\n\n    return errors;\n  }\n\n  public registerOnChange(\n    fn: (value: SkyDateRangeCalculation | undefined) => SkyDateRangeCalculation,\n  ): void {\n    this.#onChange = fn;\n  }\n\n  public registerOnTouched(fn: () => SkyDateRangeCalculation): void {\n    this.#onTouched = fn;\n  }\n\n  public setDisabledState(disabled: boolean): void {\n    this.disabled = disabled;\n  }\n\n  #setValue(\n    value: SkyDateRangeCalculation | undefined,\n    notifyChange = true,\n  ): void {\n    const isNewValue = !this.#dateRangesEqual(this.#value, value);\n\n    if (isNewValue) {\n      this.#value = value;\n      if (!value || value.calculatorId === undefined) {\n        this.#valueOrDefault = this.#defaultValue;\n      } else {\n        this.#valueOrDefault = value;\n      }\n\n      if (notifyChange) {\n        this.#onChange(this.#valueOrDefault);\n      }\n    }\n  }\n\n  #patchValue(value: any): void {\n    const newValue = Object.assign({}, this.#valueOrDefault, value);\n\n    this.#setValue(newValue);\n  }\n\n  #createForm(): void {\n    this.formGroup = this.#formBuilder.group({\n      calculatorId: new UntypedFormControl(),\n      startDate: new UntypedFormControl(),\n      endDate: new UntypedFormControl(),\n    });\n\n    if (this.disabled) {\n      this.formGroup.disable();\n    }\n  }\n\n  #showRelevantFormFields(): void {\n    const calculator = this.selectedCalculator;\n\n    let showEndDatePicker = false;\n    let showStartDatePicker = false;\n\n    switch (calculator?.type) {\n      case SkyDateRangeCalculatorType.Before:\n        showEndDatePicker = true;\n        break;\n\n      case SkyDateRangeCalculatorType.After:\n        showStartDatePicker = true;\n        break;\n\n      case SkyDateRangeCalculatorType.Range:\n        showEndDatePicker = true;\n        showStartDatePicker = true;\n        break;\n\n      default:\n        break;\n    }\n\n    this.showEndDatePicker = showEndDatePicker;\n    this.showStartDatePicker = showStartDatePicker;\n    this.#changeDetector.markForCheck();\n  }\n\n  #resetFormGroupValue(value?: SkyDateRangeCalculation): void {\n    this.formGroup?.reset(value || this.#valueOrDefault);\n  }\n\n  #addEventListeners(): void {\n    // Watch for selected calculator change.\n    this.#calculatorIdControl?.valueChanges\n      .pipe(distinctUntilChanged(), takeUntil(this.#ngUnsubscribe))\n      .subscribe((value) => {\n        if (value !== this.#valueOrDefault?.calculatorId) {\n          const id = parseInt(value, 10);\n          // if the component is disabled during form creation, null is passed\n          // as the value of the calculator id control\n          // only handle the value changes if the calculator id is a number\n          /* istanbul ignore else */\n          if (!isNaN(id)) {\n            const calculator = this.#getCalculatorById(id);\n            const newValue = calculator?.getValue();\n\n            this.#setValue(newValue);\n            this.#resetFormGroupValue(newValue);\n            this.#showRelevantFormFields();\n          }\n        }\n        this.#control?.updateValueAndValidity({ emitEvent: false });\n      });\n\n    // Watch for start date value changes.\n    this.#startDateControl?.valueChanges\n      .pipe(distinctUntilChanged(), takeUntil(this.#ngUnsubscribe))\n      .subscribe((startDate) => {\n        this.#patchValue({ startDate });\n      });\n\n    // Watch for end date value changes.\n    this.#endDateControl?.valueChanges\n      .pipe(distinctUntilChanged(), takeUntil(this.#ngUnsubscribe))\n      .subscribe((endDate) => {\n        this.#patchValue({ endDate });\n      });\n\n    // Safety check\n    /* istanbul ignore else */\n    if (this.#startDateControl && this.#endDateControl) {\n      // Detect errors from the date inputs and update ng- classes on picker.\n      combineLatest([\n        this.#startDateControl.statusChanges,\n        this.#endDateControl.statusChanges,\n      ])\n        .pipe(takeUntil(this.#ngUnsubscribe))\n        .subscribe(() => {\n          this.#updateBasedOnControls();\n        });\n    } else if (this.#startDateControl) {\n      this.#startDateControl.statusChanges\n        .pipe(takeUntil(this.#ngUnsubscribe))\n        .subscribe(() => {\n          this.#updateBasedOnControls();\n        });\n    } else if (this.#endDateControl) {\n      this.#endDateControl.statusChanges\n        .pipe(takeUntil(this.#ngUnsubscribe))\n        .subscribe(() => {\n          this.#updateBasedOnControls();\n        });\n    }\n  }\n\n  #updateBasedOnControls(): void {\n    this.#changeDetector.markForCheck();\n\n    // Wait for initial validation to complete.\n    this.#ngZone.onStable.pipe(first()).subscribe(() => {\n      this.#control?.updateValueAndValidity({ emitEvent: false });\n    });\n  }\n\n  #updateCalculators(): Promise<void> {\n    return this.#dateRangeService\n      .getCalculators(this.calculatorIds)\n      .then((calculators) => {\n        this.calculators = calculators;\n        // Ensure that any previously set value is used to determine the selected calculator\n        this.#updateSelectedCalculator();\n        this.#changeDetector.markForCheck();\n      });\n  }\n\n  #getCalculatorById(\n    id: SkyDateRangeCalculatorId,\n  ): SkyDateRangeCalculator | undefined {\n    return this.calculators.find((calculator) => {\n      return calculator.calculatorId === id;\n    });\n  }\n\n  #dateRangesEqual(\n    rangeA: SkyDateRangeCalculation | undefined,\n    rangeB: SkyDateRangeCalculation | undefined,\n  ): boolean {\n    return (\n      !!rangeA && !!rangeB && JSON.stringify(rangeA) === JSON.stringify(rangeB)\n    );\n  }\n\n  #updateSelectedCalculator(): void {\n    this.selectedCalculator = this.#valueOrDefault\n      ? this.#getCalculatorById(this.#valueOrDefault.calculatorId)\n      : undefined;\n  }\n\n  /* istanbul ignore next */\n  // eslint-disable-next-line @typescript-eslint/no-empty-function, @typescript-eslint/no-unused-vars\n  #onChange = (_: SkyDateRangeCalculation | undefined): void => {};\n  /* istanbul ignore next */\n  // eslint-disable-next-line @typescript-eslint/no-empty-function\n  #onTouched = (): void => {};\n}\n","<div\n  *ngIf=\"isReady && formGroup\"\n  class=\"sky-date-range-picker\"\n  [formGroup]=\"formGroup\"\n>\n  <div\n    class=\"sky-date-range-picker-form-group\"\n    [ngClass]=\"{\n      'sky-date-range-picker-last-input':\n        !showStartDatePicker && !showEndDatePicker\n    }\"\n  >\n    <sky-input-box\n      [disabled]=\"disabled\"\n      [helpPopoverContent]=\"helpPopoverContent\"\n      [helpPopoverTitle]=\"helpPopoverTitle\"\n      [labelText]=\"\n        label || ('skyux_date_range_picker_default_label' | skyLibResources)\n      \"\n      [hintText]=\"hintText\"\n    >\n      <select\n        class=\"sky-form-control\"\n        formControlName=\"calculatorId\"\n        [attr.id]=\"dateRangePickerId + '-select-calculator'\"\n        (blur)=\"onFieldBlur()\"\n      >\n        <option\n          *ngFor=\"let calculator of calculators\"\n          [value]=\"calculator.calculatorId\"\n        >\n          {{ calculator.shortDescription }}\n        </option>\n      </select>\n    </sky-input-box>\n  </div>\n  <div\n    class=\"sky-date-range-picker-form-group\"\n    [ngClass]=\"{\n      'sky-date-range-picker-last-input':\n        showStartDatePicker && !showEndDatePicker\n    }\"\n    [hidden]=\"!showStartDatePicker\"\n  >\n    <sky-input-box [disabled]=\"disabled\">\n      <label\n        class=\"sky-control-label\"\n        [attr.for]=\"dateRangePickerId + '-start-date'\"\n        [ngClass]=\"{ 'sky-control-label-required': startDateRequired }\"\n      >\n        {{\n          selectedCalculator?.type\n            | skyDateRangePickerStartDateResourceKey\n            | skyLibResources\n        }}\n      </label>\n\n      <sky-datepicker>\n        <input\n          class=\"sky-form-control\"\n          formControlName=\"startDate\"\n          [attr.aria-label]=\"\n            label\n              ? ('skyux_date_range_picker_default_aria_label'\n                | skyLibResources\n                  : (selectedCalculator?.type\n                      | skyDateRangePickerStartDateResourceKey\n                      | skyLibResources)\n                  : label)\n              : (selectedCalculator?.type\n                | skyDateRangePickerStartDateResourceKey\n                | skyLibResources)\n          \"\n          [attr.id]=\"dateRangePickerId + '-start-date'\"\n          [required]=\"!!startDateRequired\"\n          [dateFormat]=\"dateFormatOrDefault\"\n          (blur)=\"onFieldBlur()\"\n          skyDatepickerInput\n        />\n      </sky-datepicker>\n    </sky-input-box>\n  </div>\n\n  <div\n    class=\"sky-date-range-picker-form-group\"\n    [ngClass]=\"{ 'sky-date-range-picker-last-input': showEndDatePicker }\"\n    [hidden]=\"!showEndDatePicker\"\n  >\n    <sky-input-box [disabled]=\"disabled\">\n      <label\n        class=\"sky-control-label\"\n        [attr.for]=\"dateRangePickerId + '-end-date'\"\n        [ngClass]=\"{ 'sky-control-label-required': endDateRequired }\"\n      >\n        {{\n          selectedCalculator?.type\n            | skyDateRangePickerEndDateResourceKey\n            | skyLibResources\n        }}\n      </label>\n\n      <sky-datepicker>\n        <input\n          class=\"sky-form-control\"\n          formControlName=\"endDate\"\n          skyDatepickerInput\n          [attr.aria-label]=\"\n            label\n              ? ('skyux_date_range_picker_default_aria_label'\n                | skyLibResources\n                  : (selectedCalculator?.type\n                      | skyDateRangePickerEndDateResourceKey\n                      | skyLibResources)\n                  : label)\n              : (selectedCalculator?.type\n                | skyDateRangePickerEndDateResourceKey\n                | skyLibResources)\n          \"\n          [attr.id]=\"dateRangePickerId + '-end-date'\"\n          [dateFormat]=\"dateFormatOrDefault\"\n          [required]=\"!!endDateRequired\"\n          (blur)=\"onFieldBlur()\"\n        />\n      </sky-datepicker>\n    </sky-input-box>\n  </div>\n</div>\n"]}
441
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-range-picker.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/datetime/src/lib/modules/date-range-picker/date-range-picker.component.ts","../../../../../../../../libs/components/datetime/src/lib/modules/date-range-picker/date-range-picker.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  DoCheck,\n  HostBinding,\n  Injector,\n  Input,\n  OnDestroy,\n  OnInit,\n  TemplateRef,\n  booleanAttribute,\n  inject,\n} from '@angular/core';\nimport {\n  AbstractControl,\n  ControlValueAccessor,\n  FormBuilder,\n  FormControl,\n  FormGroup,\n  FormsModule,\n  NG_VALIDATORS,\n  NG_VALUE_ACCESSOR,\n  NgControl,\n  ReactiveFormsModule,\n  ValidationErrors,\n  Validator,\n  Validators,\n} from '@angular/forms';\nimport {\n  SkyFormFieldLabelTextRequiredService,\n  SkyInputBoxModule,\n} from '@skyux/forms';\n\nimport { Subject, distinctUntilChanged, merge, takeUntil } from 'rxjs';\n\nimport { SkyDatepickerModule } from '../datepicker/datepicker.module';\nimport { SkyDatetimeResourcesModule } from '../shared/sky-datetime-resources.module';\n\nimport { SkyDateRangePickerEndDateResourceKeyPipe } from './date-range-picker-end-date-resource-key.pipe';\nimport { SkyDateRangePickerStartDateResourceKeyPipe } from './date-range-picker-start-date-resource-key.pipe';\nimport { SkyDateRangeService } from './date-range.service';\nimport { SkyDateRangeCalculation } from './types/date-range-calculation';\nimport { SkyDateRangeCalculator } from './types/date-range-calculator';\nimport { SkyDateRangeCalculatorId } from './types/date-range-calculator-id';\nimport { SkyDateRangeCalculatorType } from './types/date-range-calculator-type';\nimport { SKY_DEFAULT_CALCULATOR_IDS } from './types/date-range-default-calculator-configs';\n\ntype DateValue = Date | string | null | undefined;\n\nfunction areDateRangesEqual(\n  rangeA: SkyDateRangeCalculation | undefined,\n  rangeB: SkyDateRangeCalculation | undefined,\n): boolean {\n  return (\n    !!rangeA &&\n    !!rangeB &&\n    rangeA.calculatorId === rangeB.calculatorId &&\n    areDatesEqual(rangeA.startDate, rangeB.startDate) &&\n    areDatesEqual(rangeA.endDate, rangeB.endDate)\n  );\n}\n\nfunction areDatesEqual(a: DateValue, b: DateValue): boolean {\n  if (typeof a !== typeof b) {\n    return false;\n  }\n\n  if (!a && !b) {\n    return true;\n  }\n\n  if (typeof a === 'string' && a === b) {\n    return true;\n  }\n\n  return a instanceof Date && b instanceof Date && a.getTime() === b.getTime();\n}\n\nfunction isNullOrUndefined(value: unknown): value is undefined | null {\n  return value === undefined || value === null;\n}\n\nfunction isPartialValue(\n  value: Partial<SkyDateRangeCalculation> | null | undefined,\n): value is Partial<SkyDateRangeCalculation> | null | undefined {\n  return (\n    isNullOrUndefined(value) ||\n    isNullOrUndefined(value.calculatorId) ||\n    !('endDate' in value) ||\n    !('startDate' in value)\n  );\n}\n\n@Component({\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [\n    CommonModule,\n    FormsModule,\n    ReactiveFormsModule,\n    SkyDatepickerModule,\n    SkyDateRangePickerEndDateResourceKeyPipe,\n    SkyDateRangePickerStartDateResourceKeyPipe,\n    SkyDatetimeResourcesModule,\n    SkyInputBoxModule,\n  ],\n  providers: [\n    {\n      provide: NG_VALIDATORS,\n      useExisting: SkyDateRangePickerComponent,\n      multi: true,\n    },\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: SkyDateRangePickerComponent,\n      multi: true,\n    },\n  ],\n  selector: 'sky-date-range-picker',\n  standalone: true,\n  styleUrl: './date-range-picker.component.scss',\n  templateUrl: './date-range-picker.component.html',\n})\nexport class SkyDateRangePickerComponent\n  implements\n    AfterViewInit,\n    ControlValueAccessor,\n    DoCheck,\n    OnDestroy,\n    OnInit,\n    Validator\n{\n  /**\n   * IDs for the date range options to include in the picker's dropdown.\n   * The options specify calculator objects that return two `Date` objects to represent date ranges.\n   * By default, this property includes all `SkyDateRangeCalculatorId` values.\n   */\n  @Input()\n  public set calculatorIds(\n    calculatorIds: SkyDateRangeCalculatorId[] | undefined,\n  ) {\n    const currentCalculatorId = this.#getValue().calculatorId;\n\n    this.#_calculatorIds = calculatorIds ?? SKY_DEFAULT_CALCULATOR_IDS;\n    this.calculators = this.#dateRangeSvc.filterCalculators(\n      this.#_calculatorIds,\n    );\n\n    // If the currently selected calculator isn't available anymore,\n    // select the first calculator in the new array.\n    if (!this.#_calculatorIds.includes(currentCalculatorId)) {\n      this.#setValue(\n        { calculatorId: this.calculatorIds[0] },\n        { emitEvent: true },\n      );\n    }\n  }\n\n  public get calculatorIds(): SkyDateRangeCalculatorId[] {\n    return this.#_calculatorIds;\n  }\n\n  /**\n   * The date format for\n   * [the `sky-datepicker` components](https://developer.blackbaud.com/skyux/components/datepicker)\n   * that make up the date range picker. The text input is a composite component of\n   * up to two `sky-datepicker` components.\n   * @default \"MM/DD/YYYY\"\n   */\n  @Input()\n  public dateFormat: string | undefined;\n\n  /**\n   * Whether to disable the date range picker on template-driven forms. Don't use\n   * this input on reactive forms because they may overwrite the input or leave\n   * the control out of sync. To set the disabled state on reactive forms,\n   * use the `FormControl` instead.\n   * @default false\n   */\n  @Input({ transform: booleanAttribute })\n  public set disabled(value: boolean) {\n    if (value) {\n      this.formGroup.disable();\n    } else {\n      this.formGroup.enable();\n    }\n  }\n\n  /**\n   * Whether to require users to specify a end date.\n   * @deprecated Use the `required` directive or Angular's `Validators.required`\n   * on the form control to mark the date range picker as required.\n   */\n  @Input({ transform: booleanAttribute })\n  public endDateRequired = false;\n\n  /**\n   * The content of the help popover. When specified along with `labelText`, a [help inline](https://developer.blackbaud.com/skyux/components/help-inline)\n   * button is added to date range picker. The help inline button displays a [popover](https://developer.blackbaud.com/skyux/components/popover)\n   * when clicked using the specified content and optional title.\n   * @preview\n   */\n  @Input()\n  public helpPopoverContent: string | TemplateRef<unknown> | undefined;\n\n  /**\n   * The title of the help popover. This property only applies when `helpPopoverContent` is\n   * also specified.\n   * @preview\n   */\n  @Input()\n  public helpPopoverTitle: string | undefined;\n\n  /**\n   * [Persistent inline help text](https://developer.blackbaud.com/skyux/design/guidelines/user-assistance#inline-help) that provides\n   * additional context to the user.\n   * @preview\n   */\n  @Input()\n  public hintText: string | undefined;\n\n  /**\n   * The label for the date range picker.\n   * @required\n   */\n  @Input()\n  public label: string | undefined;\n\n  /**\n   * Whether the date range picker requires a value.\n   */\n  @Input({ transform: booleanAttribute })\n  public required = false;\n\n  /**\n   * Whether the date range picker is stacked on another form component. When specified, the appropriate\n   * vertical spacing is automatically added to the date range picker.\n   * @preview\n   */\n  @Input({ transform: booleanAttribute })\n  @HostBinding('class.sky-margin-stacked-lg')\n  public stacked = false;\n\n  /**\n   * Whether to require users to specify a start date.\n   * @deprecated Use the `required` directive or Angular's `Validators.required`\n   * on the form control to mark the date range picker as required.\n   */\n  @Input({ transform: booleanAttribute })\n  public startDateRequired = false;\n\n  @HostBinding('style.display')\n  protected display: string | undefined;\n\n  protected calculators: SkyDateRangeCalculator[] = [];\n  protected formGroup: FormGroup;\n  protected hasErrors = false;\n  protected selectedCalculator: SkyDateRangeCalculator;\n  protected showEndDatePicker = false;\n  protected showStartDatePicker = false;\n\n  get #calculatorIdControl(): AbstractControl<SkyDateRangeCalculatorId> {\n    return this.formGroup.get(\n      'calculatorId',\n    ) as AbstractControl<SkyDateRangeCalculatorId>;\n  }\n\n  get #endDateControl(): AbstractControl<DateValue> {\n    return this.formGroup.get('endDate') as AbstractControl<DateValue>;\n  }\n\n  get #startDateControl(): AbstractControl<DateValue> {\n    return this.formGroup.get('startDate') as AbstractControl<DateValue>;\n  }\n\n  #_calculatorIds = SKY_DEFAULT_CALCULATOR_IDS;\n  #_value: SkyDateRangeCalculation;\n  #hostControl: AbstractControl | null | undefined;\n  #ngUnsubscribe = new Subject<void>();\n  #notifyChange: ((_: SkyDateRangeCalculation) => void) | undefined;\n  #notifyTouched: (() => void) | undefined;\n\n  readonly #changeDetector = inject(ChangeDetectorRef);\n  readonly #dateRangeSvc = inject(SkyDateRangeService);\n  readonly #injector = inject(Injector);\n  readonly #labelTextRequiredSvc = inject(\n    SkyFormFieldLabelTextRequiredService,\n    {\n      optional: true,\n    },\n  );\n\n  constructor() {\n    this.calculators = this.#dateRangeSvc.calculators;\n    this.selectedCalculator = this.calculators[0];\n\n    const initialValue = this.#getDefaultValue(this.selectedCalculator);\n    this.#_value = initialValue;\n\n    this.formGroup = inject(FormBuilder).group({\n      calculatorId: new FormControl<number>(initialValue.calculatorId),\n      startDate: new FormControl<DateValue>(initialValue.startDate),\n      endDate: new FormControl<DateValue>(initialValue.endDate),\n    });\n  }\n\n  public ngOnInit(): void {\n    if (this.#labelTextRequiredSvc) {\n      this.#labelTextRequiredSvc.validateLabelText(this.label);\n\n      if (!this.label) {\n        this.display = 'none';\n      }\n    }\n  }\n\n  public ngAfterViewInit(): void {\n    this.#hostControl = this.#injector.get(NgControl, null, {\n      optional: true,\n      self: true,\n    })?.control;\n\n    // Set a default value on the control if it's undefined on init.\n    // We need to use setTimeout to avoid interfering with the first\n    // validation cycle.\n    if (isPartialValue(this.#hostControl?.value)) {\n      setTimeout(() => {\n        this.#hostControl?.setValue(this.#getValue(), {\n          emitEvent: false,\n          onlySelf: true,\n        });\n      });\n    }\n\n    // Update the view when \"required\" or \"disabled\" states are changed on the\n    // host control.\n    this.#hostControl?.statusChanges\n      .pipe(distinctUntilChanged(), takeUntil(this.#ngUnsubscribe))\n      .subscribe(() => {\n        this.#changeDetector.markForCheck();\n      });\n\n    // Start listening for changes after the first change detection cycle.\n    setTimeout(() => {\n      this.formGroup.valueChanges\n        .pipe(\n          distinctUntilChanged(areDateRangesEqual),\n          takeUntil(this.#ngUnsubscribe),\n        )\n        .subscribe((value) => {\n          if (!isNullOrUndefined(value?.calculatorId)) {\n            // The select element sets the calculator ID to a string, but we\n            // need it to be a number.\n            value.calculatorId = +value.calculatorId;\n\n            // If the calculator ID is changed, we need to reset the start and\n            // end date values and wait until the next valueChanges event to\n            // notify the host control.\n            if (value.calculatorId !== this.#getValue().calculatorId) {\n              this.#setValue(\n                { calculatorId: value.calculatorId },\n                { emitEvent: true },\n              );\n\n              return;\n            }\n          }\n\n          this.#setValue(value, { emitEvent: false });\n          const newValue = this.#getValue();\n\n          // Update the host control if the value is different.\n          if (!areDateRangesEqual(this.#hostControl?.value, newValue)) {\n            this.#notifyChange?.(newValue);\n          }\n        });\n    });\n\n    // If the datepickers' statuses change, we want to retrigger the host\n    // control's validation so that their errors are reflected back to the host.\n    merge(\n      this.#startDateControl.statusChanges,\n      this.#endDateControl.statusChanges,\n    )\n      .pipe(distinctUntilChanged(), takeUntil(this.#ngUnsubscribe))\n      .subscribe(() => {\n        // Use a setTimeout to avoid an ExpressionChangedAfterChecked error,\n        // since multiple calls to updateValueAndValidity in the same\n        // cycle may collide with one another.\n        setTimeout(() => {\n          this.#hostControl?.updateValueAndValidity({\n            emitEvent: false,\n            onlySelf: true,\n          });\n        });\n      });\n\n    this.#updatePickerVisibility(this.selectedCalculator);\n  }\n\n  /**\n   * Check for touched status in ngDoCheck since Angular does not (currently)\n   * have an API to respond to touched status changes from the host control.\n   * @see https://github.com/angular/angular/issues/17736#issuecomment-310812368\n   * TODO: Angular 18 introduces a new API to respond to these statuses.\n   * @see https://github.com/angular/angular/issues/10887#issuecomment-2035267400\n   */\n  public ngDoCheck(): void {\n    const control = this.#hostControl;\n    const touched = this.formGroup.touched;\n\n    if (control) {\n      if (control.touched && !touched) {\n        this.formGroup.markAllAsTouched();\n        this.#changeDetector.markForCheck();\n      } else if (control.untouched && touched) {\n        this.formGroup.markAsUntouched();\n        this.#changeDetector.markForCheck();\n      }\n\n      this.hasErrors = !!control.errors && (control.touched || control.dirty);\n    }\n  }\n\n  public ngOnDestroy(): void {\n    this.#ngUnsubscribe.next();\n    this.#ngUnsubscribe.complete();\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  public registerOnChange(fn: (_: unknown) => void): void {\n    this.#notifyChange = fn;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  public registerOnTouched(fn: () => void): void {\n    this.#notifyTouched = fn;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  public setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  // Implemented as part of Validator.\n  public validate(control: AbstractControl): ValidationErrors | null {\n    let errors: ValidationErrors | null = null;\n\n    const calculatorErrors = this.selectedCalculator.validate(control.value);\n    const startDateErrors = this.#startDateControl.errors;\n    const endDateErrors = this.#endDateControl.errors;\n\n    if (calculatorErrors) {\n      errors = {\n        skyDateRange: {\n          calculatorId: this.#getValue().calculatorId,\n          errors: calculatorErrors,\n        },\n      };\n    }\n\n    if (this.showStartDatePicker && startDateErrors) {\n      errors ||= {};\n      errors = { ...errors, ...startDateErrors };\n    }\n\n    if (this.showEndDatePicker && endDateErrors) {\n      errors ||= {};\n      errors = { ...errors, ...endDateErrors };\n    }\n\n    // Set errors on the calculator select so that they appear beneath it.\n    this.#calculatorIdControl.setErrors(errors);\n    this.#changeDetector.markForCheck();\n\n    return errors;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  // The date range picker always has a value, so if the consumer passes in a\n  // partial value (via `patchValue`) or null, we need to update the host control's\n  // value with the complete value after it's been modified.\n  public writeValue(value: Partial<SkyDateRangeCalculation> | undefined): void {\n    this.#patchValue(value);\n\n    // Update the host control if it is set to a partial or null value.\n    if (isPartialValue(value)) {\n      this.#hostControl?.setValue(this.#getValue(), {\n        emitEvent: false,\n        onlySelf: true,\n      });\n    }\n  }\n\n  protected isRequired(): boolean {\n    return !!(\n      this.required || this.#hostControl?.hasValidator(Validators.required)\n    );\n  }\n\n  protected onBlur(): void {\n    this.#notifyTouched?.();\n  }\n\n  #getCalculator(calculatorId: number): SkyDateRangeCalculator {\n    const found = this.calculators.find((c) => c.calculatorId === calculatorId);\n\n    /*safety check: should not happen*/\n    /*istanbul ignore if*/\n    if (!found) {\n      throw new Error(\n        `A date range calculator with ID (${calculatorId}) could not be found.`,\n      );\n    }\n\n    return found;\n  }\n\n  #getDefaultValue(\n    calculator: SkyDateRangeCalculator,\n  ): SkyDateRangeCalculation {\n    return calculator.getValue();\n  }\n\n  #getValue(): SkyDateRangeCalculation {\n    return this.#_value;\n  }\n\n  #patchValue(\n    partialValue: Partial<SkyDateRangeCalculation> | null | undefined,\n  ): void {\n    this.#setValue(\n      isNullOrUndefined(partialValue)\n        ? null\n        : { ...this.#getValue(), ...partialValue },\n      { emitEvent: true },\n    );\n  }\n\n  /**\n   * Sets the value to be used by the date range picker form control.\n   */\n  #setValue(\n    value: SkyDateRangeCalculation | null | undefined,\n    options: { emitEvent: boolean },\n  ): void {\n    const oldValue = this.#getValue();\n\n    const valueOrDefault =\n      !value || isNullOrUndefined(value.calculatorId)\n        ? this.#getDefaultValue(this.calculators[0])\n        : {\n            ...this.#getDefaultValue(this.#getCalculator(value.calculatorId)),\n            ...value,\n          };\n\n    // Ensure falsy values are set to null.\n    valueOrDefault.endDate = valueOrDefault.endDate || null;\n    valueOrDefault.startDate = valueOrDefault.startDate || null;\n\n    if (!areDateRangesEqual(oldValue, valueOrDefault)) {\n      this.#_value = valueOrDefault;\n\n      this.selectedCalculator = this.#getCalculator(\n        valueOrDefault.calculatorId,\n      );\n\n      if (oldValue.calculatorId !== valueOrDefault.calculatorId) {\n        this.#updatePickerVisibility(this.selectedCalculator);\n      }\n\n      if (options?.emitEvent) {\n        this.formGroup.setValue(valueOrDefault);\n      }\n    }\n  }\n\n  #updatePickerVisibility(calculator: SkyDateRangeCalculator): void {\n    let showEndDatePicker = false;\n    let showStartDatePicker = false;\n\n    switch (calculator.type) {\n      case SkyDateRangeCalculatorType.Before:\n        showEndDatePicker = true;\n        break;\n\n      case SkyDateRangeCalculatorType.After:\n        showStartDatePicker = true;\n        break;\n\n      case SkyDateRangeCalculatorType.Range:\n        showEndDatePicker = true;\n        showStartDatePicker = true;\n        break;\n\n      default:\n        break;\n    }\n\n    this.showEndDatePicker = showEndDatePicker;\n    this.showStartDatePicker = showStartDatePicker;\n    this.#changeDetector.markForCheck();\n  }\n}\n","<div\n  class=\"sky-date-range-picker\"\n  [formGroup]=\"formGroup\"\n  (focusout)=\"onBlur()\"\n>\n  <div\n    class=\"sky-date-range-picker-form-group\"\n    [ngClass]=\"{\n      'sky-date-range-picker-last-input':\n        !showStartDatePicker && !showEndDatePicker\n    }\"\n  >\n    <sky-input-box\n      [hasErrors]=\"hasErrors\"\n      [helpPopoverContent]=\"helpPopoverContent\"\n      [helpPopoverTitle]=\"helpPopoverTitle\"\n      [hintText]=\"hintText\"\n      [labelText]=\"\n        label || ('skyux_date_range_picker_default_label' | skyLibResources)\n      \"\n    >\n      <select\n        formControlName=\"calculatorId\"\n        [required]=\"isRequired()\"\n        (blur)=\"onBlur()\"\n      >\n        <option\n          *ngFor=\"let calculator of calculators\"\n          [value]=\"calculator.calculatorId\"\n        >\n          {{\n            calculator._shortDescriptionResourceKey\n              ? (calculator._shortDescriptionResourceKey | skyLibResources)\n              : calculator.shortDescription\n          }}\n        </option>\n      </select>\n    </sky-input-box>\n  </div>\n  <div\n    class=\"sky-date-range-picker-form-group sky-date-range-datepicker-wrapper\"\n    [hidden]=\"!showStartDatePicker\"\n    [ngClass]=\"{\n      'sky-date-range-picker-last-input':\n        showStartDatePicker && !showEndDatePicker\n    }\"\n  >\n    <sky-input-box\n      [errorsScreenReaderOnly]=\"true\"\n      [hasErrors]=\"hasErrors\"\n      [labelText]=\"\n        selectedCalculator.type\n          | skyDateRangePickerStartDateResourceKey\n          | skyLibResources\n      \"\n    >\n      <sky-datepicker>\n        <input\n          formControlName=\"startDate\"\n          name=\"startDate\"\n          skyDatepickerInput\n          type=\"text\"\n          [attr.aria-label]=\"\n            label\n              ? ('skyux_date_range_picker_default_aria_label'\n                | skyLibResources\n                  : (selectedCalculator.type\n                      | skyDateRangePickerStartDateResourceKey\n                      | skyLibResources)\n                  : label)\n              : (selectedCalculator.type\n                | skyDateRangePickerStartDateResourceKey\n                | skyLibResources)\n          \"\n          [dateFormat]=\"dateFormat\"\n          [required]=\"\n            showStartDatePicker && (startDateRequired || isRequired())\n          \"\n        />\n      </sky-datepicker>\n    </sky-input-box>\n  </div>\n  <div\n    class=\"sky-date-range-picker-form-group sky-date-range-datepicker-wrapper\"\n    [hidden]=\"!showEndDatePicker\"\n    [ngClass]=\"{ 'sky-date-range-picker-last-input': showEndDatePicker }\"\n  >\n    <sky-input-box\n      [errorsScreenReaderOnly]=\"true\"\n      [hasErrors]=\"hasErrors\"\n      [labelText]=\"\n        selectedCalculator.type\n          | skyDateRangePickerEndDateResourceKey\n          | skyLibResources\n      \"\n    >\n      <sky-datepicker>\n        <input\n          formControlName=\"endDate\"\n          name=\"endDate\"\n          skyDatepickerInput\n          type=\"text\"\n          [attr.aria-label]=\"\n            label\n              ? ('skyux_date_range_picker_default_aria_label'\n                | skyLibResources\n                  : (selectedCalculator.type\n                      | skyDateRangePickerEndDateResourceKey\n                      | skyLibResources)\n                  : label)\n              : (selectedCalculator.type\n                | skyDateRangePickerEndDateResourceKey\n                | skyLibResources)\n          \"\n          [dateFormat]=\"dateFormat\"\n          [required]=\"showEndDatePicker && (endDateRequired || isRequired())\"\n        />\n      </sky-datepicker>\n    </sky-input-box>\n  </div>\n</div>\n"]}