@odx/angular 12.13.0 → 12.14.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +12 -0
- package/components/calendar/lib/services/calendar-month.service.d.ts +1 -0
- package/components/table/lib/components/header-title/header-title.component.d.ts +3 -4
- package/components/table/lib/models/index.d.ts +0 -1
- package/components/table/lib/models/sort-status.d.ts +1 -2
- package/esm2022/components/calendar/lib/services/calendar-month.service.mjs +4 -5
- package/esm2022/components/table/lib/components/header-title/header-title.component.mjs +20 -19
- package/esm2022/components/table/lib/models/index.mjs +1 -2
- package/esm2022/components/table/lib/models/sort-status.mjs +1 -1
- package/esm2022/utils/lib/helpers/index.mjs +2 -1
- package/esm2022/utils/lib/helpers/sorter.mjs +79 -0
- package/fesm2022/odx-angular-components-calendar.mjs +3 -4
- package/fesm2022/odx-angular-components-calendar.mjs.map +1 -1
- package/fesm2022/odx-angular-components-table.mjs +20 -24
- package/fesm2022/odx-angular-components-table.mjs.map +1 -1
- package/fesm2022/odx-angular-utils.mjs +80 -1
- package/fesm2022/odx-angular-utils.mjs.map +1 -1
- package/package.json +7 -7
- package/utils/lib/helpers/index.d.ts +1 -0
- package/utils/lib/helpers/sorter.d.ts +33 -0
- package/components/table/lib/models/sort-variant.d.ts +0 -6
- package/esm2022/components/table/lib/models/sort-variant.mjs +0 -6
package/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,17 @@
|
|
|
1
1
|
# @odx/angular
|
|
2
2
|
|
|
3
|
+
## 12.14.0
|
|
4
|
+
|
|
5
|
+
### Minor Changes
|
|
6
|
+
|
|
7
|
+
- 236139a: New default sort function
|
|
8
|
+
|
|
9
|
+
## 12.13.1
|
|
10
|
+
|
|
11
|
+
### Patch Changes
|
|
12
|
+
|
|
13
|
+
- 01787ab: Fixed: to avoid calendar jumps caused by the different number of days in each month, it now consistently displays 6 weeks per month
|
|
14
|
+
|
|
3
15
|
## 12.13.0
|
|
4
16
|
|
|
5
17
|
### Minor Changes
|
|
@@ -2,6 +2,7 @@ import * as i0 from "@angular/core";
|
|
|
2
2
|
/** @internal */
|
|
3
3
|
export declare class CalendarMonthService {
|
|
4
4
|
private readonly calendar;
|
|
5
|
+
private readonly weeksInView;
|
|
5
6
|
readonly weekDays$: import("rxjs").Observable<Date[]>;
|
|
6
7
|
readonly weeks$: import("rxjs").Observable<Date[][]>;
|
|
7
8
|
private generateWeekDays;
|
|
@@ -1,7 +1,6 @@
|
|
|
1
1
|
import { OnInit } from '@angular/core';
|
|
2
2
|
import { FormControl, FormGroup } from '@angular/forms';
|
|
3
3
|
import { SortStatus, TableHeaderCell } from '../../models';
|
|
4
|
-
import { TableSortVariant } from '../../models/sort-variant';
|
|
5
4
|
import * as i0 from "@angular/core";
|
|
6
5
|
/**
|
|
7
6
|
* HeaderTitleComponent provides the functionality for a table header cell, including sorting and checkbox
|
|
@@ -34,11 +33,11 @@ export declare class HeaderTitleComponent implements OnInit {
|
|
|
34
33
|
*/
|
|
35
34
|
get getSortStatus(): SortStatus;
|
|
36
35
|
/**
|
|
37
|
-
* Returns the icon
|
|
36
|
+
* Returns the icon name for the current sorting status (ascending, descending, unsorted).
|
|
38
37
|
*
|
|
39
|
-
* @returns {
|
|
38
|
+
* @returns {string}
|
|
40
39
|
*/
|
|
41
|
-
get sortIcon():
|
|
40
|
+
get sortIcon(): string;
|
|
42
41
|
/**
|
|
43
42
|
* Toggles the sorting status of the column and informs the table component of the change.
|
|
44
43
|
*/
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { inject, Injectable } from '@angular/core';
|
|
2
|
-
import { addDays,
|
|
2
|
+
import { addDays, startOfMonth, startOfWeek } from 'date-fns';
|
|
3
3
|
import { map } from 'rxjs';
|
|
4
4
|
import { CalendarService } from '../calendar.service';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
@@ -7,6 +7,7 @@ import * as i0 from "@angular/core";
|
|
|
7
7
|
export class CalendarMonthService {
|
|
8
8
|
constructor() {
|
|
9
9
|
this.calendar = inject(CalendarService);
|
|
10
|
+
this.weeksInView = 6;
|
|
10
11
|
this.weekDays$ = this.calendar.activeDate$.pipe(map((date) => this.generateWeekDays(date)));
|
|
11
12
|
this.weeks$ = this.calendar.activeDate$.pipe(map((date) => this.generateWeeks(date)));
|
|
12
13
|
}
|
|
@@ -16,12 +17,10 @@ export class CalendarMonthService {
|
|
|
16
17
|
}
|
|
17
18
|
generateWeeks(date) {
|
|
18
19
|
const startOfTheSelectedMonth = startOfMonth(date);
|
|
19
|
-
const endOfTheSelectedMonth = endOfMonth(date);
|
|
20
20
|
const startDate = startOfWeek(startOfTheSelectedMonth, { weekStartsOn: 1 });
|
|
21
|
-
const endDate = endOfWeek(endOfTheSelectedMonth);
|
|
22
21
|
const allWeeks = [];
|
|
23
22
|
let currentDate = startDate;
|
|
24
|
-
|
|
23
|
+
for (let i = this.weeksInView; i > 0; i--) {
|
|
25
24
|
allWeeks.push(this.generateDaysForWeek(currentDate, date));
|
|
26
25
|
currentDate = addDays(currentDate, 7);
|
|
27
26
|
}
|
|
@@ -36,4 +35,4 @@ export class CalendarMonthService {
|
|
|
36
35
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CalendarMonthService, decorators: [{
|
|
37
36
|
type: Injectable
|
|
38
37
|
}] });
|
|
39
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
38
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FsZW5kYXItbW9udGguc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvYW5ndWxhci9jb21wb25lbnRzL2NhbGVuZGFyL3NyYy9saWIvc2VydmljZXMvY2FsZW5kYXItbW9udGguc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNuRCxPQUFPLEVBQUUsT0FBTyxFQUFFLFlBQVksRUFBRSxXQUFXLEVBQUUsTUFBTSxVQUFVLENBQUM7QUFDOUQsT0FBTyxFQUFFLEdBQUcsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUMzQixPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0scUJBQXFCLENBQUM7O0FBRXRELGdCQUFnQjtBQUVoQixNQUFNLE9BQU8sb0JBQW9CO0lBRGpDO1FBRW1CLGFBQVEsR0FBRyxNQUFNLENBQUMsZUFBZSxDQUFDLENBQUM7UUFDbkMsZ0JBQVcsR0FBRyxDQUFDLENBQUM7UUFFakIsY0FBUyxHQUFHLElBQUksQ0FBQyxRQUFRLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQyxJQUFJLEVBQUUsRUFBRSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDdkYsV0FBTSxHQUFHLElBQUksQ0FBQyxRQUFRLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQyxJQUFJLEVBQUUsRUFBRSxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO0tBeUJsRztJQXZCUyxnQkFBZ0IsQ0FBQyxJQUFVO1FBQ2pDLE1BQU0sWUFBWSxHQUFHLFdBQVcsQ0FBQyxJQUFJLEVBQUUsRUFBRSxZQUFZLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQztRQUM1RCxPQUFPLEtBQUssQ0FBQyxJQUFJLENBQUMsRUFBRSxNQUFNLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsWUFBWSxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDdkUsQ0FBQztJQUVPLGFBQWEsQ0FBQyxJQUFVO1FBQzlCLE1BQU0sdUJBQXVCLEdBQUcsWUFBWSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQ25ELE1BQU0sU0FBUyxHQUFHLFdBQVcsQ0FBQyx1QkFBdUIsRUFBRSxFQUFFLFlBQVksRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBQzVFLE1BQU0sUUFBUSxHQUFHLEVBQUUsQ0FBQztRQUVwQixJQUFJLFdBQVcsR0FBRyxTQUFTLENBQUM7UUFFNUIsS0FBSyxJQUFJLENBQUMsR0FBRyxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQztZQUMxQyxRQUFRLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxXQUFXLEVBQUUsSUFBSSxDQUFDLENBQUMsQ0FBQztZQUMzRCxXQUFXLEdBQUcsT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUMsQ0FBQztRQUN4QyxDQUFDO1FBRUQsT0FBTyxRQUFRLENBQUM7SUFDbEIsQ0FBQztJQUVPLG1CQUFtQixDQUFDLElBQVUsRUFBRSxXQUFpQjtRQUN2RCxPQUFPLEtBQUssQ0FBQyxJQUFJLENBQUMsRUFBRSxNQUFNLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQyxFQUFFLFdBQVcsQ0FBQyxDQUFDO0lBQ2hGLENBQUM7K0dBN0JVLG9CQUFvQjttSEFBcEIsb0JBQW9COzs0RkFBcEIsb0JBQW9CO2tCQURoQyxVQUFVIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgaW5qZWN0LCBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBhZGREYXlzLCBzdGFydE9mTW9udGgsIHN0YXJ0T2ZXZWVrIH0gZnJvbSAnZGF0ZS1mbnMnO1xuaW1wb3J0IHsgbWFwIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBDYWxlbmRhclNlcnZpY2UgfSBmcm9tICcuLi9jYWxlbmRhci5zZXJ2aWNlJztcblxuLyoqIEBpbnRlcm5hbCAqL1xuQEluamVjdGFibGUoKVxuZXhwb3J0IGNsYXNzIENhbGVuZGFyTW9udGhTZXJ2aWNlIHtcbiAgcHJpdmF0ZSByZWFkb25seSBjYWxlbmRhciA9IGluamVjdChDYWxlbmRhclNlcnZpY2UpO1xuICBwcml2YXRlIHJlYWRvbmx5IHdlZWtzSW5WaWV3ID0gNjtcblxuICBwdWJsaWMgcmVhZG9ubHkgd2Vla0RheXMkID0gdGhpcy5jYWxlbmRhci5hY3RpdmVEYXRlJC5waXBlKG1hcCgoZGF0ZSkgPT4gdGhpcy5nZW5lcmF0ZVdlZWtEYXlzKGRhdGUpKSk7XG4gIHB1YmxpYyByZWFkb25seSB3ZWVrcyQgPSB0aGlzLmNhbGVuZGFyLmFjdGl2ZURhdGUkLnBpcGUobWFwKChkYXRlKSA9PiB0aGlzLmdlbmVyYXRlV2Vla3MoZGF0ZSkpKTtcblxuICBwcml2YXRlIGdlbmVyYXRlV2Vla0RheXMoZGF0ZTogRGF0ZSk6IERhdGVbXSB7XG4gICAgY29uc3QgZmlyc3RXZWVrRGF5ID0gc3RhcnRPZldlZWsoZGF0ZSwgeyB3ZWVrU3RhcnRzT246IDEgfSk7XG4gICAgcmV0dXJuIEFycmF5LmZyb20oeyBsZW5ndGg6IDcgfSwgKF8sIGkpID0+IGFkZERheXMoZmlyc3RXZWVrRGF5LCBpKSk7XG4gIH1cblxuICBwcml2YXRlIGdlbmVyYXRlV2Vla3MoZGF0ZTogRGF0ZSk6IERhdGVbXVtdIHtcbiAgICBjb25zdCBzdGFydE9mVGhlU2VsZWN0ZWRNb250aCA9IHN0YXJ0T2ZNb250aChkYXRlKTtcbiAgICBjb25zdCBzdGFydERhdGUgPSBzdGFydE9mV2VlayhzdGFydE9mVGhlU2VsZWN0ZWRNb250aCwgeyB3ZWVrU3RhcnRzT246IDEgfSk7XG4gICAgY29uc3QgYWxsV2Vla3MgPSBbXTtcblxuICAgIGxldCBjdXJyZW50RGF0ZSA9IHN0YXJ0RGF0ZTtcblxuICAgIGZvciAobGV0IGkgPSB0aGlzLndlZWtzSW5WaWV3OyBpID4gMDsgaS0tKSB7XG4gICAgICBhbGxXZWVrcy5wdXNoKHRoaXMuZ2VuZXJhdGVEYXlzRm9yV2VlayhjdXJyZW50RGF0ZSwgZGF0ZSkpO1xuICAgICAgY3VycmVudERhdGUgPSBhZGREYXlzKGN1cnJlbnREYXRlLCA3KTtcbiAgICB9XG5cbiAgICByZXR1cm4gYWxsV2Vla3M7XG4gIH1cblxuICBwcml2YXRlIGdlbmVyYXRlRGF5c0ZvcldlZWsoZGF0ZTogRGF0ZSwgY3VycmVudERhdGU6IERhdGUpOiBEYXRlW10ge1xuICAgIHJldHVybiBBcnJheS5mcm9tKHsgbGVuZ3RoOiA3IH0pLm1hcCgoXywgaSkgPT4gYWRkRGF5cyhkYXRlLCBpKSwgY3VycmVudERhdGUpO1xuICB9XG59XG4iXX0=
|
|
@@ -2,9 +2,9 @@ import { CommonModule } from '@angular/common';
|
|
|
2
2
|
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, HostBinding, inject, Input, ViewEncapsulation } from '@angular/core';
|
|
3
3
|
import { FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';
|
|
4
4
|
import { CheckboxComponent } from '@odx/angular/components/checkbox';
|
|
5
|
+
import { IconComponent } from '@odx/angular/components/icon';
|
|
5
6
|
import { untilDestroyed } from '@odx/angular/utils';
|
|
6
7
|
import { tap } from 'rxjs';
|
|
7
|
-
import { TableSortVariant } from '../../models/sort-variant';
|
|
8
8
|
import { TABLE } from '../../table.config';
|
|
9
9
|
import * as i0 from "@angular/core";
|
|
10
10
|
import * as i1 from "@angular/forms";
|
|
@@ -32,15 +32,19 @@ export class HeaderTitleComponent {
|
|
|
32
32
|
* @returns {SortStatus}
|
|
33
33
|
*/
|
|
34
34
|
get getSortStatus() {
|
|
35
|
-
return { column: this.item?.name, sortVariant:
|
|
35
|
+
return { column: this.item?.name, sortVariant: undefined };
|
|
36
36
|
}
|
|
37
37
|
/**
|
|
38
|
-
* Returns the icon
|
|
38
|
+
* Returns the icon name for the current sorting status (ascending, descending, unsorted).
|
|
39
39
|
*
|
|
40
|
-
* @returns {
|
|
40
|
+
* @returns {string}
|
|
41
41
|
*/
|
|
42
42
|
get sortIcon() {
|
|
43
|
-
|
|
43
|
+
if (this.sortStatus.sortVariant === 'asc')
|
|
44
|
+
return 'sort-down';
|
|
45
|
+
if (this.sortStatus.sortVariant === 'desc')
|
|
46
|
+
return 'sort-up';
|
|
47
|
+
return 'sortable';
|
|
44
48
|
}
|
|
45
49
|
/**
|
|
46
50
|
* Toggles the sorting status of the column and informs the table component of the change.
|
|
@@ -48,16 +52,13 @@ export class HeaderTitleComponent {
|
|
|
48
52
|
sortColumn() {
|
|
49
53
|
if (!this.item?.sortable)
|
|
50
54
|
return;
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
this.table.clearSort();
|
|
59
|
-
this.sortStatus = { ...this.sortStatus, sortVariant: TableSortVariant.DESC };
|
|
60
|
-
break;
|
|
55
|
+
if (this.sortStatus.sortVariant === 'desc' || !this.sortStatus.sortVariant) {
|
|
56
|
+
this.table.clearSort();
|
|
57
|
+
this.sortStatus = { ...this.sortStatus, sortVariant: 'asc' };
|
|
58
|
+
}
|
|
59
|
+
else if (this.sortStatus.sortVariant === 'asc') {
|
|
60
|
+
this.table.clearSort();
|
|
61
|
+
this.sortStatus = { ...this.sortStatus, sortVariant: 'desc' };
|
|
61
62
|
}
|
|
62
63
|
this.table.sorted.emit(this.sortStatus);
|
|
63
64
|
}
|
|
@@ -65,7 +66,7 @@ export class HeaderTitleComponent {
|
|
|
65
66
|
* Resets the sort status to unsorted.
|
|
66
67
|
*/
|
|
67
68
|
clearSortStatus() {
|
|
68
|
-
this.sortStatus =
|
|
69
|
+
this.sortStatus = this.getSortStatus;
|
|
69
70
|
this.check();
|
|
70
71
|
}
|
|
71
72
|
/**
|
|
@@ -112,15 +113,15 @@ export class HeaderTitleComponent {
|
|
|
112
113
|
return this.item.width ?? 'auto';
|
|
113
114
|
}
|
|
114
115
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: HeaderTitleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
115
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: HeaderTitleComponent, isStandalone: true, selector: "th[odxHeaderTitle]", inputs: { item: "item" }, host: { properties: { "style.width": "this.columnWidth" } }, ngImport: i0, template: "@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
|
|
116
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: HeaderTitleComponent, isStandalone: true, selector: "th[odxHeaderTitle]", inputs: { item: "item" }, host: { properties: { "style.width": "this.columnWidth" } }, ngImport: i0, template: "@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", dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: CheckboxComponent, selector: "odx-checkbox", inputs: ["indeterminate"], outputs: ["indeterminateChange"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: IconComponent, selector: "odx-icon", inputs: ["inline", "size", "name", "iconSet", "identifier"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
|
116
117
|
}
|
|
117
118
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: HeaderTitleComponent, decorators: [{
|
|
118
119
|
type: Component,
|
|
119
|
-
args: [{ selector: 'th[odxHeaderTitle]', standalone: true, encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, imports: [CommonModule, CheckboxComponent, ReactiveFormsModule], template: "@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
|
|
120
|
+
args: [{ selector: 'th[odxHeaderTitle]', standalone: true, encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, imports: [CommonModule, CheckboxComponent, ReactiveFormsModule, IconComponent], template: "@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" }]
|
|
120
121
|
}], propDecorators: { item: [{
|
|
121
122
|
type: Input
|
|
122
123
|
}], columnWidth: [{
|
|
123
124
|
type: HostBinding,
|
|
124
125
|
args: ['style.width']
|
|
125
126
|
}] } });
|
|
126
|
-
//# 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 { untilDestroyed } from '@odx/angular/utils';\nimport { tap } from 'rxjs';\nimport { SortStatus, TableBodyCell, TableHeaderCell } from '../../models';\nimport { TableSortVariant } from '../../models/sort-variant';\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],\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: TableSortVariant.UNSORTED };\n  }\n\n  /**\n   * Returns the icon or indicator for the current sorting status (ascending, descending, unsorted).\n   *\n   * @returns {TableSortVariant}\n   */\n  public get sortIcon(): TableSortVariant {\n    return this.sortStatus.sortVariant;\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    switch (this.sortStatus.sortVariant) {\n      case TableSortVariant.DESC:\n      case TableSortVariant.UNSORTED:\n        this.table.clearSort();\n        this.sortStatus = { ...this.sortStatus, sortVariant: TableSortVariant.ASC };\n        break;\n      case TableSortVariant.ASC:\n        this.table.clearSort();\n\n        this.sortStatus = { ...this.sortStatus, sortVariant: TableSortVariant.DESC };\n        break;\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 = { column: this.item?.name, sortVariant: TableSortVariant.UNSORTED };\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      @switch (sortIcon) {\n        @case ('unsorted') {\n          <svg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 16 16\">\n            <path d=\"m14,5H2v-2h12v2Zm-4,2H2v2h8v-2Zm-4,4H2v2h4v-2Z\" style=\"fill: var(--blue-700)\" />\n          </svg>\n        }\n        @case ('desc') {\n          <svg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 16 16\">\n            <polygon points=\"11 6 8 3 5 6 7 6 7 13 9 13 9 6 11 6\" style=\"fill: var(--blue-700)\" />\n          </svg>\n        }\n        @case ('asc') {\n          <svg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 16 16\">\n            <polygon points=\"5 10 8 13 11 10 9 10 9 3 7 3 7 10 5 10\" style=\"fill: var(--blue-700)\" />\n          </svg>\n        }\n      }\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 { 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"]}
|
|
@@ -1,6 +1,5 @@
|
|
|
1
1
|
export * from './header-cell';
|
|
2
2
|
export * from './sort-status';
|
|
3
|
-
export * from './sort-variant';
|
|
4
3
|
export * from './table-body-cell';
|
|
5
4
|
export * from './table-variant';
|
|
6
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXIvY29tcG9uZW50cy90YWJsZS9zcmMvbGliL21vZGVscy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGVBQWUsQ0FBQztBQUM5QixjQUFjLGVBQWUsQ0FBQztBQUM5QixjQUFjLG1CQUFtQixDQUFDO0FBQ2xDLGNBQWMsaUJBQWlCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2hlYWRlci1jZWxsJztcbmV4cG9ydCAqIGZyb20gJy4vc29ydC1zdGF0dXMnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS1ib2R5LWNlbGwnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS12YXJpYW50JztcbiJdfQ==
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic29ydC1zdGF0dXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXIvY29tcG9uZW50cy90YWJsZS9zcmMvbGliL21vZGVscy9zb3J0LXN0YXR1cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBTb3J0U3RhdHVzIHtcbiAgY29sdW1uOiBzdHJpbmc7XG4gIHNvcnRWYXJpYW50PzogJ2FzYycgfCAnZGVzYyc7XG59XG4iXX0=
|
|
@@ -18,6 +18,7 @@ export * from './match-url';
|
|
|
18
18
|
export * from './ng-changes';
|
|
19
19
|
export * from './provide-config';
|
|
20
20
|
export * from './queue';
|
|
21
|
+
export * from './sorter';
|
|
21
22
|
export * from './type-guards';
|
|
22
23
|
export * from './until-destroyed';
|
|
23
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
24
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL2FuZ3VsYXIvdXRpbHMvc3JjL2xpYi9oZWxwZXJzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsV0FBVyxDQUFDO0FBQzFCLGNBQWMsc0JBQXNCLENBQUM7QUFDckMsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxxQkFBcUIsQ0FBQztBQUNwQyxjQUFjLHdCQUF3QixDQUFDO0FBQ3ZDLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsT0FBTyxDQUFDO0FBQ3RCLGNBQWMsaUJBQWlCLENBQUM7QUFDaEMsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxxQkFBcUIsQ0FBQztBQUNwQyxjQUFjLHFCQUFxQixDQUFDO0FBQ3BDLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsaUJBQWlCLENBQUM7QUFDaEMsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxrQkFBa0IsQ0FBQztBQUNqQyxjQUFjLFNBQVMsQ0FBQztBQUN4QixjQUFjLFVBQVUsQ0FBQztBQUN6QixjQUFjLGVBQWUsQ0FBQztBQUM5QixjQUFjLG1CQUFtQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9hbmd1bGFyJztcbmV4cG9ydCAqIGZyb20gJy4vYW5pbWF0aW9ucy1oYW5kbGVyJztcbmV4cG9ydCAqIGZyb20gJy4vYXJyYXknO1xuZXhwb3J0ICogZnJvbSAnLi9idWlsZC11cmwnO1xuZXhwb3J0ICogZnJvbSAnLi9idWlsZC13ZWJzaXRlLXVybCc7XG5leHBvcnQgKiBmcm9tICcuL2NhY2hlLXN0b3JhZ2UtY2xpZW50JztcbmV4cG9ydCAqIGZyb20gJy4vY29lcmNpb24nO1xuZXhwb3J0ICogZnJvbSAnLi9kZWJvdW5jZSc7XG5leHBvcnQgKiBmcm9tICcuL2RlZmVyLWZuJztcbmV4cG9ydCAqIGZyb20gJy4vZG9tJztcbmV4cG9ydCAqIGZyb20gJy4vZXZlbnQtbWFuYWdlcic7XG5leHBvcnQgKiBmcm9tICcuL2dldC1heGlzJztcbmV4cG9ydCAqIGZyb20gJy4vZ2V0LWxhbmd1YWdlLWNvZGUnO1xuZXhwb3J0ICogZnJvbSAnLi9nZXQtb3Bwb3NpdGUtc2lkZSc7XG5leHBvcnQgKiBmcm9tICcuL2dldC1zaWRlJztcbmV4cG9ydCAqIGZyb20gJy4vZ2V0LXVuaXF1ZS1pZCc7XG5leHBvcnQgKiBmcm9tICcuL21hdGNoLXVybCc7XG5leHBvcnQgKiBmcm9tICcuL25nLWNoYW5nZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9wcm92aWRlLWNvbmZpZyc7XG5leHBvcnQgKiBmcm9tICcuL3F1ZXVlJztcbmV4cG9ydCAqIGZyb20gJy4vc29ydGVyJztcbmV4cG9ydCAqIGZyb20gJy4vdHlwZS1ndWFyZHMnO1xuZXhwb3J0ICogZnJvbSAnLi91bnRpbC1kZXN0cm95ZWQnO1xuIl19
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Handles comparison for null values, assuming undefined has already been handled.
|
|
3
|
+
* - `null` is considered less than any defined (non-null, non-undefined) value.
|
|
4
|
+
* @param valA First value.
|
|
5
|
+
* @param valB Second value.
|
|
6
|
+
* @returns Comparison result (-1, 0, 1) or null if both values are non-null.
|
|
7
|
+
*/
|
|
8
|
+
function handleNullValues(valA, valB) {
|
|
9
|
+
const aIsNull = valA === null;
|
|
10
|
+
const bIsNull = valB === null;
|
|
11
|
+
if (aIsNull && bIsNull)
|
|
12
|
+
return 0;
|
|
13
|
+
if (aIsNull)
|
|
14
|
+
return -1; // valA is null, valB is not -> valA is "lesser"
|
|
15
|
+
if (bIsNull)
|
|
16
|
+
return 1; // valB is null, valA is not -> valA is "greater"
|
|
17
|
+
return null; // Both are non-null (and non-undefined at this stage of the main sorter)
|
|
18
|
+
}
|
|
19
|
+
function compareNonNullValues(valA, valB, collator) {
|
|
20
|
+
if (typeof valA === 'string' && typeof valB === 'string') {
|
|
21
|
+
return collator.compare(valA, valB);
|
|
22
|
+
}
|
|
23
|
+
else if (typeof valA === 'number' && typeof valB === 'number') {
|
|
24
|
+
return valA - valB;
|
|
25
|
+
}
|
|
26
|
+
else if (typeof valA === 'boolean' && typeof valB === 'boolean') {
|
|
27
|
+
return Number(valA) - Number(valB);
|
|
28
|
+
}
|
|
29
|
+
return collator.compare(String(valA), String(valB));
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Creates a comparison function suitable for Array.prototype.sort(),
|
|
33
|
+
* based on the provided sort parameters.
|
|
34
|
+
* Uses Intl.Collator for string comparison.
|
|
35
|
+
* Can sort arrays of objects by an extracted value (via `keyFn`) or arrays of primitives directly.
|
|
36
|
+
*
|
|
37
|
+
* Behavior for `null` and `undefined` values (either direct or returned by `keyFn`):
|
|
38
|
+
* - `undefined` values are **always** sorted to the end of the array, regardless of the sort direction.
|
|
39
|
+
* - `null` values are sorted to the beginning of the array for ASC sort.
|
|
40
|
+
* - For DESC sort, `null` values are sorted to the end of the array, but before any `undefined` values.
|
|
41
|
+
*
|
|
42
|
+
* @template T The type of items in the array.
|
|
43
|
+
* @param options Optional sorting options. If `keyFn` is not provided, items are compared directly.
|
|
44
|
+
* @returns A comparison function `(a: T, b: T) => number`.
|
|
45
|
+
*/
|
|
46
|
+
export function createSorter(options) {
|
|
47
|
+
const { keyFn: userProvidedKeyFn, sortDirection = 'asc', locale, collatorOptions } = options || {};
|
|
48
|
+
if (userProvidedKeyFn !== undefined && typeof userProvidedKeyFn !== 'function') {
|
|
49
|
+
console.warn('createSorter: provided keyFn is not a function. Returning a no-op sorter.');
|
|
50
|
+
return () => 0;
|
|
51
|
+
}
|
|
52
|
+
// Default keyFn extracts the item itself, suitable for primitives or if T is the sortable value.
|
|
53
|
+
const keyFn = userProvidedKeyFn ?? ((item) => item);
|
|
54
|
+
const collator = new Intl.Collator(locale, collatorOptions);
|
|
55
|
+
return (itemA, itemB) => {
|
|
56
|
+
const valA = keyFn(itemA);
|
|
57
|
+
const valB = keyFn(itemB);
|
|
58
|
+
const aIsUndefined = valA === undefined;
|
|
59
|
+
const bIsUndefined = valB === undefined;
|
|
60
|
+
if (aIsUndefined && bIsUndefined)
|
|
61
|
+
return 0;
|
|
62
|
+
if (aIsUndefined)
|
|
63
|
+
return 1;
|
|
64
|
+
if (bIsUndefined)
|
|
65
|
+
return -1;
|
|
66
|
+
let comparison = 0;
|
|
67
|
+
const nullComparison = handleNullValues(valA, valB);
|
|
68
|
+
if (nullComparison !== null) {
|
|
69
|
+
comparison = nullComparison;
|
|
70
|
+
}
|
|
71
|
+
else {
|
|
72
|
+
comparison = compareNonNullValues(valA, // valA is known to be not null/undefined here
|
|
73
|
+
valB, // valB is known to be not null/undefined here
|
|
74
|
+
collator);
|
|
75
|
+
}
|
|
76
|
+
return sortDirection === 'desc' ? comparison * -1 : comparison;
|
|
77
|
+
};
|
|
78
|
+
}
|
|
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"]}
|
|
@@ -7,7 +7,7 @@ import { CoreModule, WindowRef, DisabledController } from '@odx/angular';
|
|
|
7
7
|
import { CSSComponent, fastDeepEqual, CSSModifier } from '@odx/angular/internal';
|
|
8
8
|
import { createConfigTokens, isString, isPresent, injectElement, hasChanged, EventManager, deferFn, untilDestroyed, isFunction } from '@odx/angular/utils';
|
|
9
9
|
import { BehaviorSubject, shareReplay, map, filter, distinctUntilChanged, merge, defer, of, withLatestFrom, combineLatest } from 'rxjs';
|
|
10
|
-
import { isSameDay, isSameMonth, isSameYear, endOfDay, endOfMonth, endOfYear, startOfDay, startOfMonth, startOfYear, isValid, toDate, isBefore, isEqual, addMonths, subMonths, isWithinInterval, startOfWeek, addDays,
|
|
10
|
+
import { isSameDay, isSameMonth, isSameYear, endOfDay, endOfMonth, endOfYear, startOfDay, startOfMonth, startOfYear, isValid, toDate, isBefore, isEqual, addMonths, subMonths, isWithinInterval, startOfWeek, addDays, eachMonthOfInterval, addYears, getYear, setYear, subDays, subWeeks, addWeeks, setMonth, getMonth } from 'date-fns';
|
|
11
11
|
import { ActionGroupComponent } from '@odx/angular/components/action-group';
|
|
12
12
|
import { ButtonComponent } from '@odx/angular/components/button';
|
|
13
13
|
import { IconComponent } from '@odx/angular/components/icon';
|
|
@@ -706,6 +706,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
706
706
|
class CalendarMonthService {
|
|
707
707
|
constructor() {
|
|
708
708
|
this.calendar = inject(CalendarService);
|
|
709
|
+
this.weeksInView = 6;
|
|
709
710
|
this.weekDays$ = this.calendar.activeDate$.pipe(map((date) => this.generateWeekDays(date)));
|
|
710
711
|
this.weeks$ = this.calendar.activeDate$.pipe(map((date) => this.generateWeeks(date)));
|
|
711
712
|
}
|
|
@@ -715,12 +716,10 @@ class CalendarMonthService {
|
|
|
715
716
|
}
|
|
716
717
|
generateWeeks(date) {
|
|
717
718
|
const startOfTheSelectedMonth = startOfMonth(date);
|
|
718
|
-
const endOfTheSelectedMonth = endOfMonth(date);
|
|
719
719
|
const startDate = startOfWeek(startOfTheSelectedMonth, { weekStartsOn: 1 });
|
|
720
|
-
const endDate = endOfWeek(endOfTheSelectedMonth);
|
|
721
720
|
const allWeeks = [];
|
|
722
721
|
let currentDate = startDate;
|
|
723
|
-
|
|
722
|
+
for (let i = this.weeksInView; i > 0; i--) {
|
|
724
723
|
allWeeks.push(this.generateDaysForWeek(currentDate, date));
|
|
725
724
|
currentDate = addDays(currentDate, 7);
|
|
726
725
|
}
|