1 | /**
|
---|
2 | * @license
|
---|
3 | * Copyright Google LLC All Rights Reserved.
|
---|
4 | *
|
---|
5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
6 | * found in the LICENSE file at https://angular.io/license
|
---|
7 | */
|
---|
8 | import { coerceBooleanProperty } from '@angular/cdk/coercion';
|
---|
9 | import { DOWN_ARROW } from '@angular/cdk/keycodes';
|
---|
10 | import { Directive, ElementRef, EventEmitter, Inject, Input, Optional, Output, } from '@angular/core';
|
---|
11 | import { DateAdapter, MAT_DATE_FORMATS, } from '@angular/material/core';
|
---|
12 | import { Subscription, Subject } from 'rxjs';
|
---|
13 | import { createMissingDateImplError } from './datepicker-errors';
|
---|
14 | /**
|
---|
15 | * An event used for datepicker input and change events. We don't always have access to a native
|
---|
16 | * input or change event because the event may have been triggered by the user clicking on the
|
---|
17 | * calendar popup. For consistency, we always use MatDatepickerInputEvent instead.
|
---|
18 | */
|
---|
19 | export class MatDatepickerInputEvent {
|
---|
20 | constructor(
|
---|
21 | /** Reference to the datepicker input component that emitted the event. */
|
---|
22 | target,
|
---|
23 | /** Reference to the native input element associated with the datepicker input. */
|
---|
24 | targetElement) {
|
---|
25 | this.target = target;
|
---|
26 | this.targetElement = targetElement;
|
---|
27 | this.value = this.target.value;
|
---|
28 | }
|
---|
29 | }
|
---|
30 | /** Base class for datepicker inputs. */
|
---|
31 | export class MatDatepickerInputBase {
|
---|
32 | constructor(_elementRef, _dateAdapter, _dateFormats) {
|
---|
33 | this._elementRef = _elementRef;
|
---|
34 | this._dateAdapter = _dateAdapter;
|
---|
35 | this._dateFormats = _dateFormats;
|
---|
36 | /** Emits when a `change` event is fired on this `<input>`. */
|
---|
37 | this.dateChange = new EventEmitter();
|
---|
38 | /** Emits when an `input` event is fired on this `<input>`. */
|
---|
39 | this.dateInput = new EventEmitter();
|
---|
40 | /** Emits when the internal state has changed */
|
---|
41 | this.stateChanges = new Subject();
|
---|
42 | this._onTouched = () => { };
|
---|
43 | this._validatorOnChange = () => { };
|
---|
44 | this._cvaOnChange = () => { };
|
---|
45 | this._valueChangesSubscription = Subscription.EMPTY;
|
---|
46 | this._localeSubscription = Subscription.EMPTY;
|
---|
47 | /** The form control validator for whether the input parses. */
|
---|
48 | this._parseValidator = () => {
|
---|
49 | return this._lastValueValid ?
|
---|
50 | null : { 'matDatepickerParse': { 'text': this._elementRef.nativeElement.value } };
|
---|
51 | };
|
---|
52 | /** The form control validator for the date filter. */
|
---|
53 | this._filterValidator = (control) => {
|
---|
54 | const controlValue = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(control.value));
|
---|
55 | return !controlValue || this._matchesFilter(controlValue) ?
|
---|
56 | null : { 'matDatepickerFilter': true };
|
---|
57 | };
|
---|
58 | /** The form control validator for the min date. */
|
---|
59 | this._minValidator = (control) => {
|
---|
60 | const controlValue = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(control.value));
|
---|
61 | const min = this._getMinDate();
|
---|
62 | return (!min || !controlValue ||
|
---|
63 | this._dateAdapter.compareDate(min, controlValue) <= 0) ?
|
---|
64 | null : { 'matDatepickerMin': { 'min': min, 'actual': controlValue } };
|
---|
65 | };
|
---|
66 | /** The form control validator for the max date. */
|
---|
67 | this._maxValidator = (control) => {
|
---|
68 | const controlValue = this._dateAdapter.getValidDateOrNull(this._dateAdapter.deserialize(control.value));
|
---|
69 | const max = this._getMaxDate();
|
---|
70 | return (!max || !controlValue ||
|
---|
71 | this._dateAdapter.compareDate(max, controlValue) >= 0) ?
|
---|
72 | null : { 'matDatepickerMax': { 'max': max, 'actual': controlValue } };
|
---|
73 | };
|
---|
74 | /** Whether the last value set on the input was valid. */
|
---|
75 | this._lastValueValid = false;
|
---|
76 | if (typeof ngDevMode === 'undefined' || ngDevMode) {
|
---|
77 | if (!this._dateAdapter) {
|
---|
78 | throw createMissingDateImplError('DateAdapter');
|
---|
79 | }
|
---|
80 | if (!this._dateFormats) {
|
---|
81 | throw createMissingDateImplError('MAT_DATE_FORMATS');
|
---|
82 | }
|
---|
83 | }
|
---|
84 | // Update the displayed date when the locale changes.
|
---|
85 | this._localeSubscription = _dateAdapter.localeChanges.subscribe(() => {
|
---|
86 | this._assignValueProgrammatically(this.value);
|
---|
87 | });
|
---|
88 | }
|
---|
89 | /** The value of the input. */
|
---|
90 | get value() {
|
---|
91 | return this._model ? this._getValueFromModel(this._model.selection) : this._pendingValue;
|
---|
92 | }
|
---|
93 | set value(value) {
|
---|
94 | this._assignValueProgrammatically(value);
|
---|
95 | }
|
---|
96 | /** Whether the datepicker-input is disabled. */
|
---|
97 | get disabled() { return !!this._disabled || this._parentDisabled(); }
|
---|
98 | set disabled(value) {
|
---|
99 | const newValue = coerceBooleanProperty(value);
|
---|
100 | const element = this._elementRef.nativeElement;
|
---|
101 | if (this._disabled !== newValue) {
|
---|
102 | this._disabled = newValue;
|
---|
103 | this.stateChanges.next(undefined);
|
---|
104 | }
|
---|
105 | // We need to null check the `blur` method, because it's undefined during SSR.
|
---|
106 | // In Ivy static bindings are invoked earlier, before the element is attached to the DOM.
|
---|
107 | // This can cause an error to be thrown in some browsers (IE/Edge) which assert that the
|
---|
108 | // element has been inserted.
|
---|
109 | if (newValue && this._isInitialized && element.blur) {
|
---|
110 | // Normally, native input elements automatically blur if they turn disabled. This behavior
|
---|
111 | // is problematic, because it would mean that it triggers another change detection cycle,
|
---|
112 | // which then causes a changed after checked error if the input element was focused before.
|
---|
113 | element.blur();
|
---|
114 | }
|
---|
115 | }
|
---|
116 | /** Gets the base validator functions. */
|
---|
117 | _getValidators() {
|
---|
118 | return [this._parseValidator, this._minValidator, this._maxValidator, this._filterValidator];
|
---|
119 | }
|
---|
120 | /** Registers a date selection model with the input. */
|
---|
121 | _registerModel(model) {
|
---|
122 | this._model = model;
|
---|
123 | this._valueChangesSubscription.unsubscribe();
|
---|
124 | if (this._pendingValue) {
|
---|
125 | this._assignValue(this._pendingValue);
|
---|
126 | }
|
---|
127 | this._valueChangesSubscription = this._model.selectionChanged.subscribe(event => {
|
---|
128 | if (this._shouldHandleChangeEvent(event)) {
|
---|
129 | const value = this._getValueFromModel(event.selection);
|
---|
130 | this._lastValueValid = this._isValidValue(value);
|
---|
131 | this._cvaOnChange(value);
|
---|
132 | this._onTouched();
|
---|
133 | this._formatValue(value);
|
---|
134 | this.dateInput.emit(new MatDatepickerInputEvent(this, this._elementRef.nativeElement));
|
---|
135 | this.dateChange.emit(new MatDatepickerInputEvent(this, this._elementRef.nativeElement));
|
---|
136 | }
|
---|
137 | });
|
---|
138 | }
|
---|
139 | ngAfterViewInit() {
|
---|
140 | this._isInitialized = true;
|
---|
141 | }
|
---|
142 | ngOnChanges(changes) {
|
---|
143 | if (dateInputsHaveChanged(changes, this._dateAdapter)) {
|
---|
144 | this.stateChanges.next(undefined);
|
---|
145 | }
|
---|
146 | }
|
---|
147 | ngOnDestroy() {
|
---|
148 | this._valueChangesSubscription.unsubscribe();
|
---|
149 | this._localeSubscription.unsubscribe();
|
---|
150 | this.stateChanges.complete();
|
---|
151 | }
|
---|
152 | /** @docs-private */
|
---|
153 | registerOnValidatorChange(fn) {
|
---|
154 | this._validatorOnChange = fn;
|
---|
155 | }
|
---|
156 | /** @docs-private */
|
---|
157 | validate(c) {
|
---|
158 | return this._validator ? this._validator(c) : null;
|
---|
159 | }
|
---|
160 | // Implemented as part of ControlValueAccessor.
|
---|
161 | writeValue(value) {
|
---|
162 | this._assignValueProgrammatically(value);
|
---|
163 | }
|
---|
164 | // Implemented as part of ControlValueAccessor.
|
---|
165 | registerOnChange(fn) {
|
---|
166 | this._cvaOnChange = fn;
|
---|
167 | }
|
---|
168 | // Implemented as part of ControlValueAccessor.
|
---|
169 | registerOnTouched(fn) {
|
---|
170 | this._onTouched = fn;
|
---|
171 | }
|
---|
172 | // Implemented as part of ControlValueAccessor.
|
---|
173 | setDisabledState(isDisabled) {
|
---|
174 | this.disabled = isDisabled;
|
---|
175 | }
|
---|
176 | _onKeydown(event) {
|
---|
177 | const isAltDownArrow = event.altKey && event.keyCode === DOWN_ARROW;
|
---|
178 | if (isAltDownArrow && !this._elementRef.nativeElement.readOnly) {
|
---|
179 | this._openPopup();
|
---|
180 | event.preventDefault();
|
---|
181 | }
|
---|
182 | }
|
---|
183 | _onInput(value) {
|
---|
184 | const lastValueWasValid = this._lastValueValid;
|
---|
185 | let date = this._dateAdapter.parse(value, this._dateFormats.parse.dateInput);
|
---|
186 | this._lastValueValid = this._isValidValue(date);
|
---|
187 | date = this._dateAdapter.getValidDateOrNull(date);
|
---|
188 | if (!this._dateAdapter.sameDate(date, this.value)) {
|
---|
189 | this._assignValue(date);
|
---|
190 | this._cvaOnChange(date);
|
---|
191 | this.dateInput.emit(new MatDatepickerInputEvent(this, this._elementRef.nativeElement));
|
---|
192 | }
|
---|
193 | else {
|
---|
194 | // Call the CVA change handler for invalid values
|
---|
195 | // since this is what marks the control as dirty.
|
---|
196 | if (value && !this.value) {
|
---|
197 | this._cvaOnChange(date);
|
---|
198 | }
|
---|
199 | if (lastValueWasValid !== this._lastValueValid) {
|
---|
200 | this._validatorOnChange();
|
---|
201 | }
|
---|
202 | }
|
---|
203 | }
|
---|
204 | _onChange() {
|
---|
205 | this.dateChange.emit(new MatDatepickerInputEvent(this, this._elementRef.nativeElement));
|
---|
206 | }
|
---|
207 | /** Handles blur events on the input. */
|
---|
208 | _onBlur() {
|
---|
209 | // Reformat the input only if we have a valid value.
|
---|
210 | if (this.value) {
|
---|
211 | this._formatValue(this.value);
|
---|
212 | }
|
---|
213 | this._onTouched();
|
---|
214 | }
|
---|
215 | /** Formats a value and sets it on the input element. */
|
---|
216 | _formatValue(value) {
|
---|
217 | this._elementRef.nativeElement.value =
|
---|
218 | value ? this._dateAdapter.format(value, this._dateFormats.display.dateInput) : '';
|
---|
219 | }
|
---|
220 | /** Assigns a value to the model. */
|
---|
221 | _assignValue(value) {
|
---|
222 | // We may get some incoming values before the model was
|
---|
223 | // assigned. Save the value so that we can assign it later.
|
---|
224 | if (this._model) {
|
---|
225 | this._assignValueToModel(value);
|
---|
226 | this._pendingValue = null;
|
---|
227 | }
|
---|
228 | else {
|
---|
229 | this._pendingValue = value;
|
---|
230 | }
|
---|
231 | }
|
---|
232 | /** Whether a value is considered valid. */
|
---|
233 | _isValidValue(value) {
|
---|
234 | return !value || this._dateAdapter.isValid(value);
|
---|
235 | }
|
---|
236 | /**
|
---|
237 | * Checks whether a parent control is disabled. This is in place so that it can be overridden
|
---|
238 | * by inputs extending this one which can be placed inside of a group that can be disabled.
|
---|
239 | */
|
---|
240 | _parentDisabled() {
|
---|
241 | return false;
|
---|
242 | }
|
---|
243 | /** Programmatically assigns a value to the input. */
|
---|
244 | _assignValueProgrammatically(value) {
|
---|
245 | value = this._dateAdapter.deserialize(value);
|
---|
246 | this._lastValueValid = this._isValidValue(value);
|
---|
247 | value = this._dateAdapter.getValidDateOrNull(value);
|
---|
248 | this._assignValue(value);
|
---|
249 | this._formatValue(value);
|
---|
250 | }
|
---|
251 | /** Gets whether a value matches the current date filter. */
|
---|
252 | _matchesFilter(value) {
|
---|
253 | const filter = this._getDateFilter();
|
---|
254 | return !filter || filter(value);
|
---|
255 | }
|
---|
256 | }
|
---|
257 | MatDatepickerInputBase.decorators = [
|
---|
258 | { type: Directive }
|
---|
259 | ];
|
---|
260 | MatDatepickerInputBase.ctorParameters = () => [
|
---|
261 | { type: ElementRef },
|
---|
262 | { type: DateAdapter, decorators: [{ type: Optional }] },
|
---|
263 | { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_DATE_FORMATS,] }] }
|
---|
264 | ];
|
---|
265 | MatDatepickerInputBase.propDecorators = {
|
---|
266 | value: [{ type: Input }],
|
---|
267 | disabled: [{ type: Input }],
|
---|
268 | dateChange: [{ type: Output }],
|
---|
269 | dateInput: [{ type: Output }]
|
---|
270 | };
|
---|
271 | /**
|
---|
272 | * Checks whether the `SimpleChanges` object from an `ngOnChanges`
|
---|
273 | * callback has any changes, accounting for date objects.
|
---|
274 | */
|
---|
275 | export function dateInputsHaveChanged(changes, adapter) {
|
---|
276 | const keys = Object.keys(changes);
|
---|
277 | for (let key of keys) {
|
---|
278 | const { previousValue, currentValue } = changes[key];
|
---|
279 | if (adapter.isDateInstance(previousValue) && adapter.isDateInstance(currentValue)) {
|
---|
280 | if (!adapter.sameDate(previousValue, currentValue)) {
|
---|
281 | return true;
|
---|
282 | }
|
---|
283 | }
|
---|
284 | else {
|
---|
285 | return true;
|
---|
286 | }
|
---|
287 | }
|
---|
288 | return false;
|
---|
289 | }
|
---|
290 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datepicker-input-base.js","sourceRoot":"","sources":["../../../../../../src/material/datepicker/datepicker-input-base.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {BooleanInput, coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {DOWN_ARROW} from '@angular/cdk/keycodes';\nimport {\n  Directive,\n  ElementRef,\n  EventEmitter,\n  Inject,\n  Input,\n  OnDestroy,\n  Optional,\n  Output,\n  AfterViewInit,\n  OnChanges,\n  SimpleChanges,\n} from '@angular/core';\nimport {\n  AbstractControl,\n  ControlValueAccessor,\n  ValidationErrors,\n  Validator,\n  ValidatorFn,\n} from '@angular/forms';\nimport {\n  DateAdapter,\n  MAT_DATE_FORMATS,\n  MatDateFormats,\n} from '@angular/material/core';\nimport {Subscription, Subject} from 'rxjs';\nimport {createMissingDateImplError} from './datepicker-errors';\nimport {\n  ExtractDateTypeFromSelection,\n  MatDateSelectionModel,\n  DateSelectionModelChange,\n} from './date-selection-model';\n\n/**\n * An event used for datepicker input and change events. We don't always have access to a native\n * input or change event because the event may have been triggered by the user clicking on the\n * calendar popup. For consistency, we always use MatDatepickerInputEvent instead.\n */\nexport class MatDatepickerInputEvent<D, S = unknown> {\n  /** The new value for the target datepicker input. */\n  value: D | null;\n\n  constructor(\n      /** Reference to the datepicker input component that emitted the event. */\n      public target: MatDatepickerInputBase<S, D>,\n      /** Reference to the native input element associated with the datepicker input. */\n      public targetElement: HTMLElement) {\n    this.value = this.target.value;\n  }\n}\n\n/** Function that can be used to filter out dates from a calendar. */\nexport type DateFilterFn<D> = (date: D | null) => boolean;\n\n/** Base class for datepicker inputs. */\n@Directive()\nexport abstract class MatDatepickerInputBase<S, D = ExtractDateTypeFromSelection<S>>\n  implements ControlValueAccessor, AfterViewInit, OnChanges, OnDestroy, Validator {\n\n  /** Whether the component has been initialized. */\n  private _isInitialized: boolean;\n\n  /** The value of the input. */\n  @Input()\n  get value(): D | null {\n    return this._model ? this._getValueFromModel(this._model.selection) : this._pendingValue;\n  }\n  set value(value: D | null) {\n    this._assignValueProgrammatically(value);\n  }\n  protected _model: MatDateSelectionModel<S, D> | undefined;\n\n  /** Whether the datepicker-input is disabled. */\n  @Input()\n  get disabled(): boolean { return !!this._disabled || this._parentDisabled(); }\n  set disabled(value: boolean) {\n    const newValue = coerceBooleanProperty(value);\n    const element = this._elementRef.nativeElement;\n\n    if (this._disabled !== newValue) {\n      this._disabled = newValue;\n      this.stateChanges.next(undefined);\n    }\n\n    // We need to null check the `blur` method, because it's undefined during SSR.\n    // In Ivy static bindings are invoked earlier, before the element is attached to the DOM.\n    // This can cause an error to be thrown in some browsers (IE/Edge) which assert that the\n    // element has been inserted.\n    if (newValue && this._isInitialized && element.blur) {\n      // Normally, native input elements automatically blur if they turn disabled. This behavior\n      // is problematic, because it would mean that it triggers another change detection cycle,\n      // which then causes a changed after checked error if the input element was focused before.\n      element.blur();\n    }\n  }\n  private _disabled: boolean;\n\n  /** Emits when a `change` event is fired on this `<input>`. */\n  @Output() readonly dateChange: EventEmitter<MatDatepickerInputEvent<D, S>> =\n      new EventEmitter<MatDatepickerInputEvent<D, S>>();\n\n  /** Emits when an `input` event is fired on this `<input>`. */\n  @Output() readonly dateInput: EventEmitter<MatDatepickerInputEvent<D, S>> =\n      new EventEmitter<MatDatepickerInputEvent<D, S>>();\n\n  /** Emits when the internal state has changed */\n  readonly stateChanges = new Subject<void>();\n\n  _onTouched = () => {};\n  _validatorOnChange = () => {};\n\n  private _cvaOnChange: (value: any) => void = () => {};\n  private _valueChangesSubscription = Subscription.EMPTY;\n  private _localeSubscription = Subscription.EMPTY;\n\n  /**\n   * Since the value is kept on the model which is assigned in an Input,\n   * we might get a value before we have a model. This property keeps track\n   * of the value until we have somewhere to assign it.\n   */\n  private _pendingValue: D | null;\n\n  /** The form control validator for whether the input parses. */\n  private _parseValidator: ValidatorFn = (): ValidationErrors | null => {\n    return this._lastValueValid ?\n        null : {'matDatepickerParse': {'text': this._elementRef.nativeElement.value}};\n  }\n\n  /** The form control validator for the date filter. */\n  private _filterValidator: ValidatorFn = (control: AbstractControl): ValidationErrors | null => {\n    const controlValue = this._dateAdapter.getValidDateOrNull(\n      this._dateAdapter.deserialize(control.value));\n    return !controlValue || this._matchesFilter(controlValue) ?\n        null : {'matDatepickerFilter': true};\n  }\n\n  /** The form control validator for the min date. */\n  private _minValidator: ValidatorFn = (control: AbstractControl): ValidationErrors | null => {\n    const controlValue = this._dateAdapter.getValidDateOrNull(\n      this._dateAdapter.deserialize(control.value));\n    const min = this._getMinDate();\n    return (!min || !controlValue ||\n        this._dateAdapter.compareDate(min, controlValue) <= 0) ?\n        null : {'matDatepickerMin': {'min': min, 'actual': controlValue}};\n  }\n\n  /** The form control validator for the max date. */\n  private _maxValidator: ValidatorFn = (control: AbstractControl): ValidationErrors | null => {\n    const controlValue = this._dateAdapter.getValidDateOrNull(\n      this._dateAdapter.deserialize(control.value));\n    const max = this._getMaxDate();\n    return (!max || !controlValue ||\n        this._dateAdapter.compareDate(max, controlValue) >= 0) ?\n        null : {'matDatepickerMax': {'max': max, 'actual': controlValue}};\n  }\n\n  /** Gets the base validator functions. */\n  protected _getValidators(): ValidatorFn[] {\n    return [this._parseValidator, this._minValidator, this._maxValidator, this._filterValidator];\n  }\n\n  /** Gets the minimum date for the input. Used for validation. */\n  abstract _getMinDate(): D | null;\n\n  /** Gets the maximum date for the input. Used for validation. */\n  abstract _getMaxDate(): D | null;\n\n  /** Gets the date filter function. Used for validation. */\n  protected abstract _getDateFilter(): DateFilterFn<D> | undefined;\n\n  /** Registers a date selection model with the input. */\n  _registerModel(model: MatDateSelectionModel<S, D>): void {\n    this._model = model;\n    this._valueChangesSubscription.unsubscribe();\n\n    if (this._pendingValue) {\n      this._assignValue(this._pendingValue);\n    }\n\n    this._valueChangesSubscription = this._model.selectionChanged.subscribe(event => {\n      if (this._shouldHandleChangeEvent(event)) {\n        const value = this._getValueFromModel(event.selection);\n        this._lastValueValid = this._isValidValue(value);\n        this._cvaOnChange(value);\n        this._onTouched();\n        this._formatValue(value);\n        this.dateInput.emit(new MatDatepickerInputEvent(this, this._elementRef.nativeElement));\n        this.dateChange.emit(new MatDatepickerInputEvent(this, this._elementRef.nativeElement));\n      }\n    });\n  }\n\n  /** Opens the popup associated with the input. */\n  protected abstract _openPopup(): void;\n\n  /** Assigns a value to the input's model. */\n  protected abstract _assignValueToModel(model: D | null): void;\n\n  /** Converts a value from the model into a native value for the input. */\n  protected abstract _getValueFromModel(modelValue: S): D | null;\n\n  /** Combined form control validator for this input. */\n  protected abstract _validator: ValidatorFn | null;\n\n  /** Predicate that determines whether the input should handle a particular change event. */\n  protected abstract _shouldHandleChangeEvent(event: DateSelectionModelChange<S>): boolean;\n\n  /** Whether the last value set on the input was valid. */\n  protected _lastValueValid = false;\n\n  constructor(\n      protected _elementRef: ElementRef<HTMLInputElement>,\n      @Optional() public _dateAdapter: DateAdapter<D>,\n      @Optional() @Inject(MAT_DATE_FORMATS) private _dateFormats: MatDateFormats) {\n\n    if (typeof ngDevMode === 'undefined' || ngDevMode) {\n      if (!this._dateAdapter) {\n        throw createMissingDateImplError('DateAdapter');\n      }\n      if (!this._dateFormats) {\n        throw createMissingDateImplError('MAT_DATE_FORMATS');\n      }\n    }\n\n    // Update the displayed date when the locale changes.\n    this._localeSubscription = _dateAdapter.localeChanges.subscribe(() => {\n      this._assignValueProgrammatically(this.value);\n    });\n  }\n\n  ngAfterViewInit() {\n    this._isInitialized = true;\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (dateInputsHaveChanged(changes, this._dateAdapter)) {\n      this.stateChanges.next(undefined);\n    }\n  }\n\n  ngOnDestroy() {\n    this._valueChangesSubscription.unsubscribe();\n    this._localeSubscription.unsubscribe();\n    this.stateChanges.complete();\n  }\n\n  /** @docs-private */\n  registerOnValidatorChange(fn: () => void): void {\n    this._validatorOnChange = fn;\n  }\n\n  /** @docs-private */\n  validate(c: AbstractControl): ValidationErrors | null {\n    return this._validator ? this._validator(c) : null;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  writeValue(value: D): void {\n    this._assignValueProgrammatically(value);\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  registerOnChange(fn: (value: any) => void): void {\n    this._cvaOnChange = fn;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  registerOnTouched(fn: () => void): void {\n    this._onTouched = fn;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  _onKeydown(event: KeyboardEvent) {\n    const isAltDownArrow = event.altKey && event.keyCode === DOWN_ARROW;\n\n    if (isAltDownArrow && !this._elementRef.nativeElement.readOnly) {\n      this._openPopup();\n      event.preventDefault();\n    }\n  }\n\n  _onInput(value: string) {\n    const lastValueWasValid = this._lastValueValid;\n    let date = this._dateAdapter.parse(value, this._dateFormats.parse.dateInput);\n    this._lastValueValid = this._isValidValue(date);\n    date = this._dateAdapter.getValidDateOrNull(date);\n\n    if (!this._dateAdapter.sameDate(date, this.value)) {\n      this._assignValue(date);\n      this._cvaOnChange(date);\n      this.dateInput.emit(new MatDatepickerInputEvent(this, this._elementRef.nativeElement));\n    } else {\n      // Call the CVA change handler for invalid values\n      // since this is what marks the control as dirty.\n      if (value && !this.value) {\n        this._cvaOnChange(date);\n      }\n\n      if (lastValueWasValid !== this._lastValueValid) {\n        this._validatorOnChange();\n      }\n    }\n  }\n\n  _onChange() {\n    this.dateChange.emit(new MatDatepickerInputEvent(this, this._elementRef.nativeElement));\n  }\n\n  /** Handles blur events on the input. */\n  _onBlur() {\n    // Reformat the input only if we have a valid value.\n    if (this.value) {\n      this._formatValue(this.value);\n    }\n\n    this._onTouched();\n  }\n\n  /** Formats a value and sets it on the input element. */\n  protected _formatValue(value: D | null) {\n    this._elementRef.nativeElement.value =\n        value ? this._dateAdapter.format(value, this._dateFormats.display.dateInput) : '';\n  }\n\n  /** Assigns a value to the model. */\n  private _assignValue(value: D | null) {\n    // We may get some incoming values before the model was\n    // assigned. Save the value so that we can assign it later.\n    if (this._model) {\n      this._assignValueToModel(value);\n      this._pendingValue = null;\n    } else {\n      this._pendingValue = value;\n    }\n  }\n\n  /** Whether a value is considered valid. */\n  private _isValidValue(value: D | null): boolean {\n    return !value || this._dateAdapter.isValid(value);\n  }\n\n  /**\n   * Checks whether a parent control is disabled. This is in place so that it can be overridden\n   * by inputs extending this one which can be placed inside of a group that can be disabled.\n   */\n  protected _parentDisabled() {\n    return false;\n  }\n\n  /** Programmatically assigns a value to the input. */\n  protected _assignValueProgrammatically(value: D | null) {\n    value = this._dateAdapter.deserialize(value);\n    this._lastValueValid = this._isValidValue(value);\n    value = this._dateAdapter.getValidDateOrNull(value);\n    this._assignValue(value);\n    this._formatValue(value);\n  }\n\n  /** Gets whether a value matches the current date filter. */\n  _matchesFilter(value: D | null): boolean {\n    const filter = this._getDateFilter();\n    return !filter || filter(value);\n  }\n\n  // Accept `any` to avoid conflicts with other directives on `<input>` that\n  // may accept different types.\n  static ngAcceptInputType_value: any;\n  static ngAcceptInputType_disabled: BooleanInput;\n}\n\n/**\n * Checks whether the `SimpleChanges` object from an `ngOnChanges`\n * callback has any changes, accounting for date objects.\n */\nexport function dateInputsHaveChanged(\n  changes: SimpleChanges,\n  adapter: DateAdapter<unknown>): boolean {\n  const keys = Object.keys(changes);\n\n  for (let key of keys) {\n    const {previousValue, currentValue} = changes[key];\n\n    if (adapter.isDateInstance(previousValue) && adapter.isDateInstance(currentValue)) {\n      if (!adapter.sameDate(previousValue, currentValue)) {\n        return true;\n      }\n    } else {\n      return true;\n    }\n  }\n\n  return false;\n}\n"]} |
---|