@angular/aria 21.0.2 → 21.0.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (81) hide show
  1. package/fesm2022/_accordion-chunk.mjs +108 -0
  2. package/fesm2022/_accordion-chunk.mjs.map +1 -0
  3. package/fesm2022/_combobox-chunk.mjs +425 -0
  4. package/fesm2022/_combobox-chunk.mjs.map +1 -0
  5. package/fesm2022/_combobox-listbox-chunk.mjs +240 -0
  6. package/fesm2022/_combobox-listbox-chunk.mjs.map +1 -0
  7. package/fesm2022/_combobox-tree-chunk.mjs +331 -0
  8. package/fesm2022/_combobox-tree-chunk.mjs.map +1 -0
  9. package/fesm2022/_deferred-content-chunk.mjs +124 -0
  10. package/fesm2022/_deferred-content-chunk.mjs.map +1 -0
  11. package/fesm2022/_expansion-chunk.mjs +41 -0
  12. package/fesm2022/_expansion-chunk.mjs.map +1 -0
  13. package/fesm2022/_list-chunk.mjs +287 -0
  14. package/fesm2022/_list-chunk.mjs.map +1 -0
  15. package/fesm2022/_list-navigation-chunk.mjs +116 -0
  16. package/fesm2022/_list-navigation-chunk.mjs.map +1 -0
  17. package/fesm2022/_menu-chunk.mjs +515 -0
  18. package/fesm2022/_menu-chunk.mjs.map +1 -0
  19. package/fesm2022/_pointer-event-manager-chunk.mjs +54 -0
  20. package/fesm2022/_pointer-event-manager-chunk.mjs.map +1 -0
  21. package/fesm2022/_signal-like-chunk.mjs +118 -0
  22. package/fesm2022/_signal-like-chunk.mjs.map +1 -0
  23. package/fesm2022/_tabs-chunk.mjs +159 -0
  24. package/fesm2022/_tabs-chunk.mjs.map +1 -0
  25. package/fesm2022/_toolbar-widget-group-chunk.mjs +148 -0
  26. package/fesm2022/_toolbar-widget-group-chunk.mjs.map +1 -0
  27. package/fesm2022/_widget-chunk.mjs +5 -246
  28. package/fesm2022/_widget-chunk.mjs.map +1 -1
  29. package/fesm2022/accordion.mjs +73 -55
  30. package/fesm2022/accordion.mjs.map +1 -1
  31. package/fesm2022/aria.mjs +1 -1
  32. package/fesm2022/aria.mjs.map +1 -1
  33. package/fesm2022/combobox.mjs +166 -153
  34. package/fesm2022/combobox.mjs.map +1 -1
  35. package/fesm2022/grid.mjs +287 -261
  36. package/fesm2022/grid.mjs.map +1 -1
  37. package/fesm2022/listbox.mjs +211 -197
  38. package/fesm2022/listbox.mjs.map +1 -1
  39. package/fesm2022/menu.mjs +308 -286
  40. package/fesm2022/menu.mjs.map +1 -1
  41. package/fesm2022/private.mjs +15 -2329
  42. package/fesm2022/private.mjs.map +1 -1
  43. package/fesm2022/tabs.mjs +221 -199
  44. package/fesm2022/tabs.mjs.map +1 -1
  45. package/fesm2022/toolbar.mjs +64 -48
  46. package/fesm2022/toolbar.mjs.map +1 -1
  47. package/fesm2022/tree.mjs +54 -44
  48. package/fesm2022/tree.mjs.map +1 -1
  49. package/package.json +2 -2
  50. package/types/_accordion-chunk.d.ts +100 -0
  51. package/types/_combobox-chunk.d.ts +194 -0
  52. package/types/_deferred-content-chunk.d.ts +42 -0
  53. package/types/_expansion-chunk.d.ts +40 -0
  54. package/types/_grid-chunk.d.ts +43 -250
  55. package/types/_keyboard-event-manager-chunk.d.ts +68 -0
  56. package/types/_list-chunk.d.ts +211 -0
  57. package/types/_list-navigation-chunk.d.ts +119 -0
  58. package/types/_listbox-chunk.d.ts +107 -0
  59. package/types/_menu-chunk.d.ts +267 -0
  60. package/types/_pointer-event-manager-chunk.d.ts +34 -0
  61. package/types/_tabs-chunk.d.ts +153 -0
  62. package/types/_toolbar-chunk.d.ts +124 -0
  63. package/types/_tree-chunk.d.ts +185 -0
  64. package/types/accordion.d.ts +65 -54
  65. package/types/aria.d.ts +1 -1
  66. package/types/combobox.d.ts +86 -56
  67. package/types/grid.d.ts +47 -32
  68. package/types/listbox.d.ts +22 -7
  69. package/types/menu.d.ts +135 -117
  70. package/types/private.d.ts +28 -1376
  71. package/types/tabs.d.ts +109 -88
  72. package/types/toolbar.d.ts +77 -66
  73. package/types/tree.d.ts +118 -104
  74. package/_adev_assets/aria-accordion.json +0 -743
  75. package/_adev_assets/aria-combobox.json +0 -603
  76. package/_adev_assets/aria-grid.json +0 -893
  77. package/_adev_assets/aria-listbox.json +0 -540
  78. package/_adev_assets/aria-menu.json +0 -1049
  79. package/_adev_assets/aria-tabs.json +0 -880
  80. package/_adev_assets/aria-toolbar.json +0 -545
  81. package/_adev_assets/aria-tree.json +0 -853
