@odx/angular 12.21.1 → 12.21.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (25) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/cdk/date-input/index.d.ts +1 -0
  3. package/cdk/date-input/lib/utils/ngx-mask-init.d.ts +3 -0
  4. package/components/datepicker/lib/directives/datepicker-input-control.directive.d.ts +4 -16
  5. package/components/daterangepicker/lib/directives/daterangepicker-input-control.directive.d.ts +4 -16
  6. package/components/timepicker/lib/directives/timepicker-input-control.directive.d.ts +6 -15
  7. package/components/timepicker/lib/timepicker.service.d.ts +5 -4
  8. package/esm2022/cdk/date-input/index.mjs +2 -1
  9. package/esm2022/cdk/date-input/lib/utils/ngx-mask-init.mjs +22 -0
  10. package/esm2022/components/datepicker/lib/datepicker.component.mjs +3 -2
  11. package/esm2022/components/datepicker/lib/directives/datepicker-input-control.directive.mjs +8 -23
  12. package/esm2022/components/daterangepicker/lib/daterangepicker.component.mjs +4 -2
  13. package/esm2022/components/daterangepicker/lib/directives/daterangepicker-input-control.directive.mjs +8 -23
  14. package/esm2022/components/timepicker/lib/directives/timepicker-input-control.directive.mjs +17 -37
  15. package/esm2022/components/timepicker/lib/timepicker.component.mjs +21 -6
  16. package/esm2022/components/timepicker/lib/timepicker.service.mjs +42 -15
  17. package/fesm2022/odx-angular-cdk-date-input.mjs +23 -2
  18. package/fesm2022/odx-angular-cdk-date-input.mjs.map +1 -1
  19. package/fesm2022/odx-angular-components-datepicker.mjs +10 -24
  20. package/fesm2022/odx-angular-components-datepicker.mjs.map +1 -1
  21. package/fesm2022/odx-angular-components-daterangepicker.mjs +11 -24
  22. package/fesm2022/odx-angular-components-daterangepicker.mjs.map +1 -1
  23. package/fesm2022/odx-angular-components-timepicker.mjs +113 -90
  24. package/fesm2022/odx-angular-components-timepicker.mjs.map +1 -1
  25. package/package.json +7 -7
@@ -2,12 +2,12 @@ import { __decorate } from "tslib";
2
2
  import { Directive, EventEmitter, HostListener, inject, Output } from '@angular/core';
3
3
  import { ReadonlyController, WithTabIndex } from '@odx/angular';
4
4
  import { InputControlDirective } from '@odx/angular/cdk/custom-form-control';
5
- import { getDateInputFormat, getDateInputMask, getDateInputValueAsDate, injectDateConfig } from '@odx/angular/cdk/date-input';
5
+ import { getDateInputFormat, getDateInputValueAsDate, injectDateConfig } from '@odx/angular/cdk/date-input';
6
6
  import { CSSComponent } from '@odx/angular/internal';
7
- import { NgxMaskPipe, provideNgxMask } from 'ngx-mask';
8
- import { distinctUntilChanged, fromEvent, map, tap } from 'rxjs';
7
+ import { NgxMaskDirective, provideNgxMask } from 'ngx-mask';
9
8
  import * as i0 from "@angular/core";
10
9
  import * as i1 from "@odx/angular";
10
+ import * as i2 from "ngx-mask";
11
11
  /**
12
12
  * Enhances an input element to support date range picking, applying an input mask for date formatting
13
13
  * and managing focus events. This directive is typically used within a date range picker to provide
@@ -19,30 +19,15 @@ import * as i1 from "@odx/angular";
19
19
  let DaterangepickerInputControlDirective = class DaterangepickerInputControlDirective extends InputControlDirective {
20
20
  constructor() {
21
21
  super(...arguments);
22
- this.maskConfig = { validation: false, leadZeroDateTime: true };
23
22
  this.readonlyController = ReadonlyController.inject();
24
23
  this.config = injectDateConfig();
25
- this.inputMask = getDateInputMask(this.config);
26
- this.ngxMaskPipe = inject(NgxMaskPipe);
24
+ this.ngxMaskDirective = inject(NgxMaskDirective);
27
25
  /**
28
26
  * Emits an event when the input gains or loses focus, facilitating external event handling.
29
27
  *
30
28
  * @emits {boolean} - Indicates whether the input is focused.
31
29
  */
32
30
  this.focused = new EventEmitter();
33
- /**
34
- * Captures and processes changes to the input element's value, applying the mask and updating
35
- * the form control.
36
- *
37
- * @emits {string} - The updated value of the input element.
38
- */
39
- this.valueChange$ = fromEvent(this.element.nativeElement, 'input').pipe(distinctUntilChanged(), tap(() => this.applyMask()), map(() => this.nativeElementValue));
40
- }
41
- /**
42
- * Applies the configured input mask to the native input element's value.
43
- */
44
- applyMask() {
45
- this.nativeElementValue = this.ngxMaskPipe.transform(this.nativeElementValue, this.inputMask, this.maskConfig);
46
31
  }
