@agorapulse/ui-components 18.0.9 → 18.0.11
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/agorapulse-ui-components-18.0.11.tgz +0 -0
- package/esm2022/datepicker/datepicker.component.mjs +1 -1
- package/esm2022/index.mjs +2 -1
- package/esm2022/input-search/input-search.component.mjs +5 -4
- package/esm2022/nav-selector/agorapulse-ui-components-nav-selector.mjs +5 -0
- package/esm2022/nav-selector/directives/tree-node-accessibility.directive.mjs +30 -0
- package/esm2022/nav-selector/nav-selector-category/nav-selector-category.component.mjs +81 -0
- package/esm2022/nav-selector/nav-selector-category/nav-selector-category.presenter.mjs +29 -0
- package/esm2022/nav-selector/nav-selector-group/nav-selector-group.component.mjs +142 -0
- package/esm2022/nav-selector/nav-selector-group/nav-selector-group.presenter.mjs +31 -0
- package/esm2022/nav-selector/nav-selector-leaf/nav-selector-leaf.component.mjs +240 -0
- package/esm2022/nav-selector/nav-selector-leaf/nav-selector-leaf.presenter.mjs +35 -0
- package/esm2022/nav-selector/nav-selector-leaf-detail/nav-selector-leaf-detail.component.mjs +39 -0
- package/esm2022/nav-selector/nav-selector-leaf-detail/nav-selector-leaf-detail.presenter.mjs +18 -0
- package/esm2022/nav-selector/nav-selector-leaf-details/nav-selector-leaf-details.component.mjs +92 -0
- package/esm2022/nav-selector/nav-selector-leaf-details/nav-selector-leaf-details.presenter.mjs +31 -0
- package/esm2022/nav-selector/nav-selector.component.mjs +123 -0
- package/esm2022/nav-selector/nav-selector.mjs +9 -0
- package/esm2022/nav-selector/nav-selector.state.mjs +173 -0
- package/esm2022/nav-selector/public_api.mjs +2 -0
- package/esm2022/nav-selector/utils/leaf.utils.mjs +10 -0
- package/esm2022/nav-selector/utils/nav-selector.accessibility.mjs +171 -0
- package/esm2022/nav-selector/utils/nav-selector.builder.mjs +263 -0
- package/esm2022/nav-selector/utils/nav-selector.filter.mjs +102 -0
- package/esm2022/nav-selector/utils/nav-selector.folding.mjs +219 -0
- package/esm2022/nav-selector/utils/nav-selector.minifying.mjs +50 -0
- package/esm2022/nav-selector/utils/nav-selector.multi-select.mjs +208 -0
- package/esm2022/nav-selector/utils/nav-selector.single-select.mjs +91 -0
- package/esm2022/nav-selector/utils/nav-selector.view-more.mjs +98 -0
- package/fesm2022/agorapulse-ui-components-datepicker.mjs.map +1 -1
- package/fesm2022/agorapulse-ui-components-input-search.mjs +4 -3
- package/fesm2022/agorapulse-ui-components-input-search.mjs.map +1 -1
- package/fesm2022/agorapulse-ui-components-nav-selector.mjs +2198 -0
- package/fesm2022/agorapulse-ui-components-nav-selector.mjs.map +1 -0
- package/fesm2022/agorapulse-ui-components.mjs +1 -0
- package/fesm2022/agorapulse-ui-components.mjs.map +1 -1
- package/index.d.ts +1 -0
- package/input-search/input-search.component.d.ts +1 -1
- package/nav-selector/directives/tree-node-accessibility.directive.d.ts +9 -0
- package/nav-selector/index.d.ts +5 -0
- package/nav-selector/nav-selector-category/nav-selector-category.component.d.ts +16 -0
- package/nav-selector/nav-selector-category/nav-selector-category.presenter.d.ts +14 -0
- package/nav-selector/nav-selector-group/nav-selector-group.component.d.ts +29 -0
- package/nav-selector/nav-selector-group/nav-selector-group.presenter.d.ts +17 -0
- package/nav-selector/nav-selector-leaf/nav-selector-leaf.component.d.ts +51 -0
- package/nav-selector/nav-selector-leaf/nav-selector-leaf.presenter.d.ts +19 -0
- package/nav-selector/nav-selector-leaf-detail/nav-selector-leaf-detail.component.d.ts +13 -0
- package/nav-selector/nav-selector-leaf-detail/nav-selector-leaf-detail.presenter.d.ts +10 -0
- package/nav-selector/nav-selector-leaf-details/nav-selector-leaf-details.component.d.ts +24 -0
- package/nav-selector/nav-selector-leaf-details/nav-selector-leaf-details.presenter.d.ts +14 -0
- package/nav-selector/nav-selector.component.d.ts +29 -0
- package/nav-selector/nav-selector.d.ts +220 -0
- package/nav-selector/nav-selector.state.d.ts +47 -0
- package/nav-selector/public_api.d.ts +2 -0
- package/nav-selector/utils/leaf.utils.d.ts +5 -0
- package/nav-selector/utils/nav-selector.accessibility.d.ts +52 -0
- package/nav-selector/utils/nav-selector.builder.d.ts +32 -0
- package/nav-selector/utils/nav-selector.filter.d.ts +30 -0
- package/nav-selector/utils/nav-selector.folding.d.ts +47 -0
- package/nav-selector/utils/nav-selector.minifying.d.ts +27 -0
- package/nav-selector/utils/nav-selector.multi-select.d.ts +54 -0
- package/nav-selector/utils/nav-selector.single-select.d.ts +15 -0
- package/nav-selector/utils/nav-selector.view-more.d.ts +30 -0
- package/package.json +13 -7
- package/agorapulse-ui-components-18.0.9.tgz +0 -0
|
@@ -0,0 +1,173 @@
|
|
|
1
|
+
import { computed, effect, Injectable, signal } from '@angular/core';
|
|
2
|
+
import { isInternalNavSelectorEntryALeaf, isInternalNavSelectorEntryANode, } from './nav-selector';
|
|
3
|
+
import { NavSelectorAccessibility } from './utils/nav-selector.accessibility';
|
|
4
|
+
import { NavSelectorBuilder } from './utils/nav-selector.builder';
|
|
5
|
+
import { NavSelectorFilter } from './utils/nav-selector.filter';
|
|
6
|
+
import { NavSelectorFolding } from './utils/nav-selector.folding';
|
|
7
|
+
import { NavSelectorMinifying } from './utils/nav-selector.minifying';
|
|
8
|
+
import { NavSelectorMultiSelect } from './utils/nav-selector.multi-select';
|
|
9
|
+
import { NavSelectorSingleSelect } from './utils/nav-selector.single-select';
|
|
10
|
+
import { NavSelectorViewMore } from './utils/nav-selector.view-more';
|
|
11
|
+
import * as i0 from "@angular/core";
|
|
12
|
+
/**
|
|
13
|
+
* NavSelectorState is a service that manages the state of the nav selector.
|
|
14
|
+
*
|
|
15
|
+
* It should be provided by the component to ensure that if many nav selectors are used in the same page, they don't share the same state.
|
|
16
|
+
*/
|
|
17
|
+
export class NavSelectorState {
|
|
18
|
+
entries = signal([]);
|
|
19
|
+
multipleModeEnabled = signal(false);
|
|
20
|
+
_texts = signal({
|
|
21
|
+
title: '',
|
|
22
|
+
only: '',
|
|
23
|
+
tokenInvalid: '',
|
|
24
|
+
featureLocked: '',
|
|
25
|
+
searchPlaceholder: '',
|
|
26
|
+
noResults: '',
|
|
27
|
+
viewMore: '',
|
|
28
|
+
viewLess: '',
|
|
29
|
+
});
|
|
30
|
+
lastSelectedUids = signal([]);
|
|
31
|
+
isMultipleModeEnabled = this.multipleModeEnabled;
|
|
32
|
+
search = signal('');
|
|
33
|
+
filteredEntries = computed(() => NavSelectorFilter.filter(this.entries(), this.search()));
|
|
34
|
+
noResults = computed(() => this.filteredEntries().every(({ hidden }) => hidden));
|
|
35
|
+
texts = this._texts.asReadonly();
|
|
36
|
+
leafActionProjection = signal(undefined);
|
|
37
|
+
expanded = signal(true);
|
|
38
|
+
expandedAfterDelay = signal(true);
|
|
39
|
+
selectedUidsChangeCallback = null;
|
|
40
|
+
constructor() {
|
|
41
|
+
effect(() => {
|
|
42
|
+
const newSelectedUids = this.collectSelectedUids(this.entries());
|
|
43
|
+
if (this.selectedUidsChangeCallback) {
|
|
44
|
+
this.selectedUidsChangeCallback(newSelectedUids);
|
|
45
|
+
}
|
|
46
|
+
}, { allowSignalWrites: true });
|
|
47
|
+
}
|
|
48
|
+
updateMultiModeEnabled(enabled) {
|
|
49
|
+
this.multipleModeEnabled.set(enabled);
|
|
50
|
+
}
|
|
51
|
+
updateEntries(entries, initialSelectedEntryUids, detailsDisplayedLimit) {
|
|
52
|
+
if (this.multipleModeEnabled()) {
|
|
53
|
+
this.entries.set(NavSelectorBuilder.buildInMultipleMode(entries, initialSelectedEntryUids, detailsDisplayedLimit));
|
|
54
|
+
}
|
|
55
|
+
else {
|
|
56
|
+
this.entries.set(NavSelectorBuilder.build(entries, initialSelectedEntryUids?.[0] ?? null, detailsDisplayedLimit));
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
onGroupToggleFolding(group) {
|
|
60
|
+
this.entries.update(entries => NavSelectorFolding.toggleFolding(entries, group.uid, !this.expanded()));
|
|
61
|
+
}
|
|
62
|
+
onNodeSelect(node) {
|
|
63
|
+
this.entries.update(entries => NavSelectorMultiSelect.selectInMultipleMode(entries, node.uid));
|
|
64
|
+
}
|
|
65
|
+
onLeafClicked(leaf) {
|
|
66
|
+
if (!leaf.selectable) {
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
if (this.multipleModeEnabled()) {
|
|
70
|
+
this.entries.update(entries => NavSelectorMultiSelect.selectInMultipleMode(entries, leaf.uid));
|
|
71
|
+
}
|
|
72
|
+
else {
|
|
73
|
+
this.entries.update(entries => NavSelectorSingleSelect.select(entries, leaf.uid));
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
onLeafToggleFolding(leaf) {
|
|
77
|
+
this.entries.update(entries => NavSelectorFolding.toggleFolding(entries, leaf.uid, !this.expanded()));
|
|
78
|
+
}
|
|
79
|
+
onLeafDetailClicked(leafDetail) {
|
|
80
|
+
this.entries.update(entries => NavSelectorSingleSelect.select(entries, leafDetail.uid));
|
|
81
|
+
}
|
|
82
|
+
updateTexts(texts) {
|
|
83
|
+
this._texts.set(texts);
|
|
84
|
+
}
|
|
85
|
+
selectOnlyLeaf(leaf) {
|
|
86
|
+
this.entries.update(entries => NavSelectorMultiSelect.selectOnlyALeafInMultipleMode(entries, leaf.uid));
|
|
87
|
+
}
|
|
88
|
+
toggleExpanded() {
|
|
89
|
+
this.expanded.update(expanded => !expanded);
|
|
90
|
+
this.entries.update(entries => {
|
|
91
|
+
if (this.expanded()) {
|
|
92
|
+
return NavSelectorMinifying.expand(entries);
|
|
93
|
+
}
|
|
94
|
+
return NavSelectorMinifying.minify(entries);
|
|
95
|
+
});
|
|
96
|
+
if (!this.expanded()) {
|
|
97
|
+
setTimeout(() => this.expandedAfterDelay.set(this.expanded()), 150);
|
|
98
|
+
}
|
|
99
|
+
else {
|
|
100
|
+
this.expandedAfterDelay.set(this.expanded());
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
onSelectionChange(uids) {
|
|
104
|
+
// Avoid infinite loop if you put unexisting uids as select will generate the same entries but with different reference
|
|
105
|
+
if (JSON.stringify(uids) === JSON.stringify(this.lastSelectedUids())) {
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
this.lastSelectedUids.set(uids);
|
|
109
|
+
const selectedUids = this.collectSelectedUids(this.entries());
|
|
110
|
+
const uidToSelect = uids.filter(uid => selectedUids.every(selectedUid => selectedUid !== uid));
|
|
111
|
+
const uidToUnselect = selectedUids.filter(uid => uids.every(selectedUid => selectedUid !== uid));
|
|
112
|
+
if (!uidToSelect.length && !uidToUnselect.length) {
|
|
113
|
+
return;
|
|
114
|
+
}
|
|
115
|
+
let entries = this.entries();
|
|
116
|
+
if (this.multipleModeEnabled()) {
|
|
117
|
+
for (const uid of [...uidToSelect, ...uidToUnselect]) {
|
|
118
|
+
entries = NavSelectorMultiSelect.selectInMultipleMode(entries, uid);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
else {
|
|
122
|
+
entries = NavSelectorSingleSelect.select(entries, uids[0]);
|
|
123
|
+
}
|
|
124
|
+
this.entries.set(entries);
|
|
125
|
+
}
|
|
126
|
+
registerOnSelectedUidsChange(callback) {
|
|
127
|
+
this.selectedUidsChangeCallback = callback;
|
|
128
|
+
}
|
|
129
|
+
onArrowDown() {
|
|
130
|
+
this.entries.update(entries => NavSelectorAccessibility.focusNext(entries));
|
|
131
|
+
}
|
|
132
|
+
onArrowUp() {
|
|
133
|
+
this.entries.update(entries => NavSelectorAccessibility.focusPrevious(entries));
|
|
134
|
+
}
|
|
135
|
+
collectSelectedUids(entries) {
|
|
136
|
+
return entries.flatMap(entry => {
|
|
137
|
+
if (isInternalNavSelectorEntryALeaf(entry)) {
|
|
138
|
+
if (entry.details.length) {
|
|
139
|
+
return entry.details.flatMap(detail => (detail.selected ? [detail.uid] : []));
|
|
140
|
+
}
|
|
141
|
+
return entry.selected ? [entry.uid] : [];
|
|
142
|
+
}
|
|
143
|
+
else if (isInternalNavSelectorEntryANode(entry)) {
|
|
144
|
+
return this.collectSelectedUids(entry.children);
|
|
145
|
+
}
|
|
146
|
+
return [];
|
|
147
|
+
});
|
|
148
|
+
}
|
|
149
|
+
updateLeafAction(leafActionProjection) {
|
|
150
|
+
this.leafActionProjection.set(leafActionProjection);
|
|
151
|
+
}
|
|
152
|
+
fold(entry) {
|
|
153
|
+
this.entries.update(entries => NavSelectorFolding.fold(entries, entry.uid, !this.expanded()));
|
|
154
|
+
}
|
|
155
|
+
unfold(entry) {
|
|
156
|
+
this.entries.update(entries => NavSelectorFolding.unfold(entries, entry.uid, !this.expanded()));
|
|
157
|
+
}
|
|
158
|
+
updateDetailsDisplayedLimit(detailsDisplayedLimit) {
|
|
159
|
+
this.entries.update(entries => NavSelectorViewMore.changeViewMoreDetailsDisplayedLimit(entries, detailsDisplayedLimit));
|
|
160
|
+
}
|
|
161
|
+
viewMore(leaf) {
|
|
162
|
+
this.entries.update(entries => NavSelectorViewMore.viewMore(entries, leaf.uid));
|
|
163
|
+
}
|
|
164
|
+
viewLess(leaf) {
|
|
165
|
+
this.entries.update(entries => NavSelectorViewMore.viewLess(entries, leaf.uid));
|
|
166
|
+
}
|
|
167
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavSelectorState, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
168
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavSelectorState });
|
|
169
|
+
}
|
|
170
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavSelectorState, decorators: [{
|
|
171
|
+
type: Injectable
|
|
172
|
+
}], ctorParameters: () => [] });
|
|
173
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nav-selector.state.js","sourceRoot":"","sources":["../../../../libs/ui-components/nav-selector/src/nav-selector.state.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAe,MAAM,eAAe,CAAC;AAClF,OAAO,EAIH,+BAA+B,EAC/B,+BAA+B,GAGlC,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,wBAAwB,EAAE,MAAM,oCAAoC,CAAC;AAC9E,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAC;AAClE,OAAO,EAAE,iBAAiB,EAAE,MAAM,6BAA6B,CAAC;AAChE,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAC;AAClE,OAAO,EAAE,oBAAoB,EAAE,MAAM,gCAAgC,CAAC;AACtE,OAAO,EAAE,sBAAsB,EAAE,MAAM,mCAAmC,CAAC;AAC3E,OAAO,EAAE,uBAAuB,EAAE,MAAM,oCAAoC,CAAC;AAC7E,OAAO,EAAE,mBAAmB,EAAE,MAAM,gCAAgC,CAAC;;AAErE;;;;GAIG;AAEH,MAAM,OAAO,gBAAgB;IACjB,OAAO,GAAG,MAAM,CAA6B,EAAE,CAAC,CAAC;IACjD,mBAAmB,GAAG,MAAM,CAAU,KAAK,CAAC,CAAC;IAC7C,MAAM,GAAG,MAAM,CAA6B;QAChD,KAAK,EAAE,EAAE;QACT,IAAI,EAAE,EAAE;QACR,YAAY,EAAE,EAAE;QAChB,aAAa,EAAE,EAAE;QACjB,iBAAiB,EAAE,EAAE;QACrB,SAAS,EAAE,EAAE;QACb,QAAQ,EAAE,EAAE;QACZ,QAAQ,EAAE,EAAE;KACf,CAAC,CAAC;IACK,gBAAgB,GAAG,MAAM,CAAW,EAAE,CAAC,CAAC;IAEhD,qBAAqB,GAAG,IAAI,CAAC,mBAAmB,CAAC;IACjD,MAAM,GAAG,MAAM,CAAS,EAAE,CAAC,CAAC;IAC5B,eAAe,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,iBAAiB,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAC1F,SAAS,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,KAAK,CAAC,CAAC,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC;IACjF,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC;IAEjC,oBAAoB,GAAG,MAAM,CAAkD,SAAS,CAAC,CAAC;IAE1F,QAAQ,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;IACxB,kBAAkB,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;IAElC,0BAA0B,GAA8C,IAAI,CAAC;IAE7E;QACI,MAAM,CACF,GAAG,EAAE;YACD,MAAM,eAAe,GAAG,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;YACjE,IAAI,IAAI,CAAC,0BAA0B,EAAE,CAAC;gBAClC,IAAI,CAAC,0BAA0B,CAAC,eAAe,CAAC,CAAC;YACrD,CAAC;QACL,CAAC,EACD,EAAE,iBAAiB,EAAE,IAAI,EAAE,CAC9B,CAAC;IACN,CAAC;IAED,sBAAsB,CAAC,OAAgB;QACnC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAC1C,CAAC;IAED,aAAa,CAAC,OAA2B,EAAE,wBAAkC,EAAE,qBAA6B;QACxG,IAAI,IAAI,CAAC,mBAAmB,EAAE,EAAE,CAAC;YAC7B,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,mBAAmB,CAAC,OAAO,EAAE,wBAAwB,EAAE,qBAAqB,CAAC,CAAC,CAAC;QACvH,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,KAAK,CAAC,OAAO,EAAE,wBAAwB,EAAE,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,qBAAqB,CAAC,CAAC,CAAC;QACtH,CAAC;IACL,CAAC;IAED,oBAAoB,CAAC,KAA+B;QAChD,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,kBAAkB,CAAC,aAAa,CAAC,OAAO,EAAE,KAAK,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;IAC3G,CAAC;IAED,YAAY,CAAC,IAA8B;QACvC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,sBAAsB,CAAC,oBAAoB,CAAC,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;IACnG,CAAC;IAED,aAAa,CAAC,IAA6B;QACvC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;YACnB,OAAO;QACX,CAAC;QACD,IAAI,IAAI,CAAC,mBAAmB,EAAE,EAAE,CAAC;YAC7B,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,sBAAsB,CAAC,oBAAoB,CAAC,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;QACnG,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,uBAAuB,CAAC,MAAM,CAAC,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;QACtF,CAAC;IACL,CAAC;IAED,mBAAmB,CAAC,IAA8B;QAC9C,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,kBAAkB,CAAC,aAAa,CAAC,OAAO,EAAE,IAAI,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;IAC1G,CAAC;IAED,mBAAmB,CAAC,UAA0C;QAC1D,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,uBAAuB,CAAC,MAAM,CAAC,OAAO,EAAE,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC;IAC5F,CAAC;IAED,WAAW,CAAC,KAAiC;QACzC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IAC3B,CAAC;IAED,cAAc,CAAC,IAA6B;QACxC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,sBAAsB,CAAC,6BAA6B,CAAC,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;IAC5G,CAAC;IAED,cAAc;QACV,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,CAAC;QAC5C,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE;YAC1B,IAAI,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;gBAClB,OAAO,oBAAoB,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAChD,CAAC;YACD,OAAO,oBAAoB,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;YACnB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,EAAE,GAAG,CAAC,CAAC;QACxE,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;QACjD,CAAC;IACL,CAAC;IAED,iBAAiB,CAAC,IAAc;QAC5B,uHAAuH;QACvH,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC,EAAE,CAAC;YACnE,OAAO;QACX,CAAC;QACD,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAChC,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QAC9D,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,YAAY,CAAC,KAAK,CAAC,WAAW,CAAC,EAAE,CAAC,WAAW,KAAK,GAAG,CAAC,CAAC,CAAC;QAC/F,MAAM,aAAa,GAAG,YAAY,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,EAAE,CAAC,WAAW,KAAK,GAAG,CAAC,CAAC,CAAC;QAEjG,IAAI,CAAC,WAAW,CAAC,MAAM,IAAI,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC;YAC/C,OAAO;QACX,CAAC;QAED,IAAI,OAAO,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;QAE7B,IAAI,IAAI,CAAC,mBAAmB,EAAE,EAAE,CAAC;YAC7B,KAAK,MAAM,GAAG,IAAI,CAAC,GAAG,WAAW,EAAE,GAAG,aAAa,CAAC,EAAE,CAAC;gBACnD,OAAO,GAAG,sBAAsB,CAAC,oBAAoB,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;YACxE,CAAC;QACL,CAAC;aAAM,CAAC;YACJ,OAAO,GAAG,uBAAuB,CAAC,MAAM,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QAC/D,CAAC;QACD,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAC9B,CAAC;IAED,4BAA4B,CAAC,QAA0C;QACnE,IAAI,CAAC,0BAA0B,GAAG,QAAQ,CAAC;IAC/C,CAAC;IAED,WAAW;QACP,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,wBAAwB,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC;IAChF,CAAC;IAED,SAAS;QACL,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,wBAAwB,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC;IACpF,CAAC;IAEO,mBAAmB,CAAC,OAAmC;QAC3D,OAAO,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YAC3B,IAAI,+BAA+B,CAAC,KAAK,CAAC,EAAE,CAAC;gBACzC,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC;oBACvB,OAAO,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;gBAClF,CAAC;gBACD,OAAO,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YAC7C,CAAC;iBAAM,IAAI,+BAA+B,CAAC,KAAK,CAAC,EAAE,CAAC;gBAChD,OAAO,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;YACpD,CAAC;YACD,OAAO,EAAE,CAAC;QACd,CAAC,CAAC,CAAC;IACP,CAAC;IAED,gBAAgB,CAAC,oBAAyD;QACtE,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC;IACxD,CAAC;IAED,IAAI,CAAC,KAA+B;QAChC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,kBAAkB,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;IAClG,CAAC;IAED,MAAM,CAAC,KAA+B;QAClC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,kBAAkB,CAAC,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;IACpG,CAAC;IAED,2BAA2B,CAAC,qBAA6B;QACrD,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,mBAAmB,CAAC,mCAAmC,CAAC,OAAO,EAAE,qBAAqB,CAAC,CAAC,CAAC;IAC5H,CAAC;IAED,QAAQ,CAAC,IAA6B;QAClC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,mBAAmB,CAAC,QAAQ,CAAC,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;IACpF,CAAC;IAED,QAAQ,CAAC,IAA6B;QAClC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,mBAAmB,CAAC,QAAQ,CAAC,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;IACpF,CAAC;uGAhLQ,gBAAgB;2GAAhB,gBAAgB;;2FAAhB,gBAAgB;kBAD5B,UAAU","sourcesContent":["import { computed, effect, Injectable, signal, TemplateRef } from '@angular/core';\nimport {\n    InternalNavSelectorEntry,\n    InternalNavSelectorLeaf,\n    InternalNavSelectorLeafDetails,\n    isInternalNavSelectorEntryALeaf,\n    isInternalNavSelectorEntryANode,\n    NavSelectorEntry,\n    NavSelectorTranslatedTexts,\n} from './nav-selector';\nimport { NavSelectorAccessibility } from './utils/nav-selector.accessibility';\nimport { NavSelectorBuilder } from './utils/nav-selector.builder';\nimport { NavSelectorFilter } from './utils/nav-selector.filter';\nimport { NavSelectorFolding } from './utils/nav-selector.folding';\nimport { NavSelectorMinifying } from './utils/nav-selector.minifying';\nimport { NavSelectorMultiSelect } from './utils/nav-selector.multi-select';\nimport { NavSelectorSingleSelect } from './utils/nav-selector.single-select';\nimport { NavSelectorViewMore } from './utils/nav-selector.view-more';\n\n/**\n * NavSelectorState is a service that manages the state of the nav selector.\n *\n * It should be provided by the component to ensure that if many nav selectors are used in the same page, they don't share the same state.\n */\n@Injectable()\nexport class NavSelectorState {\n    private entries = signal<InternalNavSelectorEntry[]>([]);\n    private multipleModeEnabled = signal<boolean>(false);\n    private _texts = signal<NavSelectorTranslatedTexts>({\n        title: '',\n        only: '',\n        tokenInvalid: '',\n        featureLocked: '',\n        searchPlaceholder: '',\n        noResults: '',\n        viewMore: '',\n        viewLess: '',\n    });\n    private lastSelectedUids = signal<string[]>([]);\n\n    isMultipleModeEnabled = this.multipleModeEnabled;\n    search = signal<string>('');\n    filteredEntries = computed(() => NavSelectorFilter.filter(this.entries(), this.search()));\n    noResults = computed(() => this.filteredEntries().every(({ hidden }) => hidden));\n    texts = this._texts.asReadonly();\n\n    leafActionProjection = signal<ReadonlyArray<TemplateRef<unknown>> | undefined>(undefined);\n\n    expanded = signal(true);\n    expandedAfterDelay = signal(true);\n\n    selectedUidsChangeCallback: ((selectedUids: string[]) => void) | null = null;\n\n    constructor() {\n        effect(\n            () => {\n                const newSelectedUids = this.collectSelectedUids(this.entries());\n                if (this.selectedUidsChangeCallback) {\n                    this.selectedUidsChangeCallback(newSelectedUids);\n                }\n            },\n            { allowSignalWrites: true }\n        );\n    }\n\n    updateMultiModeEnabled(enabled: boolean) {\n        this.multipleModeEnabled.set(enabled);\n    }\n\n    updateEntries(entries: NavSelectorEntry[], initialSelectedEntryUids: string[], detailsDisplayedLimit: number) {\n        if (this.multipleModeEnabled()) {\n            this.entries.set(NavSelectorBuilder.buildInMultipleMode(entries, initialSelectedEntryUids, detailsDisplayedLimit));\n        } else {\n            this.entries.set(NavSelectorBuilder.build(entries, initialSelectedEntryUids?.[0] ?? null, detailsDisplayedLimit));\n        }\n    }\n\n    onGroupToggleFolding(group: InternalNavSelectorEntry) {\n        this.entries.update(entries => NavSelectorFolding.toggleFolding(entries, group.uid, !this.expanded()));\n    }\n\n    onNodeSelect(node: InternalNavSelectorEntry) {\n        this.entries.update(entries => NavSelectorMultiSelect.selectInMultipleMode(entries, node.uid));\n    }\n\n    onLeafClicked(leaf: InternalNavSelectorLeaf) {\n        if (!leaf.selectable) {\n            return;\n        }\n        if (this.multipleModeEnabled()) {\n            this.entries.update(entries => NavSelectorMultiSelect.selectInMultipleMode(entries, leaf.uid));\n        } else {\n            this.entries.update(entries => NavSelectorSingleSelect.select(entries, leaf.uid));\n        }\n    }\n\n    onLeafToggleFolding(leaf: InternalNavSelectorEntry) {\n        this.entries.update(entries => NavSelectorFolding.toggleFolding(entries, leaf.uid, !this.expanded()));\n    }\n\n    onLeafDetailClicked(leafDetail: InternalNavSelectorLeafDetails) {\n        this.entries.update(entries => NavSelectorSingleSelect.select(entries, leafDetail.uid));\n    }\n\n    updateTexts(texts: NavSelectorTranslatedTexts) {\n        this._texts.set(texts);\n    }\n\n    selectOnlyLeaf(leaf: InternalNavSelectorLeaf) {\n        this.entries.update(entries => NavSelectorMultiSelect.selectOnlyALeafInMultipleMode(entries, leaf.uid));\n    }\n\n    toggleExpanded() {\n        this.expanded.update(expanded => !expanded);\n        this.entries.update(entries => {\n            if (this.expanded()) {\n                return NavSelectorMinifying.expand(entries);\n            }\n            return NavSelectorMinifying.minify(entries);\n        });\n        if (!this.expanded()) {\n            setTimeout(() => this.expandedAfterDelay.set(this.expanded()), 150);\n        } else {\n            this.expandedAfterDelay.set(this.expanded());\n        }\n    }\n\n    onSelectionChange(uids: string[]) {\n        // Avoid infinite loop if you put unexisting uids as select will generate the same entries but with different reference\n        if (JSON.stringify(uids) === JSON.stringify(this.lastSelectedUids())) {\n            return;\n        }\n        this.lastSelectedUids.set(uids);\n        const selectedUids = this.collectSelectedUids(this.entries());\n        const uidToSelect = uids.filter(uid => selectedUids.every(selectedUid => selectedUid !== uid));\n        const uidToUnselect = selectedUids.filter(uid => uids.every(selectedUid => selectedUid !== uid));\n\n        if (!uidToSelect.length && !uidToUnselect.length) {\n            return;\n        }\n\n        let entries = this.entries();\n\n        if (this.multipleModeEnabled()) {\n            for (const uid of [...uidToSelect, ...uidToUnselect]) {\n                entries = NavSelectorMultiSelect.selectInMultipleMode(entries, uid);\n            }\n        } else {\n            entries = NavSelectorSingleSelect.select(entries, uids[0]);\n        }\n        this.entries.set(entries);\n    }\n\n    registerOnSelectedUidsChange(callback: (selectedUids: string[]) => void) {\n        this.selectedUidsChangeCallback = callback;\n    }\n\n    onArrowDown() {\n        this.entries.update(entries => NavSelectorAccessibility.focusNext(entries));\n    }\n\n    onArrowUp() {\n        this.entries.update(entries => NavSelectorAccessibility.focusPrevious(entries));\n    }\n\n    private collectSelectedUids(entries: InternalNavSelectorEntry[]): string[] {\n        return entries.flatMap(entry => {\n            if (isInternalNavSelectorEntryALeaf(entry)) {\n                if (entry.details.length) {\n                    return entry.details.flatMap(detail => (detail.selected ? [detail.uid] : []));\n                }\n                return entry.selected ? [entry.uid] : [];\n            } else if (isInternalNavSelectorEntryANode(entry)) {\n                return this.collectSelectedUids(entry.children);\n            }\n            return [];\n        });\n    }\n\n    updateLeafAction(leafActionProjection: ReadonlyArray<TemplateRef<unknown>>) {\n        this.leafActionProjection.set(leafActionProjection);\n    }\n\n    fold(entry: InternalNavSelectorEntry) {\n        this.entries.update(entries => NavSelectorFolding.fold(entries, entry.uid, !this.expanded()));\n    }\n\n    unfold(entry: InternalNavSelectorEntry) {\n        this.entries.update(entries => NavSelectorFolding.unfold(entries, entry.uid, !this.expanded()));\n    }\n\n    updateDetailsDisplayedLimit(detailsDisplayedLimit: number) {\n        this.entries.update(entries => NavSelectorViewMore.changeViewMoreDetailsDisplayedLimit(entries, detailsDisplayedLimit));\n    }\n\n    viewMore(leaf: InternalNavSelectorLeaf) {\n        this.entries.update(entries => NavSelectorViewMore.viewMore(entries, leaf.uid));\n    }\n\n    viewLess(leaf: InternalNavSelectorLeaf) {\n        this.entries.update(entries => NavSelectorViewMore.viewLess(entries, leaf.uid));\n    }\n}\n"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export * from './nav-selector.component';
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL2xpYnMvdWktY29tcG9uZW50cy9uYXYtc2VsZWN0b3Ivc3JjL3B1YmxpY19hcGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBUUEsY0FBYywwQkFBMEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7XG4gICAgTmF2U2VsZWN0b3JDYXRlZ29yeSxcbiAgICBOYXZTZWxlY3Rvckdyb3VwLFxuICAgIE5hdlNlbGVjdG9yTGVhZixcbiAgICBOYXZTZWxlY3RvckxlYWZEZXRhaWxzLFxuICAgIE5hdlNlbGVjdG9yVHJhbnNsYXRlZFRleHRzLFxuICAgIE5vZGVBY2Nlc3NpYmlsaXR5LFxufSBmcm9tICcuL25hdi1zZWxlY3Rvcic7XG5leHBvcnQgKiBmcm9tICcuL25hdi1zZWxlY3Rvci5jb21wb25lbnQnO1xuIl19
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export const isInternalNavSelectorEntryALeafFocusable = (leaf, { parentFolded, minified }) => {
|
|
2
|
+
if (parentFolded) {
|
|
3
|
+
return false;
|
|
4
|
+
}
|
|
5
|
+
if (minified) {
|
|
6
|
+
return !leaf.disabled && !leaf.displayTokenInvalid && !leaf.displayFeatureLocked;
|
|
7
|
+
}
|
|
8
|
+
return !leaf.disabled;
|
|
9
|
+
};
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGVhZi51dGlscy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL2xpYnMvdWktY29tcG9uZW50cy9uYXYtc2VsZWN0b3Ivc3JjL3V0aWxzL2xlYWYudXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsTUFBTSxDQUFDLE1BQU0sd0NBQXdDLEdBQUcsQ0FDcEQsSUFBNkIsRUFDN0IsRUFBRSxZQUFZLEVBQUUsUUFBUSxFQUFnRCxFQUNqRSxFQUFFO0lBQ1QsSUFBSSxZQUFZLEVBQUUsQ0FBQztRQUNmLE9BQU8sS0FBSyxDQUFDO0lBQ2pCLENBQUM7SUFDRCxJQUFJLFFBQVEsRUFBRSxDQUFDO1FBQ1gsT0FBTyxDQUFDLElBQUksQ0FBQyxRQUFRLElBQUksQ0FBQyxJQUFJLENBQUMsbUJBQW1CLElBQUksQ0FBQyxJQUFJLENBQUMsb0JBQW9CLENBQUM7SUFDckYsQ0FBQztJQUNELE9BQU8sQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDO0FBQzFCLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEludGVybmFsTmF2U2VsZWN0b3JMZWFmIH0gZnJvbSAnLi4vbmF2LXNlbGVjdG9yJztcblxuZXhwb3J0IGNvbnN0IGlzSW50ZXJuYWxOYXZTZWxlY3RvckVudHJ5QUxlYWZGb2N1c2FibGUgPSAoXG4gICAgbGVhZjogSW50ZXJuYWxOYXZTZWxlY3RvckxlYWYsXG4gICAgeyBwYXJlbnRGb2xkZWQsIG1pbmlmaWVkIH06IHsgcGFyZW50Rm9sZGVkOiBib29sZWFuOyBtaW5pZmllZDogYm9vbGVhbiB9XG4pOiBib29sZWFuID0+IHtcbiAgICBpZiAocGFyZW50Rm9sZGVkKSB7XG4gICAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG4gICAgaWYgKG1pbmlmaWVkKSB7XG4gICAgICAgIHJldHVybiAhbGVhZi5kaXNhYmxlZCAmJiAhbGVhZi5kaXNwbGF5VG9rZW5JbnZhbGlkICYmICFsZWFmLmRpc3BsYXlGZWF0dXJlTG9ja2VkO1xuICAgIH1cbiAgICByZXR1cm4gIWxlYWYuZGlzYWJsZWQ7XG59O1xuIl19
|
|
@@ -0,0 +1,171 @@
|
|
|
1
|
+
import { isInternalNavSelectorEntryACategory, isInternalNavSelectorEntryAGroup, isInternalNavSelectorEntryALeaf, isInternalNavSelectorEntryALeafDetails, } from '../nav-selector';
|
|
2
|
+
/**
|
|
3
|
+
* NavSelectorAccessibility is a utility class that provides methods to manage the accessibility and the focusability of nav selector entries.
|
|
4
|
+
*/
|
|
5
|
+
export class NavSelectorAccessibility {
|
|
6
|
+
/**
|
|
7
|
+
* Set tabIndex to 0 for the first focusable entry and -1 for all other entries
|
|
8
|
+
* @param entries nav selector entries
|
|
9
|
+
*/
|
|
10
|
+
static resetFocus(entries) {
|
|
11
|
+
return entries.reduce((acc, entry) => {
|
|
12
|
+
if (!acc.length) {
|
|
13
|
+
return [{ ...entry, accessibility: { ...entry.accessibility, tabIndex: entry.focusable ? 0 : -1 } }];
|
|
14
|
+
}
|
|
15
|
+
if (acc.every(({ focusable }) => !focusable)) {
|
|
16
|
+
return [...acc, { ...entry, accessibility: { ...entry.accessibility, tabIndex: entry.focusable ? 0 : -1 } }];
|
|
17
|
+
}
|
|
18
|
+
return [...acc, entry];
|
|
19
|
+
}, []);
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Focus the given entry and set tabIndex to 0 for the given uid and -1 for all other entries
|
|
23
|
+
* @param entries nav selector entries
|
|
24
|
+
* @param uid the uid of the entry to focus
|
|
25
|
+
*/
|
|
26
|
+
static focus(entries, uid) {
|
|
27
|
+
return entries.map(entry => {
|
|
28
|
+
if (isInternalNavSelectorEntryAGroup(entry) || isInternalNavSelectorEntryACategory(entry)) {
|
|
29
|
+
return {
|
|
30
|
+
...this.focusAnEntry(entry, uid),
|
|
31
|
+
children: NavSelectorAccessibility.focus(entry.children, uid),
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
if (isInternalNavSelectorEntryALeaf(entry)) {
|
|
35
|
+
return {
|
|
36
|
+
...this.focusAnEntry(entry, uid),
|
|
37
|
+
details: entry.details.map(detail => this.focusAnEntry(detail, uid)),
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
return this.focusAnEntry(entry, uid);
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Unfocus the given entry and set tabIndex to -1 for the given uid and 0 for all other entries
|
|
45
|
+
* @param entries nav selector entries
|
|
46
|
+
* @param uid the uid of the entry to unfocus
|
|
47
|
+
*/
|
|
48
|
+
static unfocus(entries, uid) {
|
|
49
|
+
return entries.map(entry => {
|
|
50
|
+
if (isInternalNavSelectorEntryAGroup(entry) || isInternalNavSelectorEntryACategory(entry)) {
|
|
51
|
+
return {
|
|
52
|
+
...this.unfocusAnEntry(entry, uid),
|
|
53
|
+
children: NavSelectorAccessibility.unfocus(entry.children, uid),
|
|
54
|
+
};
|
|
55
|
+
}
|
|
56
|
+
if (isInternalNavSelectorEntryALeaf(entry)) {
|
|
57
|
+
return {
|
|
58
|
+
...this.unfocusAnEntry(entry, uid),
|
|
59
|
+
details: entry.details.map(detail => this.unfocusAnEntry(detail, uid)),
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
return this.unfocusAnEntry(entry, uid);
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Focus the previous entry focusable and set tabIndex to 0 for the previous focusable entry and -1 for all other entries
|
|
67
|
+
*
|
|
68
|
+
* If not previous entries are focusable, entries will not be modified
|
|
69
|
+
*
|
|
70
|
+
* @param entries nav selector entries
|
|
71
|
+
*/
|
|
72
|
+
static focusPrevious(entries) {
|
|
73
|
+
const flattenEntries = this.flatten(entries);
|
|
74
|
+
const idxFocusedId = flattenEntries.findIndex(({ accessibility }) => accessibility.tabIndex === 0);
|
|
75
|
+
if (idxFocusedId === 0 || this.previousEntryIsFirstAndNotFocusable(flattenEntries, idxFocusedId)) {
|
|
76
|
+
return entries;
|
|
77
|
+
}
|
|
78
|
+
const focusedEntry = flattenEntries[idxFocusedId];
|
|
79
|
+
focusedEntry.accessibility.tabIndex = -1;
|
|
80
|
+
for (let i = idxFocusedId - 1; i >= 0; i--) {
|
|
81
|
+
const entry = flattenEntries[i];
|
|
82
|
+
if (entry.focusable) {
|
|
83
|
+
entry.accessibility.tabIndex = 0;
|
|
84
|
+
break;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
return this.unflatten(flattenEntries);
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Focus the next entry focusable and set tabIndex to 0 for the next focusable entry and -1 for all other entries
|
|
91
|
+
*
|
|
92
|
+
* If not next entries are focusable, entries will not be modified
|
|
93
|
+
*
|
|
94
|
+
* @param entries nav selector entries
|
|
95
|
+
*/
|
|
96
|
+
static focusNext(entries) {
|
|
97
|
+
const flattenEntries = this.flatten(entries);
|
|
98
|
+
const focusableFlattenEntries = flattenEntries.filter(entry => entry.focusable);
|
|
99
|
+
const idxFocusedId = focusableFlattenEntries.findIndex(({ accessibility }) => accessibility.tabIndex === 0);
|
|
100
|
+
if (focusableFlattenEntries.length - 1 <= idxFocusedId) {
|
|
101
|
+
return entries;
|
|
102
|
+
}
|
|
103
|
+
focusableFlattenEntries[idxFocusedId].accessibility.tabIndex = -1;
|
|
104
|
+
focusableFlattenEntries[idxFocusedId + 1].accessibility.tabIndex = 0;
|
|
105
|
+
return this.unflatten(flattenEntries);
|
|
106
|
+
}
|
|
107
|
+
static previousEntryIsFirstAndNotFocusable(flattenEntries, idxFocusedId) {
|
|
108
|
+
if (idxFocusedId - 1 === 0) {
|
|
109
|
+
const previous = flattenEntries[0];
|
|
110
|
+
return !previous.focusable;
|
|
111
|
+
}
|
|
112
|
+
return false;
|
|
113
|
+
}
|
|
114
|
+
static focusAnEntry(entry, uid) {
|
|
115
|
+
return { ...entry, accessibility: { ...entry.accessibility, tabIndex: entry.uid === uid ? 0 : -1 } };
|
|
116
|
+
}
|
|
117
|
+
static unfocusAnEntry(entry, uid) {
|
|
118
|
+
return { ...entry, accessibility: { ...entry.accessibility, tabIndex: entry.uid === uid ? -1 : entry.accessibility.tabIndex } };
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
* Flatten the nav selector entries. It is easier to work with flat entries for the focus management as we need to know which is the previous or the next entry
|
|
122
|
+
* @param entries nav selector entries
|
|
123
|
+
*/
|
|
124
|
+
static flatten(entries) {
|
|
125
|
+
return entries.flatMap(entry => {
|
|
126
|
+
if (isInternalNavSelectorEntryAGroup(entry) || isInternalNavSelectorEntryACategory(entry)) {
|
|
127
|
+
return [{ ...entry, children: [] }, ...this.flatten(entry.children)];
|
|
128
|
+
}
|
|
129
|
+
if (isInternalNavSelectorEntryALeaf(entry)) {
|
|
130
|
+
return [{ ...entry, details: [] }, ...entry.details.map(detail => ({ ...detail }))];
|
|
131
|
+
}
|
|
132
|
+
return [{ ...entry }];
|
|
133
|
+
});
|
|
134
|
+
}
|
|
135
|
+
/**
|
|
136
|
+
* Unflatten the nav selector entries
|
|
137
|
+
* @param entries nav selector entries
|
|
138
|
+
*/
|
|
139
|
+
static unflatten(entries) {
|
|
140
|
+
return entries.reduceRight((acc, entry) => {
|
|
141
|
+
if (!acc.length) {
|
|
142
|
+
return [entry];
|
|
143
|
+
}
|
|
144
|
+
if (entry.accessibility.ariaLevel < acc[0].accessibility.ariaLevel) {
|
|
145
|
+
if (isInternalNavSelectorEntryALeaf(entry)) {
|
|
146
|
+
return [
|
|
147
|
+
{
|
|
148
|
+
...entry,
|
|
149
|
+
details: acc
|
|
150
|
+
.filter(it => isInternalNavSelectorEntryALeafDetails(it))
|
|
151
|
+
// Force destroying the reference to the original object else Angular will not detect the change
|
|
152
|
+
.map(it => ({ ...it })),
|
|
153
|
+
},
|
|
154
|
+
...acc.filter(it => !isInternalNavSelectorEntryALeafDetails(it)),
|
|
155
|
+
];
|
|
156
|
+
}
|
|
157
|
+
else if (isInternalNavSelectorEntryAGroup(entry) || isInternalNavSelectorEntryACategory(entry)) {
|
|
158
|
+
return [
|
|
159
|
+
{
|
|
160
|
+
...entry,
|
|
161
|
+
children: acc.filter(({ accessibility }) => accessibility.ariaLevel > entry.accessibility.ariaLevel),
|
|
162
|
+
},
|
|
163
|
+
...acc.filter(({ accessibility }) => accessibility.ariaLevel <= entry.accessibility.ariaLevel),
|
|
164
|
+
];
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
return [entry, ...acc];
|
|
168
|
+
}, []);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nav-selector.accessibility.js","sourceRoot":"","sources":["../../../../../libs/ui-components/nav-selector/src/utils/nav-selector.accessibility.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    InternalNavSelectorEntry,\n    InternalNavSelectorGroup,\n    InternalNavSelectorLeaf,\n    InternalNavSelectorLeafDetails,\n    isInternalNavSelectorEntryACategory,\n    isInternalNavSelectorEntryAGroup,\n    isInternalNavSelectorEntryALeaf,\n    isInternalNavSelectorEntryALeafDetails,\n} from '../nav-selector';\n\n/**\n * NavSelectorAccessibility is a utility class that provides methods to manage the accessibility and the focusability of nav selector entries.\n */\nexport class NavSelectorAccessibility {\n    /**\n     * Set tabIndex to 0 for the first focusable entry and -1 for all other entries\n     * @param entries nav selector entries\n     */\n    static resetFocus(entries: InternalNavSelectorEntry[]): InternalNavSelectorEntry[] {\n        return entries.reduce((acc, entry) => {\n            if (!acc.length) {\n                return [{ ...entry, accessibility: { ...entry.accessibility, tabIndex: entry.focusable ? 0 : -1 } }];\n            }\n\n            if (acc.every(({ focusable }) => !focusable)) {\n                return [...acc, { ...entry, accessibility: { ...entry.accessibility, tabIndex: entry.focusable ? 0 : -1 } }];\n            }\n\n            return [...acc, entry];\n        }, [] as InternalNavSelectorEntry[]);\n    }\n\n    /**\n     * Focus the given entry and set tabIndex to 0 for the given uid and -1 for all other entries\n     * @param entries nav selector entries\n     * @param uid the uid of the entry to focus\n     */\n    static focus<T extends InternalNavSelectorEntry>(entries: T[], uid: string): InternalNavSelectorEntry[] {\n        return entries.map(entry => {\n            if (isInternalNavSelectorEntryAGroup(entry) || isInternalNavSelectorEntryACategory(entry)) {\n                return {\n                    ...this.focusAnEntry(entry, uid),\n                    children: NavSelectorAccessibility.focus(entry.children, uid),\n                } as InternalNavSelectorGroup;\n            }\n            if (isInternalNavSelectorEntryALeaf(entry)) {\n                return {\n                    ...this.focusAnEntry(entry, uid),\n                    details: entry.details.map(detail => this.focusAnEntry(detail, uid)),\n                } as InternalNavSelectorLeaf;\n            }\n            return this.focusAnEntry(entry, uid);\n        });\n    }\n\n    /**\n     * Unfocus the given entry and set tabIndex to -1 for the given uid and 0 for all other entries\n     * @param entries nav selector entries\n     * @param uid the uid of the entry to unfocus\n     */\n    static unfocus(entries: InternalNavSelectorEntry[], uid: string): InternalNavSelectorEntry[] {\n        return entries.map(entry => {\n            if (isInternalNavSelectorEntryAGroup(entry) || isInternalNavSelectorEntryACategory(entry)) {\n                return {\n                    ...this.unfocusAnEntry(entry, uid),\n                    children: NavSelectorAccessibility.unfocus(entry.children, uid),\n                } as InternalNavSelectorGroup;\n            }\n            if (isInternalNavSelectorEntryALeaf(entry)) {\n                return {\n                    ...this.unfocusAnEntry(entry, uid),\n                    details: entry.details.map(detail => this.unfocusAnEntry(detail, uid)),\n                } as InternalNavSelectorLeaf;\n            }\n            return this.unfocusAnEntry(entry, uid);\n        });\n    }\n\n    /**\n     * Focus the previous entry focusable and set tabIndex to 0 for the previous focusable entry and -1 for all other entries\n     *\n     * If not previous entries are focusable, entries will not be modified\n     *\n     * @param entries nav selector entries\n     */\n    static focusPrevious<T extends InternalNavSelectorEntry>(entries: T[]): InternalNavSelectorEntry[] {\n        const flattenEntries = this.flatten(entries);\n\n        const idxFocusedId = flattenEntries.findIndex(({ accessibility }) => accessibility.tabIndex === 0);\n\n        if (idxFocusedId === 0 || this.previousEntryIsFirstAndNotFocusable(flattenEntries, idxFocusedId)) {\n            return entries;\n        }\n\n        const focusedEntry = flattenEntries[idxFocusedId];\n        focusedEntry.accessibility.tabIndex = -1;\n\n        for (let i = idxFocusedId - 1; i >= 0; i--) {\n            const entry = flattenEntries[i];\n            if (entry.focusable) {\n                entry.accessibility.tabIndex = 0;\n                break;\n            }\n        }\n\n        return this.unflatten(flattenEntries);\n    }\n\n    /**\n     * Focus the next entry focusable and set tabIndex to 0 for the next focusable entry and -1 for all other entries\n     *\n     * If not next entries are focusable, entries will not be modified\n     *\n     * @param entries nav selector entries\n     */\n    static focusNext<T extends InternalNavSelectorEntry>(entries: T[]): InternalNavSelectorEntry[] {\n        const flattenEntries = this.flatten(entries);\n\n        const focusableFlattenEntries = flattenEntries.filter(entry => entry.focusable);\n\n        const idxFocusedId = focusableFlattenEntries.findIndex(({ accessibility }) => accessibility.tabIndex === 0);\n\n        if (focusableFlattenEntries.length - 1 <= idxFocusedId) {\n            return entries;\n        }\n\n        focusableFlattenEntries[idxFocusedId].accessibility.tabIndex = -1;\n\n        focusableFlattenEntries[idxFocusedId + 1].accessibility.tabIndex = 0;\n\n        return this.unflatten(flattenEntries);\n    }\n\n    private static previousEntryIsFirstAndNotFocusable(flattenEntries: InternalNavSelectorEntry[], idxFocusedId: number): boolean {\n        if (idxFocusedId - 1 === 0) {\n            const previous = flattenEntries[0];\n            return !previous.focusable;\n        }\n        return false;\n    }\n\n    private static focusAnEntry(entry: InternalNavSelectorEntry, uid: string): InternalNavSelectorEntry {\n        return { ...entry, accessibility: { ...entry.accessibility, tabIndex: entry.uid === uid ? 0 : -1 } };\n    }\n\n    private static unfocusAnEntry(entry: InternalNavSelectorEntry, uid: string): InternalNavSelectorEntry {\n        return { ...entry, accessibility: { ...entry.accessibility, tabIndex: entry.uid === uid ? -1 : entry.accessibility.tabIndex } };\n    }\n\n    /**\n     * Flatten the nav selector entries. It is easier to work with flat entries for the focus management as we need to know which is the previous or the next entry\n     * @param entries nav selector entries\n     */\n    private static flatten(entries: InternalNavSelectorEntry[]): InternalNavSelectorEntry[] {\n        return entries.flatMap(entry => {\n            if (isInternalNavSelectorEntryAGroup(entry) || isInternalNavSelectorEntryACategory(entry)) {\n                return [{ ...entry, children: [] }, ...this.flatten(entry.children)];\n            }\n            if (isInternalNavSelectorEntryALeaf(entry)) {\n                return [{ ...entry, details: [] }, ...entry.details.map(detail => ({ ...detail }))];\n            }\n            return [{ ...entry }];\n        });\n    }\n\n    /**\n     * Unflatten the nav selector entries\n     * @param entries nav selector entries\n     */\n    private static unflatten<T extends InternalNavSelectorEntry>(entries: T[]): InternalNavSelectorEntry[] {\n        return entries.reduceRight((acc, entry) => {\n            if (!acc.length) {\n                return [entry];\n            }\n            if (entry.accessibility.ariaLevel < acc[0].accessibility.ariaLevel) {\n                if (isInternalNavSelectorEntryALeaf(entry)) {\n                    return [\n                        {\n                            ...entry,\n                            details: acc\n                                .filter(it => isInternalNavSelectorEntryALeafDetails(it))\n                                // Force destroying the reference to the original object else Angular will not detect the change\n                                .map(it => ({ ...it })) as InternalNavSelectorLeafDetails[],\n                        },\n                        ...acc.filter(it => !isInternalNavSelectorEntryALeafDetails(it)),\n                    ];\n                } else if (isInternalNavSelectorEntryAGroup(entry) || isInternalNavSelectorEntryACategory(entry)) {\n                    return [\n                        {\n                            ...entry,\n                            children: acc.filter(\n                                ({ accessibility }) => accessibility.ariaLevel > entry.accessibility.ariaLevel\n                            ) as InternalNavSelectorLeaf[],\n                        },\n                        ...acc.filter(({ accessibility }) => accessibility.ariaLevel <= entry.accessibility.ariaLevel),\n                    ];\n                }\n            }\n            return [entry, ...acc];\n        }, [] as InternalNavSelectorEntry[]);\n    }\n}\n"]}
|