@@ -0,0 +1,240 @@
1
+ import { computed, signal, Modifier, KeyboardEventManager } from './_signal-like-chunk.mjs';
2
+ import { List } from './_list-chunk.mjs';
3
+ import { PointerEventManager } from './_pointer-event-manager-chunk.mjs';
4
+
5
+ class ListboxPattern {
6
+ inputs;
7
+ listBehavior;
8
+ orientation;
9
+ disabled = computed(() => this.listBehavior.disabled());
10
+ readonly;
11
+ tabIndex = computed(() => this.listBehavior.tabIndex());
12
+ activeDescendant = computed(() => this.listBehavior.activeDescendant());
13
+ multi;
14
+ setsize = computed(() => this.inputs.items().length);
15
+ followFocus = computed(() => this.inputs.selectionMode() === 'follow');
16
+ wrap = signal(true);
17
+ prevKey = computed(() => {
18
+ if (this.inputs.orientation() === 'vertical') {
19
+ return 'ArrowUp';
20
+ }
21
+ return this.inputs.textDirection() === 'rtl' ? 'ArrowRight' : 'ArrowLeft';
22
+ });
23
+ nextKey = computed(() => {
24
+ if (this.inputs.orientation() === 'vertical') {
25
+ return 'ArrowDown';
26
+ }
27
+ return this.inputs.textDirection() === 'rtl' ? 'ArrowLeft' : 'ArrowRight';
28
+ });
29
+ dynamicSpaceKey = computed(() => this.listBehavior.isTyping() ? '' : ' ');
30
+ typeaheadRegexp = /^.$/;
31
+ keydown = computed(() => {
32
+ const manager = new KeyboardEventManager();
33
+ if (this.readonly()) {
34
+ return manager.on(this.prevKey, () => this.listBehavior.prev()).on(this.nextKey, () => this.listBehavior.next()).on('Home', () => this.listBehavior.first()).on('End', () => this.listBehavior.last()).on(this.typeaheadRegexp, e => this.listBehavior.search(e.key));
35
+ }
36
+ if (!this.followFocus()) {
37
+ manager.on(this.prevKey, () => this.listBehavior.prev()).on(this.nextKey, () => this.listBehavior.next()).on('Home', () => this.listBehavior.first()).on('End', () => this.listBehavior.last()).on(this.typeaheadRegexp, e => this.listBehavior.search(e.key));
38
+ }
39
+ if (this.followFocus()) {
40
+ manager.on(this.prevKey, () => this.listBehavior.prev({
41
+ selectOne: true
42
+ })).on(this.nextKey, () => this.listBehavior.next({
43
+ selectOne: true
44
+ })).on('Home', () => this.listBehavior.first({
45
+ selectOne: true
46
+ })).on('End', () => this.listBehavior.last({
47
+ selectOne: true
48
+ })).on(this.typeaheadRegexp, e => this.listBehavior.search(e.key, {
49
+ selectOne: true
50
+ }));
51
+ }
52
+ if (this.inputs.multi()) {
53
+ manager.on(Modifier.Any, 'Shift', () => this.listBehavior.anchor(this.listBehavior.activeIndex())).on(Modifier.Shift, this.prevKey, () => this.listBehavior.prev({
54
+ selectRange: true
55
+ })).on(Modifier.Shift, this.nextKey, () => this.listBehavior.next({
56
+ selectRange: true
57
+ })).on([Modifier.Ctrl | Modifier.Shift, Modifier.Meta | Modifier.Shift], 'Home', () => this.listBehavior.first({
58
+ selectRange: true,
59
+ anchor: false
60
+ })).on([Modifier.Ctrl | Modifier.Shift, Modifier.Meta | Modifier.Shift], 'End', () => this.listBehavior.last({
61
+ selectRange: true,
62
+ anchor: false
63
+ })).on(Modifier.Shift, 'Enter', () => this.listBehavior.updateSelection({
64
+ selectRange: true,
65
+ anchor: false
66
+ })).on(Modifier.Shift, this.dynamicSpaceKey, () => this.listBehavior.updateSelection({
67
+ selectRange: true,
68
+ anchor: false
69
+ }));
70
+ }
71
+ if (!this.followFocus() && this.inputs.multi()) {
72
+ manager.on(this.dynamicSpaceKey, () => this.listBehavior.toggle()).on('Enter', () => this.listBehavior.toggle()).on([Modifier.Ctrl, Modifier.Meta], 'A', () => this.listBehavior.toggleAll());
73
+ }
74
+ if (!this.followFocus() && !this.inputs.multi()) {
75
+ manager.on(this.dynamicSpaceKey, () => this.listBehavior.toggleOne());
76
+ manager.on('Enter', () => this.listBehavior.toggleOne());
77
+ }
78
+ if (this.inputs.multi() && this.followFocus()) {
79
+ manager.on([Modifier.Ctrl, Modifier.Meta], this.prevKey, () => this.listBehavior.prev()).on([Modifier.Ctrl, Modifier.Meta], this.nextKey, () => this.listBehavior.next()).on([Modifier.Ctrl, Modifier.Meta], ' ', () => this.listBehavior.toggle()).on([Modifier.Ctrl, Modifier.Meta], 'Enter', () => this.listBehavior.toggle()).on([Modifier.Ctrl, Modifier.Meta], 'Home', () => this.listBehavior.first()).on([Modifier.Ctrl, Modifier.Meta], 'End', () => this.listBehavior.last()).on([Modifier.Ctrl, Modifier.Meta], 'A', () => {
80
+ this.listBehavior.toggleAll();
81
+ this.listBehavior.select();
82
+ });
83
+ }
84
+ return manager;
85
+ });
86
+ pointerdown = computed(() => {
87
+ const manager = new PointerEventManager();
88
+ if (this.readonly()) {
89
+ return manager.on(e => this.listBehavior.goto(this._getItem(e)));
90
+ }
91
+ if (this.multi()) {
92
+ manager.on(Modifier.Shift, e => this.listBehavior.goto(this._getItem(e), {
93
+ selectRange: true
94
+ }));
95
+ }
96
+ if (!this.multi() && this.followFocus()) {
97
+ return manager.on(e => this.listBehavior.goto(this._getItem(e), {
98
+ selectOne: true
99
+ }));
100
+ }
101
+ if (!this.multi() && !this.followFocus()) {
102
+ return manager.on(e => this.listBehavior.goto(this._getItem(e), {
103
+ toggle: true
104
+ }));
105
+ }
106
+ if (this.multi() && this.followFocus()) {
107
+ return manager.on(e => this.listBehavior.goto(this._getItem(e), {
108
+ selectOne: true
109
+ })).on(Modifier.Ctrl, e => this.listBehavior.goto(this._getItem(e), {
110
+ toggle: true
111
+ }));
112
+ }
113
+ if (this.multi() && !this.followFocus()) {
114
+ return manager.on(e => this.listBehavior.goto(this._getItem(e), {
115
+ toggle: true
116
+ }));
117
+ }
118
+ return manager;
119
+ });
120
+ constructor(inputs) {
121
+ this.inputs = inputs;
122
+ this.readonly = inputs.readonly;
123
+ this.orientation = inputs.orientation;
124
+ this.multi = inputs.multi;
125
+ this.listBehavior = new List(inputs);
126
+ }
127
+ validate() {
128
+ const violations = [];
129
+ if (!this.inputs.multi() && this.inputs.values().length > 1) {
130
+ violations.push(`A single-select listbox should not have multiple selected options. Selected options: ${this.inputs.values().join(', ')}`);
131
+ }
132
+ return violations;
133
+ }
134
+ onKeydown(event) {
135
+ if (!this.disabled()) {
136
+ this.keydown().handle(event);
137
+ }
138
+ }
139
+ onPointerdown(event) {
140
+ if (!this.disabled()) {
141
+ this.pointerdown().handle(event);
142
+ }
143
+ }
144
+ setDefaultState() {
145
+ let firstItem = null;
146
+ for (const item of this.inputs.items()) {
147
+ if (this.listBehavior.isFocusable(item)) {
148
+ if (!firstItem) {
149
+ firstItem = item;
150
+ }
151
+ if (item.selected()) {
152
+ this.inputs.activeItem.set(item);
153
+ return;
154
+ }
155
+ }
156
+ }
157
+ if (firstItem) {
158
+ this.inputs.activeItem.set(firstItem);
159
+ }
160
+ }
161
+ _getItem(e) {
162
+ if (!(e.target instanceof HTMLElement)) {
163
+ return;
164
+ }
165
+ const element = e.target.closest('[role="option"]');
166
+ return this.inputs.items().find(i => i.element() === element);
167
+ }
168
+ }
169
+
170
+ class OptionPattern {
171
+ id;
172
+ value;
173
+ index = computed(() => this.listbox()?.inputs.items().indexOf(this) ?? -1);
174
+ active = computed(() => this.listbox()?.inputs.activeItem() === this);
175
+ selected = computed(() => this.listbox()?.inputs.values().includes(this.value()));
176
+ selectable = () => true;
177
+ disabled;
178
+ searchTerm;
179
+ listbox;
180
+ tabIndex = computed(() => this.listbox()?.listBehavior.getItemTabindex(this));
181
+ element;
182
+ constructor(args) {
183
+ this.id = args.id;
184
+ this.value = args.value;
185
+ this.listbox = args.listbox;
186
+ this.element = args.element;
187
+ this.disabled = args.disabled;
188
+ this.searchTerm = args.searchTerm;
189
+ }
190
+ }
191
+
192
+ class ComboboxListboxPattern extends ListboxPattern {
193
+ inputs;
194
+ id = computed(() => this.inputs.id());
195
+ role = computed(() => 'listbox');
196
+ activeId = computed(() => this.listBehavior.activeDescendant());
197
+ items = computed(() => this.inputs.items());
198
+ tabIndex = () => -1;
199
+ multi = computed(() => {
200
+ return this.inputs.combobox()?.readonly() ? this.inputs.multi() : false;
201
+ });
202
+ constructor(inputs) {
203
+ if (inputs.combobox()) {
204
+ inputs.focusMode = () => 'activedescendant';
205
+ inputs.element = inputs.combobox().inputs.inputEl;
206
+ }
207
+ super(inputs);
208
+ this.inputs = inputs;
209
+ }
210
+ onKeydown(_) {}
211
+ onPointerdown(_) {}
212
+ setDefaultState() {}
213
+ focus = (item, opts) => {
214
+ this.listBehavior.goto(item, opts);
215
+ };
216
+ getActiveItem = () => this.inputs.activeItem();
217
+ next = () => this.listBehavior.next();
218
+ prev = () => this.listBehavior.prev();
219
+ last = () => this.listBehavior.last();
220
+ first = () => this.listBehavior.first();
221
+ unfocus = () => this.listBehavior.unfocus();
222
+ select = item => this.listBehavior.select(item);
223
+ toggle = item => this.listBehavior.toggle(item);
224
+ clearSelection = () => this.listBehavior.deselectAll();
225
+ getItem = e => this._getItem(e);
226
+ getSelectedItems = () => {
227
+ const items = [];
228
+ for (const value of this.inputs.values()) {
229
+ const item = this.items().find(i => i.value() === value);
230
+ if (item) {
231
+ items.push(item);
232
+ }
233
+ }
234
+ return items;
235
+ };
236
+ setValue = value => this.inputs.values.set(value ? [value] : []);
237
+ }
238
+
239
+ export { ComboboxListboxPattern, ListboxPattern, OptionPattern };
240
+ //# sourceMappingURL=_combobox-listbox-chunk.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"_combobox-listbox-chunk.mjs","sources":["../../../../../darwin_arm64-fastbuild-ST-fdfa778d11ba/bin/src/aria/private/listbox/listbox.ts","../../../../../darwin_arm64-fastbuild-ST-fdfa778d11ba/bin/src/aria/private/listbox/option.ts","../../../../../darwin_arm64-fastbuild-ST-fdfa778d11ba/bin/src/aria/private/listbox/combobox-listbox.ts"],"sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.dev/license\n */\n\nimport {OptionPattern} from './option';\nimport {KeyboardEventManager, PointerEventManager, Modifier} from '../behaviors/event-manager';\nimport {computed, signal, SignalLike} from '../behaviors/signal-like/signal-like';\nimport {List, ListInputs} from '../behaviors/list/list';\n\n/** Represents the required inputs for a listbox. */\nexport type ListboxInputs<V> = ListInputs<OptionPattern<V>, V> & {\n /** A unique identifier for the listbox. */\n id: SignalLike<string>;\n\n /** Whether the listbox is readonly. */\n readonly: SignalLike<boolean>;\n};\n\n/** Controls the state of a listbox. */\nexport class ListboxPattern<V> {\n listBehavior: List<OptionPattern<V>, V>;\n\n /** Whether the list is vertically or horizontally oriented. */\n orientation: SignalLike<'vertical' | 'horizontal'>;\n\n /** Whether the listbox is disabled. */\n disabled = computed(() => this.listBehavior.disabled());\n\n /** Whether the listbox is readonly. */\n readonly: SignalLike<boolean>;\n\n /** The tab index of the listbox. */\n tabIndex: SignalLike<-1 | 0> = computed(() => this.listBehavior.tabIndex());\n\n /** The id of the current active item. */\n activeDescendant = computed(() => this.listBehavior.activeDescendant());\n\n /** Whether multiple items in the list can be selected at once. */\n multi: SignalLike<boolean>;\n\n /** The number of items in the listbox. */\n setsize = computed(() => this.inputs.items().length);\n\n /** Whether the listbox selection follows focus. */\n followFocus = computed(() => this.inputs.selectionMode() === 'follow');\n\n /** Whether the listbox should wrap. Used to disable wrapping while range selecting. */\n wrap = signal(true);\n\n /** The key used to navigate to the previous item in the list. */\n prevKey = computed(() => {\n if (this.inputs.orientation() === 'vertical') {\n return 'ArrowUp';\n }\n return this.inputs.textDirection() === 'rtl' ? 'ArrowRight' : 'ArrowLeft';\n });\n\n /** The key used to navigate to the next item in the list. */\n nextKey = computed(() => {\n if (this.inputs.orientation() === 'vertical') {\n return 'ArrowDown';\n }\n return this.inputs.textDirection() === 'rtl' ? 'ArrowLeft' : 'ArrowRight';\n });\n\n /** Represents the space key. Does nothing when the user is actively using typeahead. */\n dynamicSpaceKey = computed(() => (this.listBehavior.isTyping() ? '' : ' '));\n\n /** The regexp used to decide if a key should trigger typeahead. */\n typeaheadRegexp = /^.$/;\n\n /** The keydown event manager for the listbox. */\n keydown = computed(() => {\n const manager = new KeyboardEventManager();\n\n if (this.readonly()) {\n return manager\n .on(this.prevKey, () => this.listBehavior.prev())\n .on(this.nextKey, () => this.listBehavior.next())\n .on('Home', () => this.listBehavior.first())\n .on('End', () => this.listBehavior.last())\n .on(this.typeaheadRegexp, e => this.listBehavior.search(e.key));\n }\n\n if (!this.followFocus()) {\n manager\n .on(this.prevKey, () => this.listBehavior.prev())\n .on(this.nextKey, () => this.listBehavior.next())\n .on('Home', () => this.listBehavior.first())\n .on('End', () => this.listBehavior.last())\n .on(this.typeaheadRegexp, e => this.listBehavior.search(e.key));\n }\n\n if (this.followFocus()) {\n manager\n .on(this.prevKey, () => this.listBehavior.prev({selectOne: true}))\n .on(this.nextKey, () => this.listBehavior.next({selectOne: true}))\n .on('Home', () => this.listBehavior.first({selectOne: true}))\n .on('End', () => this.listBehavior.last({selectOne: true}))\n .on(this.typeaheadRegexp, e => this.listBehavior.search(e.key, {selectOne: true}));\n }\n\n if (this.inputs.multi()) {\n manager\n .on(Modifier.Any, 'Shift', () => this.listBehavior.anchor(this.listBehavior.activeIndex()))\n .on(Modifier.Shift, this.prevKey, () => this.listBehavior.prev({selectRange: true}))\n .on(Modifier.Shift, this.nextKey, () => this.listBehavior.next({selectRange: true}))\n .on([Modifier.Ctrl | Modifier.Shift, Modifier.Meta | Modifier.Shift], 'Home', () =>\n this.listBehavior.first({selectRange: true, anchor: false}),\n )\n .on([Modifier.Ctrl | Modifier.Shift, Modifier.Meta | Modifier.Shift], 'End', () =>\n this.listBehavior.last({selectRange: true, anchor: false}),\n )\n .on(Modifier.Shift, 'Enter', () =>\n this.listBehavior.updateSelection({selectRange: true, anchor: false}),\n )\n .on(Modifier.Shift, this.dynamicSpaceKey, () =>\n this.listBehavior.updateSelection({selectRange: true, anchor: false}),\n );\n }\n\n if (!this.followFocus() && this.inputs.multi()) {\n manager\n .on(this.dynamicSpaceKey, () => this.listBehavior.toggle())\n .on('Enter', () => this.listBehavior.toggle())\n .on([Modifier.Ctrl, Modifier.Meta], 'A', () => this.listBehavior.toggleAll());\n }\n\n if (!this.followFocus() && !this.inputs.multi()) {\n manager.on(this.dynamicSpaceKey, () => this.listBehavior.toggleOne());\n manager.on('Enter', () => this.listBehavior.toggleOne());\n }\n\n if (this.inputs.multi() && this.followFocus()) {\n manager\n .on([Modifier.Ctrl, Modifier.Meta], this.prevKey, () => this.listBehavior.prev())\n .on([Modifier.Ctrl, Modifier.Meta], this.nextKey, () => this.listBehavior.next())\n .on([Modifier.Ctrl, Modifier.Meta], ' ', () => this.listBehavior.toggle())\n .on([Modifier.Ctrl, Modifier.Meta], 'Enter', () => this.listBehavior.toggle())\n .on([Modifier.Ctrl, Modifier.Meta], 'Home', () => this.listBehavior.first())\n .on([Modifier.Ctrl, Modifier.Meta], 'End', () => this.listBehavior.last())\n .on([Modifier.Ctrl, Modifier.Meta], 'A', () => {\n this.listBehavior.toggleAll();\n this.listBehavior.select(); // Ensure the currect option remains selected.\n });\n }\n\n return manager;\n });\n\n /** The pointerdown event manager for the listbox. */\n pointerdown = computed(() => {\n const manager = new PointerEventManager();\n\n if (this.readonly()) {\n return manager.on(e => this.listBehavior.goto(this._getItem(e)!));\n }\n\n if (this.multi()) {\n manager.on(Modifier.Shift, e =>\n this.listBehavior.goto(this._getItem(e)!, {selectRange: true}),\n );\n }\n\n if (!this.multi() && this.followFocus()) {\n return manager.on(e => this.listBehavior.goto(this._getItem(e)!, {selectOne: true}));\n }\n\n if (!this.multi() && !this.followFocus()) {\n return manager.on(e => this.listBehavior.goto(this._getItem(e)!, {toggle: true}));\n }\n\n if (this.multi() && this.followFocus()) {\n return manager\n .on(e => this.listBehavior.goto(this._getItem(e)!, {selectOne: true}))\n .on(Modifier.Ctrl, e => this.listBehavior.goto(this._getItem(e)!, {toggle: true}));\n }\n\n if (this.multi() && !this.followFocus()) {\n return manager.on(e => this.listBehavior.goto(this._getItem(e)!, {toggle: true}));\n }\n\n return manager;\n });\n\n constructor(readonly inputs: ListboxInputs<V>) {\n this.readonly = inputs.readonly;\n this.orientation = inputs.orientation;\n this.multi = inputs.multi;\n this.listBehavior = new List(inputs);\n }\n\n /** Returns a set of violations */\n validate(): string[] {\n const violations: string[] = [];\n\n if (!this.inputs.multi() && this.inputs.values().length > 1) {\n violations.push(\n `A single-select listbox should not have multiple selected options. Selected options: ${this.inputs.values().join(', ')}`,\n );\n }\n\n return violations;\n }\n\n /** Handles keydown events for the listbox. */\n onKeydown(event: KeyboardEvent) {\n if (!this.disabled()) {\n this.keydown().handle(event);\n }\n }\n\n onPointerdown(event: PointerEvent) {\n if (!this.disabled()) {\n this.pointerdown().handle(event);\n }\n }\n\n /**\n * Sets the listbox to it's default initial state.\n *\n * Sets the active index of the listbox to the first focusable selected\n * item if one exists. Otherwise, sets focus to the first focusable item.\n *\n * This method should be called once the listbox and it's options are properly initialized,\n * meaning the ListboxPattern and OptionPatterns should have references to each other before this\n * is called.\n */\n setDefaultState() {\n let firstItem: OptionPattern<V> | null = null;\n\n for (const item of this.inputs.items()) {\n if (this.listBehavior.isFocusable(item)) {\n if (!firstItem) {\n firstItem = item;\n }\n if (item.selected()) {\n this.inputs.activeItem.set(item);\n return;\n }\n }\n }\n\n if (firstItem) {\n this.inputs.activeItem.set(firstItem);\n }\n }\n\n protected _getItem(e: PointerEvent) {\n if (!(e.target instanceof HTMLElement)) {\n return;\n }\n\n const element = e.target.closest('[role=\"option\"]');\n return this.inputs.items().find(i => i.element() === element);\n }\n}\n","/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.dev/license\n */\n\nimport {computed, SignalLike} from '../behaviors/signal-like/signal-like';\nimport {List, ListInputs, ListItem} from '../behaviors/list/list';\n\n/**\n * Represents the properties exposed by a listbox that need to be accessed by an option.\n * This exists to avoid circular dependency errors between the listbox and option.\n */\ninterface ListboxPattern<V> {\n inputs: ListInputs<OptionPattern<V>, V>;\n listBehavior: List<OptionPattern<V>, V>;\n}\n\n/** Represents the required inputs for an option in a listbox. */\nexport interface OptionInputs<V> extends Omit<ListItem<V>, 'index' | 'selectable'> {\n listbox: SignalLike<ListboxPattern<V> | undefined>;\n}\n\n/** Represents an option in a listbox. */\nexport class OptionPattern<V> {\n /** A unique identifier for the option. */\n id: SignalLike<string>;\n\n /** The value of the option. */\n value: SignalLike<V>;\n\n /** The position of the option in the list. */\n index = computed(() => this.listbox()?.inputs.items().indexOf(this) ?? -1);\n\n /** Whether the option is active. */\n active = computed(() => this.listbox()?.inputs.activeItem() === this);\n\n /** Whether the option is selected. */\n selected = computed(() => this.listbox()?.inputs.values().includes(this.value()));\n\n /** Whether the option is selectable. */\n selectable = () => true;\n\n /** Whether the option is disabled. */\n disabled: SignalLike<boolean>;\n\n /** The text used by the typeahead search. */\n searchTerm: SignalLike<string>;\n\n /** A reference to the parent listbox. */\n listbox: SignalLike<ListboxPattern<V> | undefined>;\n\n /** The tab index of the option. */\n tabIndex = computed(() => this.listbox()?.listBehavior.getItemTabindex(this));\n\n /** The html element that should receive focus. */\n element: SignalLike<HTMLElement | undefined>;\n\n constructor(args: OptionInputs<V>) {\n this.id = args.id;\n this.value = args.value;\n this.listbox = args.listbox;\n this.element = args.element;\n this.disabled = args.disabled;\n this.searchTerm = args.searchTerm;\n }\n}\n","/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.dev/license\n */\n\nimport {ListboxInputs, ListboxPattern} from './listbox';\nimport {SignalLike, computed} from '../behaviors/signal-like/signal-like';\nimport {OptionPattern} from './option';\nimport {ComboboxPattern, ComboboxListboxControls} from '../combobox/combobox';\n\nexport type ComboboxListboxInputs<V> = ListboxInputs<V> & {\n /** The combobox controlling the listbox. */\n combobox: SignalLike<ComboboxPattern<OptionPattern<V>, V> | undefined>;\n};\n\nexport class ComboboxListboxPattern<V>\n extends ListboxPattern<V>\n implements ComboboxListboxControls<OptionPattern<V>, V>\n{\n /** A unique identifier for the popup. */\n id = computed(() => this.inputs.id());\n\n /** The ARIA role for the listbox. */\n role = computed(() => 'listbox' as const);\n\n /** The id of the active (focused) item in the listbox. */\n activeId = computed(() => this.listBehavior.activeDescendant());\n\n /** The list of options in the listbox. */\n items: SignalLike<OptionPattern<V>[]> = computed(() => this.inputs.items());\n\n /** The tab index for the listbox. Always -1 because the combobox handles focus. */\n override tabIndex: SignalLike<-1 | 0> = () => -1;\n\n /** Whether multiple items in the list can be selected at once. */\n override multi = computed(() => {\n return this.inputs.combobox()?.readonly() ? this.inputs.multi() : false;\n });\n\n constructor(override readonly inputs: ComboboxListboxInputs<V>) {\n if (inputs.combobox()) {\n inputs.focusMode = () => 'activedescendant';\n inputs.element = inputs.combobox()!.inputs.inputEl;\n }\n\n super(inputs);\n }\n\n /** Noop. The combobox handles keydown events. */\n override onKeydown(_: KeyboardEvent): void {}\n\n /** Noop. The combobox handles pointerdown events. */\n override onPointerdown(_: PointerEvent): void {}\n\n /** Noop. The combobox controls the open state. */\n override setDefaultState(): void {}\n\n /** Navigates to the specified item in the listbox. */\n focus = (item: OptionPattern<V>, opts?: {focusElement?: boolean}) => {\n this.listBehavior.goto(item, opts);\n };\n\n /** Navigates to the previous focusable item in the listbox. */\n getActiveItem = () => this.inputs.activeItem();\n\n /** Navigates to the next focusable item in the listbox. */\n next = () => this.listBehavior.next();\n\n /** Navigates to the previous focusable item in the listbox. */\n prev = () => this.listBehavior.prev();\n\n /** Navigates to the last focusable item in the listbox. */\n last = () => this.listBehavior.last();\n\n /** Navigates to the first focusable item in the listbox. */\n first = () => this.listBehavior.first();\n\n /** Unfocuses the currently focused item in the listbox. */\n unfocus = () => this.listBehavior.unfocus();\n\n /** Selects the specified item in the listbox. */\n select = (item?: OptionPattern<V>) => this.listBehavior.select(item);\n\n /** Toggles the selection state of the given item in the listbox. */\n toggle = (item?: OptionPattern<V>) => this.listBehavior.toggle(item);\n\n /** Clears the selection in the listbox. */\n clearSelection = () => this.listBehavior.deselectAll();\n\n /** Retrieves the OptionPattern associated with a pointer event. */\n getItem = (e: PointerEvent) => this._getItem(e);\n\n /** Retrieves the currently selected items in the listbox. */\n getSelectedItems = () => {\n // NOTE: We need to do this funky for loop to preserve the order of the selected values.\n const items = [];\n for (const value of this.inputs.values()) {\n const item = this.items().find(i => i.value() === value);\n if (item) {\n items.push(item);\n }\n }\n return items;\n };\n\n /** Sets the value of the combobox listbox. */\n setValue = (value: V | undefined) => this.inputs.values.set(value ? [value] : []);\n}\n"],"names":["ListboxPattern","inputs","listBehavior","orientation","disabled","computed","readonly","tabIndex","activeDescendant","multi","setsize","items","length","followFocus","selectionMode","wrap","signal","prevKey","textDirection","nextKey","dynamicSpaceKey","isTyping","typeaheadRegexp","keydown","manager","KeyboardEventManager","on","prev","next","first","last","e","search","key","selectOne","Modifier","Any","anchor","activeIndex","Shift","selectRange","Ctrl","Meta","updateSelection","toggle","toggleAll","toggleOne","select","pointerdown","PointerEventManager","goto","_getItem","constructor","List","validate","violations","values","push","join","onKeydown","event","handle","onPointerdown","setDefaultState","firstItem","item","isFocusable","selected","activeItem","set","target","HTMLElement","element","closest","find","i","OptionPattern","id","value","index","listbox","indexOf","active","includes","selectable","searchTerm","getItemTabindex","args","ComboboxListboxPattern","role","activeId","combobox","focusMode","inputEl","_","focus","opts","getActiveItem","unfocus","clearSelection","deselectAll","getItem","getSelectedItems","setValue"],"mappings":"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"}
@@ -0,0 +1,331 @@
1
+ import { computed, Modifier, KeyboardEventManager } from './_signal-like-chunk.mjs';
2
+ import { List } from './_list-chunk.mjs';
3
+ import { ListExpansion } from './_expansion-chunk.mjs';
4
+ import { PointerEventManager } from './_pointer-event-manager-chunk.mjs';
5
+
6
+ class TreeItemPattern {
7
+ inputs;
8
+ id = () => this.inputs.id();
9
+ value = () => this.inputs.value();
10
+ element = () => this.inputs.element();
11
+ disabled = () => this.inputs.disabled();
12
+ searchTerm = () => this.inputs.searchTerm();
13
+ tree = () => this.inputs.tree();
14
+ parent = () => this.inputs.parent();
15
+ children = () => this.inputs.children();
16
+ index = computed(() => this.tree().visibleItems().indexOf(this));
17
+ expansionBehavior;
18
+ expandable = () => this.inputs.hasChildren();
19
+ selectable = () => this.inputs.selectable();
20
+ expanded;
21
+ level = computed(() => this.parent().level() + 1);
22
+ visible = computed(() => this.parent().expanded() && this.parent().visible());
23
+ setsize = computed(() => this.parent().children().length);
24
+ posinset = computed(() => this.parent().children().indexOf(this) + 1);
25
+ active = computed(() => this.tree().activeItem() === this);
26
+ tabIndex = computed(() => this.tree().listBehavior.getItemTabindex(this));
27
+ selected = computed(() => {
28
+ if (this.tree().nav()) {
29
+ return undefined;
30
+ }
31
+ if (!this.selectable()) {
32
+ return undefined;
33
+ }
34
+ return this.tree().values().includes(this.value());
35
+ });
36
+ current = computed(() => {
37
+ if (!this.tree().nav()) {
38
+ return undefined;
39
+ }
40
+ if (!this.selectable()) {
41
+ return undefined;
42
+ }
43
+ return this.tree().values().includes(this.value()) ? this.tree().currentType() : undefined;
44
+ });
45
+ constructor(inputs) {
46
+ this.inputs = inputs;
47
+ this.expanded = inputs.expanded;
48
+ this.expansionBehavior = new ListExpansion({
49
+ ...inputs,
50
+ multiExpandable: () => true,
51
+ items: this.children,
52
+ disabled: computed(() => this.tree()?.disabled() ?? false)
53
+ });
54
+ }
55
+ }
56
+ class TreePattern {
57
+ inputs;
58
+ listBehavior;
59
+ expansionBehavior;
60
+ level = () => 0;
61
+ expanded = () => true;
62
+ visible = () => true;
63
+ tabIndex = computed(() => this.listBehavior.tabIndex());
64
+ activeDescendant = computed(() => this.listBehavior.activeDescendant());
65
+ children = computed(() => this.inputs.allItems().filter(item => item.level() === this.level() + 1));
66
+ visibleItems = computed(() => this.inputs.allItems().filter(item => item.visible()));
67
+ followFocus = computed(() => this.inputs.selectionMode() === 'follow');
68
+ isRtl = computed(() => this.inputs.textDirection() === 'rtl');
69
+ prevKey = computed(() => {
70
+ if (this.inputs.orientation() === 'vertical') {
71
+ return 'ArrowUp';
72
+ }
73
+ return this.isRtl() ? 'ArrowRight' : 'ArrowLeft';
74
+ });
75
+ nextKey = computed(() => {
76
+ if (this.inputs.orientation() === 'vertical') {
77
+ return 'ArrowDown';
78
+ }
79
+ return this.isRtl() ? 'ArrowLeft' : 'ArrowRight';
80
+ });
81
+ collapseKey = computed(() => {
82
+ if (this.inputs.orientation() === 'horizontal') {
83
+ return 'ArrowUp';
84
+ }
85
+ return this.isRtl() ? 'ArrowRight' : 'ArrowLeft';
86
+ });
87
+ expandKey = computed(() => {
88
+ if (this.inputs.orientation() === 'horizontal') {
89
+ return 'ArrowDown';
90
+ }
91
+ return this.isRtl() ? 'ArrowLeft' : 'ArrowRight';
92
+ });
93
+ dynamicSpaceKey = computed(() => this.listBehavior.isTyping() ? '' : ' ');
94
+ typeaheadRegexp = /^.$/;
95
+ keydown = computed(() => {
96
+ const manager = new KeyboardEventManager();
97
+ const list = this.listBehavior;
98
+ manager.on(this.prevKey, () => list.prev({
99
+ selectOne: this.followFocus()
100
+ })).on(this.nextKey, () => list.next({
101
+ selectOne: this.followFocus()
102
+ })).on('Home', () => list.first({
103
+ selectOne: this.followFocus()
104
+ })).on('End', () => list.last({
105
+ selectOne: this.followFocus()
106
+ })).on(this.typeaheadRegexp, e => list.search(e.key, {
107
+ selectOne: this.followFocus()
108
+ })).on(this.expandKey, () => this.expand({
109
+ selectOne: this.followFocus()
110
+ })).on(this.collapseKey, () => this.collapse({
111
+ selectOne: this.followFocus()
112
+ })).on(Modifier.Shift, '*', () => this.expandSiblings());
113
+ if (this.inputs.multi()) {
114
+ manager.on(Modifier.Any, 'Shift', () => list.anchor(this.listBehavior.activeIndex())).on(Modifier.Shift, this.prevKey, () => list.prev({
115
+ selectRange: true
116
+ })).on(Modifier.Shift, this.nextKey, () => list.next({
117
+ selectRange: true
118
+ })).on([Modifier.Ctrl | Modifier.Shift, Modifier.Meta | Modifier.Shift], 'Home', () => list.first({
119
+ selectRange: true,
120
+ anchor: false
121
+ })).on([Modifier.Ctrl | Modifier.Shift, Modifier.Meta | Modifier.Shift], 'End', () => list.last({
122
+ selectRange: true,
123
+ anchor: false
124
+ })).on(Modifier.Shift, 'Enter', () => list.updateSelection({
125
+ selectRange: true,
126
+ anchor: false
127
+ })).on(Modifier.Shift, this.dynamicSpaceKey, () => list.updateSelection({
128
+ selectRange: true,
129
+ anchor: false
130
+ }));
131
+ }
132
+ if (!this.followFocus() && this.inputs.multi()) {
133
+ manager.on(this.dynamicSpaceKey, () => list.toggle()).on('Enter', () => list.toggle(), {
134
+ preventDefault: !this.nav()
135
+ }).on([Modifier.Ctrl, Modifier.Meta], 'A', () => list.toggleAll());
136
+ }
137
+ if (!this.followFocus() && !this.inputs.multi()) {
138
+ manager.on(this.dynamicSpaceKey, () => list.selectOne());
139
+ manager.on('Enter', () => list.selectOne(), {
140
+ preventDefault: !this.nav()
141
+ });
142
+ }
143
+ if (this.inputs.multi() && this.followFocus()) {
144
+ manager.on([Modifier.Ctrl, Modifier.Meta], this.prevKey, () => list.prev()).on([Modifier.Ctrl, Modifier.Meta], this.nextKey, () => list.next()).on([Modifier.Ctrl, Modifier.Meta], this.expandKey, () => this.expand()).on([Modifier.Ctrl, Modifier.Meta], this.collapseKey, () => this.collapse()).on([Modifier.Ctrl, Modifier.Meta], ' ', () => list.toggle()).on([Modifier.Ctrl, Modifier.Meta], 'Enter', () => list.toggle()).on([Modifier.Ctrl, Modifier.Meta], 'Home', () => list.first()).on([Modifier.Ctrl, Modifier.Meta], 'End', () => list.last()).on([Modifier.Ctrl, Modifier.Meta], 'A', () => {
145
+ list.toggleAll();
146
+ list.select();
147
+ });
148
+ }
149
+ return manager;
150
+ });
151
+ pointerdown = computed(() => {
152
+ const manager = new PointerEventManager();
153
+ if (this.multi()) {
154
+ manager.on(Modifier.Shift, e => this.goto(e, {
155
+ selectRange: true
156
+ }));
157
+ }
158
+ if (!this.multi()) {
159
+ return manager.on(e => this.goto(e, {
160
+ selectOne: true
161
+ }));
162
+ }
163
+ if (this.multi() && this.followFocus()) {
164
+ return manager.on(e => this.goto(e, {
165
+ selectOne: true
166
+ })).on(Modifier.Ctrl, e => this.goto(e, {
167
+ toggle: true
168
+ }));
169
+ }
170
+ if (this.multi() && !this.followFocus()) {
171
+ return manager.on(e => this.goto(e, {
172
+ toggle: true
173
+ }));
174
+ }
175
+ return manager;
176
+ });
177
+ id = () => this.inputs.id();
178
+ element = () => this.inputs.element();
179
+ nav = () => this.inputs.nav();
180
+ currentType = () => this.inputs.currentType();
181
+ allItems = () => this.inputs.allItems();
182
+ focusMode = () => this.inputs.focusMode();
183
+ disabled = () => this.inputs.disabled();
184
+ activeItem;
185
+ softDisabled = () => this.inputs.softDisabled();
186
+ wrap = () => this.inputs.wrap();
187
+ orientation = () => this.inputs.orientation();
188
+ textDirection = () => this.textDirection();
189
+ multi = computed(() => this.nav() ? false : this.inputs.multi());
190
+ selectionMode = () => this.inputs.selectionMode();
191
+ typeaheadDelay = () => this.inputs.typeaheadDelay();
192
+ values;
193
+ constructor(inputs) {
194
+ this.inputs = inputs;
195
+ this.activeItem = inputs.activeItem;
196
+ this.values = inputs.values;
197
+ this.listBehavior = new List({
198
+ ...inputs,
199
+ items: this.visibleItems,
200
+ multi: this.multi
201
+ });
202
+ this.expansionBehavior = new ListExpansion({
203
+ multiExpandable: () => true,
204
+ items: this.children,
205
+ disabled: this.disabled
206
+ });
207
+ }
208
+ setDefaultState() {
209
+ let firstItem;
210
+ for (const item of this.allItems()) {
211
+ if (!item.visible()) continue;
212
+ if (!this.listBehavior.isFocusable(item)) continue;
213
+ if (firstItem === undefined) {
214
+ firstItem = item;
215
+ }
216
+ if (item.selected()) {
217
+ this.activeItem.set(item);
218
+ return;
219
+ }
220
+ }
221
+ if (firstItem !== undefined) {
222
+ this.activeItem.set(firstItem);
223
+ }
224
+ }
225
+ onKeydown(event) {
226
+ if (!this.disabled()) {
227
+ this.keydown().handle(event);
228
+ }
229
+ }
230
+ onPointerdown(event) {
231
+ if (!this.disabled()) {
232
+ this.pointerdown().handle(event);
233
+ }
234
+ }
235
+ goto(e, opts) {
236
+ const item = this._getItem(e);
237
+ if (!item) return;
238
+ this.listBehavior.goto(item, opts);
239
+ this.toggleExpansion(item);
240
+ }
241
+ toggleExpansion(item) {
242
+ item ??= this.activeItem();
243
+ if (!item || !this.listBehavior.isFocusable(item)) return;
244
+ if (!item.expandable()) return;
245
+ if (item.expanded()) {
246
+ this.collapse();
247
+ } else {
248
+ this.expansionBehavior.open(item);
249
+ }
250
+ }
251
+ expand(opts) {
252
+ const item = this.activeItem();
253
+ if (!item || !this.listBehavior.isFocusable(item)) return;
254
+ if (item.expandable() && !item.expanded()) {
255
+ this.expansionBehavior.open(item);
256
+ } else if (item.expanded() && item.children().some(item => this.listBehavior.isFocusable(item))) {
257
+ this.listBehavior.next(opts);
258
+ }
259
+ }
260
+ expandSiblings(item) {
261
+ item ??= this.activeItem();
262
+ const siblings = item?.parent()?.children();
263
+ siblings?.forEach(item => this.expansionBehavior.open(item));
264
+ }
265
+ collapse(opts) {
266
+ const item = this.activeItem();
267
+ if (!item || !this.listBehavior.isFocusable(item)) return;
268
+ if (item.expandable() && item.expanded()) {
269
+ this.expansionBehavior.close(item);
270
+ } else if (item.parent() && item.parent() !== this) {
271
+ const parentItem = item.parent();
272
+ if (parentItem instanceof TreeItemPattern && this.listBehavior.isFocusable(parentItem)) {
273
+ this.listBehavior.goto(parentItem, opts);
274
+ }
275
+ }
276
+ }
277
+ _getItem(event) {
278
+ if (!(event.target instanceof HTMLElement)) {
279
+ return;
280
+ }
281
+ const element = event.target.closest('[role="treeitem"]');
282
+ return this.inputs.allItems().find(i => i.element() === element);
283
+ }
284
+ }
285
+
286
+ class ComboboxTreePattern extends TreePattern {
287
+ inputs;
288
+ isItemCollapsible = () => this.inputs.activeItem()?.parent() instanceof TreeItemPattern;
289
+ role = () => 'tree';
290
+ activeId = computed(() => this.listBehavior.activeDescendant());
291
+ getActiveItem = () => this.inputs.activeItem();
292
+ items = computed(() => this.inputs.allItems());
293
+ tabIndex = () => -1;
294
+ constructor(inputs) {
295
+ if (inputs.combobox()) {
296
+ inputs.multi = () => false;
297
+ inputs.focusMode = () => 'activedescendant';
298
+ inputs.element = inputs.combobox().inputs.inputEl;
299
+ }
300
+ super(inputs);
301
+ this.inputs = inputs;
302
+ }
303
+ onKeydown(_) {}
304
+ onPointerdown(_) {}
305
+ setDefaultState() {}
306
+ focus = item => this.listBehavior.goto(item);
307
+ next = () => this.listBehavior.next();
308
+ prev = () => this.listBehavior.prev();
309
+ last = () => this.listBehavior.last();
310
+ first = () => this.listBehavior.first();
311
+ unfocus = () => this.listBehavior.unfocus();
312
+ select = item => this.listBehavior.select(item);
313
+ toggle = item => this.listBehavior.toggle(item);
314
+ clearSelection = () => this.listBehavior.deselectAll();
315
+ getItem = e => this._getItem(e);
316
+ getSelectedItems = () => this.inputs.allItems().filter(item => item.selected());
317
+ setValue = value => this.inputs.values.set(value ? [value] : []);
318
+ expandItem = () => this.expand();
319
+ collapseItem = () => this.collapse();
320
+ isItemExpandable(item = this.inputs.activeItem()) {
321
+ return item ? item.expandable() : false;
322
+ }
323
+ expandAll = () => this.items().forEach(item => this.expansionBehavior.open(item));
324
+ collapseAll = () => this.items().forEach(item => item.expansionBehavior.close(item));
325
+ isItemSelectable = (item = this.inputs.activeItem()) => {
326
+ return item ? item.selectable() : false;
327
+ };
328
+ }
329
+
330
+ export { ComboboxTreePattern, TreeItemPattern, TreePattern };
331
+ //# sourceMappingURL=_combobox-tree-chunk.mjs.map