47
32
  /**
48
33
  * Converts the current input value to a Date object based on the configuration's date format.
@@ -67,7 +52,7 @@ let DaterangepickerInputControlDirective = class DaterangepickerInputControlDire
67
52
  this.focused.emit(false);
68
53
  }
69
54
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DaterangepickerInputControlDirective, deps: null, target: i0.ɵɵFactoryTarget.Directive }); }
70
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: DaterangepickerInputControlDirective, isStandalone: true, selector: "input[odxDaterangepickerControl],input[odxDaterangepickerStartDateControl], input[odxDaterangepickerEndDateControl]", outputs: { focused: "focused" }, host: { listeners: { "focusin": "handleFocusIn()", "focusout": "handleFocusOut()" }, properties: { "attr.readonly": "readonlyController?.readonly || null", "attr.placeholder": "placeholder" } }, providers: [ReadonlyController.connect(), provideNgxMask(), NgxMaskPipe], usesInheritance: true, hostDirectives: [{ directive: i1.WithTabIndex }], ngImport: i0 }); }
55
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: DaterangepickerInputControlDirective, isStandalone: true, selector: "input[odxDaterangepickerControl],input[odxDaterangepickerStartDateControl], input[odxDaterangepickerEndDateControl]", outputs: { focused: "focused" }, host: { listeners: { "focusin": "handleFocusIn()", "focusout": "handleFocusOut()" }, properties: { "attr.readonly": "readonlyController?.readonly || null", "attr.placeholder": "placeholder" } }, providers: [ReadonlyController.connect(), provideNgxMask({ validation: false, leadZeroDateTime: true })], usesInheritance: true, hostDirectives: [{ directive: i1.WithTabIndex }, { directive: i2.NgxMaskDirective }], ngImport: i0 }); }
71
56
  };
72
57
  DaterangepickerInputControlDirective = __decorate([
73
58
  CSSComponent('daterangepicker__control')
@@ -82,8 +67,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
82
67
  '[attr.readonly]': 'readonlyController?.readonly || null',
83
68
  '[attr.placeholder]': 'placeholder',
84
69
  },
85
- providers: [ReadonlyController.connect(), provideNgxMask(), NgxMaskPipe],
86
- hostDirectives: [WithTabIndex],
70
+ providers: [ReadonlyController.connect(), provideNgxMask({ validation: false, leadZeroDateTime: true })],
71
+ hostDirectives: [WithTabIndex, NgxMaskDirective],
87
72
  }]
88
73
  }], propDecorators: { focused: [{
89
74
  type: Output
@@ -94,4 +79,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
94
79
  type: HostListener,
95
80
  args: ['focusout']
96
81
  }] } });
97
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0ZXJhbmdlcGlja2VyLWlucHV0LWNvbnRyb2wuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9hbmd1bGFyL2NvbXBvbmVudHMvZGF0ZXJhbmdlcGlja2VyL3NyYy9saWIvZGlyZWN0aXZlcy9kYXRlcmFuZ2VwaWNrZXItaW5wdXQtY29udHJvbC5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLFlBQVksRUFBRSxNQUFNLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3RGLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxZQUFZLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFDaEUsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sc0NBQXNDLENBQUM7QUFDN0UsT0FBTyxFQUFFLGtCQUFrQixFQUFFLGdCQUFnQixFQUFFLHVCQUF1QixFQUFFLGdCQUFnQixFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDOUgsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQ3JELE9BQU8sRUFBaUIsV0FBVyxFQUFFLGNBQWMsRUFBRSxNQUFNLFVBQVUsQ0FBQztBQUN0RSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsU0FBUyxFQUFFLEdBQUcsRUFBRSxHQUFHLEVBQUUsTUFBTSxNQUFNLENBQUM7OztBQUVqRTs7Ozs7OztHQU9HO0FBWUksSUFBTSxvQ0FBb0MsR0FBMUMsTUFBTSxvQ0FBcUMsU0FBUSxxQkFBcUI7SUFBeEU7O1FBQ1ksZUFBVSxHQUEyQixFQUFFLFVBQVUsRUFBRSxLQUFLLEVBQUUsZ0JBQWdCLEVBQUUsSUFBSSxFQUFFLENBQUM7UUFFakYsdUJBQWtCLEdBQUcsa0JBQWtCLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDakQsV0FBTSxHQUFHLGdCQUFnQixFQUFFLENBQUM7UUFDNUIsY0FBUyxHQUFHLGdCQUFnQixDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUMxQyxnQkFBVyxHQUFHLE1BQU0sQ0FBQyxXQUFXLENBQUMsQ0FBQztRQUVyRDs7OztXQUlHO1FBRUksWUFBTyxHQUFHLElBQUksWUFBWSxFQUFXLENBQUM7UUFFN0M7Ozs7O1dBS0c7UUFDYSxpQkFBWSxHQUFHLFNBQVMsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLGFBQWEsRUFBRSxPQUFPLENBQUMsQ0FBQyxJQUFJLENBQ2hGLG9CQUFvQixFQUFFLEVBQ3RCLEdBQUcsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUMsRUFDM0IsR0FBRyxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxDQUNuQyxDQUFDO0tBb0NIO0lBbENDOztPQUVHO0lBQ0ksU0FBUztRQUNkLElBQUksQ0FBQyxrQkFBa0IsR0FBRyxJQUFJLENBQUMsV0FBVyxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsa0JBQWtCLEVBQUUsSUFBSSxDQUFDLFNBQVMsRUFBRSxJQUFJLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDakgsQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxJQUFXLFdBQVc7UUFDcEIsT0FBTyx1QkFBdUIsQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDO0lBQ3ZFLENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsSUFBVyxXQUFXO1FBQ3BCLE9BQU8sa0JBQWtCLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLFdBQVcsRUFBRSxDQUFDO0lBQ3ZELENBQUM7SUFHUyxhQUFhO1FBQ3JCLElBQUksQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzFCLENBQUM7SUFHUyxjQUFjO1FBQ3RCLElBQUksQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQzNCLENBQUM7K0dBN0RVLG9DQUFvQzttR0FBcEMsb0NBQW9DLHNZQUhwQyxDQUFDLGtCQUFrQixDQUFDLE9BQU8sRUFBRSxFQUFFLGNBQWMsRUFBRSxFQUFFLFdBQVcsQ0FBQzs7QUFHN0Qsb0NBQW9DO0lBWGhELFlBQVksQ0FBQywwQkFBMEIsQ0FBQztHQVc1QixvQ0FBb0MsQ0E4RGhEOzs0RkE5RFksb0NBQW9DO2tCQVZoRCxTQUFTO21CQUFDO29CQUNULFVBQVUsRUFBRSxJQUFJO29CQUNoQixRQUFRLEVBQUUscUhBQXFIO29CQUMvSCxJQUFJLEVBQUU7d0JBQ0osaUJBQWlCLEVBQUUsc0NBQXNDO3dCQUN6RCxvQkFBb0IsRUFBRSxhQUFhO3FCQUNwQztvQkFDRCxTQUFTLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQyxPQUFPLEVBQUUsRUFBRSxjQUFjLEVBQUUsRUFBRSxXQUFXLENBQUM7b0JBQ3hFLGNBQWMsRUFBRSxDQUFDLFlBQVksQ0FBQztpQkFDL0I7OEJBZVEsT0FBTztzQkFEYixNQUFNO2dCQXlDRyxhQUFhO3NCQUR0QixZQUFZO3VCQUFDLFNBQVM7Z0JBTWIsY0FBYztzQkFEdkIsWUFBWTt1QkFBQyxVQUFVIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlLCBFdmVudEVtaXR0ZXIsIEhvc3RMaXN0ZW5lciwgaW5qZWN0LCBPdXRwdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFJlYWRvbmx5Q29udHJvbGxlciwgV2l0aFRhYkluZGV4IH0gZnJvbSAnQG9keC9hbmd1bGFyJztcbmltcG9ydCB7IElucHV0Q29udHJvbERpcmVjdGl2ZSB9IGZyb20gJ0BvZHgvYW5ndWxhci9jZGsvY3VzdG9tLWZvcm0tY29udHJvbCc7XG5pbXBvcnQgeyBnZXREYXRlSW5wdXRGb3JtYXQsIGdldERhdGVJbnB1dE1hc2ssIGdldERhdGVJbnB1dFZhbHVlQXNEYXRlLCBpbmplY3REYXRlQ29uZmlnIH0gZnJvbSAnQG9keC9hbmd1bGFyL2Nkay9kYXRlLWlucHV0JztcbmltcG9ydCB7IENTU0NvbXBvbmVudCB9IGZyb20gJ0BvZHgvYW5ndWxhci9pbnRlcm5hbCc7XG5pbXBvcnQgeyBOZ3hNYXNrQ29uZmlnLCBOZ3hNYXNrUGlwZSwgcHJvdmlkZU5neE1hc2sgfSBmcm9tICduZ3gtbWFzayc7XG5pbXBvcnQgeyBkaXN0aW5jdFVudGlsQ2hhbmdlZCwgZnJvbUV2ZW50LCBtYXAsIHRhcCB9IGZyb20gJ3J4anMnO1xuXG4vKipcbiAqIEVuaGFuY2VzIGFuIGlucHV0IGVsZW1lbnQgdG8gc3VwcG9ydCBkYXRlIHJhbmdlIHBpY2tpbmcsIGFwcGx5aW5nIGFuIGlucHV0IG1hc2sgZm9yIGRhdGUgZm9ybWF0dGluZ1xuICogYW5kIG1hbmFnaW5nIGZvY3VzIGV2ZW50cy4gVGhpcyBkaXJlY3RpdmUgaXMgdHlwaWNhbGx5IHVzZWQgd2l0aGluIGEgZGF0ZSByYW5nZSBwaWNrZXIgdG8gcHJvdmlkZVxuICogY29uc2lzdGVudCBhbmQgY29uZmlndXJhYmxlIGlucHV0IGJlaGF2aW9yLiBFeHRlbmRzIHRoZSBgSW5wdXRDb250cm9sRGlyZWN0aXZlYCB0byBwcm92aWRlIGZvcm0gY29udHJvbC5cbiAqIEhhcyBob3N0IGRpcmVjdGl2ZSBgV2l0aFRhYkluZGV4YCB0byBtYW5hZ2UgdGhlIHRhYiBpbmRleCBhdHRyaWJ1dGUgb2YgdGhlIGlucHV0IGVsZW1lbnQuXG4gKlxuICogQHNlZSB7SW5wdXRDb250cm9sRGlyZWN0aXZlfVxuICovXG5AQ1NTQ29tcG9uZW50KCdkYXRlcmFuZ2VwaWNrZXJfX2NvbnRyb2wnKVxuQERpcmVjdGl2ZSh7XG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIHNlbGVjdG9yOiAnaW5wdXRbb2R4RGF0ZXJhbmdlcGlja2VyQ29udHJvbF0saW5wdXRbb2R4RGF0ZXJhbmdlcGlja2VyU3RhcnREYXRlQ29udHJvbF0sIGlucHV0W29keERhdGVyYW5nZXBpY2tlckVuZERhdGVDb250cm9sXScsXG4gIGhvc3Q6IHtcbiAgICAnW2F0dHIucmVhZG9ubHldJzogJ3JlYWRvbmx5Q29udHJvbGxlcj8ucmVhZG9ubHkgfHwgbnVsbCcsXG4gICAgJ1thdHRyLnBsYWNlaG9sZGVyXSc6ICdwbGFjZWhvbGRlcicsXG4gIH0sXG4gIHByb3ZpZGVyczogW1JlYWRvbmx5Q29udHJvbGxlci5jb25uZWN0KCksIHByb3ZpZGVOZ3hNYXNrKCksIE5neE1hc2tQaXBlXSxcbiAgaG9zdERpcmVjdGl2ZXM6IFtXaXRoVGFiSW5kZXhdLFxufSlcbmV4cG9ydCBjbGFzcyBEYXRlcmFuZ2VwaWNrZXJJbnB1dENvbnRyb2xEaXJlY3RpdmUgZXh0ZW5kcyBJbnB1dENvbnRyb2xEaXJlY3RpdmUge1xuICBwcml2YXRlIHJlYWRvbmx5IG1hc2tDb25maWc6IFBhcnRpYWw8Tmd4TWFza0NvbmZpZz4gPSB7IHZhbGlkYXRpb246IGZhbHNlLCBsZWFkWmVyb0RhdGVUaW1lOiB0cnVlIH07XG5cbiAgcHJvdGVjdGVkIHJlYWRvbmx5IHJlYWRvbmx5Q29udHJvbGxlciA9IFJlYWRvbmx5Q29udHJvbGxlci5pbmplY3QoKTtcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IGNvbmZpZyA9IGluamVjdERhdGVDb25maWcoKTtcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IGlucHV0TWFzayA9IGdldERhdGVJbnB1dE1hc2sodGhpcy5jb25maWcpO1xuICBwcm90ZWN0ZWQgcmVhZG9ubHkgbmd4TWFza1BpcGUgPSBpbmplY3QoTmd4TWFza1BpcGUpO1xuXG4gIC8qKlxuICAgKiBFbWl0cyBhbiBldmVudCB3aGVuIHRoZSBpbnB1dCBnYWlucyBvciBsb3NlcyBmb2N1cywgZmFjaWxpdGF0aW5nIGV4dGVybmFsIGV2ZW50IGhhbmRsaW5nLlxuICAgKlxuICAgKiBAZW1pdHMge2Jvb2xlYW59IC0gSW5kaWNhdGVzIHdoZXRoZXIgdGhlIGlucHV0IGlzIGZvY3VzZWQuXG4gICAqL1xuICBAT3V0cHV0KClcbiAgcHVibGljIGZvY3VzZWQgPSBuZXcgRXZlbnRFbWl0dGVyPGJvb2xlYW4+KCk7XG5cbiAgLyoqXG4gICAqIENhcHR1cmVzIGFuZCBwcm9jZXNzZXMgY2hhbmdlcyB0byB0aGUgaW5wdXQgZWxlbWVudCdzIHZhbHVlLCBhcHBseWluZyB0aGUgbWFzayBhbmQgdXBkYXRpbmdcbiAgICogdGhlIGZvcm0gY29udHJvbC5cbiAgICpcbiAgICogQGVtaXRzIHtzdHJpbmd9IC0gVGhlIHVwZGF0ZWQgdmFsdWUgb2YgdGhlIGlucHV0IGVsZW1lbnQuXG4gICAqL1xuICBwdWJsaWMgb3ZlcnJpZGUgdmFsdWVDaGFuZ2UkID0gZnJvbUV2ZW50KHRoaXMuZWxlbWVudC5uYXRpdmVFbGVtZW50LCAnaW5wdXQnKS5waXBlKFxuICAgIGRpc3RpbmN0VW50aWxDaGFuZ2VkKCksXG4gICAgdGFwKCgpID0+IHRoaXMuYXBwbHlNYXNrKCkpLFxuICAgIG1hcCgoKSA9PiB0aGlzLm5hdGl2ZUVsZW1lbnRWYWx1ZSksXG4gICk7XG5cbiAgLyoqXG4gICAqIEFwcGxpZXMgdGhlIGNvbmZpZ3VyZWQgaW5wdXQgbWFzayB0byB0aGUgbmF0aXZlIGlucHV0IGVsZW1lbnQncyB2YWx1ZS5cbiAgICovXG4gIHB1YmxpYyBhcHBseU1hc2soKTogdm9pZCB7XG4gICAgdGhpcy5uYXRpdmVFbGVtZW50VmFsdWUgPSB0aGlzLm5neE1hc2tQaXBlLnRyYW5zZm9ybSh0aGlzLm5hdGl2ZUVsZW1lbnRWYWx1ZSwgdGhpcy5pbnB1dE1hc2ssIHRoaXMubWFza0NvbmZpZyk7XG4gIH1cblxuICAvKipcbiAgICogQ29udmVydHMgdGhlIGN1cnJlbnQgaW5wdXQgdmFsdWUgdG8gYSBEYXRlIG9iamVjdCBiYXNlZCBvbiB0aGUgY29uZmlndXJhdGlvbidzIGRhdGUgZm9ybWF0LlxuICAgKlxuICAgKiBAcmV0dXJucyB7RGF0ZSB8IG51bGx9IFRoZSBwYXJzZWQgZGF0ZSBvYmplY3Qgb3IgbnVsbCBpZiB0aGUgaW5wdXQgZG9lcyBub3QgcmVwcmVzZW50IGEgdmFsaWQgZGF0ZS5cbiAgICovXG4gIHB1YmxpYyBnZXQgdmFsdWVBc0RhdGUoKTogRGF0ZSB8IG51bGwge1xuICAgIHJldHVybiBnZXREYXRlSW5wdXRWYWx1ZUFzRGF0ZSh0aGlzLmNvbmZpZywgdGhpcy5uYXRpdmVFbGVtZW50VmFsdWUpO1xuICB9XG5cbiAgLyoqXG4gICAqIFByb3ZpZGVzIHRoZSBwbGFjZWhvbGRlciB0ZXh0IGZvciB0aGUgaW5wdXQsIHR5cGljYWxseSB0aGUgZGF0ZSBmb3JtYXQgaW4gdXBwZXJjYXNlLlxuICAgKlxuICAgKiBAcmV0dXJucyB7c3RyaW5nfSBUaGUgcGxhY2Vob2xkZXIgdGV4dCBmb3IgdGhlIGlucHV0LlxuICAgKi9cbiAgcHVibGljIGdldCBwbGFjZWhvbGRlcigpOiBzdHJpbmcge1xuICAgIHJldHVybiBnZXREYXRlSW5wdXRGb3JtYXQodGhpcy5jb25maWcpLnRvVXBwZXJDYXNlKCk7XG4gIH1cblxuICBASG9zdExpc3RlbmVyKCdmb2N1c2luJylcbiAgcHJvdGVjdGVkIGhhbmRsZUZvY3VzSW4oKTogdm9pZCB7XG4gICAgdGhpcy5mb2N1c2VkLmVtaXQodHJ1ZSk7XG4gIH1cblxuICBASG9zdExpc3RlbmVyKCdmb2N1c291dCcpXG4gIHByb3RlY3RlZCBoYW5kbGVGb2N1c091dCgpOiB2b2lkIHtcbiAgICB0aGlzLmZvY3VzZWQuZW1pdChmYWxzZSk7XG4gIH1cbn1cbiJdfQ==
82
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0ZXJhbmdlcGlja2VyLWlucHV0LWNvbnRyb2wuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9hbmd1bGFyL2NvbXBvbmVudHMvZGF0ZXJhbmdlcGlja2VyL3NyYy9saWIvZGlyZWN0aXZlcy9kYXRlcmFuZ2VwaWNrZXItaW5wdXQtY29udHJvbC5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLFlBQVksRUFBRSxNQUFNLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3RGLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxZQUFZLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFDaEUsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sc0NBQXNDLENBQUM7QUFDN0UsT0FBTyxFQUFFLGtCQUFrQixFQUFFLHVCQUF1QixFQUFFLGdCQUFnQixFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDNUcsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQ3JELE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxjQUFjLEVBQUUsTUFBTSxVQUFVLENBQUM7Ozs7QUFFNUQ7Ozs7Ozs7R0FPRztBQVlJLElBQU0sb0NBQW9DLEdBQTFDLE1BQU0sb0NBQXFDLFNBQVEscUJBQXFCO0lBQXhFOztRQUNjLHVCQUFrQixHQUFHLGtCQUFrQixDQUFDLE1BQU0sRUFBRSxDQUFDO1FBQ2pELFdBQU0sR0FBRyxnQkFBZ0IsRUFBRSxDQUFDO1FBQy9CLHFCQUFnQixHQUFHLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO1FBRTVEOzs7O1dBSUc7UUFFSSxZQUFPLEdBQUcsSUFBSSxZQUFZLEVBQVcsQ0FBQztLQTZCOUM7SUEzQkM7Ozs7T0FJRztJQUNILElBQVcsV0FBVztRQUNwQixPQUFPLHVCQUF1QixDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLGtCQUFrQixDQUFDLENBQUM7SUFDdkUsQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxJQUFXLFdBQVc7UUFDcEIsT0FBTyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUMsV0FBVyxFQUFFLENBQUM7SUFDdkQsQ0FBQztJQUdTLGFBQWE7UUFDckIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDMUIsQ0FBQztJQUdTLGNBQWM7UUFDdEIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDM0IsQ0FBQzsrR0F2Q1Usb0NBQW9DO21HQUFwQyxvQ0FBb0Msc1lBSHBDLENBQUMsa0JBQWtCLENBQUMsT0FBTyxFQUFFLEVBQUUsY0FBYyxDQUFDLEVBQUUsVUFBVSxFQUFFLEtBQUssRUFBRSxnQkFBZ0IsRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDOztBQUc3RixvQ0FBb0M7SUFYaEQsWUFBWSxDQUFDLDBCQUEwQixDQUFDO0dBVzVCLG9DQUFvQyxDQXdDaEQ7OzRGQXhDWSxvQ0FBb0M7a0JBVmhELFNBQVM7bUJBQUM7b0JBQ1QsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLFFBQVEsRUFBRSxxSEFBcUg7b0JBQy9ILElBQUksRUFBRTt3QkFDSixpQkFBaUIsRUFBRSxzQ0FBc0M7d0JBQ3pELG9CQUFvQixFQUFFLGFBQWE7cUJBQ3BDO29CQUNELFNBQVMsRUFBRSxDQUFDLGtCQUFrQixDQUFDLE9BQU8sRUFBRSxFQUFFLGNBQWMsQ0FBQyxFQUFFLFVBQVUsRUFBRSxLQUFLLEVBQUUsZ0JBQWdCLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQztvQkFDeEcsY0FBYyxFQUFFLENBQUMsWUFBWSxFQUFFLGdCQUFnQixDQUFDO2lCQUNqRDs4QkFZUSxPQUFPO3NCQURiLE1BQU07Z0JBc0JHLGFBQWE7c0JBRHRCLFlBQVk7dUJBQUMsU0FBUztnQkFNYixjQUFjO3NCQUR2QixZQUFZO3VCQUFDLFVBQVUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIEV2ZW50RW1pdHRlciwgSG9zdExpc3RlbmVyLCBpbmplY3QsIE91dHB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgUmVhZG9ubHlDb250cm9sbGVyLCBXaXRoVGFiSW5kZXggfSBmcm9tICdAb2R4L2FuZ3VsYXInO1xuaW1wb3J0IHsgSW5wdXRDb250cm9sRGlyZWN0aXZlIH0gZnJvbSAnQG9keC9hbmd1bGFyL2Nkay9jdXN0b20tZm9ybS1jb250cm9sJztcbmltcG9ydCB7IGdldERhdGVJbnB1dEZvcm1hdCwgZ2V0RGF0ZUlucHV0VmFsdWVBc0RhdGUsIGluamVjdERhdGVDb25maWcgfSBmcm9tICdAb2R4L2FuZ3VsYXIvY2RrL2RhdGUtaW5wdXQnO1xuaW1wb3J0IHsgQ1NTQ29tcG9uZW50IH0gZnJvbSAnQG9keC9hbmd1bGFyL2ludGVybmFsJztcbmltcG9ydCB7IE5neE1hc2tEaXJlY3RpdmUsIHByb3ZpZGVOZ3hNYXNrIH0gZnJvbSAnbmd4LW1hc2snO1xuXG4vKipcbiAqIEVuaGFuY2VzIGFuIGlucHV0IGVsZW1lbnQgdG8gc3VwcG9ydCBkYXRlIHJhbmdlIHBpY2tpbmcsIGFwcGx5aW5nIGFuIGlucHV0IG1hc2sgZm9yIGRhdGUgZm9ybWF0dGluZ1xuICogYW5kIG1hbmFnaW5nIGZvY3VzIGV2ZW50cy4gVGhpcyBkaXJlY3RpdmUgaXMgdHlwaWNhbGx5IHVzZWQgd2l0aGluIGEgZGF0ZSByYW5nZSBwaWNrZXIgdG8gcHJvdmlkZVxuICogY29uc2lzdGVudCBhbmQgY29uZmlndXJhYmxlIGlucHV0IGJlaGF2aW9yLiBFeHRlbmRzIHRoZSBgSW5wdXRDb250cm9sRGlyZWN0aXZlYCB0byBwcm92aWRlIGZvcm0gY29udHJvbC5cbiAqIEhhcyBob3N0IGRpcmVjdGl2ZSBgV2l0aFRhYkluZGV4YCB0byBtYW5hZ2UgdGhlIHRhYiBpbmRleCBhdHRyaWJ1dGUgb2YgdGhlIGlucHV0IGVsZW1lbnQuXG4gKlxuICogQHNlZSB7SW5wdXRDb250cm9sRGlyZWN0aXZlfVxuICovXG5AQ1NTQ29tcG9uZW50KCdkYXRlcmFuZ2VwaWNrZXJfX2NvbnRyb2wnKVxuQERpcmVjdGl2ZSh7XG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIHNlbGVjdG9yOiAnaW5wdXRbb2R4RGF0ZXJhbmdlcGlja2VyQ29udHJvbF0saW5wdXRbb2R4RGF0ZXJhbmdlcGlja2VyU3RhcnREYXRlQ29udHJvbF0sIGlucHV0W29keERhdGVyYW5nZXBpY2tlckVuZERhdGVDb250cm9sXScsXG4gIGhvc3Q6IHtcbiAgICAnW2F0dHIucmVhZG9ubHldJzogJ3JlYWRvbmx5Q29udHJvbGxlcj8ucmVhZG9ubHkgfHwgbnVsbCcsXG4gICAgJ1thdHRyLnBsYWNlaG9sZGVyXSc6ICdwbGFjZWhvbGRlcicsXG4gIH0sXG4gIHByb3ZpZGVyczogW1JlYWRvbmx5Q29udHJvbGxlci5jb25uZWN0KCksIHByb3ZpZGVOZ3hNYXNrKHsgdmFsaWRhdGlvbjogZmFsc2UsIGxlYWRaZXJvRGF0ZVRpbWU6IHRydWUgfSldLFxuICBob3N0RGlyZWN0aXZlczogW1dpdGhUYWJJbmRleCwgTmd4TWFza0RpcmVjdGl2ZV0sXG59KVxuZXhwb3J0IGNsYXNzIERhdGVyYW5nZXBpY2tlcklucHV0Q29udHJvbERpcmVjdGl2ZSBleHRlbmRzIElucHV0Q29udHJvbERpcmVjdGl2ZSB7XG4gIHByb3RlY3RlZCByZWFkb25seSByZWFkb25seUNvbnRyb2xsZXIgPSBSZWFkb25seUNvbnRyb2xsZXIuaW5qZWN0KCk7XG4gIHByb3RlY3RlZCByZWFkb25seSBjb25maWcgPSBpbmplY3REYXRlQ29uZmlnKCk7XG4gIHB1YmxpYyByZWFkb25seSBuZ3hNYXNrRGlyZWN0aXZlID0gaW5qZWN0KE5neE1hc2tEaXJlY3RpdmUpO1xuXG4gIC8qKlxuICAgKiBFbWl0cyBhbiBldmVudCB3aGVuIHRoZSBpbnB1dCBnYWlucyBvciBsb3NlcyBmb2N1cywgZmFjaWxpdGF0aW5nIGV4dGVybmFsIGV2ZW50IGhhbmRsaW5nLlxuICAgKlxuICAgKiBAZW1pdHMge2Jvb2xlYW59IC0gSW5kaWNhdGVzIHdoZXRoZXIgdGhlIGlucHV0IGlzIGZvY3VzZWQuXG4gICAqL1xuICBAT3V0cHV0KClcbiAgcHVibGljIGZvY3VzZWQgPSBuZXcgRXZlbnRFbWl0dGVyPGJvb2xlYW4+KCk7XG5cbiAgLyoqXG4gICAqIENvbnZlcnRzIHRoZSBjdXJyZW50IGlucHV0IHZhbHVlIHRvIGEgRGF0ZSBvYmplY3QgYmFzZWQgb24gdGhlIGNvbmZpZ3VyYXRpb24ncyBkYXRlIGZvcm1hdC5cbiAgICpcbiAgICogQHJldHVybnMge0RhdGUgfCBudWxsfSBUaGUgcGFyc2VkIGRhdGUgb2JqZWN0IG9yIG51bGwgaWYgdGhlIGlucHV0IGRvZXMgbm90IHJlcHJlc2VudCBhIHZhbGlkIGRhdGUuXG4gICAqL1xuICBwdWJsaWMgZ2V0IHZhbHVlQXNEYXRlKCk6IERhdGUgfCBudWxsIHtcbiAgICByZXR1cm4gZ2V0RGF0ZUlucHV0VmFsdWVBc0RhdGUodGhpcy5jb25maWcsIHRoaXMubmF0aXZlRWxlbWVudFZhbHVlKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBQcm92aWRlcyB0aGUgcGxhY2Vob2xkZXIgdGV4dCBmb3IgdGhlIGlucHV0LCB0eXBpY2FsbHkgdGhlIGRhdGUgZm9ybWF0IGluIHVwcGVyY2FzZS5cbiAgICpcbiAgICogQHJldHVybnMge3N0cmluZ30gVGhlIHBsYWNlaG9sZGVyIHRleHQgZm9yIHRoZSBpbnB1dC5cbiAgICovXG4gIHB1YmxpYyBnZXQgcGxhY2Vob2xkZXIoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gZ2V0RGF0ZUlucHV0Rm9ybWF0KHRoaXMuY29uZmlnKS50b1VwcGVyQ2FzZSgpO1xuICB9XG5cbiAgQEhvc3RMaXN0ZW5lcignZm9jdXNpbicpXG4gIHByb3RlY3RlZCBoYW5kbGVGb2N1c0luKCk6IHZvaWQge1xuICAgIHRoaXMuZm9jdXNlZC5lbWl0KHRydWUpO1xuICB9XG5cbiAgQEhvc3RMaXN0ZW5lcignZm9jdXNvdXQnKVxuICBwcm90ZWN0ZWQgaGFuZGxlRm9jdXNPdXQoKTogdm9pZCB7XG4gICAgdGhpcy5mb2N1c2VkLmVtaXQoZmFsc2UpO1xuICB9XG59XG4iXX0=
@@ -2,14 +2,14 @@ import { __decorate } from "tslib";
2
2
  import { Directive, inject } from '@angular/core';
3
3
  import { WithDisabledState, WithTabIndex } from '@odx/angular';
4
4
  import { InputControlDirective } from '@odx/angular/cdk/custom-form-control';
5
+ import { initNgxMask, ngxMaskProviderConfig } from '@odx/angular/cdk/date-input';
5
6
  import { CSSComponent } from '@odx/angular/internal';
6
- import { NgxMaskPipe, provideNgxMask } from 'ngx-mask';
7
- import { distinctUntilChanged, fromEvent, map } from 'rxjs';
7
+ import { NgxMaskDirective, provideNgxMask } from 'ngx-mask';
8
8
  import { TimepickerService } from '../timepicker.service';
9
9
  import { TIMEPICKER_CONTROL } from '../timepicker.token';
10
- import { processInputValue } from '../utils/ngx-mask-helper';
11
10
  import * as i0 from "@angular/core";
12
11
  import * as i1 from "@odx/angular";
12
+ import * as i2 from "ngx-mask";
13
13
  /**
14
14
  * Directive to enhance a standard input element for time picking, integrating mask functionality for time format.
15
15
  * It automatically adapts to locale settings provided by the enclosing `TimepickerComponent` to display time in the appropriate format.
@@ -21,35 +21,15 @@ let TimepickerInputControlDirective = class TimepickerInputControlDirective exte
21
21
  super(...arguments);
22
22
  this.timepicker = inject(TIMEPICKER_CONTROL);
23
23
  this.timepickerService = inject(TimepickerService);
24
- this.ngxMaskPipe = inject(NgxMaskPipe);
25
- /**
26
- * Stream of value changes for the input element, applying the mask and processing the input based on locale.
27
- */
28
- this.valueChange$ = fromEvent(this.element.nativeElement, 'input').pipe(distinctUntilChanged(), map((event) => this.applyMask(event)));
24
+ this.ngxMaskDirective = inject(NgxMaskDirective, { self: true, optional: true });
29
25
  }
