@odx/angular 12.14.0 → 12.14.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.
- package/CHANGELOG.md +12 -0
- package/components/data-table/lib/models/index.d.ts +14 -2
- package/components/datepicker/lib/datepicker.component.d.ts +10 -0
- package/components/daterangepicker/lib/daterangepicker.component.d.ts +11 -0
- package/components/table/lib/components/header-title/header-title.component.d.ts +1 -1
- package/components/table/lib/models/index.d.ts +1 -0
- package/components/table/lib/models/sort-status.d.ts +2 -1
- package/components/table/lib/models/sort-variant.d.ts +21 -0
- package/components/timepicker/lib/components/timepicker-option.component.d.ts +13 -1
- package/components/timepicker/lib/timepicker.component.d.ts +7 -0
- package/components/timepicker/lib/timepicker.service.d.ts +19 -11
- package/esm2022/components/data-table/lib/models/index.mjs +14 -3
- package/esm2022/components/datepicker/lib/datepicker.component.mjs +19 -2
- package/esm2022/components/daterangepicker/lib/daterangepicker.component.mjs +27 -9
- package/esm2022/components/table/lib/components/header-title/header-title.component.mjs +9 -9
- package/esm2022/components/table/lib/models/index.mjs +2 -1
- package/esm2022/components/table/lib/models/sort-status.mjs +1 -1
- package/esm2022/components/table/lib/models/sort-variant.mjs +16 -0
- package/esm2022/components/timepicker/lib/components/timepicker-option.component.mjs +14 -2
- package/esm2022/components/timepicker/lib/timepicker.component.mjs +15 -5
- package/esm2022/components/timepicker/lib/timepicker.service.mjs +35 -13
- package/esm2022/utils/lib/helpers/sorter.mjs +15 -3
- package/fesm2022/odx-angular-components-data-table.mjs +15 -7
- package/fesm2022/odx-angular-components-data-table.mjs.map +1 -1
- package/fesm2022/odx-angular-components-datepicker.mjs +18 -1
- package/fesm2022/odx-angular-components-datepicker.mjs.map +1 -1
- package/fesm2022/odx-angular-components-daterangepicker.mjs +27 -9
- package/fesm2022/odx-angular-components-daterangepicker.mjs.map +1 -1
- package/fesm2022/odx-angular-components-table.mjs +24 -9
- package/fesm2022/odx-angular-components-table.mjs.map +1 -1
- package/fesm2022/odx-angular-components-timepicker.mjs +61 -17
- package/fesm2022/odx-angular-components-timepicker.mjs.map +1 -1
- package/fesm2022/odx-angular-utils.mjs +15 -3
- package/fesm2022/odx-angular-utils.mjs.map +1 -1
- package/package.json +7 -7
- package/utils/lib/helpers/sorter.d.ts +19 -1
- package/components/data-table/lib/models/sort-direction.d.ts +0 -6
- package/components/data-table/lib/models/sort-state.d.ts +0 -5
- package/esm2022/components/data-table/lib/models/sort-direction.mjs +0 -6
- package/esm2022/components/data-table/lib/models/sort-state.mjs +0 -2
|
@@ -3,7 +3,7 @@ import { ChangeDetectionStrategy, ChangeDetectorRef, Component, HostBinding, inj
|
|
|
3
3
|
import { FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';
|
|
4
4
|
import { CheckboxComponent } from '@odx/angular/components/checkbox';
|
|
5
5
|
import { IconComponent } from '@odx/angular/components/icon';
|
|
6
|
-
import { untilDestroyed } from '@odx/angular/utils';
|
|
6
|
+
import { SortDirection, untilDestroyed } from '@odx/angular/utils';
|
|
7
7
|
import { tap } from 'rxjs';
|
|
8
8
|
import { TABLE } from '../../table.config';
|
|
9
9
|
import * as i0 from "@angular/core";
|
|
@@ -35,14 +35,14 @@ export class HeaderTitleComponent {
|
|
|
35
35
|
return { column: this.item?.name, sortVariant: undefined };
|
|
36
36
|
}
|
|
37
37
|
/**
|
|
38
|
-
* Returns the icon name for the current sorting status (ascending, descending
|
|
38
|
+
* Returns the icon name for the current sorting status (ascending, descending).
|
|
39
39
|
*
|
|
40
40
|
* @returns {string}
|
|
41
41
|
*/
|
|
42
42
|
get sortIcon() {
|
|
43
|
-
if (this.sortStatus.sortVariant ===
|
|
43
|
+
if (this.sortStatus.sortVariant === SortDirection.ASC)
|
|
44
44
|
return 'sort-down';
|
|
45
|
-
if (this.sortStatus.sortVariant ===
|
|
45
|
+
if (this.sortStatus.sortVariant === SortDirection.DESC)
|
|
46
46
|
return 'sort-up';
|
|
47
47
|
return 'sortable';
|
|
48
48
|
}
|
|
@@ -52,13 +52,13 @@ export class HeaderTitleComponent {
|
|
|
52
52
|
sortColumn() {
|
|
53
53
|
if (!this.item?.sortable)
|
|
54
54
|
return;
|
|
55
|
-
if (this.sortStatus.sortVariant ===
|
|
55
|
+
if (this.sortStatus.sortVariant === SortDirection.DESC || !this.sortStatus.sortVariant) {
|
|
56
56
|
this.table.clearSort();
|
|
57
|
-
this.sortStatus = { ...this.sortStatus, sortVariant:
|
|
57
|
+
this.sortStatus = { ...this.sortStatus, sortVariant: SortDirection.ASC };
|
|
58
58
|
}
|
|
59
|
-
else if (this.sortStatus.sortVariant ===
|
|
59
|
+
else if (this.sortStatus.sortVariant === SortDirection.ASC) {
|
|
60
60
|
this.table.clearSort();
|
|
61
|
-
this.sortStatus = { ...this.sortStatus, sortVariant:
|
|
61
|
+
this.sortStatus = { ...this.sortStatus, sortVariant: SortDirection.DESC };
|
|
62
62
|
}
|
|
63
63
|
this.table.sorted.emit(this.sortStatus);
|
|
64
64
|
}
|
|
@@ -124,4 +124,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
124
124
|
type: HostBinding,
|
|
125
125
|
args: ['style.width']
|
|
126
126
|
}] } });
|
|
127
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"header-title.component.js","sourceRoot":"","sources":["../../../../../../../../../libs/angular/components/table/src/lib/components/header-title/header-title.component.ts","../../../../../../../../../libs/angular/components/table/src/lib/components/header-title/header-title.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { ChangeDetectionStrategy, ChangeDetectorRef, Component, HostBinding, inject, Input, OnInit, ViewEncapsulation } from '@angular/core';\nimport { FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';\nimport { CheckboxComponent } from '@odx/angular/components/checkbox';\nimport { IconComponent } from '@odx/angular/components/icon';\nimport { untilDestroyed } from '@odx/angular/utils';\nimport { tap } from 'rxjs';\nimport { SortStatus, TableBodyCell, TableHeaderCell } from '../../models';\nimport { TABLE } from '../../table.config';\n\n/**\n * HeaderTitleComponent provides the functionality for a table header cell, including sorting and checkbox\n * operations if applicable. It handles changes in sort state and emits relevant events to the table component\n * to manage sorting and selection state across the table.\n */\n@Component({\n  // eslint-disable-next-line @angular-eslint/component-selector\n  selector: 'th[odxHeaderTitle]',\n  templateUrl: './header-title.component.html',\n  standalone: true,\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [CommonModule, CheckboxComponent, ReactiveFormsModule, IconComponent],\n})\nexport class HeaderTitleComponent implements OnInit {\n  private readonly table = inject(TABLE);\n  private readonly cdr = inject(ChangeDetectorRef);\n  private readonly takeUntilDestroyed = untilDestroyed();\n  private sortStatus: SortStatus = this.getSortStatus;\n\n  /**\n   * The header cell item containing metadata like column name, sortability, and whether it should include a checkbox.\n   *\n   * @type {TableHeaderCell}\n   */\n  @Input()\n  public item!: TableHeaderCell;\n\n  /**\n   * FormGroup to manage the checkbox state within the header.\n   *\n   * @type {FormGroup}\n   */\n  public checkForm = new FormGroup({ check: new FormControl(false) });\n\n  /**\n   * Computes and returns the current sort status of the column associated with this header.\n   *\n   * @returns {SortStatus}\n   */\n  public get getSortStatus(): SortStatus {\n    return { column: this.item?.name, sortVariant: undefined };\n  }\n\n  /**\n   * Returns the icon name for the current sorting status (ascending, descending, unsorted).\n   *\n   * @returns {string}\n   */\n  public get sortIcon(): string {\n    if (this.sortStatus.sortVariant === 'asc') return 'sort-down';\n    if (this.sortStatus.sortVariant === 'desc') return 'sort-up';\n    return 'sortable';\n  }\n\n  /**\n   * Toggles the sorting status of the column and informs the table component of the change.\n   */\n  public sortColumn(): void {\n    if (!this.item?.sortable) return;\n    if (this.sortStatus.sortVariant === 'desc' || !this.sortStatus.sortVariant) {\n      this.table.clearSort();\n      this.sortStatus = { ...this.sortStatus, sortVariant: 'asc' };\n    } else if (this.sortStatus.sortVariant === 'asc') {\n      this.table.clearSort();\n      this.sortStatus = { ...this.sortStatus, sortVariant: 'desc' };\n    }\n\n    this.table.sorted.emit(this.sortStatus);\n  }\n\n  /**\n   * Resets the sort status to unsorted.\n   */\n  public clearSortStatus(): void {\n    this.sortStatus = this.getSortStatus;\n    this.check();\n  }\n\n  /**\n   * Returns the number of selected items in the column, used to determine checkbox state.\n   *\n   * @returns {number}\n   */\n  public get selected(): number {\n    return this.table.data().filter((d: TableBodyCell) => d[this.item.name]).length;\n  }\n\n  /**\n   * Updates the component view, usually after a state change.\n   */\n  public check(): void {\n    this.cdr.markForCheck();\n  }\n\n  /**\n   * Determines whether the checkbox should be marked as checked.\n   *\n   * @returns {boolean}\n   */\n  public get checked(): boolean {\n    const column = this.item.check && this.item.name;\n    return this.table.data().every((item: TableBodyCell) => column && item[column]);\n  }\n\n  public ngOnInit(): void {\n    this.checkBoxListener();\n  }\n\n  private checkBoxListener(): void {\n    if (!this.item.check) return;\n    this.checkForm\n      .get('check')\n      ?.valueChanges.pipe(\n        this.takeUntilDestroyed(),\n        tap((check) => this.table.checked.emit({ column: this.item.name, check: !!check })),\n      )\n      .subscribe();\n  }\n\n  public get indeterminate(): boolean {\n    const column = this.item.check && this.item.name;\n    const { length } = this.table.data();\n    const unchecked = length - this.table.data().filter((item: TableBodyCell) => column && item[column]).length;\n    return unchecked > 0 && unchecked < length;\n  }\n\n  @HostBinding('style.width')\n  protected get columnWidth(): string {\n    return this.item.width ?? 'auto';\n  }\n}\n","@if (item.check) {\n  <div class=\"odx-table__check\">\n    <div [formGroup]=\"checkForm\">\n      <odx-checkbox [checked]=\"checked\" [indeterminate]=\"indeterminate\" formControlName=\"check\" />\n    </div>\n  </div>\n} @else {\n  <div class=\"odx-table__header-cell-title\" (click)=\"sortColumn()\" [class.sortable]=\"item.sortable\">\n    {{ item.title }}\n    @if (item.sortable) {\n      <odx-icon iconSet=\"core\" [name]=\"sortIcon\" size=\"small\" />\n    }\n  </div>\n  <ng-content select=\"odx-form-field\" />\n}\n"]}
|
|
127
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"header-title.component.js","sourceRoot":"","sources":["../../../../../../../../../libs/angular/components/table/src/lib/components/header-title/header-title.component.ts","../../../../../../../../../libs/angular/components/table/src/lib/components/header-title/header-title.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { ChangeDetectionStrategy, ChangeDetectorRef, Component, HostBinding, inject, Input, OnInit, ViewEncapsulation } from '@angular/core';\nimport { FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';\nimport { CheckboxComponent } from '@odx/angular/components/checkbox';\nimport { IconComponent } from '@odx/angular/components/icon';\nimport { SortDirection, untilDestroyed } from '@odx/angular/utils';\nimport { tap } from 'rxjs';\nimport { SortStatus, TableBodyCell, TableHeaderCell } from '../../models';\nimport { TABLE } from '../../table.config';\n\n/**\n * HeaderTitleComponent provides the functionality for a table header cell, including sorting and checkbox\n * operations if applicable. It handles changes in sort state and emits relevant events to the table component\n * to manage sorting and selection state across the table.\n */\n@Component({\n  // eslint-disable-next-line @angular-eslint/component-selector\n  selector: 'th[odxHeaderTitle]',\n  templateUrl: './header-title.component.html',\n  standalone: true,\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [CommonModule, CheckboxComponent, ReactiveFormsModule, IconComponent],\n})\nexport class HeaderTitleComponent implements OnInit {\n  private readonly table = inject(TABLE);\n  private readonly cdr = inject(ChangeDetectorRef);\n  private readonly takeUntilDestroyed = untilDestroyed();\n  private sortStatus: SortStatus = this.getSortStatus;\n\n  /**\n   * The header cell item containing metadata like column name, sortability, and whether it should include a checkbox.\n   *\n   * @type {TableHeaderCell}\n   */\n  @Input()\n  public item!: TableHeaderCell;\n\n  /**\n   * FormGroup to manage the checkbox state within the header.\n   *\n   * @type {FormGroup}\n   */\n  public checkForm = new FormGroup({ check: new FormControl(false) });\n\n  /**\n   * Computes and returns the current sort status of the column associated with this header.\n   *\n   * @returns {SortStatus}\n   */\n  public get getSortStatus(): SortStatus {\n    return { column: this.item?.name, sortVariant: undefined };\n  }\n\n  /**\n   * Returns the icon name for the current sorting status (ascending, descending).\n   *\n   * @returns {string}\n   */\n  public get sortIcon(): string {\n    if (this.sortStatus.sortVariant === SortDirection.ASC) return 'sort-down';\n    if (this.sortStatus.sortVariant === SortDirection.DESC) return 'sort-up';\n    return 'sortable';\n  }\n\n  /**\n   * Toggles the sorting status of the column and informs the table component of the change.\n   */\n  public sortColumn(): void {\n    if (!this.item?.sortable) return;\n    if (this.sortStatus.sortVariant === SortDirection.DESC || !this.sortStatus.sortVariant) {\n      this.table.clearSort();\n      this.sortStatus = { ...this.sortStatus, sortVariant: SortDirection.ASC };\n    } else if (this.sortStatus.sortVariant === SortDirection.ASC) {\n      this.table.clearSort();\n      this.sortStatus = { ...this.sortStatus, sortVariant: SortDirection.DESC };\n    }\n\n    this.table.sorted.emit(this.sortStatus);\n  }\n\n  /**\n   * Resets the sort status to unsorted.\n   */\n  public clearSortStatus(): void {\n    this.sortStatus = this.getSortStatus;\n    this.check();\n  }\n\n  /**\n   * Returns the number of selected items in the column, used to determine checkbox state.\n   *\n   * @returns {number}\n   */\n  public get selected(): number {\n    return this.table.data().filter((d: TableBodyCell) => d[this.item.name]).length;\n  }\n\n  /**\n   * Updates the component view, usually after a state change.\n   */\n  public check(): void {\n    this.cdr.markForCheck();\n  }\n\n  /**\n   * Determines whether the checkbox should be marked as checked.\n   *\n   * @returns {boolean}\n   */\n  public get checked(): boolean {\n    const column = this.item.check && this.item.name;\n    return this.table.data().every((item: TableBodyCell) => column && item[column]);\n  }\n\n  public ngOnInit(): void {\n    this.checkBoxListener();\n  }\n\n  private checkBoxListener(): void {\n    if (!this.item.check) return;\n    this.checkForm\n      .get('check')\n      ?.valueChanges.pipe(\n        this.takeUntilDestroyed(),\n        tap((check) => this.table.checked.emit({ column: this.item.name, check: !!check })),\n      )\n      .subscribe();\n  }\n\n  public get indeterminate(): boolean {\n    const column = this.item.check && this.item.name;\n    const { length } = this.table.data();\n    const unchecked = length - this.table.data().filter((item: TableBodyCell) => column && item[column]).length;\n    return unchecked > 0 && unchecked < length;\n  }\n\n  @HostBinding('style.width')\n  protected get columnWidth(): string {\n    return this.item.width ?? 'auto';\n  }\n}\n","@if (item.check) {\n  <div class=\"odx-table__check\">\n    <div [formGroup]=\"checkForm\">\n      <odx-checkbox [checked]=\"checked\" [indeterminate]=\"indeterminate\" formControlName=\"check\" />\n    </div>\n  </div>\n} @else {\n  <div class=\"odx-table__header-cell-title\" (click)=\"sortColumn()\" [class.sortable]=\"item.sortable\">\n    {{ item.title }}\n    @if (item.sortable) {\n      <odx-icon iconSet=\"core\" [name]=\"sortIcon\" size=\"small\" />\n    }\n  </div>\n  <ng-content select=\"odx-form-field\" />\n}\n"]}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
export * from './header-cell';
|
|
2
2
|
export * from './sort-status';
|
|
3
|
+
export * from './sort-variant';
|
|
3
4
|
export * from './table-body-cell';
|
|
4
5
|
export * from './table-variant';
|
|
5
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXIvY29tcG9uZW50cy90YWJsZS9zcmMvbGliL21vZGVscy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGVBQWUsQ0FBQztBQUM5QixjQUFjLGVBQWUsQ0FBQztBQUM5QixjQUFjLGdCQUFnQixDQUFDO0FBQy9CLGNBQWMsbUJBQW1CLENBQUM7QUFDbEMsY0FBYyxpQkFBaUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vaGVhZGVyLWNlbGwnO1xuZXhwb3J0ICogZnJvbSAnLi9zb3J0LXN0YXR1cyc7XG5leHBvcnQgKiBmcm9tICcuL3NvcnQtdmFyaWFudCc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLWJvZHktY2VsbCc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLXZhcmlhbnQnO1xuIl19
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic29ydC1zdGF0dXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXIvY29tcG9uZW50cy90YWJsZS9zcmMvbGliL21vZGVscy9zb3J0LXN0YXR1cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgU29ydERpcmVjdGlvbiB9IGZyb20gJ0BvZHgvYW5ndWxhci91dGlscyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU29ydFN0YXR1cyB7XG4gIGNvbHVtbjogc3RyaW5nO1xuICBzb3J0VmFyaWFudD86IFNvcnREaXJlY3Rpb247XG59XG4iXX0=
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { SortDirection } from '@odx/angular/utils';
|
|
2
|
+
/**
|
|
3
|
+
* @deprecated Use `SortDirection` from `@odx/angular/utils` instead.
|
|
4
|
+
* Consider using `SortDirection | undefined` or `SortDirection | null` to represent an unsorted state.
|
|
5
|
+
* Constant object representing the available sort states for a table column.
|
|
6
|
+
* @property {string} UNSORTED - The column is not sorted.
|
|
7
|
+
* @property {string} ASC - The column is sorted in ascending order (equivalent to `SortDirection.ASC`).
|
|
8
|
+
* @property {string} DESC - The column is sorted in descending order (equivalent to `SortDirection.DESC`).
|
|
9
|
+
* @readonly
|
|
10
|
+
*/
|
|
11
|
+
export const TableSortVariant = {
|
|
12
|
+
UNSORTED: 'unsorted',
|
|
13
|
+
// ASC and DESC values are now inherited from SortDirection
|
|
14
|
+
...SortDirection,
|
|
15
|
+
};
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic29ydC12YXJpYW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9hbmd1bGFyL2NvbXBvbmVudHMvdGFibGUvc3JjL2xpYi9tb2RlbHMvc29ydC12YXJpYW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQVNuRDs7Ozs7Ozs7R0FRRztBQUNILE1BQU0sQ0FBQyxNQUFNLGdCQUFnQixHQUFHO0lBQzlCLFFBQVEsRUFBRSxVQUFVO0lBQ3BCLDJEQUEyRDtJQUMzRCxHQUFHLGFBQWE7Q0FDUixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgU29ydERpcmVjdGlvbiB9IGZyb20gJ0BvZHgvYW5ndWxhci91dGlscyc7XG5cbi8qKlxuICogQGRlcHJlY2F0ZWQgVXNlIGBTb3J0RGlyZWN0aW9uYCBmcm9tIGBAb2R4L2FuZ3VsYXIvdXRpbHNgIGluc3RlYWQuXG4gKiBDb25zaWRlciB1c2luZyBgU29ydERpcmVjdGlvbiB8IHVuZGVmaW5lZGAgb3IgYFNvcnREaXJlY3Rpb24gfCBudWxsYCB0byByZXByZXNlbnQgYW4gdW5zb3J0ZWQgc3RhdGUuXG4gKiBEZWZpbmVzIHRoZSBwb3NzaWJsZSBzb3J0IHN0YXRlcyBmb3IgYSB0YWJsZSBjb2x1bW4uXG4gKi9cbmV4cG9ydCB0eXBlIFRhYmxlU29ydFZhcmlhbnQgPSAndW5zb3J0ZWQnIHwgU29ydERpcmVjdGlvbjtcblxuLyoqXG4gKiBAZGVwcmVjYXRlZCBVc2UgYFNvcnREaXJlY3Rpb25gIGZyb20gYEBvZHgvYW5ndWxhci91dGlsc2AgaW5zdGVhZC5cbiAqIENvbnNpZGVyIHVzaW5nIGBTb3J0RGlyZWN0aW9uIHwgdW5kZWZpbmVkYCBvciBgU29ydERpcmVjdGlvbiB8IG51bGxgIHRvIHJlcHJlc2VudCBhbiB1bnNvcnRlZCBzdGF0ZS5cbiAqIENvbnN0YW50IG9iamVjdCByZXByZXNlbnRpbmcgdGhlIGF2YWlsYWJsZSBzb3J0IHN0YXRlcyBmb3IgYSB0YWJsZSBjb2x1bW4uXG4gKiBAcHJvcGVydHkge3N0cmluZ30gVU5TT1JURUQgLSBUaGUgY29sdW1uIGlzIG5vdCBzb3J0ZWQuXG4gKiBAcHJvcGVydHkge3N0cmluZ30gQVNDIC0gVGhlIGNvbHVtbiBpcyBzb3J0ZWQgaW4gYXNjZW5kaW5nIG9yZGVyIChlcXVpdmFsZW50IHRvIGBTb3J0RGlyZWN0aW9uLkFTQ2ApLlxuICogQHByb3BlcnR5IHtzdHJpbmd9IERFU0MgLSBUaGUgY29sdW1uIGlzIHNvcnRlZCBpbiBkZXNjZW5kaW5nIG9yZGVyIChlcXVpdmFsZW50IHRvIGBTb3J0RGlyZWN0aW9uLkRFU0NgKS5cbiAqIEByZWFkb25seVxuICovXG5leHBvcnQgY29uc3QgVGFibGVTb3J0VmFyaWFudCA9IHtcbiAgVU5TT1JURUQ6ICd1bnNvcnRlZCcsXG4gIC8vIEFTQyBhbmQgREVTQyB2YWx1ZXMgYXJlIG5vdyBpbmhlcml0ZWQgZnJvbSBTb3J0RGlyZWN0aW9uXG4gIC4uLlNvcnREaXJlY3Rpb24sXG59IGFzIGNvbnN0O1xuIl19
|
|
@@ -26,7 +26,18 @@ let TimepickerOptionComponent = class TimepickerOptionComponent extends OptionCo
|
|
|
26
26
|
super();
|
|
27
27
|
this.timepicker = inject(TIMEPICKER_CONTROL);
|
|
28
28
|
this.disabledController = DisabledController.inject();
|
|
29
|
+
/**
|
|
30
|
+
* Indicates whether this time option is currently selected.
|
|
31
|
+
* @type {boolean}
|
|
32
|
+
* @default false
|
|
33
|
+
*/
|
|
29
34
|
this.isSelected = false;
|
|
35
|
+
/**
|
|
36
|
+
* Indicates whether this time option is currently active (e.g., highlighted by keyboard navigation).
|
|
37
|
+
* This is an override from the base `OptionControl`.
|
|
38
|
+
* @type {boolean}
|
|
39
|
+
* @default false
|
|
40
|
+
*/
|
|
30
41
|
this.isActive = false;
|
|
31
42
|
/**
|
|
32
43
|
* Emits an event when the option is selected, allowing the timepicker component to update its value
|
|
@@ -41,7 +52,8 @@ let TimepickerOptionComponent = class TimepickerOptionComponent extends OptionCo
|
|
|
41
52
|
this.isSelected = this.timepicker.isTimeOptionSelected(this);
|
|
42
53
|
}
|
|
43
54
|
/**
|
|
44
|
-
* Sets the active styles for the option, indicating that it is
|
|
55
|
+
* Sets the active styles for the option, typically indicating that it is
|
|
56
|
+
* focused or highlighted via keyboard navigation.
|
|
45
57
|
*/
|
|
46
58
|
setActiveStyles() {
|
|
47
59
|
this.isActive = true;
|
|
@@ -71,4 +83,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
71
83
|
}], ctorParameters: () => [], propDecorators: { selected: [{
|
|
72
84
|
type: Output
|
|
73
85
|
}] } });
|
|
74
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
86
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGltZXBpY2tlci1vcHRpb24uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9hbmd1bGFyL2NvbXBvbmVudHMvdGltZXBpY2tlci9zcmMvbGliL2NvbXBvbmVudHMvdGltZXBpY2tlci1vcHRpb24uY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBVSxNQUFNLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzVILE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxpQkFBaUIsRUFBRSx1QkFBdUIsRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUM5RixPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFDaEUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQ3JELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLHFCQUFxQixDQUFDOzs7QUFFekQ7Ozs7OztHQU1HO0FBV0ksSUFBTSx5QkFBeUIsR0FBL0IsTUFBTSx5QkFBMEIsU0FBUSxhQUFxQjtJQWtCbEU7Ozs7T0FJRztJQUNILElBQVcsUUFBUTtRQUNqQixPQUFPLENBQUMsQ0FBQyxJQUFJLENBQUMsa0JBQWtCLEVBQUUsUUFBUSxDQUFDO0lBQzdDLENBQUM7SUFXRDtRQUNFLEtBQUssRUFBRSxDQUFDO1FBcENPLGVBQVUsR0FBRyxNQUFNLENBQUMsa0JBQWtCLENBQUMsQ0FBQztRQUN0Qyx1QkFBa0IsR0FBRyxrQkFBa0IsQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUVwRTs7OztXQUlHO1FBQ0ksZUFBVSxHQUFHLEtBQUssQ0FBQztRQUMxQjs7Ozs7V0FLRztRQUNhLGFBQVEsR0FBRyxLQUFLLENBQUM7UUFXakM7Ozs7O1dBS0c7UUFFSSxhQUFRLEdBQUcsSUFBSSxZQUFZLEVBQTZCLENBQUM7UUFJOUQsdUJBQXVCLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsa0JBQWtCLEVBQUUsQ0FBQyxDQUFDLFNBQVMsRUFBRSxDQUFDO0lBQ3ZGLENBQUM7SUFFTSxRQUFRO1FBQ2IsSUFBSSxDQUFDLFVBQVUsR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLG9CQUFvQixDQUFDLElBQUksQ0FBQyxDQUFDO0lBQy9ELENBQUM7SUFFRDs7O09BR0c7SUFDSSxlQUFlO1FBQ3BCLElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxDQUFDO0lBQ3ZCLENBQUM7SUFFUyxZQUFZO1FBQ3BCLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzNCLENBQUM7K0dBdkRVLHlCQUF5QjttR0FBekIseUJBQXlCLHVHQUh6QixDQUFDLGtCQUFrQixDQUFDLE9BQU8sRUFBRSxDQUFDLHdHQUovQixnQkFBZ0I7O0FBT2YseUJBQXlCO0lBVnJDLFlBQVksQ0FBQyxtQkFBbUIsQ0FBQzs7R0FVckIseUJBQXlCLENBd0RyQzs7NEZBeERZLHlCQUF5QjtrQkFUckMsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsdUJBQXVCO29CQUNqQyxRQUFRLEVBQUUsZ0JBQWdCO29CQUMxQixVQUFVLEVBQUUsSUFBSTtvQkFDaEIsYUFBYSxFQUFFLGlCQUFpQixDQUFDLElBQUk7b0JBQ3JDLGVBQWUsRUFBRSx1QkFBdUIsQ0FBQyxNQUFNO29CQUMvQyxTQUFTLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQyxPQUFPLEVBQUUsQ0FBQztvQkFDekMsY0FBYyxFQUFFLENBQUMsaUJBQWlCLENBQUM7aUJBQ3BDO3dEQW1DUSxRQUFRO3NCQURkLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSwgQ29tcG9uZW50LCBFdmVudEVtaXR0ZXIsIE9uSW5pdCwgT3V0cHV0LCBWaWV3RW5jYXBzdWxhdGlvbiwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBEaXNhYmxlZENvbnRyb2xsZXIsIFdpdGhEaXNhYmxlZFN0YXRlLCBkZXRlY3RDb250cm9sbGVyQ2hhbmdlcyB9IGZyb20gJ0BvZHgvYW5ndWxhcic7XG5pbXBvcnQgeyBPcHRpb25Db250cm9sIH0gZnJvbSAnQG9keC9hbmd1bGFyL2Nkay9vcHRpb24tY29udHJvbCc7XG5pbXBvcnQgeyBDU1NDb21wb25lbnQgfSBmcm9tICdAb2R4L2FuZ3VsYXIvaW50ZXJuYWwnO1xuaW1wb3J0IHsgVElNRVBJQ0tFUl9DT05UUk9MIH0gZnJvbSAnLi4vdGltZXBpY2tlci50b2tlbic7XG5cbi8qKlxuICogUmVwcmVzZW50cyBhIHNlbGVjdGFibGUgb3B0aW9uIHdpdGhpbiBhIHRpbWVwaWNrZXIgY29tcG9uZW50LiBUaGlzIGNvbXBvbmVudCBhbGxvd3MgZm9yXG4gKiBpbmRpdmlkdWFsIHRpbWUgdmFsdWVzIHRvIGJlIHNlbGVjdGVkIGZyb20gYSBkcm9wZG93biBtZW51IGFzIHBhcnQgb2YgdGhlIHRpbWVwaWNrZXIgaW50ZXJmYWNlLlxuICogSXQgaW50ZWdyYXRlcyB3aXRoIHRoZSB0aW1lcGlja2VyJ3MgY29udHJvbCB0byBtYW5hZ2Ugc2VsZWN0aW9uIHN0YXRlIGFuZCBhY2Nlc3NpYmlsaXR5LlxuICpcbiAqIEBleHRlbmRzIHtPcHRpb25Db250cm9sPHN0cmluZz59XG4gKi9cbkBDU1NDb21wb25lbnQoJ3RpbWVwaWNrZXItb3B0aW9uJylcbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ29keC10aW1lcGlja2VyLW9wdGlvbicsXG4gIHRlbXBsYXRlOiBgPG5nLWNvbnRlbnQgLz5gLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBlbmNhcHN1bGF0aW9uOiBWaWV3RW5jYXBzdWxhdGlvbi5Ob25lLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgcHJvdmlkZXJzOiBbRGlzYWJsZWRDb250cm9sbGVyLmNvbm5lY3QoKV0sXG4gIGhvc3REaXJlY3RpdmVzOiBbV2l0aERpc2FibGVkU3RhdGVdLFxufSlcbmV4cG9ydCBjbGFzcyBUaW1lcGlja2VyT3B0aW9uQ29tcG9uZW50IGV4dGVuZHMgT3B0aW9uQ29udHJvbDxzdHJpbmc+IGltcGxlbWVudHMgT25Jbml0IHtcbiAgcHJpdmF0ZSByZWFkb25seSB0aW1lcGlja2VyID0gaW5qZWN0KFRJTUVQSUNLRVJfQ09OVFJPTCk7XG4gIHByb3RlY3RlZCByZWFkb25seSBkaXNhYmxlZENvbnRyb2xsZXIgPSBEaXNhYmxlZENvbnRyb2xsZXIuaW5qZWN0KCk7XG5cbiAgLyoqXG4gICAqIEluZGljYXRlcyB3aGV0aGVyIHRoaXMgdGltZSBvcHRpb24gaXMgY3VycmVudGx5IHNlbGVjdGVkLlxuICAgKiBAdHlwZSB7Ym9vbGVhbn1cbiAgICogQGRlZmF1bHQgZmFsc2VcbiAgICovXG4gIHB1YmxpYyBpc1NlbGVjdGVkID0gZmFsc2U7XG4gIC8qKlxuICAgKiBJbmRpY2F0ZXMgd2hldGhlciB0aGlzIHRpbWUgb3B0aW9uIGlzIGN1cnJlbnRseSBhY3RpdmUgKGUuZy4sIGhpZ2hsaWdodGVkIGJ5IGtleWJvYXJkIG5hdmlnYXRpb24pLlxuICAgKiBUaGlzIGlzIGFuIG92ZXJyaWRlIGZyb20gdGhlIGJhc2UgYE9wdGlvbkNvbnRyb2xgLlxuICAgKiBAdHlwZSB7Ym9vbGVhbn1cbiAgICogQGRlZmF1bHQgZmFsc2VcbiAgICovXG4gIHB1YmxpYyBvdmVycmlkZSBpc0FjdGl2ZSA9IGZhbHNlO1xuXG4gIC8qKlxuICAgKiBXaGV0aGVyIHRoZSBvcHRpb24gaXMgZGlzYWJsZWQuIFRoaXMgaXMgdXNlZCB0byBkZXRlcm1pbmUgd2hldGhlciB0aGUgb3B0aW9uIHNob3VsZCBiZSBzZWxlY3RhYmxlLlxuICAgKlxuICAgKiBAdHlwZSB7Ym9vbGVhbn1cbiAgICovXG4gIHB1YmxpYyBnZXQgZGlzYWJsZWQoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuICEhdGhpcy5kaXNhYmxlZENvbnRyb2xsZXI/LmRpc2FibGVkO1xuICB9XG5cbiAgLyoqXG4gICAqIEVtaXRzIGFuIGV2ZW50IHdoZW4gdGhlIG9wdGlvbiBpcyBzZWxlY3RlZCwgYWxsb3dpbmcgdGhlIHRpbWVwaWNrZXIgY29tcG9uZW50IHRvIHVwZGF0ZSBpdHMgdmFsdWVcbiAgICogYW5kIGNsb3NlIHRoZSBkcm9wZG93bi5cbiAgICpcbiAgICogQGVtaXRzIHtUaW1lcGlja2VyT3B0aW9uQ29tcG9uZW50fVxuICAgKi9cbiAgQE91dHB1dCgpXG4gIHB1YmxpYyBzZWxlY3RlZCA9IG5ldyBFdmVudEVtaXR0ZXI8VGltZXBpY2tlck9wdGlvbkNvbXBvbmVudD4oKTtcblxuICBjb25zdHJ1Y3RvcigpIHtcbiAgICBzdXBlcigpO1xuICAgIGRldGVjdENvbnRyb2xsZXJDaGFuZ2VzKHRoaXMudGltZXBpY2tlcikucGlwZSh0aGlzLnRha2VVbnRpbERlc3Ryb3llZCgpKS5zdWJzY3JpYmUoKTtcbiAgfVxuXG4gIHB1YmxpYyBuZ09uSW5pdCgpOiB2b2lkIHtcbiAgICB0aGlzLmlzU2VsZWN0ZWQgPSB0aGlzLnRpbWVwaWNrZXIuaXNUaW1lT3B0aW9uU2VsZWN0ZWQodGhpcyk7XG4gIH1cblxuICAvKipcbiAgICogU2V0cyB0aGUgYWN0aXZlIHN0eWxlcyBmb3IgdGhlIG9wdGlvbiwgdHlwaWNhbGx5IGluZGljYXRpbmcgdGhhdCBpdCBpc1xuICAgKiBmb2N1c2VkIG9yIGhpZ2hsaWdodGVkIHZpYSBrZXlib2FyZCBuYXZpZ2F0aW9uLlxuICAgKi9cbiAgcHVibGljIHNldEFjdGl2ZVN0eWxlcygpOiB2b2lkIHtcbiAgICB0aGlzLmlzQWN0aXZlID0gdHJ1ZTtcbiAgfVxuXG4gIHByb3RlY3RlZCBzZWxlY3RPcHRpb24oKTogdm9pZCB7XG4gICAgdGhpcy5zZWxlY3RlZC5lbWl0KHRoaXMpO1xuICB9XG59XG4iXX0=
|
|
@@ -120,6 +120,17 @@ let TimepickerComponent = class TimepickerComponent extends CustomFormControl {
|
|
|
120
120
|
get timeStampsList() {
|
|
121
121
|
return generateTimeStamps(this.step, this.useLocale);
|
|
122
122
|
}
|
|
123
|
+
/**
|
|
124
|
+
* @internal
|
|
125
|
+
* Writes a new value to the element.
|
|
126
|
+
* Part of the ControlValueAccessor interface.
|
|
127
|
+
* @param {string | null} value - The new value.
|
|
128
|
+
*/
|
|
129
|
+
writeValue(value) {
|
|
130
|
+
super.writeValue(value);
|
|
131
|
+
if (value === null || this.timepickerService.isValidTime(value))
|
|
132
|
+
this.updateInputValue();
|
|
133
|
+
}
|
|
123
134
|
onOpen() {
|
|
124
135
|
this.keyManager = new ActiveDescendantKeyManager(this.options).withHomeAndEnd();
|
|
125
136
|
this.setActiveOptionBasedOnCurrentValue();
|
|
@@ -169,10 +180,9 @@ let TimepickerComponent = class TimepickerComponent extends CustomFormControl {
|
|
|
169
180
|
return availableTimeSlots.find((option) => option.value === nearestTimeValue);
|
|
170
181
|
}
|
|
171
182
|
updateInputValue() {
|
|
172
|
-
if (this.
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
}
|
|
183
|
+
if (!this.dateField)
|
|
184
|
+
return;
|
|
185
|
+
this.dateField.nativeElementValue = this.timepickerService.getLocalizedTimeFormat(this.value, this.useLocale);
|
|
176
186
|
}
|
|
177
187
|
scrollToActiveOption(option, _opts = {}) {
|
|
178
188
|
if (isFunction(option.element.nativeElement.scrollIntoView)) {
|
|
@@ -239,4 +249,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
239
249
|
type: HostListener,
|
|
240
250
|
args: ['keydown', ['$event']]
|
|
241
251
|
}] } });
|
|
242
|
-
//# 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  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\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.value && this.dateField) {\n      const time = this.timepickerService.getLocalizedTimeFormat(this.value, this.useLocale);\n      this.dateField.nativeElementValue = time;\n    }\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"]}
|
|
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"]}
|
|
@@ -13,7 +13,7 @@ export class TimepickerService {
|
|
|
13
13
|
/**
|
|
14
14
|
* Generates a placeholder string for time input fields.
|
|
15
15
|
*
|
|
16
|
-
* @param {boolean} apm - Specifies if the placeholder should include AM/PM notation.
|
|
16
|
+
* @param {boolean} [apm=false] - Specifies if the placeholder should include AM/PM notation.
|
|
17
17
|
* @returns {string} The placeholder string.
|
|
18
18
|
*/
|
|
19
19
|
getPlaceholder(apm = false) {
|
|
@@ -24,18 +24,32 @@ export class TimepickerService {
|
|
|
24
24
|
*
|
|
25
25
|
* @param {string} time - The time to validate.
|
|
26
26
|
* @param {string} max - The maximum allowable time.
|
|
27
|
-
* @returns {boolean} True if the time is valid, false otherwise.
|
|
27
|
+
* @returns {boolean} True if the time is valid (i.e., not after max), false otherwise.
|
|
28
28
|
*/
|
|
29
29
|
maxValidation(time, max) {
|
|
30
30
|
const [targetValue, maxValue] = this.convertToDates([time, max]);
|
|
31
31
|
return isBefore(targetValue, maxValue) || isEqual(targetValue, maxValue);
|
|
32
32
|
}
|
|
33
|
+
/**
|
|
34
|
+
* Checks if the provided value is a valid time string in HH:mm or HH:mm AM/PM format.
|
|
35
|
+
* This method also acts as a type guard.
|
|
36
|
+
*
|
|
37
|
+
* @param {unknown} time - The value to validate.
|
|
38
|
+
* @returns {time is string} True if the value is a valid time string, false otherwise.
|
|
39
|
+
*/
|
|
40
|
+
isValidTime(time) {
|
|
41
|
+
if (!time || typeof time !== 'string')
|
|
42
|
+
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());
|
|
46
|
+
}
|
|
33
47
|
/**
|
|
34
48
|
* Validates if the given time is greater than or equal to a minimum time constraint.
|
|
35
49
|
*
|
|
36
50
|
* @param {string} time - The time to validate.
|
|
37
51
|
* @param {string} min - The minimum allowable time.
|
|
38
|
-
* @returns {boolean} True if the time is valid, false otherwise.
|
|
52
|
+
* @returns {boolean} True if the time is valid (i.e., not before min), false otherwise.
|
|
39
53
|
*/
|
|
40
54
|
minValidation(time, min) {
|
|
41
55
|
const [targetValue, minValue] = this.convertToDates([time, min]);
|
|
@@ -44,26 +58,34 @@ export class TimepickerService {
|
|
|
44
58
|
/**
|
|
45
59
|
* Finds the closest time to a target time from a list of times.
|
|
46
60
|
*
|
|
47
|
-
* @param {string[]} timeStamps - The list of
|
|
48
|
-
* @param {string} targetTime - The target time to find the closest match for.
|
|
49
|
-
* @returns {string} The closest time to the target.
|
|
61
|
+
* @param {string[]} timeStamps - The list of time strings to search through (e.g., ["10:00", "10:30", "11:00"]).
|
|
62
|
+
* @param {string} targetTime - The target time string to find the closest match for (e.g., "10:35").
|
|
63
|
+
* @returns {string} The closest time string from the list to the target, formatted according to locale.
|
|
50
64
|
*/
|
|
51
65
|
findClosestDate(timeStamps, targetTime) {
|
|
52
66
|
const [target] = this.convertToDates([targetTime]);
|
|
53
67
|
const datesArray = this.convertToDates(timeStamps);
|
|
68
|
+
if (datesArray.length === 0)
|
|
69
|
+
return this.getLocalizedTimeFormat(targetTime);
|
|
54
70
|
const closestDate = datesArray.reduce((prev, curr) => {
|
|
55
|
-
|
|
71
|
+
const prevDiff = Math.abs(prev.getTime() - target.getTime());
|
|
72
|
+
const currDiff = Math.abs(curr.getTime() - target.getTime());
|
|
73
|
+
return currDiff < prevDiff ? curr : prev;
|
|
56
74
|
});
|
|
57
|
-
|
|
75
|
+
const hours = closestDate.getHours().toString().padStart(2, '0');
|
|
76
|
+
const minutes = closestDate.getMinutes().toString().padStart(2, '0');
|
|
77
|
+
return this.getLocalizedTimeFormat(`${hours}:${minutes}`);
|
|
58
78
|
}
|
|
59
79
|
/**
|
|
60
|
-
* Formats a time string to a localized time format.
|
|
80
|
+
* Formats a time string to a localized time format using Intl.DateTimeFormat.
|
|
61
81
|
*
|
|
62
|
-
* @param {
|
|
63
|
-
* @param {boolean} hour12 - Specifies if the output should use 12-hour format with AM/PM notation.
|
|
64
|
-
* @returns {string} The formatted time string.
|
|
82
|
+
* @param {unknown} time - The time string to format (e.g., "14:30").
|
|
83
|
+
* @param {boolean} [hour12=false] - Specifies if the output should use 12-hour format with AM/PM notation.
|
|
84
|
+
* @returns {string} The formatted time string, or an empty string if the input time is invalid.
|
|
65
85
|
*/
|
|
66
86
|
getLocalizedTimeFormat(time, hour12 = false) {
|
|
87
|
+
if (!this.isValidTime(time))
|
|
88
|
+
return '';
|
|
67
89
|
const locale = hour12 ? 'en-US' : 'en-GB';
|
|
68
90
|
const [date] = this.convertToDates([time]);
|
|
69
91
|
return new this.windowRef.nativeWindow.Intl.DateTimeFormat(locale, {
|
|
@@ -82,4 +104,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
82
104
|
type: Injectable,
|
|
83
105
|
args: [{ providedIn: 'root' }]
|
|
84
106
|
}] });
|
|
85
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
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"]}
|
|
@@ -1,3 +1,13 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Constant object representing the available sort directions.
|
|
3
|
+
* @property {string} ASC - Sort in ascending order.
|
|
4
|
+
* @property {string} DESC - Sort in descending order.
|
|
5
|
+
* @readonly
|
|
6
|
+
*/
|
|
7
|
+
export const SortDirection = {
|
|
8
|
+
ASC: 'asc',
|
|
9
|
+
DESC: 'desc',
|
|
10
|
+
};
|
|
1
11
|
/**
|
|
2
12
|
* Handles comparison for null values, assuming undefined has already been handled.
|
|
3
13
|
* - `null` is considered less than any defined (non-null, non-undefined) value.
|
|
@@ -44,7 +54,9 @@ function compareNonNullValues(valA, valB, collator) {
|
|
|
44
54
|
* @returns A comparison function `(a: T, b: T) => number`.
|
|
45
55
|
*/
|
|
46
56
|
export function createSorter(options) {
|
|
47
|
-
const { keyFn: userProvidedKeyFn, sortDirection
|
|
57
|
+
const { keyFn: userProvidedKeyFn, sortDirection, locale, collatorOptions } = options || {};
|
|
58
|
+
if (!sortDirection || !Object.values(SortDirection).includes(sortDirection))
|
|
59
|
+
return () => 0;
|
|
48
60
|
if (userProvidedKeyFn !== undefined && typeof userProvidedKeyFn !== 'function') {
|
|
49
61
|
console.warn('createSorter: provided keyFn is not a function. Returning a no-op sorter.');
|
|
50
62
|
return () => 0;
|
|
@@ -73,7 +85,7 @@ export function createSorter(options) {
|
|
|
73
85
|
valB, // valB is known to be not null/undefined here
|
|
74
86
|
collator);
|
|
75
87
|
}
|
|
76
|
-
return sortDirection ===
|
|
88
|
+
return sortDirection === SortDirection.DESC ? comparison * -1 : comparison;
|
|
77
89
|
};
|
|
78
90
|
}
|
|
79
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sorter.js","sourceRoot":"","sources":["../../../../../../../libs/angular/utils/src/lib/helpers/sorter.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Options for creating a sorter function. Can be used for arrays of objects (with `keyFn`)\n * or arrays of primitives (if `keyFn` is omitted).\n * @template T The type of items to be sorted.\n */\nexport interface SorterOptions<T> {\n  sortDirection?: 'asc' | 'desc';\n  locale?: string;\n  collatorOptions?: Intl.CollatorOptions;\n  /**\n   * Optional function to extract the sortable value from an item.\n   * If not provided, the item itself is used for comparison (suitable for arrays of primitives).\n   * The extracted value should be a string, number, boolean, null, or undefined.\n   * @default (item: T) => item\n   */\n  keyFn?: (item: T) => string | number | boolean | null | undefined;\n}\n\n/**\n * Handles comparison for null values, assuming undefined has already been handled.\n * - `null` is considered less than any defined (non-null, non-undefined) value.\n * @param valA First value.\n * @param valB Second value.\n * @returns Comparison result (-1, 0, 1) or null if both values are non-null.\n */\nfunction handleNullValues<V>(valA: V, valB: V): number | null {\n  const aIsNull = valA === null;\n  const bIsNull = valB === null;\n\n  if (aIsNull && bIsNull) return 0;\n  if (aIsNull) return -1; // valA is null, valB is not -> valA is \"lesser\"\n  if (bIsNull) return 1; // valB is null, valA is not -> valA is \"greater\"\n\n  return null; // Both are non-null (and non-undefined at this stage of the main sorter)\n}\n\nfunction compareNonNullValues<V>(valA: NonNullable<V>, valB: NonNullable<V>, collator: Intl.Collator): number {\n  if (typeof valA === 'string' && typeof valB === 'string') {\n    return collator.compare(valA, valB);\n  } else if (typeof valA === 'number' && typeof valB === 'number') {\n    return valA - valB;\n  } else if (typeof valA === 'boolean' && typeof valB === 'boolean') {\n    return Number(valA) - Number(valB);\n  }\n  return collator.compare(String(valA), String(valB));\n}\n\n/**\n * Creates a comparison function suitable for Array.prototype.sort(),\n * based on the provided sort parameters.\n * Uses Intl.Collator for string comparison.\n * Can sort arrays of objects by an extracted value (via `keyFn`) or arrays of primitives directly.\n *\n * Behavior for `null` and `undefined` values (either direct or returned by `keyFn`):\n * - `undefined` values are **always** sorted to the end of the array, regardless of the sort direction.\n * - `null` values are sorted to the beginning of the array for ASC sort.\n * - For DESC sort, `null` values are sorted to the end of the array, but before any `undefined` values.\n *\n * @template T The type of items in the array.\n * @param options Optional sorting options. If `keyFn` is not provided, items are compared directly.\n * @returns A comparison function `(a: T, b: T) => number`.\n */\nexport function createSorter<T>(options?: SorterOptions<T>): (a: T, b: T) => number {\n  const { keyFn: userProvidedKeyFn, sortDirection = 'asc', locale, collatorOptions } = options || {};\n\n  if (userProvidedKeyFn !== undefined && typeof userProvidedKeyFn !== 'function') {\n    console.warn('createSorter: provided keyFn is not a function. Returning a no-op sorter.');\n    return () => 0;\n  }\n\n  // Default keyFn extracts the item itself, suitable for primitives or if T is the sortable value.\n  const keyFn = userProvidedKeyFn ?? ((item: T) => item as unknown as string | number | boolean | null | undefined);\n  const collator = new Intl.Collator(locale, collatorOptions);\n\n  return (itemA: T, itemB: T): number => {\n    const valA = keyFn(itemA);\n    const valB = keyFn(itemB);\n\n    const aIsUndefined = valA === undefined;\n    const bIsUndefined = valB === undefined;\n\n    if (aIsUndefined && bIsUndefined) return 0;\n    if (aIsUndefined) return 1;\n    if (bIsUndefined) return -1;\n\n    let comparison = 0;\n    const nullComparison = handleNullValues(valA, valB);\n\n    if (nullComparison !== null) {\n      comparison = nullComparison;\n    } else {\n      comparison = compareNonNullValues(\n        valA as NonNullable<string | number | boolean>, // valA is known to be not null/undefined here\n        valB as NonNullable<string | number | boolean>, // valB is known to be not null/undefined here\n        collator,\n      );\n    }\n\n    return sortDirection === 'desc' ? comparison * -1 : comparison;\n  };\n}\n"]}
|
|
91
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sorter.js","sourceRoot":"","sources":["../../../../../../../libs/angular/utils/src/lib/helpers/sorter.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Options for creating a sorter function. Can be used for arrays of objects (with `keyFn`)\n * or arrays of primitives (if `keyFn` is omitted).\n * @template T The type of items to be sorted.\n */\nexport interface SorterOptions<T> {\n  sortDirection?: SortDirection;\n  locale?: string;\n  collatorOptions?: Intl.CollatorOptions;\n  /**\n   * Optional function to extract the sortable value from an item.\n   * If not provided, the item itself is used for comparison (suitable for arrays of primitives).\n   * The extracted value should be a string, number, boolean, null, or undefined.\n   * @default (item: T) => item\n   */\n  keyFn?: (item: T) => string | number | boolean | null | undefined;\n}\n\n/**\n * Defines the possible directions for sorting.\n * It is used to specify how elements should be ordered.\n *\n * - `ASC`: Represents \"ascending\" order (e.g., A to Z, 1 to 10).\n * - `DESC`: Represents \"descending\" order (e.g., Z to A, 10 to 1).\n */\nexport type SortDirection = (typeof SortDirection)[keyof typeof SortDirection];\n\n/**\n * Constant object representing the available sort directions.\n * @property {string} ASC - Sort in ascending order.\n * @property {string} DESC - Sort in descending order.\n * @readonly\n */\nexport const SortDirection = {\n  ASC: 'asc',\n  DESC: 'desc',\n} as const;\n\n/**\n * Handles comparison for null values, assuming undefined has already been handled.\n * - `null` is considered less than any defined (non-null, non-undefined) value.\n * @param valA First value.\n * @param valB Second value.\n * @returns Comparison result (-1, 0, 1) or null if both values are non-null.\n */\nfunction handleNullValues<V>(valA: V, valB: V): number | null {\n  const aIsNull = valA === null;\n  const bIsNull = valB === null;\n\n  if (aIsNull && bIsNull) return 0;\n  if (aIsNull) return -1; // valA is null, valB is not -> valA is \"lesser\"\n  if (bIsNull) return 1; // valB is null, valA is not -> valA is \"greater\"\n\n  return null; // Both are non-null (and non-undefined at this stage of the main sorter)\n}\n\nfunction compareNonNullValues<V>(valA: NonNullable<V>, valB: NonNullable<V>, collator: Intl.Collator): number {\n  if (typeof valA === 'string' && typeof valB === 'string') {\n    return collator.compare(valA, valB);\n  } else if (typeof valA === 'number' && typeof valB === 'number') {\n    return valA - valB;\n  } else if (typeof valA === 'boolean' && typeof valB === 'boolean') {\n    return Number(valA) - Number(valB);\n  }\n  return collator.compare(String(valA), String(valB));\n}\n\n/**\n * Creates a comparison function suitable for Array.prototype.sort(),\n * based on the provided sort parameters.\n * Uses Intl.Collator for string comparison.\n * Can sort arrays of objects by an extracted value (via `keyFn`) or arrays of primitives directly.\n *\n * Behavior for `null` and `undefined` values (either direct or returned by `keyFn`):\n * - `undefined` values are **always** sorted to the end of the array, regardless of the sort direction.\n * - `null` values are sorted to the beginning of the array for ASC sort.\n * - For DESC sort, `null` values are sorted to the end of the array, but before any `undefined` values.\n *\n * @template T The type of items in the array.\n * @param options Optional sorting options. If `keyFn` is not provided, items are compared directly.\n * @returns A comparison function `(a: T, b: T) => number`.\n */\nexport function createSorter<T>(options?: SorterOptions<T>): (a: T, b: T) => number {\n  const { keyFn: userProvidedKeyFn, sortDirection, locale, collatorOptions } = options || {};\n\n  if (!sortDirection || !Object.values(SortDirection).includes(sortDirection)) return () => 0;\n\n  if (userProvidedKeyFn !== undefined && typeof userProvidedKeyFn !== 'function') {\n    console.warn('createSorter: provided keyFn is not a function. Returning a no-op sorter.');\n    return () => 0;\n  }\n\n  // Default keyFn extracts the item itself, suitable for primitives or if T is the sortable value.\n  const keyFn = userProvidedKeyFn ?? ((item: T) => item as unknown as string | number | boolean | null | undefined);\n  const collator = new Intl.Collator(locale, collatorOptions);\n\n  return (itemA: T, itemB: T): number => {\n    const valA = keyFn(itemA);\n    const valB = keyFn(itemB);\n\n    const aIsUndefined = valA === undefined;\n    const bIsUndefined = valB === undefined;\n\n    if (aIsUndefined && bIsUndefined) return 0;\n    if (aIsUndefined) return 1;\n    if (bIsUndefined) return -1;\n\n    let comparison = 0;\n    const nullComparison = handleNullValues(valA, valB);\n\n    if (nullComparison !== null) {\n      comparison = nullComparison;\n    } else {\n      comparison = compareNonNullValues(\n        valA as NonNullable<string | number | boolean>, // valA is known to be not null/undefined here\n        valB as NonNullable<string | number | boolean>, // valB is known to be not null/undefined here\n        collator,\n      );\n    }\n\n    return sortDirection === SortDirection.DESC ? comparison * -1 : comparison;\n  };\n}\n"]}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import { NgTemplateOutlet } from '@angular/common';
|
|
2
2
|
import * as i0 from '@angular/core';
|
|
3
3
|
import { inject, TemplateRef, Directive, contentChild, input, contentChildren, computed, Component, ChangeDetectionStrategy, ViewEncapsulation, NgModule } from '@angular/core';
|
|
4
|
+
export { SortDirection } from '@odx/angular/utils';
|
|
4
5
|
import { CoreModule } from '@odx/angular';
|
|
5
6
|
|
|
6
7
|
class CellDef {
|
|
@@ -97,17 +98,24 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
97
98
|
args: [{ selector: 'odx-table-empty', standalone: true }]
|
|
98
99
|
}] });
|
|
99
100
|
|
|
100
|
-
const SortDirection = {
|
|
101
|
-
UNSORTED: 'unsorted',
|
|
102
|
-
ASC: 'asc',
|
|
103
|
-
DESC: 'desc',
|
|
104
|
-
};
|
|
105
|
-
|
|
106
101
|
const TableVariant = {
|
|
107
102
|
DEFAULT: 'default',
|
|
108
103
|
STRIPED: 'striped',
|
|
109
104
|
};
|
|
110
105
|
|
|
106
|
+
/**
|
|
107
|
+
* @deprecated Use `SortDirection` from `@odx/angular/utils` directly instead.
|
|
108
|
+
* Will be removed in a future version.
|
|
109
|
+
*
|
|
110
|
+
* ```typescript
|
|
111
|
+
* // Instead of:
|
|
112
|
+
* // import { SortDirection } from '@odx/angular/components/data-table';
|
|
113
|
+
*
|
|
114
|
+
* // Use:
|
|
115
|
+
* import { SortDirection } from '@odx/angular/utils';
|
|
116
|
+
* ```
|
|
117
|
+
*/
|
|
118
|
+
|
|
111
119
|
/**
|
|
112
120
|
* DataTableComponent is a dynamic table structure built for displaying content.
|
|
113
121
|
*/
|
|
@@ -172,5 +180,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
172
180
|
* Generated bundle index. Do not edit.
|
|
173
181
|
*/
|
|
174
182
|
|
|
175
|
-
export { CellDef, CellDirective, ColumnDef, DataTableComponent, DataTableModule, HeaderCellDef, HeaderCellDirective,
|
|
183
|
+
export { CellDef, CellDirective, ColumnDef, DataTableComponent, DataTableModule, HeaderCellDef, HeaderCellDirective, TableEmptyDirective, TableVariant };
|
|
176
184
|
//# sourceMappingURL=odx-angular-components-data-table.mjs.map
|