@odx/angular 12.21.1 → 12.21.3
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.
- package/CHANGELOG.md +12 -0
- package/cdk/date-input/index.d.ts +1 -0
- package/cdk/date-input/lib/utils/ngx-mask-init.d.ts +3 -0
- package/components/autocomplete/lib/autocomplete.component.d.ts +1 -0
- package/components/datepicker/lib/directives/datepicker-input-control.directive.d.ts +4 -16
- package/components/daterangepicker/lib/directives/daterangepicker-input-control.directive.d.ts +4 -16
- package/components/timepicker/lib/directives/timepicker-input-control.directive.d.ts +6 -15
- package/components/timepicker/lib/timepicker.service.d.ts +5 -4
- package/esm2022/cdk/date-input/index.mjs +2 -1
- package/esm2022/cdk/date-input/lib/utils/ngx-mask-init.mjs +22 -0
- package/esm2022/components/autocomplete/lib/autocomplete.component.mjs +24 -8
- package/esm2022/components/datepicker/lib/datepicker.component.mjs +3 -2
- package/esm2022/components/datepicker/lib/directives/datepicker-input-control.directive.mjs +8 -23
- package/esm2022/components/daterangepicker/lib/daterangepicker.component.mjs +4 -2
- package/esm2022/components/daterangepicker/lib/directives/daterangepicker-input-control.directive.mjs +8 -23
- package/esm2022/components/timepicker/lib/directives/timepicker-input-control.directive.mjs +17 -37
- package/esm2022/components/timepicker/lib/timepicker.component.mjs +21 -6
- package/esm2022/components/timepicker/lib/timepicker.service.mjs +42 -15
- package/fesm2022/odx-angular-cdk-date-input.mjs +23 -2
- package/fesm2022/odx-angular-cdk-date-input.mjs.map +1 -1
- package/fesm2022/odx-angular-components-autocomplete.mjs +23 -7
- package/fesm2022/odx-angular-components-autocomplete.mjs.map +1 -1
- package/fesm2022/odx-angular-components-datepicker.mjs +10 -24
- package/fesm2022/odx-angular-components-datepicker.mjs.map +1 -1
- package/fesm2022/odx-angular-components-daterangepicker.mjs +11 -24
- package/fesm2022/odx-angular-components-daterangepicker.mjs.map +1 -1
- package/fesm2022/odx-angular-components-timepicker.mjs +113 -90
- package/fesm2022/odx-angular-components-timepicker.mjs.map +1 -1
- package/package.json +7 -7
|
@@ -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
|
|
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?.
|
|
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
|
|
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()
|
|
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
|
-
*
|
|
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 =
|
|
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 (
|
|
40
|
+
if (typeof time !== 'string')
|
|
41
|
+
return false;
|
|
42
|
+
const trimmedTime = time.trim();
|
|
43
|
+
if (!trimmedTime)
|
|
42
44
|
return false;
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
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
|
|
89
|
+
getLocalizedTimeFormat(time, hour12) {
|
|
87
90
|
if (!this.isValidTime(time))
|
|
88
91
|
return '';
|
|
89
|
-
const
|
|
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) =>
|
|
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":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AACzC,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,UAAU,CAAC;;AAEtD;;;GAGG;AAEH,MAAM,OAAO,iBAAiB;IAD9B;QAEmB,cAAS,GAAG,MAAM,CAAC,SAAS,CAAC,CAAC;KA+FhD;IA7FC;;;;;OAKG;IACI,cAAc,CAAC,GAAG,GAAG,KAAK;QAC/B,OAAO,GAAG,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,OAAO,CAAC;IACpC,CAAC;IAED;;;;;;OAMG;IACI,aAAa,CAAC,IAAY,EAAE,GAAW;QAC5C,MAAM,CAAC,WAAW,EAAE,QAAQ,CAAC,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC,CAAC;QACjE,OAAO,QAAQ,CAAC,WAAW,EAAE,QAAQ,CAAC,IAAI,OAAO,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;IAC3E,CAAC;IAED;;;;;;OAMG;IACI,WAAW,CAAC,IAAa;QAC9B,IAAI,CAAC,IAAI,IAAI,OAAO,IAAI,KAAK,QAAQ;YAAE,OAAO,KAAK,CAAC;QACpD,+CAA+C;QAC/C,MAAM,KAAK,GAAG,gDAAgD,CAAC;QAC/D,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;IACjC,CAAC;IAED;;;;;;OAMG;IACI,aAAa,CAAC,IAAY,EAAE,GAAW;QAC5C,MAAM,CAAC,WAAW,EAAE,QAAQ,CAAC,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC,CAAC;QACjE,OAAO,OAAO,CAAC,WAAW,EAAE,QAAQ,CAAC,IAAI,OAAO,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;IAC1E,CAAC;IAED;;;;;;OAMG;IACI,eAAe,CAAC,UAAoB,EAAE,UAAkB;QAC7D,MAAM,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC;QACnD,MAAM,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC;QAEnD,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,IAAI,CAAC,sBAAsB,CAAC,UAAU,CAAC,CAAC;QAE5E,MAAM,WAAW,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,IAAI,EAAE,EAAE;YACnD,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,MAAM,CAAC,OAAO,EAAE,CAAC,CAAC;YAC7D,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,MAAM,CAAC,OAAO,EAAE,CAAC,CAAC;YAC7D,OAAO,QAAQ,GAAG,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC;QAC3C,CAAC,CAAC,CAAC;QAEH,MAAM,KAAK,GAAG,WAAW,CAAC,QAAQ,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;QACjE,MAAM,OAAO,GAAG,WAAW,CAAC,UAAU,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;QACrE,OAAO,IAAI,CAAC,sBAAsB,CAAC,GAAG,KAAK,IAAI,OAAO,EAAE,CAAC,CAAC;IAC5D,CAAC;IAED;;;;;;OAMG;IACI,sBAAsB,CAAC,IAAa,EAAE,MAAM,GAAG,KAAK;QACzD,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC;YAAE,OAAO,EAAE,CAAC;QACvC,MAAM,MAAM,GAAG,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC;QAC1C,MAAM,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QAC3C,OAAO,IAAI,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC,IAAI,CAAC,cAAc,CAAC,MAAM,EAAE;YACjE,IAAI,EAAE,SAAS;YACf,MAAM,EAAE,SAAS;YACjB,MAAM;SACP,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IAClB,CAAC;IAEO,cAAc,CAAC,KAAe;QACpC,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,IAAI,CAAC,GAAG,IAAI,aAAa,CAAC,CAAC,CAAC;IAC7D,CAAC;+GA/FU,iBAAiB;mHAAjB,iBAAiB,cADJ,MAAM;;4FACnB,iBAAiB;kBAD7B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE","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
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"odx-angular-cdk-date-input.mjs","sources":["../../../../libs/angular/cdk/date-input/src/lib/models/input-date-order.ts","../../../../libs/angular/cdk/date-input/src/lib/date.config.ts","../../../../libs/angular/cdk/date-input/src/lib/utils/get-date-input-pattern.ts","../../../../libs/angular/cdk/date-input/src/lib/utils/get-input-value-as-date.ts","../../../../libs/angular/cdk/date-input/src/odx-angular-cdk-date-input.ts"],"sourcesContent":["export type InputDateOrder = (typeof InputDateOrder)[keyof typeof InputDateOrder];\n\nexport const InputDateOrder = {\n DMY: 'DMY',\n MDY: 'MDY',\n YMD: 'YMD',\n};\n\nexport type DateInputPattern = typeof DateInputFormat | typeof DateInputMask;\n\nexport const DateInputFormat = {\n DAYS: 'dd',\n MONTH: 'MM',\n YEAR: 'yyyy',\n};\n\nexport const DateInputMask = {\n DAYS: 'd0',\n MONTH: 'M0',\n YEAR: '0000',\n};\n","import { createConfigTokens } from '@odx/angular/utils';\nimport { InputDateOrder } from './models/input-date-order';\n\nexport interface DateConfig {\n /**\n * Date's order ('DMY' | 'MDY' | 'YMD').\n * @default InputDateOrder.DMY\n */\n inputDateOrder: InputDateOrder;\n /**\n * Single-character date's separator (dot, slash etc.).\n * @default '.'\n */\n inputDateSeparator: string;\n}\n\n/**\n * Tools for injecting and providing the date configuration with default configuration for the date.\n *\n * @example\n * // Providing custom date configuration.\n * ```ts\n * @Component({\n * providers: [provideDateConfig({ inputDateOrder: InputDateOrder.MDY, inputDateSeparator: '/'})]\n * })\n * export class MyComponent {}\n *\n * // Injecting the date configuration.\n * ```ts\n * @Component({})\n * export class MyComponent {\n * constructor(@Inject(injectDateConfig()) private readonly dateConfig: DateConfig) {}\n * }\n * ```\n */\nexport const { DateConfig, DateDefaultConfig, injectDateConfig, provideDateConfig } = createConfigTokens('Date', '@odx/angular/components/date', {\n inputDateOrder: InputDateOrder.DMY,\n inputDateSeparator: '.',\n});\n","import { DateConfig } from '../date.config';\nimport { DateInputFormat, DateInputMask, DateInputPattern, InputDateOrder } from '../models/input-date-order';\n\nfunction getDateInputPattern({ inputDateOrder, inputDateSeparator }: DateConfig, pattern: DateInputPattern): string {\n const patternList: Record<InputDateOrder, string> = {\n [InputDateOrder.DMY]: `${pattern.DAYS}${inputDateSeparator}${pattern.MONTH}${inputDateSeparator}${pattern.YEAR}`,\n [InputDateOrder.MDY]: `${pattern.MONTH}${inputDateSeparator}${pattern.DAYS}${inputDateSeparator}${pattern.YEAR}`,\n [InputDateOrder.YMD]: `${pattern.YEAR}${inputDateSeparator}${pattern.MONTH}${inputDateSeparator}${pattern.DAYS}`,\n };\n\n return patternList[inputDateOrder];\n}\n\n/** @internal */\nexport function getDateInputFormat(config: DateConfig): string {\n return getDateInputPattern(config, DateInputFormat);\n}\n\n/** @internal */\nexport function getDateInputMask(config: DateConfig): string {\n return getDateInputPattern(config, DateInputMask);\n}\n","import { isValid, toDate } from 'date-fns';\nimport { DateConfig } from '../date.config';\nimport { DateInputFormat, InputDateOrder } from '../models/input-date-order';\n\ninterface DateCandidateInterface {\n month: string;\n day: string;\n year: string;\n}\n\nconst functionList: Record<InputDateOrder, (splittedValue: string[]) => DateCandidateInterface> = {\n [InputDateOrder.DMY]: (splittedValue: string[]) => {\n const [day, month, year] = splittedValue;\n return { month, day, year };\n },\n [InputDateOrder.MDY]: (splittedValue: string[]) => {\n const [month, day, year] = splittedValue;\n return { month, day, year };\n },\n [InputDateOrder.YMD]: (splittedValue: string[]) => {\n const [year, month, day] = splittedValue;\n return { month, day, year };\n },\n};\n\nfunction getDateCandidate({ inputDateOrder, inputDateSeparator }: DateConfig, value: string): DateCandidateInterface | null {\n const splittedValue = value.split(inputDateSeparator);\n\n const { month, day, year }: DateCandidateInterface = functionList[inputDateOrder](splittedValue);\n\n if (\n !month ||\n !day ||\n !year ||\n month.length !== DateInputFormat.MONTH.length ||\n day.length !== DateInputFormat.DAYS.length ||\n year.length !== DateInputFormat.YEAR.length\n ) {\n return null;\n }\n\n return { month, day, year };\n}\n\n/** @internal */\nexport function getDateInputValueAsDate(config: DateConfig, value: string): Date | null {\n const dateCandidate: DateCandidateInterface | null = getDateCandidate(config, value);\n\n if (!dateCandidate) return null;\n\n const date = toDate(`${dateCandidate.month}.${dateCandidate.day}.${dateCandidate.year}`);\n\n return isValid(date) ? date : null;\n}\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"odx-angular-cdk-date-input.mjs","sources":["../../../../libs/angular/cdk/date-input/src/lib/models/input-date-order.ts","../../../../libs/angular/cdk/date-input/src/lib/date.config.ts","../../../../libs/angular/cdk/date-input/src/lib/utils/get-date-input-pattern.ts","../../../../libs/angular/cdk/date-input/src/lib/utils/get-input-value-as-date.ts","../../../../libs/angular/cdk/date-input/src/lib/utils/ngx-mask-init.ts","../../../../libs/angular/cdk/date-input/src/odx-angular-cdk-date-input.ts"],"sourcesContent":["export type InputDateOrder = (typeof InputDateOrder)[keyof typeof InputDateOrder];\n\nexport const InputDateOrder = {\n DMY: 'DMY',\n MDY: 'MDY',\n YMD: 'YMD',\n};\n\nexport type DateInputPattern = typeof DateInputFormat | typeof DateInputMask;\n\nexport const DateInputFormat = {\n DAYS: 'dd',\n MONTH: 'MM',\n YEAR: 'yyyy',\n};\n\nexport const DateInputMask = {\n DAYS: 'd0',\n MONTH: 'M0',\n YEAR: '0000',\n};\n","import { createConfigTokens } from '@odx/angular/utils';\nimport { InputDateOrder } from './models/input-date-order';\n\nexport interface DateConfig {\n /**\n * Date's order ('DMY' | 'MDY' | 'YMD').\n * @default InputDateOrder.DMY\n */\n inputDateOrder: InputDateOrder;\n /**\n * Single-character date's separator (dot, slash etc.).\n * @default '.'\n */\n inputDateSeparator: string;\n}\n\n/**\n * Tools for injecting and providing the date configuration with default configuration for the date.\n *\n * @example\n * // Providing custom date configuration.\n * ```ts\n * @Component({\n * providers: [provideDateConfig({ inputDateOrder: InputDateOrder.MDY, inputDateSeparator: '/'})]\n * })\n * export class MyComponent {}\n *\n * // Injecting the date configuration.\n * ```ts\n * @Component({})\n * export class MyComponent {\n * constructor(@Inject(injectDateConfig()) private readonly dateConfig: DateConfig) {}\n * }\n * ```\n */\nexport const { DateConfig, DateDefaultConfig, injectDateConfig, provideDateConfig } = createConfigTokens('Date', '@odx/angular/components/date', {\n inputDateOrder: InputDateOrder.DMY,\n inputDateSeparator: '.',\n});\n","import { DateConfig } from '../date.config';\nimport { DateInputFormat, DateInputMask, DateInputPattern, InputDateOrder } from '../models/input-date-order';\n\nfunction getDateInputPattern({ inputDateOrder, inputDateSeparator }: DateConfig, pattern: DateInputPattern): string {\n const patternList: Record<InputDateOrder, string> = {\n [InputDateOrder.DMY]: `${pattern.DAYS}${inputDateSeparator}${pattern.MONTH}${inputDateSeparator}${pattern.YEAR}`,\n [InputDateOrder.MDY]: `${pattern.MONTH}${inputDateSeparator}${pattern.DAYS}${inputDateSeparator}${pattern.YEAR}`,\n [InputDateOrder.YMD]: `${pattern.YEAR}${inputDateSeparator}${pattern.MONTH}${inputDateSeparator}${pattern.DAYS}`,\n };\n\n return patternList[inputDateOrder];\n}\n\n/** @internal */\nexport function getDateInputFormat(config: DateConfig): string {\n return getDateInputPattern(config, DateInputFormat);\n}\n\n/** @internal */\nexport function getDateInputMask(config: DateConfig): string {\n return getDateInputPattern(config, DateInputMask);\n}\n","import { isValid, toDate } from 'date-fns';\nimport { DateConfig } from '../date.config';\nimport { DateInputFormat, InputDateOrder } from '../models/input-date-order';\n\ninterface DateCandidateInterface {\n month: string;\n day: string;\n year: string;\n}\n\nconst functionList: Record<InputDateOrder, (splittedValue: string[]) => DateCandidateInterface> = {\n [InputDateOrder.DMY]: (splittedValue: string[]) => {\n const [day, month, year] = splittedValue;\n return { month, day, year };\n },\n [InputDateOrder.MDY]: (splittedValue: string[]) => {\n const [month, day, year] = splittedValue;\n return { month, day, year };\n },\n [InputDateOrder.YMD]: (splittedValue: string[]) => {\n const [year, month, day] = splittedValue;\n return { month, day, year };\n },\n};\n\nfunction getDateCandidate({ inputDateOrder, inputDateSeparator }: DateConfig, value: string): DateCandidateInterface | null {\n const splittedValue = value.split(inputDateSeparator);\n\n const { month, day, year }: DateCandidateInterface = functionList[inputDateOrder](splittedValue);\n\n if (\n !month ||\n !day ||\n !year ||\n month.length !== DateInputFormat.MONTH.length ||\n day.length !== DateInputFormat.DAYS.length ||\n year.length !== DateInputFormat.YEAR.length\n ) {\n return null;\n }\n\n return { month, day, year };\n}\n\n/** @internal */\nexport function getDateInputValueAsDate(config: DateConfig, value: string): Date | null {\n const dateCandidate: DateCandidateInterface | null = getDateCandidate(config, value);\n\n if (!dateCandidate) return null;\n\n const date = toDate(`${dateCandidate.month}.${dateCandidate.day}.${dateCandidate.year}`);\n\n return isValid(date) ? date : null;\n}\n","import { isFunction } from '@odx/angular/utils';\nimport { initialConfig, NgxMaskConfig, NgxMaskDirective } from 'ngx-mask';\n\nexport function initNgxMask(ngxMask: NgxMaskDirective | undefined = undefined, mask: string): void {\n if (!ngxMask) return;\n if (!Object.hasOwn(ngxMask, '_maskValue')) {\n throw new Error('[DatepickerInputControlDirective] initNgxMask: ngx-mask internal API changed, mask signal not available.');\n }\n const maskSignal = ngxMask['_maskValue'];\n isFunction(maskSignal.set) && maskSignal.set(mask);\n ngxMask._maskService.maskExpression = mask;\n}\n\nexport const ngxMaskProviderConfig: Partial<NgxMaskConfig> = {\n validation: false,\n leadZeroDateTime: true,\n patterns: {\n ...initialConfig.patterns,\n A: { pattern: new RegExp('[AaPp]') },\n M: { pattern: new RegExp('[mM]') },\n },\n};\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":[],"mappings":"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"}
|
|
@@ -8,7 +8,6 @@ import { DropdownDirective } from '@odx/angular/components/dropdown';
|
|
|
8
8
|
import { LoadingSpinnerModule } from '@odx/angular/components/loading-spinner';
|
|
9
9
|
import { CSSComponent } from '@odx/angular/internal';
|
|
10
10
|
import { deferFn } from '@odx/angular/utils';
|
|
11
|
-
import { tap, filter } from 'rxjs';
|
|
12
11
|
import { OptionControl } from '@odx/angular/cdk/option-control';
|
|
13
12
|
import { InputControlDirective } from '@odx/angular/cdk/custom-form-control';
|
|
14
13
|
|
|
@@ -92,6 +91,7 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
92
91
|
super(...arguments);
|
|
93
92
|
this.defaultActiveOptionIndex = 0;
|
|
94
93
|
this.patchValueFlag = false;
|
|
94
|
+
this.shouldOpenOnOptionUpdate = false;
|
|
95
95
|
this.clickOutsideDirective = inject(ClickOutsideDirective, { host: true });
|
|
96
96
|
/** Text to display when no options are found.
|
|
97
97
|
*
|
|
@@ -133,14 +133,22 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
133
133
|
this.searchField?.reset();
|
|
134
134
|
}
|
|
135
135
|
handleQueryListOption(options) {
|
|
136
|
-
options.changes
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
136
|
+
options.changes.pipe(this.takeUntilDestroyed()).subscribe(() => {
|
|
137
|
+
this.initKeyManager(options.toArray());
|
|
138
|
+
deferFn(() => {
|
|
139
|
+
this.updateDropdownState();
|
|
140
|
+
if (this.isOpen && this.hasOptions) {
|
|
141
|
+
this.activateSelectedOption();
|
|
142
|
+
}
|
|
143
|
+
});
|
|
140
144
|
});
|
|
141
145
|
}
|
|
142
146
|
handleSearchFieldChanges() {
|
|
143
|
-
this.searchField?.valueChange$.pipe(this.takeUntilDestroyed()).subscribe(() =>
|
|
147
|
+
this.searchField?.valueChange$.pipe(this.takeUntilDestroyed()).subscribe(() => {
|
|
148
|
+
this.shouldOpenOnOptionUpdate = true;
|
|
149
|
+
this.triggerControllerChange();
|
|
150
|
+
this.updateDropdownState();
|
|
151
|
+
});
|
|
144
152
|
}
|
|
145
153
|
handleClickOutside() {
|
|
146
154
|
this.clickOutsideDirective.odxClickOutside.pipe(this.takeUntilDestroyed()).subscribe(() => this.clickOutside());
|
|
@@ -154,8 +162,9 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
154
162
|
this.patchValueFlag = false;
|
|
155
163
|
return;
|
|
156
164
|
}
|
|
157
|
-
if (!this.isOpen && (this.hasOptions || !!this.notFoundText())) {
|
|
165
|
+
if (!this.isOpen && this.shouldOpenOnOptionUpdate && (this.hasOptions || !!this.notFoundText() || this.isLoading)) {
|
|
158
166
|
this.openDropdown();
|
|
167
|
+
this.shouldOpenOnOptionUpdate = false;
|
|
159
168
|
return;
|
|
160
169
|
}
|
|
161
170
|
if (this.isOpen && !this.hasOptions && !this.notFoundText()) {
|
|
@@ -170,8 +179,10 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
170
179
|
}
|
|
171
180
|
if (this.isReadonly || this.isDisabled)
|
|
172
181
|
return;
|
|
182
|
+
this.shouldOpenOnOptionUpdate = true;
|
|
173
183
|
if (!this.isOpen && this.hasOptions) {
|
|
174
184
|
this.openDropdown();
|
|
185
|
+
this.shouldOpenOnOptionUpdate = false;
|
|
175
186
|
}
|
|
176
187
|
}
|
|
177
188
|
async handleKeyboardEvent(event) {
|
|
@@ -182,6 +193,10 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
182
193
|
this.blurSelectSearchField();
|
|
183
194
|
return;
|
|
184
195
|
}
|
|
196
|
+
if (!this.isOpen && ['ArrowDown', 'ArrowUp', 'PageDown', 'PageUp', 'Home', 'End'].includes(event.key)) {
|
|
197
|
+
this.shouldOpenOnOptionUpdate = true;
|
|
198
|
+
this.updateDropdownState();
|
|
199
|
+
}
|
|
185
200
|
if (this.isOpen && this.hasOptions) {
|
|
186
201
|
if (event.key === 'Enter' || event.key === 'Tab') {
|
|
187
202
|
event.preventDefault();
|
|
@@ -204,6 +219,7 @@ let AutocompleteComponent = class AutocompleteComponent extends AutocompleteCont
|
|
|
204
219
|
}
|
|
205
220
|
onDropdownClosed() {
|
|
206
221
|
super.onDropdownClosed();
|
|
222
|
+
this.shouldOpenOnOptionUpdate = false;
|
|
207
223
|
!this.searchField?.nativeElementValue && this.blurSelectSearchField();
|
|
208
224
|
}
|
|
209
225
|
updateSearchField(value) {
|