30
26
  /**
31
- * Configuration for the mask applied to the timepicker input. Adjusts based on locale settings.
32
- *
33
- * @returns {Partial<NgxMaskConfig>} The mask configuration for the timepicker input.
34
- */
35
- get maskConfig() {
36
- return {
37
- validation: false,
38
- apm: this.timepicker.useLocale,
39
- leadZeroDateTime: !this.timepicker.useLocale,
40
- patterns: {
41
- A: { pattern: new RegExp('[AaPp]') },
42
- M: { pattern: new RegExp('M') },
43
- },
44
- };
45
- }
46
- /**
47
- * Computes and returns the placeholder text for the input based on locale settings.
27
+ * Returns the placeholder text for the input based on locale settings.
48
28
  *
49
29
  * @returns {string} The placeholder text for the input.
50
30
  */
51
31
  get placeholder() {
52
- return this.timepickerService.getPlaceholder(this.timepicker.useLocale);
32
+ return this.timepickerService.useLocale() ? '--:-- --' : '--:--';
53
33
  }
54
34
  /**
55
35
  * Determines whether the timepicker input is readonly, based on the state of the parent timepicker component.
@@ -65,17 +45,17 @@ let TimepickerInputControlDirective = class TimepickerInputControlDirective exte
65
45
  * @param {Event} event - The input event triggering the mask application.
66
46
  * @returns {string} The masked input value.
67
47
  */
68
- applyMask(event) {
69
- const mask = this.timepicker.useLocale ? 'Hh:m0 AM' : 'Hh:m0';
70
- let value = this.ngxMaskPipe.transform(this.nativeElementValue, mask, this.maskConfig);
71
- if (this.timepicker.useLocale) {
72
- value = processInputValue(value, event);
48
+ applyMask(useLocale) {
49
+ if (!this.ngxMaskDirective)
50
+ return;
51
+ const mask = useLocale ? 'Hh:m0 AM' : 'Hh:m0';
52
+ useLocale !== this.ngxMaskDirective._maskService.apm && (this.ngxMaskDirective._maskService.apm = useLocale);
53
+ if (Object.hasOwn(this.ngxMaskDirective, '_maskValue') && this.ngxMaskDirective['_maskValue'] !== mask) {
54
+ initNgxMask(this.ngxMaskDirective, mask);
73
55
  }
74
- this.nativeElementValue = value;
75
- return value;
76
56
  }
77
57
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TimepickerInputControlDirective, deps: null, target: i0.ɵɵFactoryTarget.Directive }); }
78
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: TimepickerInputControlDirective, isStandalone: true, selector: "input[odxTimepickerControl]", host: { properties: { "attr.readonly": "isReadonly || null", "attr.placeholder": "placeholder" } }, providers: [provideNgxMask(), NgxMaskPipe], usesInheritance: true, hostDirectives: [{ directive: i1.WithTabIndex }, { directive: i1.WithDisabledState }], ngImport: i0 }); }
58
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: TimepickerInputControlDirective, isStandalone: true, selector: "input[odxTimepickerControl]", host: { properties: { "attr.readonly": "isReadonly || null", "attr.placeholder": "placeholder" } }, providers: [provideNgxMask(ngxMaskProviderConfig)], usesInheritance: true, hostDirectives: [{ directive: i1.WithTabIndex }, { directive: i1.WithDisabledState }, { directive: i2.NgxMaskDirective }], ngImport: i0 }); }
79
59
  };
80
60
  TimepickerInputControlDirective = __decorate([
81
61
  CSSComponent('timepicker__control')
@@ -90,8 +70,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
90
70
  '[attr.readonly]': 'isReadonly || null',
91
71
  '[attr.placeholder]': 'placeholder',
92
72
  },
93
- providers: [provideNgxMask(), NgxMaskPipe],
94
- hostDirectives: [WithTabIndex, WithDisabledState],
73
+ providers: [provideNgxMask(ngxMaskProviderConfig)],
74
+ hostDirectives: [WithTabIndex, WithDisabledState, NgxMaskDirective],
95
75
  }]
96
76
  }] });
97
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGltZXBpY2tlci1pbnB1dC1jb250cm9sLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvYW5ndWxhci9jb21wb25lbnRzL3RpbWVwaWNrZXIvc3JjL2xpYi9kaXJlY3RpdmVzL3RpbWVwaWNrZXItaW5wdXQtY29udHJvbC5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2xELE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxZQUFZLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFDL0QsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sc0NBQXNDLENBQUM7QUFDN0UsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQ3JELE9BQU8sRUFBaUIsV0FBVyxFQUFFLGNBQWMsRUFBRSxNQUFNLFVBQVUsQ0FBQztBQUN0RSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsU0FBUyxFQUFFLEdBQUcsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUM1RCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUMxRCxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUN6RCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQzs7O0FBRTdEOzs7OztHQUtHO0FBWUksSUFBTSwrQkFBK0IsR0FBckMsTUFBTSwrQkFBZ0MsU0FBUSxxQkFBcUI7SUFBbkU7O1FBQ1ksZUFBVSxHQUFHLE1BQU0sQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDO1FBQ3hDLHNCQUFpQixHQUFHLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO1FBQzVDLGdCQUFXLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBb0JyRDs7V0FFRztRQUNhLGlCQUFZLEdBQUcsU0FBUyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsYUFBYSxFQUFFLE9BQU8sQ0FBQyxDQUFDLElBQUksQ0FDaEYsb0JBQW9CLEVBQUUsRUFDdEIsR0FBRyxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQ3RDLENBQUM7S0FtQ0g7SUEzREM7Ozs7T0FJRztJQUNILElBQWMsVUFBVTtRQUN0QixPQUFPO1lBQ0wsVUFBVSxFQUFFLEtBQUs7WUFDakIsR0FBRyxFQUFFLElBQUksQ0FBQyxVQUFVLENBQUMsU0FBUztZQUM5QixnQkFBZ0IsRUFBRSxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsU0FBUztZQUU1QyxRQUFRLEVBQUU7Z0JBQ1IsQ0FBQyxFQUFFLEVBQUUsT0FBTyxFQUFFLElBQUksTUFBTSxDQUFDLFFBQVEsQ0FBQyxFQUFFO2dCQUNwQyxDQUFDLEVBQUUsRUFBRSxPQUFPLEVBQUUsSUFBSSxNQUFNLENBQUMsR0FBRyxDQUFDLEVBQUU7YUFDaEM7U0FDRixDQUFDO0lBQ0osQ0FBQztJQVVEOzs7O09BSUc7SUFDSCxJQUFXLFdBQVc7UUFDcEIsT0FBTyxJQUFJLENBQUMsaUJBQWlCLENBQUMsY0FBYyxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDMUUsQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxJQUFjLFVBQVU7UUFDdEIsT0FBTyxJQUFJLENBQUMsVUFBVSxDQUFDLFVBQVUsQ0FBQztJQUNwQyxDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSyxTQUFTLENBQUMsS0FBWTtRQUM1QixNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUM7UUFDOUQsSUFBSSxLQUFLLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLGtCQUFrQixFQUFFLElBQUksRUFBRSxJQUFJLENBQUMsVUFBVSxDQUFDLENBQUM7UUFDdkYsSUFBSSxJQUFJLENBQUMsVUFBVSxDQUFDLFNBQVMsRUFBRSxDQUFDO1lBQzlCLEtBQUssR0FBRyxpQkFBaUIsQ0FBQyxLQUFLLEVBQUUsS0FBbUIsQ0FBQyxDQUFDO1FBQ3hELENBQUM7UUFDRCxJQUFJLENBQUMsa0JBQWtCLEdBQUcsS0FBSyxDQUFDO1FBQ2hDLE9BQU8sS0FBSyxDQUFDO0lBQ2YsQ0FBQzsrR0EvRFUsK0JBQStCO21HQUEvQiwrQkFBK0IsOEtBSC9CLENBQUMsY0FBYyxFQUFFLEVBQUUsV0FBVyxDQUFDOztBQUcvQiwrQkFBK0I7SUFYM0MsWUFBWSxDQUFDLHFCQUFxQixDQUFDO0dBV3ZCLCtCQUErQixDQWdFM0M7OzRGQWhFWSwrQkFBK0I7a0JBVjNDLFNBQVM7bUJBQUM7b0JBQ1QsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLFFBQVEsRUFBRSw2QkFBNkI7b0JBQ3ZDLElBQUksRUFBRTt3QkFDSixpQkFBaUIsRUFBRSxvQkFBb0I7d0JBQ3ZDLG9CQUFvQixFQUFFLGFBQWE7cUJBQ3BDO29CQUNELFNBQVMsRUFBRSxDQUFDLGNBQWMsRUFBRSxFQUFFLFdBQVcsQ0FBQztvQkFDMUMsY0FBYyxFQUFFLENBQUMsWUFBWSxFQUFFLGlCQUFpQixDQUFDO2lCQUNsRCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBXaXRoRGlzYWJsZWRTdGF0ZSwgV2l0aFRhYkluZGV4IH0gZnJvbSAnQG9keC9hbmd1bGFyJztcbmltcG9ydCB7IElucHV0Q29udHJvbERpcmVjdGl2ZSB9IGZyb20gJ0BvZHgvYW5ndWxhci9jZGsvY3VzdG9tLWZvcm0tY29udHJvbCc7XG5pbXBvcnQgeyBDU1NDb21wb25lbnQgfSBmcm9tICdAb2R4L2FuZ3VsYXIvaW50ZXJuYWwnO1xuaW1wb3J0IHsgTmd4TWFza0NvbmZpZywgTmd4TWFza1BpcGUsIHByb3ZpZGVOZ3hNYXNrIH0gZnJvbSAnbmd4LW1hc2snO1xuaW1wb3J0IHsgZGlzdGluY3RVbnRpbENoYW5nZWQsIGZyb21FdmVudCwgbWFwIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBUaW1lcGlja2VyU2VydmljZSB9IGZyb20gJy4uL3RpbWVwaWNrZXIuc2VydmljZSc7XG5pbXBvcnQgeyBUSU1FUElDS0VSX0NPTlRST0wgfSBmcm9tICcuLi90aW1lcGlja2VyLnRva2VuJztcbmltcG9ydCB7IHByb2Nlc3NJbnB1dFZhbHVlIH0gZnJvbSAnLi4vdXRpbHMvbmd4LW1hc2staGVscGVyJztcblxuLyoqXG4gKiBEaXJlY3RpdmUgdG8gZW5oYW5jZSBhIHN0YW5kYXJkIGlucHV0IGVsZW1lbnQgZm9yIHRpbWUgcGlja2luZywgaW50ZWdyYXRpbmcgbWFzayBmdW5jdGlvbmFsaXR5IGZvciB0aW1lIGZvcm1hdC5cbiAqIEl0IGF1dG9tYXRpY2FsbHkgYWRhcHRzIHRvIGxvY2FsZSBzZXR0aW5ncyBwcm92aWRlZCBieSB0aGUgZW5jbG9zaW5nIGBUaW1lcGlja2VyQ29tcG9uZW50YCB0byBkaXNwbGF5IHRpbWUgaW4gdGhlIGFwcHJvcHJpYXRlIGZvcm1hdC5cbiAqXG4gKiBAZXh0ZW5kcyB7SW5wdXRDb250cm9sRGlyZWN0aXZlfVxuICovXG5AQ1NTQ29tcG9uZW50KCd0aW1lcGlja2VyX19jb250cm9sJylcbkBEaXJlY3RpdmUoe1xuICBzdGFuZGFsb25lOiB0cnVlLFxuICBzZWxlY3RvcjogJ2lucHV0W29keFRpbWVwaWNrZXJDb250cm9sXScsXG4gIGhvc3Q6IHtcbiAgICAnW2F0dHIucmVhZG9ubHldJzogJ2lzUmVhZG9ubHkgfHwgbnVsbCcsXG4gICAgJ1thdHRyLnBsYWNlaG9sZGVyXSc6ICdwbGFjZWhvbGRlcicsXG4gIH0sXG4gIHByb3ZpZGVyczogW3Byb3ZpZGVOZ3hNYXNrKCksIE5neE1hc2tQaXBlXSxcbiAgaG9zdERpcmVjdGl2ZXM6IFtXaXRoVGFiSW5kZXgsIFdpdGhEaXNhYmxlZFN0YXRlXSxcbn0pXG5leHBvcnQgY2xhc3MgVGltZXBpY2tlcklucHV0Q29udHJvbERpcmVjdGl2ZSBleHRlbmRzIElucHV0Q29udHJvbERpcmVjdGl2ZSB7XG4gIHByaXZhdGUgcmVhZG9ubHkgdGltZXBpY2tlciA9IGluamVjdChUSU1FUElDS0VSX0NPTlRST0wpO1xuICBwcml2YXRlIHJlYWRvbmx5IHRpbWVwaWNrZXJTZXJ2aWNlID0gaW5qZWN0KFRpbWVwaWNrZXJTZXJ2aWNlKTtcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IG5neE1hc2tQaXBlID0gaW5qZWN0KE5neE1hc2tQaXBlKTtcblxuICAvKipcbiAgICogQ29uZmlndXJhdGlvbiBmb3IgdGhlIG1hc2sgYXBwbGllZCB0byB0aGUgdGltZXBpY2tlciBpbnB1dC4gQWRqdXN0cyBiYXNlZCBvbiBsb2NhbGUgc2V0dGluZ3MuXG4gICAqXG4gICAqIEByZXR1cm5zIHtQYXJ0aWFsPE5neE1hc2tDb25maWc+fSBUaGUgbWFzayBjb25maWd1cmF0aW9uIGZvciB0aGUgdGltZXBpY2tlciBpbnB1dC5cbiAgICovXG4gIHByb3RlY3RlZCBnZXQgbWFza0NvbmZpZygpOiBQYXJ0aWFsPE5neE1hc2tDb25maWc+IHtcbiAgICByZXR1cm4ge1xuICAgICAgdmFsaWRhdGlvbjogZmFsc2UsXG4gICAgICBhcG06IHRoaXMudGltZXBpY2tlci51c2VMb2NhbGUsXG4gICAgICBsZWFkWmVyb0RhdGVUaW1lOiAhdGhpcy50aW1lcGlja2VyLnVzZUxvY2FsZSxcblxuICAgICAgcGF0dGVybnM6IHtcbiAgICAgICAgQTogeyBwYXR0ZXJuOiBuZXcgUmVnRXhwKCdbQWFQcF0nKSB9LFxuICAgICAgICBNOiB7IHBhdHRlcm46IG5ldyBSZWdFeHAoJ00nKSB9LFxuICAgICAgfSxcbiAgICB9O1xuICB9XG5cbiAgLyoqXG4gICAqIFN0cmVhbSBvZiB2YWx1ZSBjaGFuZ2VzIGZvciB0aGUgaW5wdXQgZWxlbWVudCwgYXBwbHlpbmcgdGhlIG1hc2sgYW5kIHByb2Nlc3NpbmcgdGhlIGlucHV0IGJhc2VkIG9uIGxvY2FsZS5cbiAgICovXG4gIHB1YmxpYyBvdmVycmlkZSB2YWx1ZUNoYW5nZSQgPSBmcm9tRXZlbnQodGhpcy5lbGVtZW50Lm5hdGl2ZUVsZW1lbnQsICdpbnB1dCcpLnBpcGUoXG4gICAgZGlzdGluY3RVbnRpbENoYW5nZWQoKSxcbiAgICBtYXAoKGV2ZW50KSA9PiB0aGlzLmFwcGx5TWFzayhldmVudCkpLFxuICApO1xuXG4gIC8qKlxuICAgKiBDb21wdXRlcyBhbmQgcmV0dXJucyB0aGUgcGxhY2Vob2xkZXIgdGV4dCBmb3IgdGhlIGlucHV0IGJhc2VkIG9uIGxvY2FsZSBzZXR0aW5ncy5cbiAgICpcbiAgICogQHJldHVybnMge3N0cmluZ30gVGhlIHBsYWNlaG9sZGVyIHRleHQgZm9yIHRoZSBpbnB1dC5cbiAgICovXG4gIHB1YmxpYyBnZXQgcGxhY2Vob2xkZXIoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gdGhpcy50aW1lcGlja2VyU2VydmljZS5nZXRQbGFjZWhvbGRlcih0aGlzLnRpbWVwaWNrZXIudXNlTG9jYWxlKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBEZXRlcm1pbmVzIHdoZXRoZXIgdGhlIHRpbWVwaWNrZXIgaW5wdXQgaXMgcmVhZG9ubHksIGJhc2VkIG9uIHRoZSBzdGF0ZSBvZiB0aGUgcGFyZW50IHRpbWVwaWNrZXIgY29tcG9uZW50LlxuICAgKlxuICAgKiBAcmV0dXJucyB7Ym9vbGVhbn0gYHRydWVgIGlmIHRoZSBpbnB1dCBzaG91bGQgYmUgcmVhZG9ubHk7IG90aGVyd2lzZSwgYGZhbHNlYC5cbiAgICovXG4gIHByb3RlY3RlZCBnZXQgaXNSZWFkb25seSgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy50aW1lcGlja2VyLmlzUmVhZG9ubHk7XG4gIH1cblxuICAvKipcbiAgICogQXBwbGllcyB0aGUgdGltZSBmb3JtYXQgbWFzayB0byB0aGUgaW5wdXQgdmFsdWUsIHRyYW5zZm9ybWluZyBhbmQgcmV0dXJuaW5nIHRoZSBtYXNrZWQgdmFsdWUuXG4gICAqXG4gICAqIEBwYXJhbSB7RXZlbnR9IGV2ZW50IC0gVGhlIGlucHV0IGV2ZW50IHRyaWdnZXJpbmcgdGhlIG1hc2sgYXBwbGljYXRpb24uXG4gICAqIEByZXR1cm5zIHtzdHJpbmd9IFRoZSBtYXNrZWQgaW5wdXQgdmFsdWUuXG4gICAqL1xuICBwcml2YXRlIGFwcGx5TWFzayhldmVudDogRXZlbnQpOiBzdHJpbmcge1xuICAgIGNvbnN0IG1hc2sgPSB0aGlzLnRpbWVwaWNrZXIudXNlTG9jYWxlID8gJ0hoOm0wIEFNJyA6ICdIaDptMCc7XG4gICAgbGV0IHZhbHVlID0gdGhpcy5uZ3hNYXNrUGlwZS50cmFuc2Zvcm0odGhpcy5uYXRpdmVFbGVtZW50VmFsdWUsIG1hc2ssIHRoaXMubWFza0NvbmZpZyk7XG4gICAgaWYgKHRoaXMudGltZXBpY2tlci51c2VMb2NhbGUpIHtcbiAgICAgIHZhbHVlID0gcHJvY2Vzc0lucHV0VmFsdWUodmFsdWUsIGV2ZW50IGFzIElucHV0RXZlbnQpO1xuICAgIH1cbiAgICB0aGlzLm5hdGl2ZUVsZW1lbnRWYWx1ZSA9IHZhbHVlO1xuICAgIHJldHVybiB2YWx1ZTtcbiAgfVxufVxuIl19
77
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGltZXBpY2tlci1pbnB1dC1jb250cm9sLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvYW5ndWxhci9jb21wb25lbnRzL3RpbWVwaWNrZXIvc3JjL2xpYi9kaXJlY3RpdmVzL3RpbWVwaWNrZXItaW5wdXQtY29udHJvbC5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2xELE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxZQUFZLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFDL0QsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sc0NBQXNDLENBQUM7QUFDN0UsT0FBTyxFQUFFLFdBQVcsRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBQ2pGLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUNyRCxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsY0FBYyxFQUFFLE1BQU0sVUFBVSxDQUFDO0FBQzVELE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQzFELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLHFCQUFxQixDQUFDOzs7O0FBRXpEOzs7OztHQUtHO0FBWUksSUFBTSwrQkFBK0IsR0FBckMsTUFBTSwrQkFBZ0MsU0FBUSxxQkFBcUI7SUFBbkU7O1FBQ1ksZUFBVSxHQUFHLE1BQU0sQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDO1FBQ3hDLHNCQUFpQixHQUFHLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO1FBQy9DLHFCQUFnQixHQUFHLE1BQU0sQ0FBQyxnQkFBZ0IsRUFBRSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7S0FrQzdGO0lBaENDOzs7O09BSUc7SUFDSCxJQUFXLFdBQVc7UUFDcEIsT0FBTyxJQUFJLENBQUMsaUJBQWlCLENBQUMsU0FBUyxFQUFFLENBQUMsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDO0lBQ25FLENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsSUFBYyxVQUFVO1FBQ3RCLE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQyxVQUFVLENBQUM7SUFDcEMsQ0FBQztJQUVEOzs7OztPQUtHO0lBQ0ksU0FBUyxDQUFDLFNBQWtCO1FBQ2pDLElBQUksQ0FBQyxJQUFJLENBQUMsZ0JBQWdCO1lBQUUsT0FBTztRQUNuQyxNQUFNLElBQUksR0FBRyxTQUFTLENBQUMsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDO1FBQzlDLFNBQVMsS0FBSyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsWUFBWSxDQUFDLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxZQUFZLENBQUMsR0FBRyxHQUFHLFNBQVMsQ0FBQyxDQUFDO1FBQzdHLElBQUksTUFBTSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLEVBQUUsWUFBWSxDQUFDLElBQUksSUFBSSxDQUFDLGdCQUFnQixDQUFDLFlBQVksQ0FBQyxLQUFLLElBQUksRUFBRSxDQUFDO1lBQ3ZHLFdBQVcsQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFDM0MsQ0FBQztJQUNILENBQUM7K0dBcENVLCtCQUErQjttR0FBL0IsK0JBQStCLDhLQUgvQixDQUFDLGNBQWMsQ0FBQyxxQkFBcUIsQ0FBQyxDQUFDOztBQUd2QywrQkFBK0I7SUFYM0MsWUFBWSxDQUFDLHFCQUFxQixDQUFDO0dBV3ZCLCtCQUErQixDQXFDM0M7OzRGQXJDWSwrQkFBK0I7a0JBVjNDLFNBQVM7bUJBQUM7b0JBQ1QsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLFFBQVEsRUFBRSw2QkFBNkI7b0JBQ3ZDLElBQUksRUFBRTt3QkFDSixpQkFBaUIsRUFBRSxvQkFBb0I7d0JBQ3ZDLG9CQUFvQixFQUFFLGFBQWE7cUJBQ3BDO29CQUNELFNBQVMsRUFBRSxDQUFDLGNBQWMsQ0FBQyxxQkFBcUIsQ0FBQyxDQUFDO29CQUNsRCxjQUFjLEVBQUUsQ0FBQyxZQUFZLEVBQUUsaUJBQWlCLEVBQUUsZ0JBQWdCLENBQUM7aUJBQ3BFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlLCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFdpdGhEaXNhYmxlZFN0YXRlLCBXaXRoVGFiSW5kZXggfSBmcm9tICdAb2R4L2FuZ3VsYXInO1xuaW1wb3J0IHsgSW5wdXRDb250cm9sRGlyZWN0aXZlIH0gZnJvbSAnQG9keC9hbmd1bGFyL2Nkay9jdXN0b20tZm9ybS1jb250cm9sJztcbmltcG9ydCB7IGluaXROZ3hNYXNrLCBuZ3hNYXNrUHJvdmlkZXJDb25maWcgfSBmcm9tICdAb2R4L2FuZ3VsYXIvY2RrL2RhdGUtaW5wdXQnO1xuaW1wb3J0IHsgQ1NTQ29tcG9uZW50IH0gZnJvbSAnQG9keC9hbmd1bGFyL2ludGVybmFsJztcbmltcG9ydCB7IE5neE1hc2tEaXJlY3RpdmUsIHByb3ZpZGVOZ3hNYXNrIH0gZnJvbSAnbmd4LW1hc2snO1xuaW1wb3J0IHsgVGltZXBpY2tlclNlcnZpY2UgfSBmcm9tICcuLi90aW1lcGlja2VyLnNlcnZpY2UnO1xuaW1wb3J0IHsgVElNRVBJQ0tFUl9DT05UUk9MIH0gZnJvbSAnLi4vdGltZXBpY2tlci50b2tlbic7XG5cbi8qKlxuICogRGlyZWN0aXZlIHRvIGVuaGFuY2UgYSBzdGFuZGFyZCBpbnB1dCBlbGVtZW50IGZvciB0aW1lIHBpY2tpbmcsIGludGVncmF0aW5nIG1hc2sgZnVuY3Rpb25hbGl0eSBmb3IgdGltZSBmb3JtYXQuXG4gKiBJdCBhdXRvbWF0aWNhbGx5IGFkYXB0cyB0byBsb2NhbGUgc2V0dGluZ3MgcHJvdmlkZWQgYnkgdGhlIGVuY2xvc2luZyBgVGltZXBpY2tlckNvbXBvbmVudGAgdG8gZGlzcGxheSB0aW1lIGluIHRoZSBhcHByb3ByaWF0ZSBmb3JtYXQuXG4gKlxuICogQGV4dGVuZHMge0lucHV0Q29udHJvbERpcmVjdGl2ZX1cbiAqL1xuQENTU0NvbXBvbmVudCgndGltZXBpY2tlcl9fY29udHJvbCcpXG5ARGlyZWN0aXZlKHtcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgc2VsZWN0b3I6ICdpbnB1dFtvZHhUaW1lcGlja2VyQ29udHJvbF0nLFxuICBob3N0OiB7XG4gICAgJ1thdHRyLnJlYWRvbmx5XSc6ICdpc1JlYWRvbmx5IHx8IG51bGwnLFxuICAgICdbYXR0ci5wbGFjZWhvbGRlcl0nOiAncGxhY2Vob2xkZXInLFxuICB9LFxuICBwcm92aWRlcnM6IFtwcm92aWRlTmd4TWFzayhuZ3hNYXNrUHJvdmlkZXJDb25maWcpXSxcbiAgaG9zdERpcmVjdGl2ZXM6IFtXaXRoVGFiSW5kZXgsIFdpdGhEaXNhYmxlZFN0YXRlLCBOZ3hNYXNrRGlyZWN0aXZlXSxcbn0pXG5leHBvcnQgY2xhc3MgVGltZXBpY2tlcklucHV0Q29udHJvbERpcmVjdGl2ZSBleHRlbmRzIElucHV0Q29udHJvbERpcmVjdGl2ZSB7XG4gIHByaXZhdGUgcmVhZG9ubHkgdGltZXBpY2tlciA9IGluamVjdChUSU1FUElDS0VSX0NPTlRST0wpO1xuICBwcml2YXRlIHJlYWRvbmx5IHRpbWVwaWNrZXJTZXJ2aWNlID0gaW5qZWN0KFRpbWVwaWNrZXJTZXJ2aWNlKTtcbiAgcHVibGljIHJlYWRvbmx5IG5neE1hc2tEaXJlY3RpdmUgPSBpbmplY3QoTmd4TWFza0RpcmVjdGl2ZSwgeyBzZWxmOiB0cnVlLCBvcHRpb25hbDogdHJ1ZSB9KTtcblxuICAvKipcbiAgICogUmV0dXJucyB0aGUgcGxhY2Vob2xkZXIgdGV4dCBmb3IgdGhlIGlucHV0IGJhc2VkIG9uIGxvY2FsZSBzZXR0aW5ncy5cbiAgICpcbiAgICogQHJldHVybnMge3N0cmluZ30gVGhlIHBsYWNlaG9sZGVyIHRleHQgZm9yIHRoZSBpbnB1dC5cbiAgICovXG4gIHB1YmxpYyBnZXQgcGxhY2Vob2xkZXIoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gdGhpcy50aW1lcGlja2VyU2VydmljZS51c2VMb2NhbGUoKSA/ICctLTotLSAtLScgOiAnLS06LS0nO1xuICB9XG5cbiAgLyoqXG4gICAqIERldGVybWluZXMgd2hldGhlciB0aGUgdGltZXBpY2tlciBpbnB1dCBpcyByZWFkb25seSwgYmFzZWQgb24gdGhlIHN0YXRlIG9mIHRoZSBwYXJlbnQgdGltZXBpY2tlciBjb21wb25lbnQuXG4gICAqXG4gICAqIEByZXR1cm5zIHtib29sZWFufSBgdHJ1ZWAgaWYgdGhlIGlucHV0IHNob3VsZCBiZSByZWFkb25seTsgb3RoZXJ3aXNlLCBgZmFsc2VgLlxuICAgKi9cbiAgcHJvdGVjdGVkIGdldCBpc1JlYWRvbmx5KCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLnRpbWVwaWNrZXIuaXNSZWFkb25seTtcbiAgfVxuXG4gIC8qKlxuICAgKiBBcHBsaWVzIHRoZSB0aW1lIGZvcm1hdCBtYXNrIHRvIHRoZSBpbnB1dCB2YWx1ZSwgdHJhbnNmb3JtaW5nIGFuZCByZXR1cm5pbmcgdGhlIG1hc2tlZCB2YWx1ZS5cbiAgICpcbiAgICogQHBhcmFtIHtFdmVudH0gZXZlbnQgLSBUaGUgaW5wdXQgZXZlbnQgdHJpZ2dlcmluZyB0aGUgbWFzayBhcHBsaWNhdGlvbi5cbiAgICogQHJldHVybnMge3N0cmluZ30gVGhlIG1hc2tlZCBpbnB1dCB2YWx1ZS5cbiAgICovXG4gIHB1YmxpYyBhcHBseU1hc2sodXNlTG9jYWxlOiBib29sZWFuKTogdm9pZCB7XG4gICAgaWYgKCF0aGlzLm5neE1hc2tEaXJlY3RpdmUpIHJldHVybjtcbiAgICBjb25zdCBtYXNrID0gdXNlTG9jYWxlID8gJ0hoOm0wIEFNJyA6ICdIaDptMCc7XG4gICAgdXNlTG9jYWxlICE9PSB0aGlzLm5neE1hc2tEaXJlY3RpdmUuX21hc2tTZXJ2aWNlLmFwbSAmJiAodGhpcy5uZ3hNYXNrRGlyZWN0aXZlLl9tYXNrU2VydmljZS5hcG0gPSB1c2VMb2NhbGUpO1xuICAgIGlmIChPYmplY3QuaGFzT3duKHRoaXMubmd4TWFza0RpcmVjdGl2ZSwgJ19tYXNrVmFsdWUnKSAmJiB0aGlzLm5neE1hc2tEaXJlY3RpdmVbJ19tYXNrVmFsdWUnXSAhPT0gbWFzaykge1xuICAgICAgaW5pdE5neE1hc2sodGhpcy5uZ3hNYXNrRGlyZWN0aXZlLCBtYXNrKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
@@ -31,8 +31,10 @@ let TimepickerComponent = class TimepickerComponent extends CustomFormControl {
31
31
  * @default false
32
32
  */
33
33
  set useLocale(val) {
34
+ this.timepickerService.useLocale.set(val);
35
+ this.dateField?.applyMask(val);
34
36
  if (this.value && this.dateField) {
35
- const time = this.timepickerService.getLocalizedTimeFormat(this.value, val);
37
+ const time = this.timepickerService.getLocalizedTimeFormat(this.value);
36
38
  this.updateValue(time);
37
39
  this.dateField.nativeElementValue = time;
38
40
  }
@@ -128,8 +130,9 @@ let TimepickerComponent = class TimepickerComponent extends CustomFormControl {
128
130
  */
129
131
  writeValue(value) {
130
132
  super.writeValue(value);
131
- if (value === null || this.timepickerService.isValidTime(value))
133
+ if (value === null || this.timepickerService.isValidTime(value)) {
132
134
  this.updateInputValue();
135
+ }
133
136
  }
134
137
  onOpen() {
135
138
  this.keyManager = new ActiveDescendantKeyManager(this.options).withHomeAndEnd();
@@ -147,7 +150,17 @@ let TimepickerComponent = class TimepickerComponent extends CustomFormControl {
147
150
  }
148
151
  }
149
152
  handleDateFieldChanges() {
150
- this.dateField?.valueChange$.pipe(this.takeUntilDestroyed()).subscribe((time) => this.updateValue(time ?? null));
153
+ this.dateField?.applyMask(this.useLocale);
154
+ this.dateField?.valueChange$.pipe(this.takeUntilDestroyed()).subscribe((time) => {
155
+ const normalized = typeof time === 'string' ? time.trim() : '';
156
+ if (!normalized) {
157
+ this.updateValue(null);
158
+ return;
159
+ }
160
+ if (this.timepickerService.is12HourFormat(normalized) === this._useLocale) {
161
+ this.updateValue(this.timepickerService.getLocalizedTimeFormat(normalized));
162
+ }
163
+ });
151
164
  }
152
165
  resetValue(e) {
153
166
  e.stopImmediatePropagation();
@@ -182,7 +195,7 @@ let TimepickerComponent = class TimepickerComponent extends CustomFormControl {
182
195
  updateInputValue() {
183
196
  if (!this.dateField)
184
197
  return;
185
- this.dateField.nativeElementValue = this.timepickerService.getLocalizedTimeFormat(this.value, this.useLocale);
198
+ this.dateField.nativeElementValue = this.timepickerService.getLocalizedTimeFormat(this.value);
186
199
  }
187
200
  scrollToActiveOption(option, _opts = {}) {
188
201
  if (isFunction(option.element.nativeElement.scrollIntoView)) {
@@ -192,7 +205,7 @@ let TimepickerComponent = class TimepickerComponent extends CustomFormControl {
192
205
  setOption(option) {
193
206
  if (!option || option.disabled)
194
207
  return;
195
- const time = this.timepickerService.getLocalizedTimeFormat(option.getLabel(), this.useLocale);
208
+ const time = this.timepickerService.getLocalizedTimeFormat(option.getLabel());
196
209
  this.dateField && (this.dateField.nativeElementValue = time);
197
210
  this.updateValue(time);
198
211
  }
@@ -204,6 +217,7 @@ let TimepickerComponent = class TimepickerComponent extends CustomFormControl {
204
217
  provide: TIMEPICKER_CONTROL,
205
218
  useExisting: forwardRef(() => TimepickerComponent),
206
219
  },
220
+ TimepickerService,
207
221
  ], viewQueries: [{ propertyName: "dropdown", first: true, predicate: DropdownDirective, descendants: true }, { propertyName: "dateField", first: true, predicate: TimepickerInputControlDirective, descendants: true }, { propertyName: "options", predicate: TimepickerOptionComponent, descendants: true }], usesInheritance: true, ngImport: i0, template: "<div class=\"odx-timepicker__wrapper\">\n <input [value]=\"value\" odxTimepickerControl type=\"text\" />\n</div>\n\n<odx-action-group class=\"odx-timepicker__trigger-wrapper odx-no-margin\">\n @if (clearable() && value) {\n <button class=\"odx-timepicker__clear\" odxButton size=\"small\" aria-label=\"Reset time\" (click)=\"resetValue($event)\">\n <odx-icon name=\"close\" iconSet=\"core\" />\n </button>\n }\n <button\n #dropdownTrigger\n class=\"odx-timepicker__trigger\"\n odxButton\n size=\"small\"\n variant=\"ghost\"\n aria-label=\"Select time\"\n [odxDropdown]=\"timeList\"\n [odxDropdownOptions]=\"{ matchReferenceWidth: true, position: 'bottom-start' }\"\n [odxDropdownTriggerElement]=\"dropdownTrigger.element.nativeElement\"\n [odxDropdownReferenceElement]=\"element.nativeElement\"\n (odxDropdownBeforeClose)=\"onTouched()\"\n (odxDropdownAfterOpen)=\"onOpen()\"\n >\n <odx-icon name=\"chevron-down\" />\n </button>\n</odx-action-group>\n\n<ng-template #timeList>\n <div class=\"odx-timepicker__option-list\" role=\"listbox\">\n @for (time of timeStampsList; track $index) {\n <odx-timepicker-option [value]=\"time\" [disabled]=\"!inTimeRange(time)\" (selected)=\"timeSelected($event)\">{{ time }}</odx-timepicker-option>\n }\n </div>\n</ng-template>\n", dependencies: [{ kind: "ngmodule", type: A11yModule }, { kind: "component", type: ActionGroupComponent, selector: "odx-action-group", inputs: ["reverse"] }, { kind: "component", type: ButtonComponent, selector: "button[odxButton], a[odxButton]", inputs: ["variant", "size"] }, { kind: "ngmodule", type: DropdownModule }, { kind: "directive", type: i1.DisabledController, selector: "[disabled]", inputs: ["disabled"] }, { kind: "directive", type: i2.DropdownDirective, selector: "[odxDropdown]", inputs: ["odxDropdown", "odxDropdownDisabled", "odxDropdownShowLoader", "odxDropdownClickOutsideActive", "odxDropdownOptions", "odxDropdownReferenceElement", "odxDropdownTriggerElement", "odxDropdownHost", "odxDropdownOpenTrigger", "odxDropdownCloseTrigger"], outputs: ["odxDropdownBeforeOpen", "odxDropdownAfterOpen", "odxDropdownBeforeClose", "odxDropdownAfterClose"], exportAs: ["odxDropdown"] }, { kind: "component", type: TimepickerOptionComponent, selector: "odx-timepicker-option", outputs: ["selected"] }, { kind: "component", type: IconComponent, selector: "odx-icon", inputs: ["inline", "size", "name", "iconSet", "identifier"] }, { kind: "directive", type: TimepickerInputControlDirective, selector: "input[odxTimepickerControl]" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
208
222
  };
209
223
  TimepickerComponent = __decorate([
@@ -220,6 +234,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
220
234
  provide: TIMEPICKER_CONTROL,
221
235
  useExisting: forwardRef(() => TimepickerComponent),
222
236
  },
237
+ TimepickerService,
223
238
  ], encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, imports: [A11yModule, ActionGroupComponent, ButtonComponent, DropdownModule, TimepickerOptionComponent, IconComponent, TimepickerInputControlDirective], host: {
224
239
  '[attr.readonly]': 'readonlyController?.readonly || null',
225
240
  }, template: "<div class=\"odx-timepicker__wrapper\">\n <input [value]=\"value\" odxTimepickerControl type=\"text\" />\n</div>\n\n<odx-action-group class=\"odx-timepicker__trigger-wrapper odx-no-margin\">\n @if (clearable() && value) {\n <button class=\"odx-timepicker__clear\" odxButton size=\"small\" aria-label=\"Reset time\" (click)=\"resetValue($event)\">\n <odx-icon name=\"close\" iconSet=\"core\" />\n </button>\n }\n <button\n #dropdownTrigger\n class=\"odx-timepicker__trigger\"\n odxButton\n size=\"small\"\n variant=\"ghost\"\n aria-label=\"Select time\"\n [odxDropdown]=\"timeList\"\n [odxDropdownOptions]=\"{ matchReferenceWidth: true, position: 'bottom-start' }\"\n [odxDropdownTriggerElement]=\"dropdownTrigger.element.nativeElement\"\n [odxDropdownReferenceElement]=\"element.nativeElement\"\n (odxDropdownBeforeClose)=\"onTouched()\"\n (odxDropdownAfterOpen)=\"onOpen()\"\n >\n <odx-icon name=\"chevron-down\" />\n </button>\n</odx-action-group>\n\n<ng-template #timeList>\n <div class=\"odx-timepicker__option-list\" role=\"listbox\">\n @for (time of timeStampsList; track $index) {\n <odx-timepicker-option [value]=\"time\" [disabled]=\"!inTimeRange(time)\" (selected)=\"timeSelected($event)\">{{ time }}</odx-timepicker-option>\n }\n </div>\n</ng-template>\n" }]
@@ -249,4 +264,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
249
264
  type: HostListener,
250
265
  args: ['keydown', ['$event']]
251
266
  }] } });
252
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"timepicker.component.js","sourceRoot":"","sources":["../../../../../../../libs/angular/components/timepicker/src/lib/timepicker.component.ts","../../../../../../../libs/angular/components/timepicker/src/lib/timepicker.component.html"],"names":[],"mappings":"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","sourcesContent":["import { A11yModule, ActiveDescendantKeyManager } from '@angular/cdk/a11y';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  HostListener,\n  Input,\n  QueryList,\n  ViewChild,\n  ViewChildren,\n  ViewEncapsulation,\n  booleanAttribute,\n  forwardRef,\n  inject,\n  input,\n  numberAttribute,\n} from '@angular/core';\nimport { DisabledController, ReadonlyController } from '@odx/angular';\nimport { CustomFormControl } from '@odx/angular/cdk/custom-form-control';\nimport { OptionControl } from '@odx/angular/cdk/option-control';\nimport { ActionGroupComponent } from '@odx/angular/components/action-group';\nimport { ButtonComponent } from '@odx/angular/components/button';\nimport { DropdownDirective, DropdownModule } from '@odx/angular/components/dropdown';\nimport { IconComponent } from '@odx/angular/components/icon';\nimport { CSSComponent } from '@odx/angular/internal';\nimport { injectElement, isFunction, untilDestroyed } from '@odx/angular/utils';\nimport { TimepickerOptionComponent } from './components/timepicker-option.component';\nimport { TimepickerInputControlDirective } from './directives/timepicker-input-control.directive';\nimport { TimepickerService } from './timepicker.service';\nimport { TIMEPICKER_CONTROL } from './timepicker.token';\nimport { generateTimeStamps } from './utils/generate-time-stamps';\n\n/**\n * Represents a time picker component allowing users to select a time from a dropdown list.\n * This component integrates with Angular forms and supports customization for locale, time range, and step intervals.\n *\n * @see {CustomFormControl}\n */\n@CSSComponent('timepicker')\n@Component({\n  standalone: true,\n  selector: 'odx-timepicker',\n  templateUrl: 'timepicker.component.html',\n  providers: [\n    DisabledController.connect(),\n    ReadonlyController.connect(),\n    {\n      provide: TIMEPICKER_CONTROL,\n      useExisting: forwardRef(() => TimepickerComponent),\n    },\n  ],\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [A11yModule, ActionGroupComponent, ButtonComponent, DropdownModule, TimepickerOptionComponent, IconComponent, TimepickerInputControlDirective],\n  host: {\n    '[attr.readonly]': 'readonlyController?.readonly || null',\n  },\n})\nexport class TimepickerComponent extends CustomFormControl<string | null> implements AfterViewInit {\n  private readonly timepickerService = inject(TimepickerService);\n  private _useLocale = false;\n  protected keyManager?: ActiveDescendantKeyManager<OptionControl<string>>;\n\n  protected readonly takeUntilDestroyed = untilDestroyed();\n\n  @ViewChild(DropdownDirective)\n  protected readonly dropdown?: DropdownDirective;\n\n  @ViewChildren(TimepickerOptionComponent, { emitDistinctChangesOnly: true })\n  protected options!: QueryList<TimepickerOptionComponent>;\n\n  public readonly element = injectElement();\n\n  /**\n   * Controls whether the timepicker should use locale-specific time formats AM/PM.\n   *\n   * @type {boolean}\n   * @default false\n   */\n  @Input({ transform: booleanAttribute })\n  public set useLocale(val: boolean) {\n    if (this.value && this.dateField) {\n      const time = this.timepickerService.getLocalizedTimeFormat(this.value, val);\n      this.updateValue(time);\n      this.dateField.nativeElementValue = time;\n    }\n    this._useLocale = val;\n  }\n\n  /**\n   * Gets a boolean value indicating whether the locale is being used.\n   *\n   * @returns {boolean} A boolean value indicating whether the locale is being used.\n   */\n  public get useLocale(): boolean {\n    return this._useLocale;\n  }\n\n  /**\n   * When set to true, the select will display a reset button.\n   *\n   * @type {boolean}\n   * @default false\n   */\n  public clearable = input(false, { transform: booleanAttribute });\n\n  /**\n   * Controls the step interval between time options in minutes.\n   *\n   * @type {number}\n   * @default 30\n   */\n  @Input({ transform: numberAttribute })\n  public step = 30;\n\n  /**\n   * Specifies the minimum time value that can be selected ('05:00' or '05:00 AM').\n   *\n   * @type {string}\n   * @default '00:00'\n   */\n  @Input()\n  public min = '00:00';\n\n  /**\n   * Specifies the maximum time value that can be selected ('22:00' or '10:00 PM').\n   *\n   * @type {string}\n   * @default '23:59'\n   */\n  @Input()\n  public max = '23:59';\n\n  /**\n   * The directive for the timepicker input control.\n   *\n   * @type {TimepickerInputControlDirective | undefined}\n   */\n  @ViewChild(TimepickerInputControlDirective)\n  public dateField?: TimepickerInputControlDirective;\n\n  constructor() {\n    super(null);\n  }\n\n  public ngAfterViewInit(): void {\n    this.handleDateFieldChanges();\n    this.updateInputValue();\n  }\n\n  /**\n   * Checks if the given time option is selected.\n   *\n   * @param {TimepickerOptionComponent} option - The time option to check.\n   * @returns {boolean} True if the option is selected, false otherwise.\n   */\n  public isTimeOptionSelected(option: TimepickerOptionComponent): boolean {\n    return this.value === option.value;\n  }\n\n  /**\n   * Handles the selection of a time option from the dropdown, updating the input field and closing the dropdown.\n   *\n   * @param {TimepickerOptionComponent | undefined} option - The selected time option component.\n   */\n  public timeSelected(option?: TimepickerOptionComponent): void {\n    this.setOption(option);\n    this.dropdown?.isOpen && this.dropdown?.close();\n  }\n\n  /**\n   * Determines whether the specified time is within the allowed time range.\n   *\n   * @param {string} time - The time to check, in 'HH:mm' or 'HH:mm AM/PM' format.\n   * @returns {boolean} True if the time is within the range; otherwise, false.\n   */\n  public inTimeRange(time: string): boolean {\n    return this.timepickerService.maxValidation(time, this.max) && this.timepickerService.minValidation(time, this.min);\n  }\n\n  /**\n   * Generates the list of time options based on the configured step interval and locale settings AM/PM.\n   *\n   * @returns {string[]} An array of time strings in 'HH:mm' format.\n   */\n  public get timeStampsList(): string[] {\n    return generateTimeStamps(this.step, this.useLocale);\n  }\n\n  /**\n   * @internal\n   * Writes a new value to the element.\n   * Part of the ControlValueAccessor interface.\n   * @param {string | null} value - The new value.\n   */\n  public override writeValue(value: string | null): void {\n    super.writeValue(value);\n    if (value === null || this.timepickerService.isValidTime(value)) this.updateInputValue();\n  }\n\n  protected onOpen(): void {\n    this.keyManager = new ActiveDescendantKeyManager<OptionControl<string>>(this.options).withHomeAndEnd();\n    this.setActiveOptionBasedOnCurrentValue();\n  }\n\n  @HostListener('click', ['$event'])\n  @HostListener('keydown', ['$event'])\n  protected handleControllerEvent(event: KeyboardEvent) {\n    if (this.readonlyController?.readonly) return;\n    this.keyManager?.onKeydown(event);\n    const activeOption = this.keyManager?.activeItem as TimepickerOptionComponent;\n    activeOption && this.scrollToActiveOption(activeOption, { behavior: 'smooth' });\n    if (event.key === 'Enter' || event.key === ' ' || event.key === 'Spacebar') {\n      event.stopImmediatePropagation();\n      this.setOption(activeOption);\n    }\n  }\n\n  protected handleDateFieldChanges(): void {\n    this.dateField?.valueChange$.pipe(this.takeUntilDestroyed()).subscribe((time) => this.updateValue(time ?? null));\n  }\n\n  protected resetValue(e: Event): void {\n    e.stopImmediatePropagation();\n    this.updateValue(null);\n  }\n\n  private setActiveOptionBasedOnCurrentValue(): void {\n    const isSelected = this.options.find((option) => option.isSelected);\n    if (isSelected) {\n      this.activateOption(isSelected);\n      return;\n    }\n    this.activateNearestTimeOption();\n  }\n\n  private activateOption(option: TimepickerOptionComponent): void {\n    this.keyManager && this.keyManager.setActiveItem(option);\n    this.scrollToActiveOption(option);\n  }\n\n  private activateNearestTimeOption(): void {\n    if (!this.value) {\n      const currentDate = new Date();\n      const isNearest = this.findNearestTimeOption(currentDate);\n      if (isNearest) {\n        this.activateOption(isNearest);\n      }\n    }\n  }\n\n  private findNearestTimeOption(currentDate: Date): TimepickerOptionComponent | undefined {\n    const availableTimeSlots = this.options.filter((option) => !option.disabled);\n    const nearestTimeValue = this.timepickerService.findClosestDate(\n      availableTimeSlots.map((option) => option.value) as string[],\n      `${currentDate.getHours()}:${currentDate.getMinutes()}`,\n    );\n\n    return availableTimeSlots.find((option) => option.value === nearestTimeValue);\n  }\n\n  private updateInputValue(): void {\n    if (!this.dateField) return;\n    this.dateField.nativeElementValue = this.timepickerService.getLocalizedTimeFormat(this.value, this.useLocale);\n  }\n\n  private scrollToActiveOption(option: TimepickerOptionComponent, _opts: ScrollIntoViewOptions = {}): void {\n    if (isFunction(option.element.nativeElement.scrollIntoView)) {\n      option.element.nativeElement.scrollIntoView({ block: 'center', ..._opts });\n    }\n  }\n\n  private setOption(option?: TimepickerOptionComponent): void {\n    if (!option || option.disabled) return;\n    const time = this.timepickerService.getLocalizedTimeFormat(option.getLabel(), this.useLocale);\n    this.dateField && (this.dateField.nativeElementValue = time);\n    this.updateValue(time);\n  }\n}\n","<div class=\"odx-timepicker__wrapper\">\n  <input [value]=\"value\" odxTimepickerControl type=\"text\" />\n</div>\n\n<odx-action-group class=\"odx-timepicker__trigger-wrapper odx-no-margin\">\n  @if (clearable() && value) {\n    <button class=\"odx-timepicker__clear\" odxButton size=\"small\" aria-label=\"Reset time\" (click)=\"resetValue($event)\">\n      <odx-icon name=\"close\" iconSet=\"core\" />\n    </button>\n  }\n  <button\n    #dropdownTrigger\n    class=\"odx-timepicker__trigger\"\n    odxButton\n    size=\"small\"\n    variant=\"ghost\"\n    aria-label=\"Select time\"\n    [odxDropdown]=\"timeList\"\n    [odxDropdownOptions]=\"{ matchReferenceWidth: true, position: 'bottom-start' }\"\n    [odxDropdownTriggerElement]=\"dropdownTrigger.element.nativeElement\"\n    [odxDropdownReferenceElement]=\"element.nativeElement\"\n    (odxDropdownBeforeClose)=\"onTouched()\"\n    (odxDropdownAfterOpen)=\"onOpen()\"\n  >\n    <odx-icon name=\"chevron-down\" />\n  </button>\n</odx-action-group>\n\n<ng-template #timeList>\n  <div class=\"odx-timepicker__option-list\" role=\"listbox\">\n    @for (time of timeStampsList; track $index) {\n      <odx-timepicker-option [value]=\"time\" [disabled]=\"!inTimeRange(time)\" (selected)=\"timeSelected($event)\">{{ time }}</odx-timepicker-option>\n    }\n  </div>\n</ng-template>\n"]}
267
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"timepicker.component.js","sourceRoot":"","sources":["../../../../../../../libs/angular/components/timepicker/src/lib/timepicker.component.ts","../../../../../../../libs/angular/components/timepicker/src/lib/timepicker.component.html"],"names":[],"mappings":"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","sourcesContent":["import { A11yModule, ActiveDescendantKeyManager } from '@angular/cdk/a11y';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  HostListener,\n  Input,\n  QueryList,\n  ViewChild,\n  ViewChildren,\n  ViewEncapsulation,\n  booleanAttribute,\n  forwardRef,\n  inject,\n  input,\n  numberAttribute,\n} from '@angular/core';\nimport { DisabledController, ReadonlyController } from '@odx/angular';\nimport { CustomFormControl } from '@odx/angular/cdk/custom-form-control';\nimport { OptionControl } from '@odx/angular/cdk/option-control';\nimport { ActionGroupComponent } from '@odx/angular/components/action-group';\nimport { ButtonComponent } from '@odx/angular/components/button';\nimport { DropdownDirective, DropdownModule } from '@odx/angular/components/dropdown';\nimport { IconComponent } from '@odx/angular/components/icon';\nimport { CSSComponent } from '@odx/angular/internal';\nimport { injectElement, isFunction, untilDestroyed } from '@odx/angular/utils';\nimport { TimepickerOptionComponent } from './components/timepicker-option.component';\nimport { TimepickerInputControlDirective } from './directives/timepicker-input-control.directive';\nimport { TimepickerService } from './timepicker.service';\nimport { TIMEPICKER_CONTROL } from './timepicker.token';\nimport { generateTimeStamps } from './utils/generate-time-stamps';\n\n/**\n * Represents a time picker component allowing users to select a time from a dropdown list.\n * This component integrates with Angular forms and supports customization for locale, time range, and step intervals.\n *\n * @see {CustomFormControl}\n */\n@CSSComponent('timepicker')\n@Component({\n  standalone: true,\n  selector: 'odx-timepicker',\n  templateUrl: 'timepicker.component.html',\n  providers: [\n    DisabledController.connect(),\n    ReadonlyController.connect(),\n    {\n      provide: TIMEPICKER_CONTROL,\n      useExisting: forwardRef(() => TimepickerComponent),\n    },\n    TimepickerService,\n  ],\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [A11yModule, ActionGroupComponent, ButtonComponent, DropdownModule, TimepickerOptionComponent, IconComponent, TimepickerInputControlDirective],\n  host: {\n    '[attr.readonly]': 'readonlyController?.readonly || null',\n  },\n})\nexport class TimepickerComponent extends CustomFormControl<string | null> implements AfterViewInit {\n  private readonly timepickerService = inject(TimepickerService);\n  private _useLocale = false;\n  protected keyManager?: ActiveDescendantKeyManager<OptionControl<string>>;\n\n  protected readonly takeUntilDestroyed = untilDestroyed();\n\n  @ViewChild(DropdownDirective)\n  protected readonly dropdown?: DropdownDirective;\n\n  @ViewChildren(TimepickerOptionComponent, { emitDistinctChangesOnly: true })\n  protected options!: QueryList<TimepickerOptionComponent>;\n\n  public readonly element = injectElement();\n\n  /**\n   * Controls whether the timepicker should use locale-specific time formats AM/PM.\n   *\n   * @type {boolean}\n   * @default false\n   */\n  @Input({ transform: booleanAttribute })\n  public set useLocale(val: boolean) {\n    this.timepickerService.useLocale.set(val);\n    this.dateField?.applyMask(val);\n    if (this.value && this.dateField) {\n      const time = this.timepickerService.getLocalizedTimeFormat(this.value);\n      this.updateValue(time);\n      this.dateField.nativeElementValue = time;\n    }\n    this._useLocale = val;\n  }\n\n  /**\n   * Gets a boolean value indicating whether the locale is being used.\n   *\n   * @returns {boolean} A boolean value indicating whether the locale is being used.\n   */\n  public get useLocale(): boolean {\n    return this._useLocale;\n  }\n\n  /**\n   * When set to true, the select will display a reset button.\n   *\n   * @type {boolean}\n   * @default false\n   */\n  public clearable = input(false, { transform: booleanAttribute });\n\n  /**\n   * Controls the step interval between time options in minutes.\n   *\n   * @type {number}\n   * @default 30\n   */\n  @Input({ transform: numberAttribute })\n  public step = 30;\n\n  /**\n   * Specifies the minimum time value that can be selected ('05:00' or '05:00 AM').\n   *\n   * @type {string}\n   * @default '00:00'\n   */\n  @Input()\n  public min = '00:00';\n\n  /**\n   * Specifies the maximum time value that can be selected ('22:00' or '10:00 PM').\n   *\n   * @type {string}\n   * @default '23:59'\n   */\n  @Input()\n  public max = '23:59';\n\n  /**\n   * The directive for the timepicker input control.\n   *\n   * @type {TimepickerInputControlDirective | undefined}\n   */\n  @ViewChild(TimepickerInputControlDirective)\n  public dateField?: TimepickerInputControlDirective;\n\n  constructor() {\n    super(null);\n  }\n\n  public ngAfterViewInit(): void {\n    this.handleDateFieldChanges();\n    this.updateInputValue();\n  }\n\n  /**\n   * Checks if the given time option is selected.\n   *\n   * @param {TimepickerOptionComponent} option - The time option to check.\n   * @returns {boolean} True if the option is selected, false otherwise.\n   */\n  public isTimeOptionSelected(option: TimepickerOptionComponent): boolean {\n    return this.value === option.value;\n  }\n\n  /**\n   * Handles the selection of a time option from the dropdown, updating the input field and closing the dropdown.\n   *\n   * @param {TimepickerOptionComponent | undefined} option - The selected time option component.\n   */\n  public timeSelected(option?: TimepickerOptionComponent): void {\n    this.setOption(option);\n    this.dropdown?.isOpen && this.dropdown?.close();\n  }\n\n  /**\n   * Determines whether the specified time is within the allowed time range.\n   *\n   * @param {string} time - The time to check, in 'HH:mm' or 'HH:mm AM/PM' format.\n   * @returns {boolean} True if the time is within the range; otherwise, false.\n   */\n  public inTimeRange(time: string): boolean {\n    return this.timepickerService.maxValidation(time, this.max) && this.timepickerService.minValidation(time, this.min);\n  }\n\n  /**\n   * Generates the list of time options based on the configured step interval and locale settings AM/PM.\n   *\n   * @returns {string[]} An array of time strings in 'HH:mm' format.\n   */\n  public get timeStampsList(): string[] {\n    return generateTimeStamps(this.step, this.useLocale);\n  }\n\n  /**\n   * @internal\n   * Writes a new value to the element.\n   * Part of the ControlValueAccessor interface.\n   * @param {string | null} value - The new value.\n   */\n  public override writeValue(value: string | null): void {\n    super.writeValue(value);\n    if (value === null || this.timepickerService.isValidTime(value)) {\n      this.updateInputValue();\n    }\n  }\n\n  protected onOpen(): void {\n    this.keyManager = new ActiveDescendantKeyManager<OptionControl<string>>(this.options).withHomeAndEnd();\n    this.setActiveOptionBasedOnCurrentValue();\n  }\n\n  @HostListener('click', ['$event'])\n  @HostListener('keydown', ['$event'])\n  protected handleControllerEvent(event: KeyboardEvent) {\n    if (this.readonlyController?.readonly) return;\n    this.keyManager?.onKeydown(event);\n    const activeOption = this.keyManager?.activeItem as TimepickerOptionComponent;\n    activeOption && this.scrollToActiveOption(activeOption, { behavior: 'smooth' });\n    if (event.key === 'Enter' || event.key === ' ' || event.key === 'Spacebar') {\n      event.stopImmediatePropagation();\n      this.setOption(activeOption);\n    }\n  }\n\n  protected handleDateFieldChanges(): void {\n    this.dateField?.applyMask(this.useLocale);\n    this.dateField?.valueChange$.pipe(this.takeUntilDestroyed()).subscribe((time) => {\n      const normalized = typeof time === 'string' ? time.trim() : '';\n      if (!normalized) {\n        this.updateValue(null);\n        return;\n      }\n      if (this.timepickerService.is12HourFormat(normalized) === this._useLocale) {\n        this.updateValue(this.timepickerService.getLocalizedTimeFormat(normalized));\n      }\n    });\n  }\n\n  protected resetValue(e: Event): void {\n    e.stopImmediatePropagation();\n    this.updateValue(null);\n  }\n\n  private setActiveOptionBasedOnCurrentValue(): void {\n    const isSelected = this.options.find((option) => option.isSelected);\n    if (isSelected) {\n      this.activateOption(isSelected);\n      return;\n    }\n    this.activateNearestTimeOption();\n  }\n\n  private activateOption(option: TimepickerOptionComponent): void {\n    this.keyManager && this.keyManager.setActiveItem(option);\n    this.scrollToActiveOption(option);\n  }\n\n  private activateNearestTimeOption(): void {\n    if (!this.value) {\n      const currentDate = new Date();\n      const isNearest = this.findNearestTimeOption(currentDate);\n      if (isNearest) {\n        this.activateOption(isNearest);\n      }\n    }\n  }\n\n  private findNearestTimeOption(currentDate: Date): TimepickerOptionComponent | undefined {\n    const availableTimeSlots = this.options.filter((option) => !option.disabled);\n    const nearestTimeValue = this.timepickerService.findClosestDate(\n      availableTimeSlots.map((option) => option.value) as string[],\n      `${currentDate.getHours()}:${currentDate.getMinutes()}`,\n    );\n\n    return availableTimeSlots.find((option) => option.value === nearestTimeValue);\n  }\n\n  private updateInputValue(): void {\n    if (!this.dateField) return;\n    this.dateField.nativeElementValue = this.timepickerService.getLocalizedTimeFormat(this.value);\n  }\n\n  private scrollToActiveOption(option: TimepickerOptionComponent, _opts: ScrollIntoViewOptions = {}): void {\n    if (isFunction(option.element.nativeElement.scrollIntoView)) {\n      option.element.nativeElement.scrollIntoView({ block: 'center', ..._opts });\n    }\n  }\n\n  private setOption(option?: TimepickerOptionComponent): void {\n    if (!option || option.disabled) return;\n    const time = this.timepickerService.getLocalizedTimeFormat(option.getLabel());\n    this.dateField && (this.dateField.nativeElementValue = time);\n    this.updateValue(time);\n  }\n}\n","<div class=\"odx-timepicker__wrapper\">\n  <input [value]=\"value\" odxTimepickerControl type=\"text\" />\n</div>\n\n<odx-action-group class=\"odx-timepicker__trigger-wrapper odx-no-margin\">\n  @if (clearable() && value) {\n    <button class=\"odx-timepicker__clear\" odxButton size=\"small\" aria-label=\"Reset time\" (click)=\"resetValue($event)\">\n      <odx-icon name=\"close\" iconSet=\"core\" />\n    </button>\n  }\n  <button\n    #dropdownTrigger\n    class=\"odx-timepicker__trigger\"\n    odxButton\n    size=\"small\"\n    variant=\"ghost\"\n    aria-label=\"Select time\"\n    [odxDropdown]=\"timeList\"\n    [odxDropdownOptions]=\"{ matchReferenceWidth: true, position: 'bottom-start' }\"\n    [odxDropdownTriggerElement]=\"dropdownTrigger.element.nativeElement\"\n    [odxDropdownReferenceElement]=\"element.nativeElement\"\n    (odxDropdownBeforeClose)=\"onTouched()\"\n    (odxDropdownAfterOpen)=\"onOpen()\"\n  >\n    <odx-icon name=\"chevron-down\" />\n  </button>\n</odx-action-group>\n\n<ng-template #timeList>\n  <div class=\"odx-timepicker__option-list\" role=\"listbox\">\n    @for (time of timeStampsList; track $index) {\n      <odx-timepicker-option [value]=\"time\" [disabled]=\"!inTimeRange(time)\" (selected)=\"timeSelected($event)\">{{ time }}</odx-timepicker-option>\n    }\n  </div>\n</ng-template>\n"]}
@@ -1,4 +1,4 @@
1
- import { Injectable, inject } from '@angular/core';
1
+ import { Injectable, inject, signal } from '@angular/core';
2
2
  import { WindowRef } from '@odx/angular';
3
3
  import { isAfter, isBefore, isEqual } from 'date-fns';
4
4
  import * as i0 from "@angular/core";
@@ -9,14 +9,13 @@ import * as i0 from "@angular/core";
9
9
  export class TimepickerService {
10
10
  constructor() {
11
11
  this.windowRef = inject(WindowRef);
12
+ this.useLocale = signal(false);
12
13
  }
13
14
  /**
14
- * Generates a placeholder string for time input fields.
15
- *
16
- * @param {boolean} [apm=false] - Specifies if the placeholder should include AM/PM notation.
17
- * @returns {string} The placeholder string.
15
+ * @deprecated Will be removed in a future major version. Use `useLocale` signal instead.
16
+ * Generates a placeholder string for time input fields, preserving legacy API.
18
17
  */
19
- getPlaceholder(apm = false) {
18
+ getPlaceholder(apm = this.useLocale()) {
20
19
  return apm ? '--:-- --' : '--:--';
21
20
  }
22
21
  /**
@@ -38,11 +37,15 @@ export class TimepickerService {
38
37
  * @returns {time is string} True if the value is a valid time string, false otherwise.
39
38
  */
40
39
  isValidTime(time) {
41
- if (!time || typeof time !== 'string')
40
+ if (typeof time !== 'string')
41
+ return false;
42
+ const trimmedTime = time.trim();
43
+ if (!trimmedTime)
42
44
  return false;
43
- // Matches "HH:mm" (24h) or "HH:mm AM/PM" (12h)
44
- const regex = /^([01]\d|2[0-3]):([0-5]\d)(\s?(AM|PM|am|pm))?$/;
45
- return regex.test(time.trim());
45
+ return Boolean(this.parseTimeString(trimmedTime.toUpperCase()));
46
+ }
47
+ is12HourFormat(value) {
48
+ return this.isValidTime(value) && !!this.parseTimeString(value.toUpperCase())?.hour12;
46
49
  }
47
50
  /**
48
51
  * Validates if the given time is greater than or equal to a minimum time constraint.
@@ -83,19 +86,43 @@ export class TimepickerService {
83
86
  * @param {boolean} [hour12=false] - Specifies if the output should use 12-hour format with AM/PM notation.
84
87
  * @returns {string} The formatted time string, or an empty string if the input time is invalid.
85
88
  */
86
- getLocalizedTimeFormat(time, hour12 = false) {
89
+ getLocalizedTimeFormat(time, hour12) {
87
90
  if (!this.isValidTime(time))
88
91
  return '';
89
- const locale = hour12 ? 'en-US' : 'en-GB';
92
+ const shouldUseHour12 = typeof hour12 === 'boolean' ? hour12 : this.useLocale();
93
+ const locale = shouldUseHour12 ? 'en-US' : 'en-GB';
90
94
  const [date] = this.convertToDates([time]);
91
95
  return new this.windowRef.nativeWindow.Intl.DateTimeFormat(locale, {
92
96
  hour: '2-digit',
93
97
  minute: '2-digit',
94
- hour12,
98
+ hour12: shouldUseHour12,
95
99
  }).format(date);
96
100
  }
97
101
  convertToDates(times) {
98
- return times.map((time) => new Date(`${time} 2022-12-19`));
102
+ return times.map((time) => {
103
+ if (typeof time !== 'string')
104
+ return new Date(NaN);
105
+ const parsed = this.parseTimeString(time.toUpperCase());
106
+ if (!parsed)
107
+ return new Date(NaN);
108
+ const { hours, minutes } = parsed;
109
+ return new Date(2022, 11, 19, hours, minutes, 0, 0);
110
+ });
111
+ }
112
+ parseTimeString(time) {
113
+ const match = time.trim().match(/^([01]?\d|2[0-3]):([0-5]\d)(?:\s?(AM|PM))?$/);
114
+ if (!match)
115
+ return null;
116
+ let hours = Number(match[1]);
117
+ const minutes = Number(match[2]);
118
+ const period = match[3]?.toUpperCase();
119
+ if (period) {
120
+ if (period === 'PM' && hours < 12)
121
+ hours += 12;
122
+ if (period === 'AM' && hours === 12)
123
+ hours = 0;
124
+ }
125
+ return { hours, minutes, hour12: Boolean(period) };
99
126
  }
100
127
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TimepickerService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
101
128
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TimepickerService, providedIn: 'root' }); }
@@ -104,4 +131,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
104
131
  type: Injectable,
105
132
  args: [{ providedIn: 'root' }]
106
133
  }] });
107
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"timepicker.service.js","sourceRoot":"","sources":["../../../../../../../libs/angular/components/timepicker/src/lib/timepicker.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { WindowRef } from '@odx/angular';\nimport { isAfter, isBefore, isEqual } from 'date-fns';\n\n/**\n * Service to provide utility functions for timepicker components, including validation, placeholder generation, and finding the closest time.\n * It also handles locale-based time formatting.\n */\n@Injectable({ providedIn: 'root' })\nexport class TimepickerService {\n  private readonly windowRef = inject(WindowRef);\n\n  /**\n   * Generates a placeholder string for time input fields.\n   *\n   * @param {boolean} [apm=false] - Specifies if the placeholder should include AM/PM notation.\n   * @returns {string} The placeholder string.\n   */\n  public getPlaceholder(apm = false): string {\n    return apm ? '--:-- --' : '--:--';\n  }\n\n  /**\n   * Validates if the given time is less than or equal to a maximum time constraint.\n   *\n   * @param {string} time - The time to validate.\n   * @param {string} max - The maximum allowable time.\n   * @returns {boolean} True if the time is valid (i.e., not after max), false otherwise.\n   */\n  public maxValidation(time: string, max: string): boolean {\n    const [targetValue, maxValue] = this.convertToDates([time, max]);\n    return isBefore(targetValue, maxValue) || isEqual(targetValue, maxValue);\n  }\n\n  /**\n   * Checks if the provided value is a valid time string in HH:mm or HH:mm AM/PM format.\n   * This method also acts as a type guard.\n   *\n   * @param {unknown} time - The value to validate.\n   * @returns {time is string} True if the value is a valid time string, false otherwise.\n   */\n  public isValidTime(time: unknown): time is string {\n    if (!time || typeof time !== 'string') return false;\n    // Matches \"HH:mm\" (24h) or \"HH:mm AM/PM\" (12h)\n    const regex = /^([01]\\d|2[0-3]):([0-5]\\d)(\\s?(AM|PM|am|pm))?$/;\n    return regex.test(time.trim());\n  }\n\n  /**\n   * Validates if the given time is greater than or equal to a minimum time constraint.\n   *\n   * @param {string} time - The time to validate.\n   * @param {string} min - The minimum allowable time.\n   * @returns {boolean} True if the time is valid (i.e., not before min), false otherwise.\n   */\n  public minValidation(time: string, min: string): boolean {\n    const [targetValue, minValue] = this.convertToDates([time, min]);\n    return isAfter(targetValue, minValue) || isEqual(targetValue, minValue);\n  }\n\n  /**\n   * Finds the closest time to a target time from a list of times.\n   *\n   * @param {string[]} timeStamps - The list of time strings to search through (e.g., [\"10:00\", \"10:30\", \"11:00\"]).\n   * @param {string} targetTime - The target time string to find the closest match for (e.g., \"10:35\").\n   * @returns {string} The closest time string from the list to the target, formatted according to locale.\n   */\n  public findClosestDate(timeStamps: string[], targetTime: string): string {\n    const [target] = this.convertToDates([targetTime]);\n    const datesArray = this.convertToDates(timeStamps);\n\n    if (datesArray.length === 0) return this.getLocalizedTimeFormat(targetTime);\n\n    const closestDate = datesArray.reduce((prev, curr) => {\n      const prevDiff = Math.abs(prev.getTime() - target.getTime());\n      const currDiff = Math.abs(curr.getTime() - target.getTime());\n      return currDiff < prevDiff ? curr : prev;\n    });\n\n    const hours = closestDate.getHours().toString().padStart(2, '0');\n    const minutes = closestDate.getMinutes().toString().padStart(2, '0');\n    return this.getLocalizedTimeFormat(`${hours}:${minutes}`);\n  }\n\n  /**\n   * Formats a time string to a localized time format using Intl.DateTimeFormat.\n   *\n   * @param {unknown} time - The time string to format (e.g., \"14:30\").\n   * @param {boolean} [hour12=false] - Specifies if the output should use 12-hour format with AM/PM notation.\n   * @returns {string} The formatted time string, or an empty string if the input time is invalid.\n   */\n  public getLocalizedTimeFormat(time: unknown, hour12 = false): string {\n    if (!this.isValidTime(time)) return '';\n    const locale = hour12 ? 'en-US' : 'en-GB';\n    const [date] = this.convertToDates([time]);\n    return new this.windowRef.nativeWindow.Intl.DateTimeFormat(locale, {\n      hour: '2-digit',\n      minute: '2-digit',\n      hour12,\n    }).format(date);\n  }\n\n  private convertToDates(times: string[]): Date[] {\n    return times.map((time) => new Date(`${time} 2022-12-19`));\n  }\n}\n"]}
134
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"timepicker.service.js","sourceRoot":"","sources":["../../../../../../../libs/angular/components/timepicker/src/lib/timepicker.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject, signal } from '@angular/core';\nimport { WindowRef } from '@odx/angular';\nimport { isAfter, isBefore, isEqual } from 'date-fns';\n\n/**\n * Service to provide utility functions for timepicker components, including validation, placeholder generation, and finding the closest time.\n * It also handles locale-based time formatting.\n */\n@Injectable({ providedIn: 'root' })\nexport class TimepickerService {\n  private readonly windowRef = inject(WindowRef);\n\n  public useLocale = signal(false);\n\n  /**\n   * @deprecated Will be removed in a future major version. Use `useLocale` signal instead.\n   * Generates a placeholder string for time input fields, preserving legacy API.\n   */\n  public getPlaceholder(apm = this.useLocale()): string {\n    return apm ? '--:-- --' : '--:--';\n  }\n\n  /**\n   * Validates if the given time is less than or equal to a maximum time constraint.\n   *\n   * @param {string} time - The time to validate.\n   * @param {string} max - The maximum allowable time.\n   * @returns {boolean} True if the time is valid (i.e., not after max), false otherwise.\n   */\n  public maxValidation(time: string, max: string): boolean {\n    const [targetValue, maxValue] = this.convertToDates([time, max]);\n    return isBefore(targetValue, maxValue) || isEqual(targetValue, maxValue);\n  }\n\n  /**\n   * Checks if the provided value is a valid time string in HH:mm or HH:mm AM/PM format.\n   * This method also acts as a type guard.\n   *\n   * @param {unknown} time - The value to validate.\n   * @returns {time is string} True if the value is a valid time string, false otherwise.\n   */\n  public isValidTime(time: unknown): time is string {\n    if (typeof time !== 'string') return false;\n    const trimmedTime = time.trim();\n    if (!trimmedTime) return false;\n    return Boolean(this.parseTimeString(trimmedTime.toUpperCase()));\n  }\n\n  public is12HourFormat(value: string): boolean {\n    return this.isValidTime(value) && !!this.parseTimeString(value.toUpperCase())?.hour12;\n  }\n\n  /**\n   * Validates if the given time is greater than or equal to a minimum time constraint.\n   *\n   * @param {string} time - The time to validate.\n   * @param {string} min - The minimum allowable time.\n   * @returns {boolean} True if the time is valid (i.e., not before min), false otherwise.\n   */\n  public minValidation(time: string, min: string): boolean {\n    const [targetValue, minValue] = this.convertToDates([time, min]);\n    return isAfter(targetValue, minValue) || isEqual(targetValue, minValue);\n  }\n\n  /**\n   * Finds the closest time to a target time from a list of times.\n   *\n   * @param {string[]} timeStamps - The list of time strings to search through (e.g., [\"10:00\", \"10:30\", \"11:00\"]).\n   * @param {string} targetTime - The target time string to find the closest match for (e.g., \"10:35\").\n   * @returns {string} The closest time string from the list to the target, formatted according to locale.\n   */\n  public findClosestDate(timeStamps: string[], targetTime: string): string {\n    const [target] = this.convertToDates([targetTime]);\n    const datesArray = this.convertToDates(timeStamps);\n\n    if (datesArray.length === 0) return this.getLocalizedTimeFormat(targetTime);\n\n    const closestDate = datesArray.reduce((prev, curr) => {\n      const prevDiff = Math.abs(prev.getTime() - target.getTime());\n      const currDiff = Math.abs(curr.getTime() - target.getTime());\n      return currDiff < prevDiff ? curr : prev;\n    });\n\n    const hours = closestDate.getHours().toString().padStart(2, '0');\n    const minutes = closestDate.getMinutes().toString().padStart(2, '0');\n    return this.getLocalizedTimeFormat(`${hours}:${minutes}`);\n  }\n\n  /**\n   * Formats a time string to a localized time format using Intl.DateTimeFormat.\n   *\n   * @param {unknown} time - The time string to format (e.g., \"14:30\").\n   * @param {boolean} [hour12=false] - Specifies if the output should use 12-hour format with AM/PM notation.\n   * @returns {string} The formatted time string, or an empty string if the input time is invalid.\n   */\n  public getLocalizedTimeFormat(time: unknown, hour12?: boolean): string {\n    if (!this.isValidTime(time)) return '';\n    const shouldUseHour12 = typeof hour12 === 'boolean' ? hour12 : this.useLocale();\n    const locale = shouldUseHour12 ? 'en-US' : 'en-GB';\n    const [date] = this.convertToDates([time]);\n    return new this.windowRef.nativeWindow.Intl.DateTimeFormat(locale, {\n      hour: '2-digit',\n      minute: '2-digit',\n      hour12: shouldUseHour12,\n    }).format(date);\n  }\n\n  private convertToDates(times: string[]): Date[] {\n    return times.map((time) => {\n      if (typeof time !== 'string') return new Date(NaN);\n      const parsed = this.parseTimeString(time.toUpperCase());\n      if (!parsed) return new Date(NaN);\n      const { hours, minutes } = parsed;\n      return new Date(2022, 11, 19, hours, minutes, 0, 0);\n    });\n  }\n\n  private parseTimeString(time: string): { hours: number; minutes: number; hour12: boolean } | null {\n    const match = time.trim().match(/^([01]?\\d|2[0-3]):([0-5]\\d)(?:\\s?(AM|PM))?$/);\n    if (!match) return null;\n\n    let hours = Number(match[1]);\n    const minutes = Number(match[2]);\n    const period = match[3]?.toUpperCase();\n\n    if (period) {\n      if (period === 'PM' && hours < 12) hours += 12;\n      if (period === 'AM' && hours === 12) hours = 0;\n    }\n\n    return { hours, minutes, hour12: Boolean(period) };\n  }\n}\n"]}
@@ -1,5 +1,6 @@
1
- import { createConfigTokens } from '@odx/angular/utils';
1
+ import { createConfigTokens, isFunction } from '@odx/angular/utils';
2
2
  import { toDate, isValid } from 'date-fns';
3
+ import { initialConfig } from 'ngx-mask';
3
4
 
4
5
  const InputDateOrder = {
5
6
  DMY: 'DMY',
@@ -94,9 +95,29 @@ function getDateInputValueAsDate(config, value) {
94
95
  return isValid(date) ? date : null;
95
96
  }
96
97
 
98
+ function initNgxMask(ngxMask = undefined, mask) {
99
+ if (!ngxMask)
100
+ return;
101
+ if (!Object.hasOwn(ngxMask, '_maskValue')) {
102
+ throw new Error('[DatepickerInputControlDirective] initNgxMask: ngx-mask internal API changed, mask signal not available.');
103
+ }
104
+ const maskSignal = ngxMask['_maskValue'];
105
+ isFunction(maskSignal.set) && maskSignal.set(mask);
106
+ ngxMask._maskService.maskExpression = mask;
107
+ }
108
+ const ngxMaskProviderConfig = {
109
+ validation: false,
110
+ leadZeroDateTime: true,
111
+ patterns: {
112
+ ...initialConfig.patterns,
113
+ A: { pattern: new RegExp('[AaPp]') },
114
+ M: { pattern: new RegExp('[mM]') },
115
+ },
116
+ };
117
+
97
118
  /**
98
119
  * Generated bundle index. Do not edit.
99
120
  */
100
121
 
101
- export { DateConfig, DateDefaultConfig, DateInputFormat, DateInputMask, InputDateOrder, getDateInputFormat, getDateInputMask, getDateInputValueAsDate, injectDateConfig, provideDateConfig };
122
+ export { DateConfig, DateDefaultConfig, DateInputFormat, DateInputMask, InputDateOrder, getDateInputFormat, getDateInputMask, getDateInputValueAsDate, initNgxMask, injectDateConfig, ngxMaskProviderConfig, provideDateConfig };
102
123
  //# sourceMappingURL=odx-angular-cdk-date-input.mjs.map