@angular/cdk 18.1.1 → 18.2.0-next.1
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/a11y/index.d.ts +283 -2
- package/coercion/private/index.d.ts +9 -0
- package/drag-drop/index.d.ts +12 -1
- package/esm2022/a11y/key-manager/list-key-manager.mjs +18 -38
- package/esm2022/a11y/key-manager/noop-tree-key-manager.mjs +94 -0
- package/esm2022/a11y/key-manager/tree-key-manager-strategy.mjs +9 -0
- package/esm2022/a11y/key-manager/tree-key-manager.mjs +345 -0
- package/esm2022/a11y/key-manager/typeahead.mjs +91 -0
- package/esm2022/a11y/public-api.mjs +4 -1
- package/esm2022/coercion/private/index.mjs +9 -0
- package/esm2022/coercion/private/observable.mjs +19 -0
- package/esm2022/coercion/private/private_public_index.mjs +5 -0
- package/esm2022/drag-drop/directives/drag.mjs +16 -3
- package/esm2022/drag-drop/drag-ref.mjs +8 -2
- package/esm2022/drag-drop/sorting/single-axis-sort-strategy.mjs +4 -3
- package/esm2022/tree/control/base-tree-control.mjs +7 -2
- package/esm2022/tree/control/flat-tree-control.mjs +8 -2
- package/esm2022/tree/control/nested-tree-control.mjs +11 -2
- package/esm2022/tree/control/tree-control.mjs +1 -1
- package/esm2022/tree/nested-node.mjs +6 -15
- package/esm2022/tree/padding.mjs +2 -4
- package/esm2022/tree/toggle.mjs +15 -8
- package/esm2022/tree/tree-errors.mjs +7 -6
- package/esm2022/tree/tree.mjs +817 -63
- package/esm2022/version.mjs +1 -1
- package/fesm2022/a11y.mjs +520 -40
- package/fesm2022/a11y.mjs.map +1 -1
- package/fesm2022/cdk.mjs +1 -1
- package/fesm2022/cdk.mjs.map +1 -1
- package/fesm2022/coercion/private.mjs +19 -0
- package/fesm2022/coercion/private.mjs.map +1 -0
- package/fesm2022/drag-drop.mjs +25 -5
- package/fesm2022/drag-drop.mjs.map +1 -1
- package/fesm2022/tree.mjs +858 -94
- package/fesm2022/tree.mjs.map +1 -1
- package/package.json +9 -3
- package/schematics/ng-add/index.js +1 -1
- package/schematics/ng-add/index.mjs +1 -1
- package/tree/index.d.ts +304 -25
package/esm2022/tree/tree.mjs
CHANGED
|
@@ -1,11 +1,22 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Google LLC All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
+
* found in the LICENSE file at https://angular.io/license
|
|
7
|
+
*/
|
|
8
|
+
import { TREE_KEY_MANAGER, } from '@angular/cdk/a11y';
|
|
9
|
+
import { Directionality } from '@angular/cdk/bidi';
|
|
10
|
+
import { isDataSource, SelectionModel, } from '@angular/cdk/collections';
|
|
11
|
+
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChildren, Directive, ElementRef, EventEmitter, Input, IterableDiffers, Output, QueryList, ViewChild, ViewEncapsulation, numberAttribute, inject, booleanAttribute, } from '@angular/core';
|
|
12
|
+
import { coerceObservable } from '@angular/cdk/coercion/private';
|
|
13
|
+
import { BehaviorSubject, combineLatest, concat, EMPTY, Subject, isObservable, of as observableOf, } from 'rxjs';
|
|
14
|
+
import { distinctUntilChanged, concatMap, map, reduce, startWith, switchMap, take, takeUntil, tap, } from 'rxjs/operators';
|
|
5
15
|
import { CdkTreeNodeDef, CdkTreeNodeOutletContext } from './node';
|
|
6
16
|
import { CdkTreeNodeOutlet } from './outlet';
|
|
7
|
-
import {
|
|
17
|
+
import { getMultipleTreeControlsError, getTreeControlMissingError, getTreeMissingMatchingNodeDefError, getTreeMultipleDefaultNodeDefsError, getTreeNoValidDataSourceError, } from './tree-errors';
|
|
8
18
|
import * as i0 from "@angular/core";
|
|
19
|
+
import * as i1 from "@angular/cdk/bidi";
|
|
9
20
|
/**
|
|
10
21
|
* CDK tree component that connects with a data source to retrieve data of type `T` and renders
|
|
11
22
|
* dataNodes with hierarchy. Updates the dataNodes when new data is provided by the data source.
|
|
@@ -24,13 +35,25 @@ export class CdkTree {
|
|
|
24
35
|
this._switchDataSource(dataSource);
|
|
25
36
|
}
|
|
26
37
|
}
|
|
27
|
-
constructor(_differs, _changeDetectorRef) {
|
|
38
|
+
constructor(_differs, _changeDetectorRef, _dir) {
|
|
28
39
|
this._differs = _differs;
|
|
29
40
|
this._changeDetectorRef = _changeDetectorRef;
|
|
41
|
+
this._dir = _dir;
|
|
30
42
|
/** Subject that emits when the component has been destroyed. */
|
|
31
43
|
this._onDestroy = new Subject();
|
|
32
44
|
/** Level of nodes */
|
|
33
45
|
this._levels = new Map();
|
|
46
|
+
/** The immediate parents for a node. This is `null` if there is no parent. */
|
|
47
|
+
this._parents = new Map();
|
|
48
|
+
/**
|
|
49
|
+
* Nodes grouped into each set, which is a list of nodes displayed together in the DOM.
|
|
50
|
+
*
|
|
51
|
+
* Lookup key is the parent of a set. Root nodes have key of null.
|
|
52
|
+
*
|
|
53
|
+
* Values is a 'set' of tree nodes. Each tree node maps to a treeitem element. Sets are in the
|
|
54
|
+
* order that it is rendered. Each set maps directly to aria-posinset and aria-setsize attributes.
|
|
55
|
+
*/
|
|
56
|
+
this._ariaSets = new Map();
|
|
34
57
|
// TODO(tinayuangao): Setup a listener for scrolling, emit the calculated view to viewChange.
|
|
35
58
|
// Remove the MAX_VALUE in viewChange
|
|
36
59
|
/**
|
|
@@ -41,12 +64,31 @@ export class CdkTree {
|
|
|
41
64
|
start: 0,
|
|
42
65
|
end: Number.MAX_VALUE,
|
|
43
66
|
});
|
|
67
|
+
/**
|
|
68
|
+
* Maintain a synchronous cache of flattened data nodes. This will only be
|
|
69
|
+
* populated after initial render, and in certain cases, will be delayed due to
|
|
70
|
+
* relying on Observable `getChildren` calls.
|
|
71
|
+
*/
|
|
72
|
+
this._flattenedNodes = new BehaviorSubject([]);
|
|
73
|
+
/** The automatically determined node type for the tree. */
|
|
74
|
+
this._nodeType = new BehaviorSubject(null);
|
|
75
|
+
/** The mapping between data and the node that is rendered. */
|
|
76
|
+
this._nodes = new BehaviorSubject(new Map());
|
|
77
|
+
/**
|
|
78
|
+
* Synchronous cache of nodes for the `TreeKeyManager`. This is separate
|
|
79
|
+
* from `_flattenedNodes` so they can be independently updated at different
|
|
80
|
+
* times.
|
|
81
|
+
*/
|
|
82
|
+
this._keyManagerNodes = new BehaviorSubject([]);
|
|
83
|
+
this._keyManagerFactory = inject(TREE_KEY_MANAGER);
|
|
84
|
+
this._viewInit = false;
|
|
44
85
|
}
|
|
45
|
-
|
|
46
|
-
this.
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
86
|
+
ngAfterContentInit() {
|
|
87
|
+
this._initializeKeyManager();
|
|
88
|
+
}
|
|
89
|
+
ngAfterContentChecked() {
|
|
90
|
+
this._updateDefaultNodeDefinition();
|
|
91
|
+
this._subscribeToDataChanges();
|
|
50
92
|
}
|
|
51
93
|
ngOnDestroy() {
|
|
52
94
|
this._nodeOutlet.viewContainer.clear();
|
|
@@ -60,19 +102,35 @@ export class CdkTree {
|
|
|
60
102
|
this._dataSubscription.unsubscribe();
|
|
61
103
|
this._dataSubscription = null;
|
|
62
104
|
}
|
|
105
|
+
// In certain tests, the tree might be destroyed before this is initialized
|
|
106
|
+
// in `ngAfterContentInit`.
|
|
107
|
+
this._keyManager?.destroy();
|
|
63
108
|
}
|
|
64
|
-
|
|
109
|
+
ngOnInit() {
|
|
110
|
+
this._checkTreeControlUsage();
|
|
111
|
+
this._initializeDataDiffer();
|
|
112
|
+
}
|
|
113
|
+
ngAfterViewInit() {
|
|
114
|
+
this._viewInit = true;
|
|
115
|
+
}
|
|
116
|
+
_updateDefaultNodeDefinition() {
|
|
65
117
|
const defaultNodeDefs = this._nodeDefs.filter(def => !def.when);
|
|
66
118
|
if (defaultNodeDefs.length > 1 && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
|
67
119
|
throw getTreeMultipleDefaultNodeDefsError();
|
|
68
120
|
}
|
|
69
121
|
this._defaultNodeDef = defaultNodeDefs[0];
|
|
70
|
-
|
|
71
|
-
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Sets the node type for the tree, if it hasn't been set yet.
|
|
125
|
+
*
|
|
126
|
+
* This will be called by the first node that's rendered in order for the tree
|
|
127
|
+
* to determine what data transformations are required.
|
|
128
|
+
*/
|
|
129
|
+
_setNodeTypeIfUnset(nodeType) {
|
|
130
|
+
if (this._nodeType.value === null) {
|
|
131
|
+
this._nodeType.next(nodeType);
|
|
72
132
|
}
|
|
73
133
|
}
|
|
74
|
-
// TODO(tinayuangao): Work on keyboard traversal and actions, make sure it's working for RTL
|
|
75
|
-
// and nested trees.
|
|
76
134
|
/**
|
|
77
135
|
* Switch to the provided data source by resetting the data and unsubscribing from the current
|
|
78
136
|
* render change subscription if one exists. If the data source is null, interpret this by
|
|
@@ -92,11 +150,21 @@ export class CdkTree {
|
|
|
92
150
|
}
|
|
93
151
|
this._dataSource = dataSource;
|
|
94
152
|
if (this._nodeDefs) {
|
|
95
|
-
this.
|
|
153
|
+
this._subscribeToDataChanges();
|
|
96
154
|
}
|
|
97
155
|
}
|
|
156
|
+
_getExpansionModel() {
|
|
157
|
+
if (!this.treeControl) {
|
|
158
|
+
this._expansionModel ??= new SelectionModel(true);
|
|
159
|
+
return this._expansionModel;
|
|
160
|
+
}
|
|
161
|
+
return this.treeControl.expansionModel;
|
|
162
|
+
}
|
|
98
163
|
/** Set up a subscription for the data provided by the data source. */
|
|
99
|
-
|
|
164
|
+
_subscribeToDataChanges() {
|
|
165
|
+
if (this._dataSubscription) {
|
|
166
|
+
return;
|
|
167
|
+
}
|
|
100
168
|
let dataStream;
|
|
101
169
|
if (isDataSource(this._dataSource)) {
|
|
102
170
|
dataStream = this._dataSource.connect(this);
|
|
@@ -107,34 +175,140 @@ export class CdkTree {
|
|
|
107
175
|
else if (Array.isArray(this._dataSource)) {
|
|
108
176
|
dataStream = observableOf(this._dataSource);
|
|
109
177
|
}
|
|
110
|
-
if (dataStream) {
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
178
|
+
if (!dataStream) {
|
|
179
|
+
if (typeof ngDevMode === 'undefined' || ngDevMode) {
|
|
180
|
+
throw getTreeNoValidDataSourceError();
|
|
181
|
+
}
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
this._dataSubscription = this._getRenderData(dataStream)
|
|
185
|
+
.pipe(takeUntil(this._onDestroy))
|
|
186
|
+
.subscribe(renderingData => {
|
|
187
|
+
this._renderDataChanges(renderingData);
|
|
188
|
+
});
|
|
189
|
+
}
|
|
190
|
+
/** Given an Observable containing a stream of the raw data, returns an Observable containing the RenderingData */
|
|
191
|
+
_getRenderData(dataStream) {
|
|
192
|
+
const expansionModel = this._getExpansionModel();
|
|
193
|
+
return combineLatest([
|
|
194
|
+
dataStream,
|
|
195
|
+
this._nodeType,
|
|
196
|
+
// We don't use the expansion data directly, however we add it here to essentially
|
|
197
|
+
// trigger data rendering when expansion changes occur.
|
|
198
|
+
expansionModel.changed.pipe(startWith(null), tap(expansionChanges => {
|
|
199
|
+
this._emitExpansionChanges(expansionChanges);
|
|
200
|
+
})),
|
|
201
|
+
]).pipe(switchMap(([data, nodeType]) => {
|
|
202
|
+
if (nodeType === null) {
|
|
203
|
+
return observableOf({ renderNodes: data, flattenedNodes: null, nodeType });
|
|
204
|
+
}
|
|
205
|
+
// If we're here, then we know what our node type is, and therefore can
|
|
206
|
+
// perform our usual rendering pipeline, which necessitates converting the data
|
|
207
|
+
return this._computeRenderingData(data, nodeType).pipe(map(convertedData => ({ ...convertedData, nodeType })));
|
|
208
|
+
}));
|
|
209
|
+
}
|
|
210
|
+
_renderDataChanges(data) {
|
|
211
|
+
if (data.nodeType === null) {
|
|
212
|
+
this.renderNodeChanges(data.renderNodes);
|
|
213
|
+
return;
|
|
114
214
|
}
|
|
115
|
-
|
|
116
|
-
|
|
215
|
+
// If we're here, then we know what our node type is, and therefore can
|
|
216
|
+
// perform our usual rendering pipeline.
|
|
217
|
+
this._updateCachedData(data.flattenedNodes);
|
|
218
|
+
this.renderNodeChanges(data.renderNodes);
|
|
219
|
+
this._updateKeyManagerItems(data.flattenedNodes);
|
|
220
|
+
}
|
|
221
|
+
_emitExpansionChanges(expansionChanges) {
|
|
222
|
+
if (!expansionChanges) {
|
|
223
|
+
return;
|
|
224
|
+
}
|
|
225
|
+
const nodes = this._nodes.value;
|
|
226
|
+
for (const added of expansionChanges.added) {
|
|
227
|
+
const node = nodes.get(added);
|
|
228
|
+
node?._emitExpansionState(true);
|
|
229
|
+
}
|
|
230
|
+
for (const removed of expansionChanges.removed) {
|
|
231
|
+
const node = nodes.get(removed);
|
|
232
|
+
node?._emitExpansionState(false);
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
_initializeKeyManager() {
|
|
236
|
+
const items = combineLatest([this._keyManagerNodes, this._nodes]).pipe(map(([keyManagerNodes, renderNodes]) => keyManagerNodes.reduce((items, data) => {
|
|
237
|
+
const node = renderNodes.get(this._getExpansionKey(data));
|
|
238
|
+
if (node) {
|
|
239
|
+
items.push(node);
|
|
240
|
+
}
|
|
241
|
+
return items;
|
|
242
|
+
}, [])));
|
|
243
|
+
const keyManagerOptions = {
|
|
244
|
+
trackBy: node => this._getExpansionKey(node.data),
|
|
245
|
+
skipPredicate: node => !!node.isDisabled,
|
|
246
|
+
typeAheadDebounceInterval: true,
|
|
247
|
+
horizontalOrientation: this._dir.value,
|
|
248
|
+
};
|
|
249
|
+
this._keyManager = this._keyManagerFactory(items, keyManagerOptions);
|
|
250
|
+
}
|
|
251
|
+
_initializeDataDiffer() {
|
|
252
|
+
// Provide a default trackBy based on `_getExpansionKey` if one isn't provided.
|
|
253
|
+
const trackBy = this.trackBy ?? ((_index, item) => this._getExpansionKey(item));
|
|
254
|
+
this._dataDiffer = this._differs.find([]).create(trackBy);
|
|
255
|
+
}
|
|
256
|
+
_checkTreeControlUsage() {
|
|
257
|
+
if (typeof ngDevMode === 'undefined' || ngDevMode) {
|
|
258
|
+
// Verify that Tree follows API contract of using one of TreeControl, levelAccessor or
|
|
259
|
+
// childrenAccessor. Throw an appropriate error if contract is not met.
|
|
260
|
+
let numTreeControls = 0;
|
|
261
|
+
if (this.treeControl) {
|
|
262
|
+
numTreeControls++;
|
|
263
|
+
}
|
|
264
|
+
if (this.levelAccessor) {
|
|
265
|
+
numTreeControls++;
|
|
266
|
+
}
|
|
267
|
+
if (this.childrenAccessor) {
|
|
268
|
+
numTreeControls++;
|
|
269
|
+
}
|
|
270
|
+
if (!numTreeControls) {
|
|
271
|
+
throw getTreeControlMissingError();
|
|
272
|
+
}
|
|
273
|
+
else if (numTreeControls > 1) {
|
|
274
|
+
throw getMultipleTreeControlsError();
|
|
275
|
+
}
|
|
117
276
|
}
|
|
118
277
|
}
|
|
119
278
|
/** Check for changes made in the data and render each change (node added/removed/moved). */
|
|
120
279
|
renderNodeChanges(data, dataDiffer = this._dataDiffer, viewContainer = this._nodeOutlet.viewContainer, parentData) {
|
|
121
280
|
const changes = dataDiffer.diff(data);
|
|
122
|
-
|
|
281
|
+
// Some tree consumers expect change detection to propagate to nodes
|
|
282
|
+
// even when the array itself hasn't changed; we explicitly detect changes
|
|
283
|
+
// anyways in order for nodes to update their data.
|
|
284
|
+
//
|
|
285
|
+
// However, if change detection is called while the component's view is
|
|
286
|
+
// still initing, then the order of child views initing will be incorrect;
|
|
287
|
+
// to prevent this, we only exit early if the view hasn't initialized yet.
|
|
288
|
+
if (!changes && !this._viewInit) {
|
|
123
289
|
return;
|
|
124
290
|
}
|
|
125
|
-
changes
|
|
291
|
+
changes?.forEachOperation((item, adjustedPreviousIndex, currentIndex) => {
|
|
126
292
|
if (item.previousIndex == null) {
|
|
127
293
|
this.insertNode(data[currentIndex], currentIndex, viewContainer, parentData);
|
|
128
294
|
}
|
|
129
295
|
else if (currentIndex == null) {
|
|
130
296
|
viewContainer.remove(adjustedPreviousIndex);
|
|
131
|
-
this._levels.delete(item.item);
|
|
132
297
|
}
|
|
133
298
|
else {
|
|
134
299
|
const view = viewContainer.get(adjustedPreviousIndex);
|
|
135
300
|
viewContainer.move(view, currentIndex);
|
|
136
301
|
}
|
|
137
302
|
});
|
|
303
|
+
// If the data itself changes, but keeps the same trackBy, we need to update the templates'
|
|
304
|
+
// context to reflect the new object.
|
|
305
|
+
changes?.forEachIdentityChange((record) => {
|
|
306
|
+
const newData = record.item;
|
|
307
|
+
if (record.currentIndex != undefined) {
|
|
308
|
+
const view = viewContainer.get(record.currentIndex);
|
|
309
|
+
view.context.$implicit = newData;
|
|
310
|
+
}
|
|
311
|
+
});
|
|
138
312
|
// TODO: change to `this._changeDetectorRef.markForCheck()`, or just switch this component to
|
|
139
313
|
// use signals.
|
|
140
314
|
this._changeDetectorRef.detectChanges();
|
|
@@ -160,21 +334,24 @@ export class CdkTree {
|
|
|
160
334
|
* within the data node view container.
|
|
161
335
|
*/
|
|
162
336
|
insertNode(nodeData, index, viewContainer, parentData) {
|
|
337
|
+
const levelAccessor = this._getLevelAccessor();
|
|
163
338
|
const node = this._getNodeDef(nodeData, index);
|
|
339
|
+
const key = this._getExpansionKey(nodeData);
|
|
164
340
|
// Node context that will be provided to created embedded view
|
|
165
341
|
const context = new CdkTreeNodeOutletContext(nodeData);
|
|
342
|
+
parentData ??= this._parents.get(key) ?? undefined;
|
|
166
343
|
// If the tree is flat tree, then use the `getLevel` function in flat tree control
|
|
167
344
|
// Otherwise, use the level of parent node.
|
|
168
|
-
if (
|
|
169
|
-
context.level =
|
|
345
|
+
if (levelAccessor) {
|
|
346
|
+
context.level = levelAccessor(nodeData);
|
|
170
347
|
}
|
|
171
|
-
else if (
|
|
172
|
-
context.level = this._levels.get(parentData) + 1;
|
|
348
|
+
else if (parentData !== undefined && this._levels.has(this._getExpansionKey(parentData))) {
|
|
349
|
+
context.level = this._levels.get(this._getExpansionKey(parentData)) + 1;
|
|
173
350
|
}
|
|
174
351
|
else {
|
|
175
352
|
context.level = 0;
|
|
176
353
|
}
|
|
177
|
-
this._levels.set(
|
|
354
|
+
this._levels.set(key, context.level);
|
|
178
355
|
// Use default tree nodeOutlet, or nested node's nodeOutlet
|
|
179
356
|
const container = viewContainer ? viewContainer : this._nodeOutlet.viewContainer;
|
|
180
357
|
container.createEmbeddedView(node.template, context, index);
|
|
@@ -185,8 +362,427 @@ export class CdkTree {
|
|
|
185
362
|
CdkTreeNode.mostRecentTreeNode.data = nodeData;
|
|
186
363
|
}
|
|
187
364
|
}
|
|
188
|
-
|
|
189
|
-
|
|
365
|
+
/** Whether the data node is expanded or collapsed. Returns true if it's expanded. */
|
|
366
|
+
isExpanded(dataNode) {
|
|
367
|
+
return !!(this.treeControl?.isExpanded(dataNode) ||
|
|
368
|
+
this._expansionModel?.isSelected(this._getExpansionKey(dataNode)));
|
|
369
|
+
}
|
|
370
|
+
/** If the data node is currently expanded, collapse it. Otherwise, expand it. */
|
|
371
|
+
toggle(dataNode) {
|
|
372
|
+
if (this.treeControl) {
|
|
373
|
+
this.treeControl.toggle(dataNode);
|
|
374
|
+
}
|
|
375
|
+
else if (this._expansionModel) {
|
|
376
|
+
this._expansionModel.toggle(this._getExpansionKey(dataNode));
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
/** Expand the data node. If it is already expanded, does nothing. */
|
|
380
|
+
expand(dataNode) {
|
|
381
|
+
if (this.treeControl) {
|
|
382
|
+
this.treeControl.expand(dataNode);
|
|
383
|
+
}
|
|
384
|
+
else if (this._expansionModel) {
|
|
385
|
+
this._expansionModel.select(this._getExpansionKey(dataNode));
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
/** Collapse the data node. If it is already collapsed, does nothing. */
|
|
389
|
+
collapse(dataNode) {
|
|
390
|
+
if (this.treeControl) {
|
|
391
|
+
this.treeControl.collapse(dataNode);
|
|
392
|
+
}
|
|
393
|
+
else if (this._expansionModel) {
|
|
394
|
+
this._expansionModel.deselect(this._getExpansionKey(dataNode));
|
|
395
|
+
}
|
|
396
|
+
}
|
|
397
|
+
/**
|
|
398
|
+
* If the data node is currently expanded, collapse it and all its descendants.
|
|
399
|
+
* Otherwise, expand it and all its descendants.
|
|
400
|
+
*/
|
|
401
|
+
toggleDescendants(dataNode) {
|
|
402
|
+
if (this.treeControl) {
|
|
403
|
+
this.treeControl.toggleDescendants(dataNode);
|
|
404
|
+
}
|
|
405
|
+
else if (this._expansionModel) {
|
|
406
|
+
if (this.isExpanded(dataNode)) {
|
|
407
|
+
this.collapseDescendants(dataNode);
|
|
408
|
+
}
|
|
409
|
+
else {
|
|
410
|
+
this.expandDescendants(dataNode);
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
/**
|
|
415
|
+
* Expand the data node and all its descendants. If they are already expanded, does nothing.
|
|
416
|
+
*/
|
|
417
|
+
expandDescendants(dataNode) {
|
|
418
|
+
if (this.treeControl) {
|
|
419
|
+
this.treeControl.expandDescendants(dataNode);
|
|
420
|
+
}
|
|
421
|
+
else if (this._expansionModel) {
|
|
422
|
+
const expansionModel = this._expansionModel;
|
|
423
|
+
expansionModel.select(this._getExpansionKey(dataNode));
|
|
424
|
+
this._getDescendants(dataNode)
|
|
425
|
+
.pipe(take(1), takeUntil(this._onDestroy))
|
|
426
|
+
.subscribe(children => {
|
|
427
|
+
expansionModel.select(...children.map(child => this._getExpansionKey(child)));
|
|
428
|
+
});
|
|
429
|
+
}
|
|
430
|
+
}
|
|
431
|
+
/** Collapse the data node and all its descendants. If it is already collapsed, does nothing. */
|
|
432
|
+
collapseDescendants(dataNode) {
|
|
433
|
+
if (this.treeControl) {
|
|
434
|
+
this.treeControl.collapseDescendants(dataNode);
|
|
435
|
+
}
|
|
436
|
+
else if (this._expansionModel) {
|
|
437
|
+
const expansionModel = this._expansionModel;
|
|
438
|
+
expansionModel.deselect(this._getExpansionKey(dataNode));
|
|
439
|
+
this._getDescendants(dataNode)
|
|
440
|
+
.pipe(take(1), takeUntil(this._onDestroy))
|
|
441
|
+
.subscribe(children => {
|
|
442
|
+
expansionModel.deselect(...children.map(child => this._getExpansionKey(child)));
|
|
443
|
+
});
|
|
444
|
+
}
|
|
445
|
+
}
|
|
446
|
+
/** Expands all data nodes in the tree. */
|
|
447
|
+
expandAll() {
|
|
448
|
+
if (this.treeControl) {
|
|
449
|
+
this.treeControl.expandAll();
|
|
450
|
+
}
|
|
451
|
+
else if (this._expansionModel) {
|
|
452
|
+
const expansionModel = this._expansionModel;
|
|
453
|
+
expansionModel.select(...this._flattenedNodes.value.map(child => this._getExpansionKey(child)));
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
/** Collapse all data nodes in the tree. */
|
|
457
|
+
collapseAll() {
|
|
458
|
+
if (this.treeControl) {
|
|
459
|
+
this.treeControl.collapseAll();
|
|
460
|
+
}
|
|
461
|
+
else if (this._expansionModel) {
|
|
462
|
+
const expansionModel = this._expansionModel;
|
|
463
|
+
expansionModel.deselect(...this._flattenedNodes.value.map(child => this._getExpansionKey(child)));
|
|
464
|
+
}
|
|
465
|
+
}
|
|
466
|
+
/** Level accessor, used for compatibility between the old Tree and new Tree */
|
|
467
|
+
_getLevelAccessor() {
|
|
468
|
+
return this.treeControl?.getLevel?.bind(this.treeControl) ?? this.levelAccessor;
|
|
469
|
+
}
|
|
470
|
+
/** Children accessor, used for compatibility between the old Tree and new Tree */
|
|
471
|
+
_getChildrenAccessor() {
|
|
472
|
+
return this.treeControl?.getChildren?.bind(this.treeControl) ?? this.childrenAccessor;
|
|
473
|
+
}
|
|
474
|
+
/**
|
|
475
|
+
* Gets the direct children of a node; used for compatibility between the old tree and the
|
|
476
|
+
* new tree.
|
|
477
|
+
*/
|
|
478
|
+
_getDirectChildren(dataNode) {
|
|
479
|
+
const levelAccessor = this._getLevelAccessor();
|
|
480
|
+
const expansionModel = this._expansionModel ?? this.treeControl?.expansionModel;
|
|
481
|
+
if (!expansionModel) {
|
|
482
|
+
return observableOf([]);
|
|
483
|
+
}
|
|
484
|
+
const key = this._getExpansionKey(dataNode);
|
|
485
|
+
const isExpanded = expansionModel.changed.pipe(switchMap(changes => {
|
|
486
|
+
if (changes.added.includes(key)) {
|
|
487
|
+
return observableOf(true);
|
|
488
|
+
}
|
|
489
|
+
else if (changes.removed.includes(key)) {
|
|
490
|
+
return observableOf(false);
|
|
491
|
+
}
|
|
492
|
+
return EMPTY;
|
|
493
|
+
}), startWith(this.isExpanded(dataNode)));
|
|
494
|
+
if (levelAccessor) {
|
|
495
|
+
return combineLatest([isExpanded, this._flattenedNodes]).pipe(map(([expanded, flattenedNodes]) => {
|
|
496
|
+
if (!expanded) {
|
|
497
|
+
return [];
|
|
498
|
+
}
|
|
499
|
+
return this._findChildrenByLevel(levelAccessor, flattenedNodes, dataNode, 1);
|
|
500
|
+
}));
|
|
501
|
+
}
|
|
502
|
+
const childrenAccessor = this._getChildrenAccessor();
|
|
503
|
+
if (childrenAccessor) {
|
|
504
|
+
return coerceObservable(childrenAccessor(dataNode) ?? []);
|
|
505
|
+
}
|
|
506
|
+
throw getTreeControlMissingError();
|
|
507
|
+
}
|
|
508
|
+
/**
|
|
509
|
+
* Given the list of flattened nodes, the level accessor, and the level range within
|
|
510
|
+
* which to consider children, finds the children for a given node.
|
|
511
|
+
*
|
|
512
|
+
* For example, for direct children, `levelDelta` would be 1. For all descendants,
|
|
513
|
+
* `levelDelta` would be Infinity.
|
|
514
|
+
*/
|
|
515
|
+
_findChildrenByLevel(levelAccessor, flattenedNodes, dataNode, levelDelta) {
|
|
516
|
+
const key = this._getExpansionKey(dataNode);
|
|
517
|
+
const startIndex = flattenedNodes.findIndex(node => this._getExpansionKey(node) === key);
|
|
518
|
+
const dataNodeLevel = levelAccessor(dataNode);
|
|
519
|
+
const expectedLevel = dataNodeLevel + levelDelta;
|
|
520
|
+
const results = [];
|
|
521
|
+
// Goes through flattened tree nodes in the `flattenedNodes` array, and get all
|
|
522
|
+
// descendants within a certain level range.
|
|
523
|
+
//
|
|
524
|
+
// If we reach a node whose level is equal to or less than the level of the tree node,
|
|
525
|
+
// we hit a sibling or parent's sibling, and should stop.
|
|
526
|
+
for (let i = startIndex + 1; i < flattenedNodes.length; i++) {
|
|
527
|
+
const currentLevel = levelAccessor(flattenedNodes[i]);
|
|
528
|
+
if (currentLevel <= dataNodeLevel) {
|
|
529
|
+
break;
|
|
530
|
+
}
|
|
531
|
+
if (currentLevel <= expectedLevel) {
|
|
532
|
+
results.push(flattenedNodes[i]);
|
|
533
|
+
}
|
|
534
|
+
}
|
|
535
|
+
return results;
|
|
536
|
+
}
|
|
537
|
+
/**
|
|
538
|
+
* Adds the specified node component to the tree's internal registry.
|
|
539
|
+
*
|
|
540
|
+
* This primarily facilitates keyboard navigation.
|
|
541
|
+
*/
|
|
542
|
+
_registerNode(node) {
|
|
543
|
+
this._nodes.value.set(this._getExpansionKey(node.data), node);
|
|
544
|
+
this._nodes.next(this._nodes.value);
|
|
545
|
+
}
|
|
546
|
+
/** Removes the specified node component from the tree's internal registry. */
|
|
547
|
+
_unregisterNode(node) {
|
|
548
|
+
this._nodes.value.delete(this._getExpansionKey(node.data));
|
|
549
|
+
this._nodes.next(this._nodes.value);
|
|
550
|
+
}
|
|
551
|
+
/**
|
|
552
|
+
* For the given node, determine the level where this node appears in the tree.
|
|
553
|
+
*
|
|
554
|
+
* This is intended to be used for `aria-level` but is 0-indexed.
|
|
555
|
+
*/
|
|
556
|
+
_getLevel(node) {
|
|
557
|
+
return this._levels.get(this._getExpansionKey(node));
|
|
558
|
+
}
|
|
559
|
+
/**
|
|
560
|
+
* For the given node, determine the size of the parent's child set.
|
|
561
|
+
*
|
|
562
|
+
* This is intended to be used for `aria-setsize`.
|
|
563
|
+
*/
|
|
564
|
+
_getSetSize(dataNode) {
|
|
565
|
+
const set = this._getAriaSet(dataNode);
|
|
566
|
+
return set.length;
|
|
567
|
+
}
|
|
568
|
+
/**
|
|
569
|
+
* For the given node, determine the index (starting from 1) of the node in its parent's child set.
|
|
570
|
+
*
|
|
571
|
+
* This is intended to be used for `aria-posinset`.
|
|
572
|
+
*/
|
|
573
|
+
_getPositionInSet(dataNode) {
|
|
574
|
+
const set = this._getAriaSet(dataNode);
|
|
575
|
+
const key = this._getExpansionKey(dataNode);
|
|
576
|
+
return set.findIndex(node => this._getExpansionKey(node) === key) + 1;
|
|
577
|
+
}
|
|
578
|
+
/** Given a CdkTreeNode, gets the node that renders that node's parent's data. */
|
|
579
|
+
_getNodeParent(node) {
|
|
580
|
+
const parent = this._parents.get(this._getExpansionKey(node.data));
|
|
581
|
+
return parent && this._nodes.value.get(this._getExpansionKey(parent));
|
|
582
|
+
}
|
|
583
|
+
/** Given a CdkTreeNode, gets the nodes that renders that node's child data. */
|
|
584
|
+
_getNodeChildren(node) {
|
|
585
|
+
return this._getDirectChildren(node.data).pipe(map(children => children.reduce((nodes, child) => {
|
|
586
|
+
const value = this._nodes.value.get(this._getExpansionKey(child));
|
|
587
|
+
if (value) {
|
|
588
|
+
nodes.push(value);
|
|
589
|
+
}
|
|
590
|
+
return nodes;
|
|
591
|
+
}, [])));
|
|
592
|
+
}
|
|
593
|
+
/** `keydown` event handler; this just passes the event to the `TreeKeyManager`. */
|
|
594
|
+
_sendKeydownToKeyManager(event) {
|
|
595
|
+
this._keyManager.onKeydown(event);
|
|
596
|
+
}
|
|
597
|
+
/** Gets all nested descendants of a given node. */
|
|
598
|
+
_getDescendants(dataNode) {
|
|
599
|
+
if (this.treeControl) {
|
|
600
|
+
return observableOf(this.treeControl.getDescendants(dataNode));
|
|
601
|
+
}
|
|
602
|
+
if (this.levelAccessor) {
|
|
603
|
+
const results = this._findChildrenByLevel(this.levelAccessor, this._flattenedNodes.value, dataNode, Infinity);
|
|
604
|
+
return observableOf(results);
|
|
605
|
+
}
|
|
606
|
+
if (this.childrenAccessor) {
|
|
607
|
+
return this._getAllChildrenRecursively(dataNode).pipe(reduce((allChildren, nextChildren) => {
|
|
608
|
+
allChildren.push(...nextChildren);
|
|
609
|
+
return allChildren;
|
|
610
|
+
}, []));
|
|
611
|
+
}
|
|
612
|
+
throw getTreeControlMissingError();
|
|
613
|
+
}
|
|
614
|
+
/**
|
|
615
|
+
* Gets all children and sub-children of the provided node.
|
|
616
|
+
*
|
|
617
|
+
* This will emit multiple times, in the order that the children will appear
|
|
618
|
+
* in the tree, and can be combined with a `reduce` operator.
|
|
619
|
+
*/
|
|
620
|
+
_getAllChildrenRecursively(dataNode) {
|
|
621
|
+
if (!this.childrenAccessor) {
|
|
622
|
+
return observableOf([]);
|
|
623
|
+
}
|
|
624
|
+
return coerceObservable(this.childrenAccessor(dataNode)).pipe(take(1), switchMap(children => {
|
|
625
|
+
// Here, we cache the parents of a particular child so that we can compute the levels.
|
|
626
|
+
for (const child of children) {
|
|
627
|
+
this._parents.set(this._getExpansionKey(child), dataNode);
|
|
628
|
+
}
|
|
629
|
+
return observableOf(...children).pipe(concatMap(child => concat(observableOf([child]), this._getAllChildrenRecursively(child))));
|
|
630
|
+
}));
|
|
631
|
+
}
|
|
632
|
+
_getExpansionKey(dataNode) {
|
|
633
|
+
// In the case that a key accessor function was not provided by the
|
|
634
|
+
// tree user, we'll default to using the node object itself as the key.
|
|
635
|
+
//
|
|
636
|
+
// This cast is safe since:
|
|
637
|
+
// - if an expansionKey is provided, TS will infer the type of K to be
|
|
638
|
+
// the return type.
|
|
639
|
+
// - if it's not, then K will be defaulted to T.
|
|
640
|
+
return this.expansionKey?.(dataNode) ?? dataNode;
|
|
641
|
+
}
|
|
642
|
+
_getAriaSet(node) {
|
|
643
|
+
const key = this._getExpansionKey(node);
|
|
644
|
+
const parent = this._parents.get(key);
|
|
645
|
+
const parentKey = parent ? this._getExpansionKey(parent) : null;
|
|
646
|
+
const set = this._ariaSets.get(parentKey);
|
|
647
|
+
return set ?? [node];
|
|
648
|
+
}
|
|
649
|
+
/**
|
|
650
|
+
* Finds the parent for the given node. If this is a root node, this
|
|
651
|
+
* returns null. If we're unable to determine the parent, for example,
|
|
652
|
+
* if we don't have cached node data, this returns undefined.
|
|
653
|
+
*/
|
|
654
|
+
_findParentForNode(node, index, cachedNodes) {
|
|
655
|
+
// In all cases, we have a mapping from node to level; all we need to do here is backtrack in
|
|
656
|
+
// our flattened list of nodes to determine the first node that's of a level lower than the
|
|
657
|
+
// provided node.
|
|
658
|
+
if (!cachedNodes.length) {
|
|
659
|
+
return null;
|
|
660
|
+
}
|
|
661
|
+
const currentLevel = this._levels.get(this._getExpansionKey(node)) ?? 0;
|
|
662
|
+
for (let parentIndex = index - 1; parentIndex >= 0; parentIndex--) {
|
|
663
|
+
const parentNode = cachedNodes[parentIndex];
|
|
664
|
+
const parentLevel = this._levels.get(this._getExpansionKey(parentNode)) ?? 0;
|
|
665
|
+
if (parentLevel < currentLevel) {
|
|
666
|
+
return parentNode;
|
|
667
|
+
}
|
|
668
|
+
}
|
|
669
|
+
return null;
|
|
670
|
+
}
|
|
671
|
+
/**
|
|
672
|
+
* Given a set of root nodes and the current node level, flattens any nested
|
|
673
|
+
* nodes into a single array.
|
|
674
|
+
*
|
|
675
|
+
* If any nodes are not expanded, then their children will not be added into the array.
|
|
676
|
+
* This will still traverse all nested children in order to build up our internal data
|
|
677
|
+
* models, but will not include them in the returned array.
|
|
678
|
+
*/
|
|
679
|
+
_flattenNestedNodesWithExpansion(nodes, level = 0) {
|
|
680
|
+
const childrenAccessor = this._getChildrenAccessor();
|
|
681
|
+
// If we're using a level accessor, we don't need to flatten anything.
|
|
682
|
+
if (!childrenAccessor) {
|
|
683
|
+
return observableOf([...nodes]);
|
|
684
|
+
}
|
|
685
|
+
return observableOf(...nodes).pipe(concatMap(node => {
|
|
686
|
+
const parentKey = this._getExpansionKey(node);
|
|
687
|
+
if (!this._parents.has(parentKey)) {
|
|
688
|
+
this._parents.set(parentKey, null);
|
|
689
|
+
}
|
|
690
|
+
this._levels.set(parentKey, level);
|
|
691
|
+
const children = coerceObservable(childrenAccessor(node));
|
|
692
|
+
return concat(observableOf([node]), children.pipe(take(1), tap(childNodes => {
|
|
693
|
+
this._ariaSets.set(parentKey, [...(childNodes ?? [])]);
|
|
694
|
+
for (const child of childNodes ?? []) {
|
|
695
|
+
const childKey = this._getExpansionKey(child);
|
|
696
|
+
this._parents.set(childKey, node);
|
|
697
|
+
this._levels.set(childKey, level + 1);
|
|
698
|
+
}
|
|
699
|
+
}), switchMap(childNodes => {
|
|
700
|
+
if (!childNodes) {
|
|
701
|
+
return observableOf([]);
|
|
702
|
+
}
|
|
703
|
+
return this._flattenNestedNodesWithExpansion(childNodes, level + 1).pipe(map(nestedNodes => (this.isExpanded(node) ? nestedNodes : [])));
|
|
704
|
+
})));
|
|
705
|
+
}), reduce((results, children) => {
|
|
706
|
+
results.push(...children);
|
|
707
|
+
return results;
|
|
708
|
+
}, []));
|
|
709
|
+
}
|
|
710
|
+
/**
|
|
711
|
+
* Converts children for certain tree configurations.
|
|
712
|
+
*
|
|
713
|
+
* This also computes parent, level, and group data.
|
|
714
|
+
*/
|
|
715
|
+
_computeRenderingData(nodes, nodeType) {
|
|
716
|
+
// The only situations where we have to convert children types is when
|
|
717
|
+
// they're mismatched; i.e. if the tree is using a childrenAccessor and the
|
|
718
|
+
// nodes are flat, or if the tree is using a levelAccessor and the nodes are
|
|
719
|
+
// nested.
|
|
720
|
+
if (this.childrenAccessor && nodeType === 'flat') {
|
|
721
|
+
// This flattens children into a single array.
|
|
722
|
+
this._ariaSets.set(null, [...nodes]);
|
|
723
|
+
return this._flattenNestedNodesWithExpansion(nodes).pipe(map(flattenedNodes => ({
|
|
724
|
+
renderNodes: flattenedNodes,
|
|
725
|
+
flattenedNodes,
|
|
726
|
+
})));
|
|
727
|
+
}
|
|
728
|
+
else if (this.levelAccessor && nodeType === 'nested') {
|
|
729
|
+
// In the nested case, we only look for root nodes. The CdkNestedNode
|
|
730
|
+
// itself will handle rendering each individual node's children.
|
|
731
|
+
const levelAccessor = this.levelAccessor;
|
|
732
|
+
return observableOf(nodes.filter(node => levelAccessor(node) === 0)).pipe(map(rootNodes => ({
|
|
733
|
+
renderNodes: rootNodes,
|
|
734
|
+
flattenedNodes: nodes,
|
|
735
|
+
})), tap(({ flattenedNodes }) => {
|
|
736
|
+
this._calculateParents(flattenedNodes);
|
|
737
|
+
}));
|
|
738
|
+
}
|
|
739
|
+
else if (nodeType === 'flat') {
|
|
740
|
+
// In the case of a TreeControl, we know that the node type matches up
|
|
741
|
+
// with the TreeControl, and so no conversions are necessary. Otherwise,
|
|
742
|
+
// we've already confirmed that the data model matches up with the
|
|
743
|
+
// desired node type here.
|
|
744
|
+
return observableOf({ renderNodes: nodes, flattenedNodes: nodes }).pipe(tap(({ flattenedNodes }) => {
|
|
745
|
+
this._calculateParents(flattenedNodes);
|
|
746
|
+
}));
|
|
747
|
+
}
|
|
748
|
+
else {
|
|
749
|
+
// For nested nodes, we still need to perform the node flattening in order
|
|
750
|
+
// to maintain our caches for various tree operations.
|
|
751
|
+
this._ariaSets.set(null, [...nodes]);
|
|
752
|
+
return this._flattenNestedNodesWithExpansion(nodes).pipe(map(flattenedNodes => ({
|
|
753
|
+
renderNodes: nodes,
|
|
754
|
+
flattenedNodes,
|
|
755
|
+
})));
|
|
756
|
+
}
|
|
757
|
+
}
|
|
758
|
+
_updateCachedData(flattenedNodes) {
|
|
759
|
+
this._flattenedNodes.next(flattenedNodes);
|
|
760
|
+
}
|
|
761
|
+
_updateKeyManagerItems(flattenedNodes) {
|
|
762
|
+
this._keyManagerNodes.next(flattenedNodes);
|
|
763
|
+
}
|
|
764
|
+
/** Traverse the flattened node data and compute parents, levels, and group data. */
|
|
765
|
+
_calculateParents(flattenedNodes) {
|
|
766
|
+
const levelAccessor = this._getLevelAccessor();
|
|
767
|
+
if (!levelAccessor) {
|
|
768
|
+
return;
|
|
769
|
+
}
|
|
770
|
+
this._parents.clear();
|
|
771
|
+
this._ariaSets.clear();
|
|
772
|
+
for (let index = 0; index < flattenedNodes.length; index++) {
|
|
773
|
+
const dataNode = flattenedNodes[index];
|
|
774
|
+
const key = this._getExpansionKey(dataNode);
|
|
775
|
+
this._levels.set(key, levelAccessor(dataNode));
|
|
776
|
+
const parent = this._findParentForNode(dataNode, index, flattenedNodes);
|
|
777
|
+
this._parents.set(key, parent);
|
|
778
|
+
const parentKey = parent ? this._getExpansionKey(parent) : null;
|
|
779
|
+
const group = this._ariaSets.get(parentKey) ?? [];
|
|
780
|
+
group.splice(index, 0, dataNode);
|
|
781
|
+
this._ariaSets.set(parentKey, group);
|
|
782
|
+
}
|
|
783
|
+
}
|
|
784
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: CdkTree, deps: [{ token: i0.IterableDiffers }, { token: i0.ChangeDetectorRef }, { token: i1.Directionality }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
785
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.1.0", type: CdkTree, isStandalone: true, selector: "cdk-tree", inputs: { dataSource: "dataSource", treeControl: "treeControl", levelAccessor: "levelAccessor", childrenAccessor: "childrenAccessor", trackBy: "trackBy", expansionKey: "expansionKey" }, host: { attributes: { "role": "tree" }, listeners: { "keydown": "_sendKeydownToKeyManager($event)" }, classAttribute: "cdk-tree" }, queries: [{ propertyName: "_nodeDefs", predicate: CdkTreeNodeDef, descendants: true }], viewQueries: [{ propertyName: "_nodeOutlet", first: true, predicate: CdkTreeNodeOutlet, descendants: true, static: true }], exportAs: ["cdkTree"], ngImport: i0, template: `<ng-container cdkTreeNodeOutlet></ng-container>`, isInline: true, dependencies: [{ kind: "directive", type: CdkTreeNodeOutlet, selector: "[cdkTreeNodeOutlet]" }], changeDetection: i0.ChangeDetectionStrategy.Default, encapsulation: i0.ViewEncapsulation.None }); }
|
|
190
786
|
}
|
|
191
787
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: CdkTree, decorators: [{
|
|
192
788
|
type: Component,
|
|
@@ -197,6 +793,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
|
|
|
197
793
|
host: {
|
|
198
794
|
'class': 'cdk-tree',
|
|
199
795
|
'role': 'tree',
|
|
796
|
+
'(keydown)': '_sendKeydownToKeyManager($event)',
|
|
200
797
|
},
|
|
201
798
|
encapsulation: ViewEncapsulation.None,
|
|
202
799
|
// The "OnPush" status for the `CdkTree` component is effectively a noop, so we are removing it.
|
|
@@ -207,12 +804,18 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
|
|
|
207
804
|
standalone: true,
|
|
208
805
|
imports: [CdkTreeNodeOutlet],
|
|
209
806
|
}]
|
|
210
|
-
}], ctorParameters: () => [{ type: i0.IterableDiffers }, { type: i0.ChangeDetectorRef }], propDecorators: { dataSource: [{
|
|
807
|
+
}], ctorParameters: () => [{ type: i0.IterableDiffers }, { type: i0.ChangeDetectorRef }, { type: i1.Directionality }], propDecorators: { dataSource: [{
|
|
211
808
|
type: Input
|
|
212
809
|
}], treeControl: [{
|
|
213
810
|
type: Input
|
|
811
|
+
}], levelAccessor: [{
|
|
812
|
+
type: Input
|
|
813
|
+
}], childrenAccessor: [{
|
|
814
|
+
type: Input
|
|
214
815
|
}], trackBy: [{
|
|
215
816
|
type: Input
|
|
817
|
+
}], expansionKey: [{
|
|
818
|
+
type: Input
|
|
216
819
|
}], _nodeOutlet: [{
|
|
217
820
|
type: ViewChild,
|
|
218
821
|
args: [CdkTreeNodeOutlet, { static: true }]
|
|
@@ -230,16 +833,42 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
|
|
|
230
833
|
export class CdkTreeNode {
|
|
231
834
|
/**
|
|
232
835
|
* The role of the tree node.
|
|
233
|
-
*
|
|
234
|
-
*
|
|
235
|
-
*
|
|
836
|
+
*
|
|
837
|
+
* @deprecated This will be ignored; the tree will automatically determine the appropriate role
|
|
838
|
+
* for tree node. This input will be removed in a future version.
|
|
839
|
+
* @breaking-change 21.0.0
|
|
236
840
|
*/
|
|
237
841
|
get role() {
|
|
238
842
|
return 'treeitem';
|
|
239
843
|
}
|
|
240
844
|
set role(_role) {
|
|
241
|
-
//
|
|
242
|
-
|
|
845
|
+
// ignore any role setting, we handle this internally.
|
|
846
|
+
}
|
|
847
|
+
/**
|
|
848
|
+
* Whether or not this node is expandable.
|
|
849
|
+
*
|
|
850
|
+
* If not using `FlatTreeControl`, or if `isExpandable` is not provided to
|
|
851
|
+
* `NestedTreeControl`, this should be provided for correct node a11y.
|
|
852
|
+
*/
|
|
853
|
+
get isExpandable() {
|
|
854
|
+
return this._isExpandable();
|
|
855
|
+
}
|
|
856
|
+
set isExpandable(isExpandable) {
|
|
857
|
+
this._inputIsExpandable = isExpandable;
|
|
858
|
+
}
|
|
859
|
+
get isExpanded() {
|
|
860
|
+
return this._tree.isExpanded(this._data);
|
|
861
|
+
}
|
|
862
|
+
set isExpanded(isExpanded) {
|
|
863
|
+
if (isExpanded) {
|
|
864
|
+
this.expand();
|
|
865
|
+
}
|
|
866
|
+
else {
|
|
867
|
+
this.collapse();
|
|
868
|
+
}
|
|
869
|
+
}
|
|
870
|
+
getLabel() {
|
|
871
|
+
return this.typeaheadLabel || this._elementRef.nativeElement.textContent?.trim() || '';
|
|
243
872
|
}
|
|
244
873
|
/**
|
|
245
874
|
* The most recently created `CdkTreeNode`. We save it in static variable so we can retrieve it
|
|
@@ -253,40 +882,101 @@ export class CdkTreeNode {
|
|
|
253
882
|
set data(value) {
|
|
254
883
|
if (value !== this._data) {
|
|
255
884
|
this._data = value;
|
|
256
|
-
this._setRoleFromData();
|
|
257
885
|
this._dataChanges.next();
|
|
258
886
|
}
|
|
259
887
|
}
|
|
260
|
-
|
|
261
|
-
|
|
888
|
+
/* If leaf node, return true to not assign aria-expanded attribute */
|
|
889
|
+
get isLeafNode() {
|
|
890
|
+
// If flat tree node data returns false for expandable property, it's a leaf node
|
|
891
|
+
if (this._tree.treeControl?.isExpandable !== undefined &&
|
|
892
|
+
!this._tree.treeControl.isExpandable(this._data)) {
|
|
893
|
+
return true;
|
|
894
|
+
// If nested tree node data returns 0 descendants, it's a leaf node
|
|
895
|
+
}
|
|
896
|
+
else if (this._tree.treeControl?.isExpandable === undefined &&
|
|
897
|
+
this._tree.treeControl?.getDescendants(this._data).length === 0) {
|
|
898
|
+
return true;
|
|
899
|
+
}
|
|
900
|
+
return false;
|
|
262
901
|
}
|
|
263
902
|
get level() {
|
|
264
|
-
// If the
|
|
903
|
+
// If the tree has a levelAccessor, use it to get the level. Otherwise read the
|
|
265
904
|
// aria-level off the parent node and use it as the level for this node (note aria-level is
|
|
266
905
|
// 1-indexed, while this property is 0-indexed, so we don't need to increment).
|
|
267
|
-
return this._tree.
|
|
268
|
-
|
|
269
|
-
|
|
906
|
+
return this._tree._getLevel(this._data) ?? this._parentNodeAriaLevel;
|
|
907
|
+
}
|
|
908
|
+
/** Determines if the tree node is expandable. */
|
|
909
|
+
_isExpandable() {
|
|
910
|
+
if (this._tree.treeControl) {
|
|
911
|
+
if (this.isLeafNode) {
|
|
912
|
+
return false;
|
|
913
|
+
}
|
|
914
|
+
// For compatibility with trees created using TreeControl before we added
|
|
915
|
+
// CdkTreeNode#isExpandable.
|
|
916
|
+
return true;
|
|
917
|
+
}
|
|
918
|
+
return this._inputIsExpandable;
|
|
919
|
+
}
|
|
920
|
+
/**
|
|
921
|
+
* Determines the value for `aria-expanded`.
|
|
922
|
+
*
|
|
923
|
+
* For non-expandable nodes, this is `null`.
|
|
924
|
+
*/
|
|
925
|
+
_getAriaExpanded() {
|
|
926
|
+
if (!this._isExpandable()) {
|
|
927
|
+
return null;
|
|
928
|
+
}
|
|
929
|
+
return String(this.isExpanded);
|
|
930
|
+
}
|
|
931
|
+
/**
|
|
932
|
+
* Determines the size of this node's parent's child set.
|
|
933
|
+
*
|
|
934
|
+
* This is intended to be used for `aria-setsize`.
|
|
935
|
+
*/
|
|
936
|
+
_getSetSize() {
|
|
937
|
+
return this._tree._getSetSize(this._data);
|
|
938
|
+
}
|
|
939
|
+
/**
|
|
940
|
+
* Determines the index (starting from 1) of this node in its parent's child set.
|
|
941
|
+
*
|
|
942
|
+
* This is intended to be used for `aria-posinset`.
|
|
943
|
+
*/
|
|
944
|
+
_getPositionInSet() {
|
|
945
|
+
return this._tree._getPositionInSet(this._data);
|
|
270
946
|
}
|
|
271
947
|
constructor(_elementRef, _tree) {
|
|
272
948
|
this._elementRef = _elementRef;
|
|
273
949
|
this._tree = _tree;
|
|
950
|
+
this._tabindex = -1;
|
|
951
|
+
/** This emits when the node has been programatically activated or activated by keyboard. */
|
|
952
|
+
this.activation = new EventEmitter();
|
|
953
|
+
/** This emits when the node's expansion status has been changed. */
|
|
954
|
+
this.expandedChange = new EventEmitter();
|
|
274
955
|
/** Subject that emits when the component has been destroyed. */
|
|
275
956
|
this._destroyed = new Subject();
|
|
276
957
|
/** Emits when the node's data has changed. */
|
|
277
958
|
this._dataChanges = new Subject();
|
|
959
|
+
this._inputIsExpandable = false;
|
|
960
|
+
/**
|
|
961
|
+
* Flag used to determine whether or not we should be focusing the actual element based on
|
|
962
|
+
* some user interaction (click or focus). On click, we don't forcibly focus the element
|
|
963
|
+
* since the click could trigger some other component that wants to grab its own focus
|
|
964
|
+
* (e.g. menu, dialog).
|
|
965
|
+
*/
|
|
966
|
+
this._shouldFocus = true;
|
|
278
967
|
this._changeDetectorRef = inject(ChangeDetectorRef);
|
|
279
968
|
CdkTreeNode.mostRecentTreeNode = this;
|
|
280
|
-
this.role = 'treeitem';
|
|
281
969
|
}
|
|
282
970
|
ngOnInit() {
|
|
283
971
|
this._parentNodeAriaLevel = getParentNodeAriaLevel(this._elementRef.nativeElement);
|
|
284
|
-
this.
|
|
285
|
-
|
|
286
|
-
.pipe(map(() => this.isExpanded), distinctUntilChanged())
|
|
972
|
+
this._tree
|
|
973
|
+
._getExpansionModel()
|
|
974
|
+
.changed.pipe(map(() => this.isExpanded), distinctUntilChanged())
|
|
287
975
|
.subscribe(() => {
|
|
288
976
|
this._changeDetectorRef.markForCheck();
|
|
289
977
|
});
|
|
978
|
+
this._tree._setNodeTypeIfUnset('flat');
|
|
979
|
+
this._tree._registerNode(this);
|
|
290
980
|
}
|
|
291
981
|
ngOnDestroy() {
|
|
292
982
|
// If this is the last tree node being destroyed,
|
|
@@ -298,21 +988,63 @@ export class CdkTreeNode {
|
|
|
298
988
|
this._destroyed.next();
|
|
299
989
|
this._destroyed.complete();
|
|
300
990
|
}
|
|
301
|
-
|
|
991
|
+
getParent() {
|
|
992
|
+
return this._tree._getNodeParent(this) ?? null;
|
|
993
|
+
}
|
|
994
|
+
getChildren() {
|
|
995
|
+
return this._tree._getNodeChildren(this);
|
|
996
|
+
}
|
|
997
|
+
/** Focuses this data node. Implemented for TreeKeyManagerItem. */
|
|
302
998
|
focus() {
|
|
303
|
-
this.
|
|
999
|
+
this._tabindex = 0;
|
|
1000
|
+
if (this._shouldFocus) {
|
|
1001
|
+
this._elementRef.nativeElement.focus();
|
|
1002
|
+
}
|
|
1003
|
+
this._changeDetectorRef.markForCheck();
|
|
1004
|
+
}
|
|
1005
|
+
/** Defocus this data node. */
|
|
1006
|
+
unfocus() {
|
|
1007
|
+
this._tabindex = -1;
|
|
1008
|
+
this._changeDetectorRef.markForCheck();
|
|
1009
|
+
}
|
|
1010
|
+
/** Emits an activation event. Implemented for TreeKeyManagerItem. */
|
|
1011
|
+
activate() {
|
|
1012
|
+
if (this.isDisabled) {
|
|
1013
|
+
return;
|
|
1014
|
+
}
|
|
1015
|
+
this.activation.next(this._data);
|
|
1016
|
+
}
|
|
1017
|
+
/** Collapses this data node. Implemented for TreeKeyManagerItem. */
|
|
1018
|
+
collapse() {
|
|
1019
|
+
if (this.isExpandable) {
|
|
1020
|
+
this._tree.collapse(this._data);
|
|
1021
|
+
}
|
|
1022
|
+
}
|
|
1023
|
+
/** Expands this data node. Implemented for TreeKeyManagerItem. */
|
|
1024
|
+
expand() {
|
|
1025
|
+
if (this.isExpandable) {
|
|
1026
|
+
this._tree.expand(this._data);
|
|
1027
|
+
}
|
|
304
1028
|
}
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
1029
|
+
_focusItem() {
|
|
1030
|
+
if (this.isDisabled) {
|
|
1031
|
+
return;
|
|
1032
|
+
}
|
|
1033
|
+
this._tree._keyManager.focusItem(this);
|
|
1034
|
+
}
|
|
1035
|
+
_setActiveItem() {
|
|
1036
|
+
if (this.isDisabled) {
|
|
1037
|
+
return;
|
|
311
1038
|
}
|
|
312
|
-
this.
|
|
1039
|
+
this._shouldFocus = false;
|
|
1040
|
+
this._tree._keyManager.focusItem(this);
|
|
1041
|
+
this._shouldFocus = true;
|
|
1042
|
+
}
|
|
1043
|
+
_emitExpansionState(expanded) {
|
|
1044
|
+
this.expandedChange.emit(expanded);
|
|
313
1045
|
}
|
|
314
1046
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: CdkTreeNode, deps: [{ token: i0.ElementRef }, { token: CdkTree }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
315
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "
|
|
1047
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "16.1.0", version: "18.1.0", type: CdkTreeNode, isStandalone: true, selector: "cdk-tree-node", inputs: { role: "role", isExpandable: ["isExpandable", "isExpandable", booleanAttribute], isExpanded: "isExpanded", isDisabled: ["isDisabled", "isDisabled", booleanAttribute], typeaheadLabel: ["cdkTreeNodeTypeaheadLabel", "typeaheadLabel"] }, outputs: { activation: "activation", expandedChange: "expandedChange" }, host: { attributes: { "role": "treeitem" }, listeners: { "click": "_setActiveItem()", "focus": "_focusItem()" }, properties: { "attr.aria-expanded": "_getAriaExpanded()", "attr.aria-level": "level + 1", "attr.aria-posinset": "_getPositionInSet()", "attr.aria-setsize": "_getSetSize()", "tabindex": "_tabindex" }, classAttribute: "cdk-tree-node" }, exportAs: ["cdkTreeNode"], ngImport: i0 }); }
|
|
316
1048
|
}
|
|
317
1049
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: CdkTreeNode, decorators: [{
|
|
318
1050
|
type: Directive,
|
|
@@ -321,12 +1053,34 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
|
|
|
321
1053
|
exportAs: 'cdkTreeNode',
|
|
322
1054
|
host: {
|
|
323
1055
|
'class': 'cdk-tree-node',
|
|
324
|
-
'[attr.aria-expanded]': '
|
|
1056
|
+
'[attr.aria-expanded]': '_getAriaExpanded()',
|
|
1057
|
+
'[attr.aria-level]': 'level + 1',
|
|
1058
|
+
'[attr.aria-posinset]': '_getPositionInSet()',
|
|
1059
|
+
'[attr.aria-setsize]': '_getSetSize()',
|
|
1060
|
+
'[tabindex]': '_tabindex',
|
|
1061
|
+
'role': 'treeitem',
|
|
1062
|
+
'(click)': '_setActiveItem()',
|
|
1063
|
+
'(focus)': '_focusItem()',
|
|
325
1064
|
},
|
|
326
1065
|
standalone: true,
|
|
327
1066
|
}]
|
|
328
1067
|
}], ctorParameters: () => [{ type: i0.ElementRef }, { type: CdkTree }], propDecorators: { role: [{
|
|
329
1068
|
type: Input
|
|
1069
|
+
}], isExpandable: [{
|
|
1070
|
+
type: Input,
|
|
1071
|
+
args: [{ transform: booleanAttribute }]
|
|
1072
|
+
}], isExpanded: [{
|
|
1073
|
+
type: Input
|
|
1074
|
+
}], isDisabled: [{
|
|
1075
|
+
type: Input,
|
|
1076
|
+
args: [{ transform: booleanAttribute }]
|
|
1077
|
+
}], typeaheadLabel: [{
|
|
1078
|
+
type: Input,
|
|
1079
|
+
args: ['cdkTreeNodeTypeaheadLabel']
|
|
1080
|
+
}], activation: [{
|
|
1081
|
+
type: Output
|
|
1082
|
+
}], expandedChange: [{
|
|
1083
|
+
type: Output
|
|
330
1084
|
}] } });
|
|
331
1085
|
function getParentNodeAriaLevel(nodeElement) {
|
|
332
1086
|
let parent = nodeElement.parentElement;
|
|
@@ -353,4 +1107,4 @@ function isNodeElement(element) {
|
|
|
353
1107
|
const classList = element.classList;
|
|
354
1108
|
return !!(classList?.contains('cdk-nested-tree-node') || classList?.contains('cdk-tree'));
|
|
355
1109
|
}
|
|
356
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree.js","sourceRoot":"","sources":["../../../../../../src/cdk/tree/tree.ts"],"names":[],"mappings":"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","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.io/license\n */\nimport {FocusableOption} from '@angular/cdk/a11y';\nimport {CollectionViewer, DataSource, isDataSource} from '@angular/cdk/collections';\nimport {\n  AfterContentChecked,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  Directive,\n  ElementRef,\n  Input,\n  IterableChangeRecord,\n  IterableDiffer,\n  IterableDiffers,\n  OnDestroy,\n  OnInit,\n  QueryList,\n  TrackByFunction,\n  ViewChild,\n  ViewContainerRef,\n  ViewEncapsulation,\n  inject,\n  numberAttribute,\n} from '@angular/core';\nimport {\n  BehaviorSubject,\n  Observable,\n  Subject,\n  Subscription,\n  isObservable,\n  of as observableOf,\n} from 'rxjs';\nimport {distinctUntilChanged, map, takeUntil} from 'rxjs/operators';\nimport {TreeControl} from './control/tree-control';\nimport {CdkTreeNodeDef, CdkTreeNodeOutletContext} from './node';\nimport {CdkTreeNodeOutlet} from './outlet';\nimport {\n  getTreeControlFunctionsMissingError,\n  getTreeControlMissingError,\n  getTreeMissingMatchingNodeDefError,\n  getTreeMultipleDefaultNodeDefsError,\n  getTreeNoValidDataSourceError,\n} from './tree-errors';\n\n/**\n * CDK tree component that connects with a data source to retrieve data of type `T` and renders\n * dataNodes with hierarchy. Updates the dataNodes when new data is provided by the data source.\n */\n@Component({\n  selector: 'cdk-tree',\n  exportAs: 'cdkTree',\n  template: `<ng-container cdkTreeNodeOutlet></ng-container>`,\n  host: {\n    'class': 'cdk-tree',\n    'role': 'tree',\n  },\n  encapsulation: ViewEncapsulation.None,\n  // The \"OnPush\" status for the `CdkTree` component is effectively a noop, so we are removing it.\n  // The view for `CdkTree` consists entirely of templates declared in other views. As they are\n  // declared elsewhere, they are checked when their declaration points are checked.\n  // tslint:disable-next-line:validate-decorators\n  changeDetection: ChangeDetectionStrategy.Default,\n  standalone: true,\n  imports: [CdkTreeNodeOutlet],\n})\nexport class CdkTree<T, K = T> implements AfterContentChecked, CollectionViewer, OnDestroy, OnInit {\n  /** Subject that emits when the component has been destroyed. */\n  private readonly _onDestroy = new Subject<void>();\n\n  /** Differ used to find the changes in the data provided by the data source. */\n  private _dataDiffer: IterableDiffer<T>;\n\n  /** Stores the node definition that does not have a when predicate. */\n  private _defaultNodeDef: CdkTreeNodeDef<T> | null;\n\n  /** Data subscription */\n  private _dataSubscription: Subscription | null;\n\n  /** Level of nodes */\n  private _levels: Map<T, number> = new Map<T, number>();\n\n  /**\n   * Provides a stream containing the latest data array to render. Influenced by the tree's\n   * stream of view window (what dataNodes are currently on screen).\n   * Data source can be an observable of data array, or a data array to render.\n   */\n  @Input()\n  get dataSource(): DataSource<T> | Observable<T[]> | T[] {\n    return this._dataSource;\n  }\n  set dataSource(dataSource: DataSource<T> | Observable<T[]> | T[]) {\n    if (this._dataSource !== dataSource) {\n      this._switchDataSource(dataSource);\n    }\n  }\n  private _dataSource: DataSource<T> | Observable<T[]> | T[];\n\n  /** The tree controller */\n  @Input() treeControl: TreeControl<T, K>;\n\n  /**\n   * Tracking function that will be used to check the differences in data changes. Used similarly\n   * to `ngFor` `trackBy` function. Optimize node operations by identifying a node based on its data\n   * relative to the function to know if a node should be added/removed/moved.\n   * Accepts a function that takes two parameters, `index` and `item`.\n   */\n  @Input() trackBy: TrackByFunction<T>;\n\n  // Outlets within the tree's template where the dataNodes will be inserted.\n  @ViewChild(CdkTreeNodeOutlet, {static: true}) _nodeOutlet: CdkTreeNodeOutlet;\n\n  /** The tree node template for the tree */\n  @ContentChildren(CdkTreeNodeDef, {\n    // We need to use `descendants: true`, because Ivy will no longer match\n    // indirect descendants if it's left as false.\n    descendants: true,\n  })\n  _nodeDefs: QueryList<CdkTreeNodeDef<T>>;\n\n  // TODO(tinayuangao): Setup a listener for scrolling, emit the calculated view to viewChange.\n  //     Remove the MAX_VALUE in viewChange\n  /**\n   * Stream containing the latest information on what rows are being displayed on screen.\n   * Can be used by the data source to as a heuristic of what data should be provided.\n   */\n  readonly viewChange = new BehaviorSubject<{start: number; end: number}>({\n    start: 0,\n    end: Number.MAX_VALUE,\n  });\n\n  constructor(\n    private _differs: IterableDiffers,\n    private _changeDetectorRef: ChangeDetectorRef,\n  ) {}\n\n  ngOnInit() {\n    this._dataDiffer = this._differs.find([]).create(this.trackBy);\n    if (!this.treeControl && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw getTreeControlMissingError();\n    }\n  }\n\n  ngOnDestroy() {\n    this._nodeOutlet.viewContainer.clear();\n\n    this.viewChange.complete();\n    this._onDestroy.next();\n    this._onDestroy.complete();\n\n    if (this._dataSource && typeof (this._dataSource as DataSource<T>).disconnect === 'function') {\n      (this.dataSource as DataSource<T>).disconnect(this);\n    }\n\n    if (this._dataSubscription) {\n      this._dataSubscription.unsubscribe();\n      this._dataSubscription = null;\n    }\n  }\n\n  ngAfterContentChecked() {\n    const defaultNodeDefs = this._nodeDefs.filter(def => !def.when);\n    if (defaultNodeDefs.length > 1 && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw getTreeMultipleDefaultNodeDefsError();\n    }\n    this._defaultNodeDef = defaultNodeDefs[0];\n\n    if (this.dataSource && this._nodeDefs && !this._dataSubscription) {\n      this._observeRenderChanges();\n    }\n  }\n\n  // TODO(tinayuangao): Work on keyboard traversal and actions, make sure it's working for RTL\n  //     and nested trees.\n\n  /**\n   * Switch to the provided data source by resetting the data and unsubscribing from the current\n   * render change subscription if one exists. If the data source is null, interpret this by\n   * clearing the node outlet. Otherwise start listening for new data.\n   */\n  private _switchDataSource(dataSource: DataSource<T> | Observable<T[]> | T[]) {\n    if (this._dataSource && typeof (this._dataSource as DataSource<T>).disconnect === 'function') {\n      (this.dataSource as DataSource<T>).disconnect(this);\n    }\n\n    if (this._dataSubscription) {\n      this._dataSubscription.unsubscribe();\n      this._dataSubscription = null;\n    }\n\n    // Remove the all dataNodes if there is now no data source\n    if (!dataSource) {\n      this._nodeOutlet.viewContainer.clear();\n    }\n\n    this._dataSource = dataSource;\n    if (this._nodeDefs) {\n      this._observeRenderChanges();\n    }\n  }\n\n  /** Set up a subscription for the data provided by the data source. */\n  private _observeRenderChanges() {\n    let dataStream: Observable<readonly T[]> | undefined;\n\n    if (isDataSource(this._dataSource)) {\n      dataStream = this._dataSource.connect(this);\n    } else if (isObservable(this._dataSource)) {\n      dataStream = this._dataSource;\n    } else if (Array.isArray(this._dataSource)) {\n      dataStream = observableOf(this._dataSource);\n    }\n\n    if (dataStream) {\n      this._dataSubscription = dataStream\n        .pipe(takeUntil(this._onDestroy))\n        .subscribe(data => this.renderNodeChanges(data));\n    } else if (typeof ngDevMode === 'undefined' || ngDevMode) {\n      throw getTreeNoValidDataSourceError();\n    }\n  }\n\n  /** Check for changes made in the data and render each change (node added/removed/moved). */\n  renderNodeChanges(\n    data: readonly T[],\n    dataDiffer: IterableDiffer<T> = this._dataDiffer,\n    viewContainer: ViewContainerRef = this._nodeOutlet.viewContainer,\n    parentData?: T,\n  ) {\n    const changes = dataDiffer.diff(data);\n    if (!changes) {\n      return;\n    }\n\n    changes.forEachOperation(\n      (\n        item: IterableChangeRecord<T>,\n        adjustedPreviousIndex: number | null,\n        currentIndex: number | null,\n      ) => {\n        if (item.previousIndex == null) {\n          this.insertNode(data[currentIndex!], currentIndex!, viewContainer, parentData);\n        } else if (currentIndex == null) {\n          viewContainer.remove(adjustedPreviousIndex!);\n          this._levels.delete(item.item);\n        } else {\n          const view = viewContainer.get(adjustedPreviousIndex!);\n          viewContainer.move(view!, currentIndex);\n        }\n      },\n    );\n\n    // TODO: change to `this._changeDetectorRef.markForCheck()`, or just switch this component to\n    // use signals.\n    this._changeDetectorRef.detectChanges();\n  }\n\n  /**\n   * Finds the matching node definition that should be used for this node data. If there is only\n   * one node definition, it is returned. Otherwise, find the node definition that has a when\n   * predicate that returns true with the data. If none return true, return the default node\n   * definition.\n   */\n  _getNodeDef(data: T, i: number): CdkTreeNodeDef<T> {\n    if (this._nodeDefs.length === 1) {\n      return this._nodeDefs.first!;\n    }\n\n    const nodeDef =\n      this._nodeDefs.find(def => def.when && def.when(i, data)) || this._defaultNodeDef;\n\n    if (!nodeDef && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw getTreeMissingMatchingNodeDefError();\n    }\n\n    return nodeDef!;\n  }\n\n  /**\n   * Create the embedded view for the data node template and place it in the correct index location\n   * within the data node view container.\n   */\n  insertNode(nodeData: T, index: number, viewContainer?: ViewContainerRef, parentData?: T) {\n    const node = this._getNodeDef(nodeData, index);\n\n    // Node context that will be provided to created embedded view\n    const context = new CdkTreeNodeOutletContext<T>(nodeData);\n\n    // If the tree is flat tree, then use the `getLevel` function in flat tree control\n    // Otherwise, use the level of parent node.\n    if (this.treeControl.getLevel) {\n      context.level = this.treeControl.getLevel(nodeData);\n    } else if (typeof parentData !== 'undefined' && this._levels.has(parentData)) {\n      context.level = this._levels.get(parentData)! + 1;\n    } else {\n      context.level = 0;\n    }\n    this._levels.set(nodeData, context.level);\n\n    // Use default tree nodeOutlet, or nested node's nodeOutlet\n    const container = viewContainer ? viewContainer : this._nodeOutlet.viewContainer;\n    container.createEmbeddedView(node.template, context, index);\n\n    // Set the data to just created `CdkTreeNode`.\n    // The `CdkTreeNode` created from `createEmbeddedView` will be saved in static variable\n    //     `mostRecentTreeNode`. We get it from static variable and pass the node data to it.\n    if (CdkTreeNode.mostRecentTreeNode) {\n      CdkTreeNode.mostRecentTreeNode.data = nodeData;\n    }\n  }\n}\n\n/**\n * Tree node for CdkTree. It contains the data in the tree node.\n */\n@Directive({\n  selector: 'cdk-tree-node',\n  exportAs: 'cdkTreeNode',\n  host: {\n    'class': 'cdk-tree-node',\n    '[attr.aria-expanded]': 'isExpanded',\n  },\n  standalone: true,\n})\nexport class CdkTreeNode<T, K = T> implements FocusableOption, OnDestroy, OnInit {\n  /**\n   * The role of the tree node.\n   * @deprecated The correct role is 'treeitem', 'group' should not be used. This input will be\n   *   removed in a future version.\n   * @breaking-change 12.0.0 Remove this input\n   */\n  @Input() get role(): 'treeitem' | 'group' {\n    return 'treeitem';\n  }\n\n  set role(_role: 'treeitem' | 'group') {\n    // TODO: move to host after View Engine deprecation\n    this._elementRef.nativeElement.setAttribute('role', _role);\n  }\n\n  /**\n   * The most recently created `CdkTreeNode`. We save it in static variable so we can retrieve it\n   * in `CdkTree` and set the data to it.\n   */\n  static mostRecentTreeNode: CdkTreeNode<any> | null = null;\n\n  /** Subject that emits when the component has been destroyed. */\n  protected readonly _destroyed = new Subject<void>();\n\n  /** Emits when the node's data has changed. */\n  readonly _dataChanges = new Subject<void>();\n\n  private _parentNodeAriaLevel: number;\n\n  /** The tree node's data. */\n  get data(): T {\n    return this._data;\n  }\n  set data(value: T) {\n    if (value !== this._data) {\n      this._data = value;\n      this._setRoleFromData();\n      this._dataChanges.next();\n    }\n  }\n  protected _data: T;\n\n  get isExpanded(): boolean {\n    return this._tree.treeControl.isExpanded(this._data);\n  }\n\n  get level(): number {\n    // If the treeControl has a getLevel method, use it to get the level. Otherwise read the\n    // aria-level off the parent node and use it as the level for this node (note aria-level is\n    // 1-indexed, while this property is 0-indexed, so we don't need to increment).\n    return this._tree.treeControl.getLevel\n      ? this._tree.treeControl.getLevel(this._data)\n      : this._parentNodeAriaLevel;\n  }\n\n  private _changeDetectorRef = inject(ChangeDetectorRef);\n\n  constructor(\n    protected _elementRef: ElementRef<HTMLElement>,\n    protected _tree: CdkTree<T, K>,\n  ) {\n    CdkTreeNode.mostRecentTreeNode = this as CdkTreeNode<T, K>;\n    this.role = 'treeitem';\n  }\n\n  ngOnInit(): void {\n    this._parentNodeAriaLevel = getParentNodeAriaLevel(this._elementRef.nativeElement);\n    this._elementRef.nativeElement.setAttribute('aria-level', `${this.level + 1}`);\n    this._tree.treeControl.expansionModel.changed\n      .pipe(\n        map(() => this.isExpanded),\n        distinctUntilChanged(),\n      )\n      .subscribe(() => {\n        this._changeDetectorRef.markForCheck();\n      });\n  }\n\n  ngOnDestroy() {\n    // If this is the last tree node being destroyed,\n    // clear out the reference to avoid leaking memory.\n    if (CdkTreeNode.mostRecentTreeNode === this) {\n      CdkTreeNode.mostRecentTreeNode = null;\n    }\n\n    this._dataChanges.complete();\n    this._destroyed.next();\n    this._destroyed.complete();\n  }\n\n  /** Focuses the menu item. Implements for FocusableOption. */\n  focus(): void {\n    this._elementRef.nativeElement.focus();\n  }\n\n  // TODO: role should eventually just be set in the component host\n  protected _setRoleFromData(): void {\n    if (\n      !this._tree.treeControl.isExpandable &&\n      !this._tree.treeControl.getChildren &&\n      (typeof ngDevMode === 'undefined' || ngDevMode)\n    ) {\n      throw getTreeControlFunctionsMissingError();\n    }\n    this.role = 'treeitem';\n  }\n}\n\nfunction getParentNodeAriaLevel(nodeElement: HTMLElement): number {\n  let parent = nodeElement.parentElement;\n  while (parent && !isNodeElement(parent)) {\n    parent = parent.parentElement;\n  }\n  if (!parent) {\n    if (typeof ngDevMode === 'undefined' || ngDevMode) {\n      throw Error('Incorrect tree structure containing detached node.');\n    } else {\n      return -1;\n    }\n  } else if (parent.classList.contains('cdk-nested-tree-node')) {\n    return numberAttribute(parent.getAttribute('aria-level')!);\n  } else {\n    // The ancestor element is the cdk-tree itself\n    return 0;\n  }\n}\n\nfunction isNodeElement(element: HTMLElement) {\n  const classList = element.classList;\n  return !!(classList?.contains('cdk-nested-tree-node') || classList?.contains('cdk-tree'));\n}\n"]}
|
|
1110
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree.js","sourceRoot":"","sources":["../../../../../../src/cdk/tree/tree.ts"],"names":[],"mappings":"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","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.io/license\n */\nimport {\n  TREE_KEY_MANAGER,\n  TreeKeyManagerFactory,\n  TreeKeyManagerItem,\n  TreeKeyManagerOptions,\n  TreeKeyManagerStrategy,\n} from '@angular/cdk/a11y';\nimport {Directionality} from '@angular/cdk/bidi';\nimport {\n  CollectionViewer,\n  DataSource,\n  isDataSource,\n  SelectionChange,\n  SelectionModel,\n} from '@angular/cdk/collections';\nimport {\n  AfterContentChecked,\n  AfterContentInit,\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  EmbeddedViewRef,\n  Input,\n  IterableChangeRecord,\n  IterableDiffer,\n  IterableDiffers,\n  OnDestroy,\n  OnInit,\n  Output,\n  QueryList,\n  TrackByFunction,\n  ViewChild,\n  ViewContainerRef,\n  ViewEncapsulation,\n  numberAttribute,\n  inject,\n  booleanAttribute,\n} from '@angular/core';\nimport {coerceObservable} from '@angular/cdk/coercion/private';\nimport {\n  BehaviorSubject,\n  combineLatest,\n  concat,\n  EMPTY,\n  Observable,\n  Subject,\n  Subscription,\n  isObservable,\n  of as observableOf,\n} from 'rxjs';\nimport {\n  distinctUntilChanged,\n  concatMap,\n  map,\n  reduce,\n  startWith,\n  switchMap,\n  take,\n  takeUntil,\n  tap,\n} from 'rxjs/operators';\nimport {TreeControl} from './control/tree-control';\nimport {CdkTreeNodeDef, CdkTreeNodeOutletContext} from './node';\nimport {CdkTreeNodeOutlet} from './outlet';\nimport {\n  getMultipleTreeControlsError,\n  getTreeControlMissingError,\n  getTreeMissingMatchingNodeDefError,\n  getTreeMultipleDefaultNodeDefsError,\n  getTreeNoValidDataSourceError,\n} from './tree-errors';\n\ntype RenderingData<T> =\n  | {\n      flattenedNodes: null;\n      nodeType: null;\n      renderNodes: readonly T[];\n    }\n  | {\n      flattenedNodes: readonly T[];\n      nodeType: 'nested' | 'flat';\n      renderNodes: readonly T[];\n    };\n\n/**\n * CDK tree component that connects with a data source to retrieve data of type `T` and renders\n * dataNodes with hierarchy. Updates the dataNodes when new data is provided by the data source.\n */\n@Component({\n  selector: 'cdk-tree',\n  exportAs: 'cdkTree',\n  template: `<ng-container cdkTreeNodeOutlet></ng-container>`,\n  host: {\n    'class': 'cdk-tree',\n    'role': 'tree',\n    '(keydown)': '_sendKeydownToKeyManager($event)',\n  },\n  encapsulation: ViewEncapsulation.None,\n  // The \"OnPush\" status for the `CdkTree` component is effectively a noop, so we are removing it.\n  // The view for `CdkTree` consists entirely of templates declared in other views. As they are\n  // declared elsewhere, they are checked when their declaration points are checked.\n  // tslint:disable-next-line:validate-decorators\n  changeDetection: ChangeDetectionStrategy.Default,\n  standalone: true,\n  imports: [CdkTreeNodeOutlet],\n})\nexport class CdkTree<T, K = T>\n  implements\n    AfterContentChecked,\n    AfterContentInit,\n    AfterViewInit,\n    CollectionViewer,\n    OnDestroy,\n    OnInit\n{\n  /** Subject that emits when the component has been destroyed. */\n  private readonly _onDestroy = new Subject<void>();\n\n  /** Differ used to find the changes in the data provided by the data source. */\n  private _dataDiffer: IterableDiffer<T>;\n\n  /** Stores the node definition that does not have a when predicate. */\n  private _defaultNodeDef: CdkTreeNodeDef<T> | null;\n\n  /** Data subscription */\n  private _dataSubscription: Subscription | null;\n\n  /** Level of nodes */\n  private _levels: Map<K, number> = new Map<K, number>();\n\n  /** The immediate parents for a node. This is `null` if there is no parent. */\n  private _parents: Map<K, T | null> = new Map<K, T | null>();\n\n  /**\n   * Nodes grouped into each set, which is a list of nodes displayed together in the DOM.\n   *\n   * Lookup key is the parent of a set. Root nodes have key of null.\n   *\n   * Values is a 'set' of tree nodes. Each tree node maps to a treeitem element. Sets are in the\n   * order that it is rendered. Each set maps directly to aria-posinset and aria-setsize attributes.\n   */\n  private _ariaSets: Map<K | null, T[]> = new Map<K | null, T[]>();\n\n  /**\n   * Provides a stream containing the latest data array to render. Influenced by the tree's\n   * stream of view window (what dataNodes are currently on screen).\n   * Data source can be an observable of data array, or a data array to render.\n   */\n  @Input()\n  get dataSource(): DataSource<T> | Observable<T[]> | T[] {\n    return this._dataSource;\n  }\n  set dataSource(dataSource: DataSource<T> | Observable<T[]> | T[]) {\n    if (this._dataSource !== dataSource) {\n      this._switchDataSource(dataSource);\n    }\n  }\n  private _dataSource: DataSource<T> | Observable<T[]> | T[];\n\n  /**\n   * The tree controller\n   *\n   * @deprecated Use one of `levelAccessor` or `childrenAccessor` instead. To be removed in a\n   * future version.\n   * @breaking-change 21.0.0\n   */\n  @Input() treeControl?: TreeControl<T, K>;\n\n  /**\n   * Given a data node, determines what tree level the node is at.\n   *\n   * One of levelAccessor or childrenAccessor must be specified, not both.\n   * This is enforced at run-time.\n   */\n  @Input() levelAccessor?: (dataNode: T) => number;\n\n  /**\n   * Given a data node, determines what the children of that node are.\n   *\n   * One of levelAccessor or childrenAccessor must be specified, not both.\n   * This is enforced at run-time.\n   */\n  @Input() childrenAccessor?: (dataNode: T) => T[] | Observable<T[]>;\n\n  /**\n   * Tracking function that will be used to check the differences in data changes. Used similarly\n   * to `ngFor` `trackBy` function. Optimize node operations by identifying a node based on its data\n   * relative to the function to know if a node should be added/removed/moved.\n   * Accepts a function that takes two parameters, `index` and `item`.\n   */\n  @Input() trackBy: TrackByFunction<T>;\n\n  /**\n   * Given a data node, determines the key by which we determine whether or not this node is expanded.\n   */\n  @Input() expansionKey?: (dataNode: T) => K;\n\n  // Outlets within the tree's template where the dataNodes will be inserted.\n  @ViewChild(CdkTreeNodeOutlet, {static: true}) _nodeOutlet: CdkTreeNodeOutlet;\n\n  /** The tree node template for the tree */\n  @ContentChildren(CdkTreeNodeDef, {\n    // We need to use `descendants: true`, because Ivy will no longer match\n    // indirect descendants if it's left as false.\n    descendants: true,\n  })\n  _nodeDefs: QueryList<CdkTreeNodeDef<T>>;\n\n  // TODO(tinayuangao): Setup a listener for scrolling, emit the calculated view to viewChange.\n  //     Remove the MAX_VALUE in viewChange\n  /**\n   * Stream containing the latest information on what rows are being displayed on screen.\n   * Can be used by the data source to as a heuristic of what data should be provided.\n   */\n  readonly viewChange = new BehaviorSubject<{start: number; end: number}>({\n    start: 0,\n    end: Number.MAX_VALUE,\n  });\n\n  /** Keep track of which nodes are expanded. */\n  private _expansionModel?: SelectionModel<K>;\n\n  /**\n   * Maintain a synchronous cache of flattened data nodes. This will only be\n   * populated after initial render, and in certain cases, will be delayed due to\n   * relying on Observable `getChildren` calls.\n   */\n  private _flattenedNodes: BehaviorSubject<readonly T[]> = new BehaviorSubject<readonly T[]>([]);\n\n  /** The automatically determined node type for the tree. */\n  private _nodeType: BehaviorSubject<'flat' | 'nested' | null> = new BehaviorSubject<\n    'flat' | 'nested' | null\n  >(null);\n\n  /** The mapping between data and the node that is rendered. */\n  private _nodes: BehaviorSubject<Map<K, CdkTreeNode<T, K>>> = new BehaviorSubject(\n    new Map<K, CdkTreeNode<T, K>>(),\n  );\n\n  /**\n   * Synchronous cache of nodes for the `TreeKeyManager`. This is separate\n   * from `_flattenedNodes` so they can be independently updated at different\n   * times.\n   */\n  private _keyManagerNodes: BehaviorSubject<readonly T[]> = new BehaviorSubject<readonly T[]>([]);\n\n  private _keyManagerFactory = inject(TREE_KEY_MANAGER) as TreeKeyManagerFactory<CdkTreeNode<T, K>>;\n\n  /** The key manager for this tree. Handles focus and activation based on user keyboard input. */\n  _keyManager: TreeKeyManagerStrategy<CdkTreeNode<T, K>>;\n  private _viewInit = false;\n\n  constructor(\n    private _differs: IterableDiffers,\n    private _changeDetectorRef: ChangeDetectorRef,\n    private _dir: Directionality,\n  ) {}\n\n  ngAfterContentInit() {\n    this._initializeKeyManager();\n  }\n\n  ngAfterContentChecked() {\n    this._updateDefaultNodeDefinition();\n    this._subscribeToDataChanges();\n  }\n\n  ngOnDestroy() {\n    this._nodeOutlet.viewContainer.clear();\n\n    this.viewChange.complete();\n    this._onDestroy.next();\n    this._onDestroy.complete();\n\n    if (this._dataSource && typeof (this._dataSource as DataSource<T>).disconnect === 'function') {\n      (this.dataSource as DataSource<T>).disconnect(this);\n    }\n\n    if (this._dataSubscription) {\n      this._dataSubscription.unsubscribe();\n      this._dataSubscription = null;\n    }\n\n    // In certain tests, the tree might be destroyed before this is initialized\n    // in `ngAfterContentInit`.\n    this._keyManager?.destroy();\n  }\n\n  ngOnInit() {\n    this._checkTreeControlUsage();\n    this._initializeDataDiffer();\n  }\n\n  ngAfterViewInit() {\n    this._viewInit = true;\n  }\n\n  private _updateDefaultNodeDefinition() {\n    const defaultNodeDefs = this._nodeDefs.filter(def => !def.when);\n    if (defaultNodeDefs.length > 1 && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw getTreeMultipleDefaultNodeDefsError();\n    }\n    this._defaultNodeDef = defaultNodeDefs[0];\n  }\n\n  /**\n   * Sets the node type for the tree, if it hasn't been set yet.\n   *\n   * This will be called by the first node that's rendered in order for the tree\n   * to determine what data transformations are required.\n   */\n  _setNodeTypeIfUnset(nodeType: 'flat' | 'nested') {\n    if (this._nodeType.value === null) {\n      this._nodeType.next(nodeType);\n    }\n  }\n\n  /**\n   * Switch to the provided data source by resetting the data and unsubscribing from the current\n   * render change subscription if one exists. If the data source is null, interpret this by\n   * clearing the node outlet. Otherwise start listening for new data.\n   */\n  private _switchDataSource(dataSource: DataSource<T> | Observable<T[]> | T[]) {\n    if (this._dataSource && typeof (this._dataSource as DataSource<T>).disconnect === 'function') {\n      (this.dataSource as DataSource<T>).disconnect(this);\n    }\n\n    if (this._dataSubscription) {\n      this._dataSubscription.unsubscribe();\n      this._dataSubscription = null;\n    }\n\n    // Remove the all dataNodes if there is now no data source\n    if (!dataSource) {\n      this._nodeOutlet.viewContainer.clear();\n    }\n\n    this._dataSource = dataSource;\n    if (this._nodeDefs) {\n      this._subscribeToDataChanges();\n    }\n  }\n\n  _getExpansionModel() {\n    if (!this.treeControl) {\n      this._expansionModel ??= new SelectionModel<K>(true);\n      return this._expansionModel;\n    }\n    return this.treeControl.expansionModel;\n  }\n\n  /** Set up a subscription for the data provided by the data source. */\n  private _subscribeToDataChanges() {\n    if (this._dataSubscription) {\n      return;\n    }\n\n    let dataStream: Observable<readonly T[]> | undefined;\n\n    if (isDataSource(this._dataSource)) {\n      dataStream = this._dataSource.connect(this);\n    } else if (isObservable(this._dataSource)) {\n      dataStream = this._dataSource;\n    } else if (Array.isArray(this._dataSource)) {\n      dataStream = observableOf(this._dataSource);\n    }\n\n    if (!dataStream) {\n      if (typeof ngDevMode === 'undefined' || ngDevMode) {\n        throw getTreeNoValidDataSourceError();\n      }\n      return;\n    }\n\n    this._dataSubscription = this._getRenderData(dataStream)\n      .pipe(takeUntil(this._onDestroy))\n      .subscribe(renderingData => {\n        this._renderDataChanges(renderingData);\n      });\n  }\n\n  /** Given an Observable containing a stream of the raw data, returns an Observable containing the RenderingData */\n  private _getRenderData(dataStream: Observable<readonly T[]>): Observable<RenderingData<T>> {\n    const expansionModel = this._getExpansionModel();\n    return combineLatest([\n      dataStream,\n      this._nodeType,\n      // We don't use the expansion data directly, however we add it here to essentially\n      // trigger data rendering when expansion changes occur.\n      expansionModel.changed.pipe(\n        startWith(null),\n        tap(expansionChanges => {\n          this._emitExpansionChanges(expansionChanges);\n        }),\n      ),\n    ]).pipe(\n      switchMap(([data, nodeType]) => {\n        if (nodeType === null) {\n          return observableOf({renderNodes: data, flattenedNodes: null, nodeType} as const);\n        }\n\n        // If we're here, then we know what our node type is, and therefore can\n        // perform our usual rendering pipeline, which necessitates converting the data\n        return this._computeRenderingData(data, nodeType).pipe(\n          map(convertedData => ({...convertedData, nodeType}) as const),\n        );\n      }),\n    );\n  }\n\n  private _renderDataChanges(data: RenderingData<T>) {\n    if (data.nodeType === null) {\n      this.renderNodeChanges(data.renderNodes);\n      return;\n    }\n\n    // If we're here, then we know what our node type is, and therefore can\n    // perform our usual rendering pipeline.\n    this._updateCachedData(data.flattenedNodes);\n    this.renderNodeChanges(data.renderNodes);\n    this._updateKeyManagerItems(data.flattenedNodes);\n  }\n\n  private _emitExpansionChanges(expansionChanges: SelectionChange<K> | null) {\n    if (!expansionChanges) {\n      return;\n    }\n\n    const nodes = this._nodes.value;\n    for (const added of expansionChanges.added) {\n      const node = nodes.get(added);\n      node?._emitExpansionState(true);\n    }\n    for (const removed of expansionChanges.removed) {\n      const node = nodes.get(removed);\n      node?._emitExpansionState(false);\n    }\n  }\n\n  private _initializeKeyManager() {\n    const items = combineLatest([this._keyManagerNodes, this._nodes]).pipe(\n      map(([keyManagerNodes, renderNodes]) =>\n        keyManagerNodes.reduce<CdkTreeNode<T, K>[]>((items, data) => {\n          const node = renderNodes.get(this._getExpansionKey(data));\n          if (node) {\n            items.push(node);\n          }\n          return items;\n        }, []),\n      ),\n    );\n\n    const keyManagerOptions: TreeKeyManagerOptions<CdkTreeNode<T, K>> = {\n      trackBy: node => this._getExpansionKey(node.data),\n      skipPredicate: node => !!node.isDisabled,\n      typeAheadDebounceInterval: true,\n      horizontalOrientation: this._dir.value,\n    };\n\n    this._keyManager = this._keyManagerFactory(items, keyManagerOptions);\n  }\n\n  private _initializeDataDiffer() {\n    // Provide a default trackBy based on `_getExpansionKey` if one isn't provided.\n    const trackBy = this.trackBy ?? ((_index: number, item: T) => this._getExpansionKey(item));\n    this._dataDiffer = this._differs.find([]).create(trackBy);\n  }\n\n  private _checkTreeControlUsage() {\n    if (typeof ngDevMode === 'undefined' || ngDevMode) {\n      // Verify that Tree follows API contract of using one of TreeControl, levelAccessor or\n      // childrenAccessor. Throw an appropriate error if contract is not met.\n      let numTreeControls = 0;\n\n      if (this.treeControl) {\n        numTreeControls++;\n      }\n      if (this.levelAccessor) {\n        numTreeControls++;\n      }\n      if (this.childrenAccessor) {\n        numTreeControls++;\n      }\n\n      if (!numTreeControls) {\n        throw getTreeControlMissingError();\n      } else if (numTreeControls > 1) {\n        throw getMultipleTreeControlsError();\n      }\n    }\n  }\n\n  /** Check for changes made in the data and render each change (node added/removed/moved). */\n  renderNodeChanges(\n    data: readonly T[],\n    dataDiffer: IterableDiffer<T> = this._dataDiffer,\n    viewContainer: ViewContainerRef = this._nodeOutlet.viewContainer,\n    parentData?: T,\n  ) {\n    const changes = dataDiffer.diff(data);\n\n    // Some tree consumers expect change detection to propagate to nodes\n    // even when the array itself hasn't changed; we explicitly detect changes\n    // anyways in order for nodes to update their data.\n    //\n    // However, if change detection is called while the component's view is\n    // still initing, then the order of child views initing will be incorrect;\n    // to prevent this, we only exit early if the view hasn't initialized yet.\n    if (!changes && !this._viewInit) {\n      return;\n    }\n\n    changes?.forEachOperation(\n      (\n        item: IterableChangeRecord<T>,\n        adjustedPreviousIndex: number | null,\n        currentIndex: number | null,\n      ) => {\n        if (item.previousIndex == null) {\n          this.insertNode(data[currentIndex!], currentIndex!, viewContainer, parentData);\n        } else if (currentIndex == null) {\n          viewContainer.remove(adjustedPreviousIndex!);\n        } else {\n          const view = viewContainer.get(adjustedPreviousIndex!);\n          viewContainer.move(view!, currentIndex);\n        }\n      },\n    );\n\n    // If the data itself changes, but keeps the same trackBy, we need to update the templates'\n    // context to reflect the new object.\n    changes?.forEachIdentityChange((record: IterableChangeRecord<T>) => {\n      const newData = record.item;\n      if (record.currentIndex != undefined) {\n        const view = viewContainer.get(record.currentIndex);\n        (view as EmbeddedViewRef<any>).context.$implicit = newData;\n      }\n    });\n\n    // TODO: change to `this._changeDetectorRef.markForCheck()`, or just switch this component to\n    // use signals.\n    this._changeDetectorRef.detectChanges();\n  }\n\n  /**\n   * Finds the matching node definition that should be used for this node data. If there is only\n   * one node definition, it is returned. Otherwise, find the node definition that has a when\n   * predicate that returns true with the data. If none return true, return the default node\n   * definition.\n   */\n  _getNodeDef(data: T, i: number): CdkTreeNodeDef<T> {\n    if (this._nodeDefs.length === 1) {\n      return this._nodeDefs.first!;\n    }\n\n    const nodeDef =\n      this._nodeDefs.find(def => def.when && def.when(i, data)) || this._defaultNodeDef;\n\n    if (!nodeDef && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw getTreeMissingMatchingNodeDefError();\n    }\n\n    return nodeDef!;\n  }\n\n  /**\n   * Create the embedded view for the data node template and place it in the correct index location\n   * within the data node view container.\n   */\n  insertNode(nodeData: T, index: number, viewContainer?: ViewContainerRef, parentData?: T) {\n    const levelAccessor = this._getLevelAccessor();\n\n    const node = this._getNodeDef(nodeData, index);\n    const key = this._getExpansionKey(nodeData);\n\n    // Node context that will be provided to created embedded view\n    const context = new CdkTreeNodeOutletContext<T>(nodeData);\n\n    parentData ??= this._parents.get(key) ?? undefined;\n    // If the tree is flat tree, then use the `getLevel` function in flat tree control\n    // Otherwise, use the level of parent node.\n    if (levelAccessor) {\n      context.level = levelAccessor(nodeData);\n    } else if (parentData !== undefined && this._levels.has(this._getExpansionKey(parentData))) {\n      context.level = this._levels.get(this._getExpansionKey(parentData))! + 1;\n    } else {\n      context.level = 0;\n    }\n    this._levels.set(key, context.level);\n\n    // Use default tree nodeOutlet, or nested node's nodeOutlet\n    const container = viewContainer ? viewContainer : this._nodeOutlet.viewContainer;\n    container.createEmbeddedView(node.template, context, index);\n\n    // Set the data to just created `CdkTreeNode`.\n    // The `CdkTreeNode` created from `createEmbeddedView` will be saved in static variable\n    //     `mostRecentTreeNode`. We get it from static variable and pass the node data to it.\n    if (CdkTreeNode.mostRecentTreeNode) {\n      CdkTreeNode.mostRecentTreeNode.data = nodeData;\n    }\n  }\n\n  /** Whether the data node is expanded or collapsed. Returns true if it's expanded. */\n  isExpanded(dataNode: T): boolean {\n    return !!(\n      this.treeControl?.isExpanded(dataNode) ||\n      this._expansionModel?.isSelected(this._getExpansionKey(dataNode))\n    );\n  }\n\n  /** If the data node is currently expanded, collapse it. Otherwise, expand it. */\n  toggle(dataNode: T): void {\n    if (this.treeControl) {\n      this.treeControl.toggle(dataNode);\n    } else if (this._expansionModel) {\n      this._expansionModel.toggle(this._getExpansionKey(dataNode));\n    }\n  }\n\n  /** Expand the data node. If it is already expanded, does nothing. */\n  expand(dataNode: T): void {\n    if (this.treeControl) {\n      this.treeControl.expand(dataNode);\n    } else if (this._expansionModel) {\n      this._expansionModel.select(this._getExpansionKey(dataNode));\n    }\n  }\n\n  /** Collapse the data node. If it is already collapsed, does nothing. */\n  collapse(dataNode: T): void {\n    if (this.treeControl) {\n      this.treeControl.collapse(dataNode);\n    } else if (this._expansionModel) {\n      this._expansionModel.deselect(this._getExpansionKey(dataNode));\n    }\n  }\n\n  /**\n   * If the data node is currently expanded, collapse it and all its descendants.\n   * Otherwise, expand it and all its descendants.\n   */\n  toggleDescendants(dataNode: T): void {\n    if (this.treeControl) {\n      this.treeControl.toggleDescendants(dataNode);\n    } else if (this._expansionModel) {\n      if (this.isExpanded(dataNode)) {\n        this.collapseDescendants(dataNode);\n      } else {\n        this.expandDescendants(dataNode);\n      }\n    }\n  }\n\n  /**\n   * Expand the data node and all its descendants. If they are already expanded, does nothing.\n   */\n  expandDescendants(dataNode: T): void {\n    if (this.treeControl) {\n      this.treeControl.expandDescendants(dataNode);\n    } else if (this._expansionModel) {\n      const expansionModel = this._expansionModel;\n      expansionModel.select(this._getExpansionKey(dataNode));\n      this._getDescendants(dataNode)\n        .pipe(take(1), takeUntil(this._onDestroy))\n        .subscribe(children => {\n          expansionModel.select(...children.map(child => this._getExpansionKey(child)));\n        });\n    }\n  }\n\n  /** Collapse the data node and all its descendants. If it is already collapsed, does nothing. */\n  collapseDescendants(dataNode: T): void {\n    if (this.treeControl) {\n      this.treeControl.collapseDescendants(dataNode);\n    } else if (this._expansionModel) {\n      const expansionModel = this._expansionModel;\n      expansionModel.deselect(this._getExpansionKey(dataNode));\n      this._getDescendants(dataNode)\n        .pipe(take(1), takeUntil(this._onDestroy))\n        .subscribe(children => {\n          expansionModel.deselect(...children.map(child => this._getExpansionKey(child)));\n        });\n    }\n  }\n\n  /** Expands all data nodes in the tree. */\n  expandAll(): void {\n    if (this.treeControl) {\n      this.treeControl.expandAll();\n    } else if (this._expansionModel) {\n      const expansionModel = this._expansionModel;\n      expansionModel.select(\n        ...this._flattenedNodes.value.map(child => this._getExpansionKey(child)),\n      );\n    }\n  }\n\n  /** Collapse all data nodes in the tree. */\n  collapseAll(): void {\n    if (this.treeControl) {\n      this.treeControl.collapseAll();\n    } else if (this._expansionModel) {\n      const expansionModel = this._expansionModel;\n      expansionModel.deselect(\n        ...this._flattenedNodes.value.map(child => this._getExpansionKey(child)),\n      );\n    }\n  }\n\n  /** Level accessor, used for compatibility between the old Tree and new Tree */\n  _getLevelAccessor() {\n    return this.treeControl?.getLevel?.bind(this.treeControl) ?? this.levelAccessor;\n  }\n\n  /** Children accessor, used for compatibility between the old Tree and new Tree */\n  _getChildrenAccessor() {\n    return this.treeControl?.getChildren?.bind(this.treeControl) ?? this.childrenAccessor;\n  }\n\n  /**\n   * Gets the direct children of a node; used for compatibility between the old tree and the\n   * new tree.\n   */\n  _getDirectChildren(dataNode: T): Observable<T[]> {\n    const levelAccessor = this._getLevelAccessor();\n    const expansionModel = this._expansionModel ?? this.treeControl?.expansionModel;\n    if (!expansionModel) {\n      return observableOf([]);\n    }\n\n    const key = this._getExpansionKey(dataNode);\n\n    const isExpanded = expansionModel.changed.pipe(\n      switchMap(changes => {\n        if (changes.added.includes(key)) {\n          return observableOf(true);\n        } else if (changes.removed.includes(key)) {\n          return observableOf(false);\n        }\n        return EMPTY;\n      }),\n      startWith(this.isExpanded(dataNode)),\n    );\n\n    if (levelAccessor) {\n      return combineLatest([isExpanded, this._flattenedNodes]).pipe(\n        map(([expanded, flattenedNodes]) => {\n          if (!expanded) {\n            return [];\n          }\n          return this._findChildrenByLevel(\n            levelAccessor,\n            flattenedNodes,\n\n            dataNode,\n            1,\n          );\n        }),\n      );\n    }\n    const childrenAccessor = this._getChildrenAccessor();\n    if (childrenAccessor) {\n      return coerceObservable(childrenAccessor(dataNode) ?? []);\n    }\n    throw getTreeControlMissingError();\n  }\n\n  /**\n   * Given the list of flattened nodes, the level accessor, and the level range within\n   * which to consider children, finds the children for a given node.\n   *\n   * For example, for direct children, `levelDelta` would be 1. For all descendants,\n   * `levelDelta` would be Infinity.\n   */\n  private _findChildrenByLevel(\n    levelAccessor: (node: T) => number,\n    flattenedNodes: readonly T[],\n    dataNode: T,\n    levelDelta: number,\n  ): T[] {\n    const key = this._getExpansionKey(dataNode);\n    const startIndex = flattenedNodes.findIndex(node => this._getExpansionKey(node) === key);\n    const dataNodeLevel = levelAccessor(dataNode);\n    const expectedLevel = dataNodeLevel + levelDelta;\n    const results: T[] = [];\n\n    // Goes through flattened tree nodes in the `flattenedNodes` array, and get all\n    // descendants within a certain level range.\n    //\n    // If we reach a node whose level is equal to or less than the level of the tree node,\n    // we hit a sibling or parent's sibling, and should stop.\n    for (let i = startIndex + 1; i < flattenedNodes.length; i++) {\n      const currentLevel = levelAccessor(flattenedNodes[i]);\n      if (currentLevel <= dataNodeLevel) {\n        break;\n      }\n      if (currentLevel <= expectedLevel) {\n        results.push(flattenedNodes[i]);\n      }\n    }\n    return results;\n  }\n\n  /**\n   * Adds the specified node component to the tree's internal registry.\n   *\n   * This primarily facilitates keyboard navigation.\n   */\n  _registerNode(node: CdkTreeNode<T, K>) {\n    this._nodes.value.set(this._getExpansionKey(node.data), node);\n    this._nodes.next(this._nodes.value);\n  }\n\n  /** Removes the specified node component from the tree's internal registry. */\n  _unregisterNode(node: CdkTreeNode<T, K>) {\n    this._nodes.value.delete(this._getExpansionKey(node.data));\n    this._nodes.next(this._nodes.value);\n  }\n\n  /**\n   * For the given node, determine the level where this node appears in the tree.\n   *\n   * This is intended to be used for `aria-level` but is 0-indexed.\n   */\n  _getLevel(node: T) {\n    return this._levels.get(this._getExpansionKey(node));\n  }\n\n  /**\n   * For the given node, determine the size of the parent's child set.\n   *\n   * This is intended to be used for `aria-setsize`.\n   */\n  _getSetSize(dataNode: T) {\n    const set = this._getAriaSet(dataNode);\n    return set.length;\n  }\n\n  /**\n   * For the given node, determine the index (starting from 1) of the node in its parent's child set.\n   *\n   * This is intended to be used for `aria-posinset`.\n   */\n  _getPositionInSet(dataNode: T) {\n    const set = this._getAriaSet(dataNode);\n    const key = this._getExpansionKey(dataNode);\n    return set.findIndex(node => this._getExpansionKey(node) === key) + 1;\n  }\n\n  /** Given a CdkTreeNode, gets the node that renders that node's parent's data. */\n  _getNodeParent(node: CdkTreeNode<T, K>) {\n    const parent = this._parents.get(this._getExpansionKey(node.data));\n    return parent && this._nodes.value.get(this._getExpansionKey(parent));\n  }\n\n  /** Given a CdkTreeNode, gets the nodes that renders that node's child data. */\n  _getNodeChildren(node: CdkTreeNode<T, K>) {\n    return this._getDirectChildren(node.data).pipe(\n      map(children =>\n        children.reduce<CdkTreeNode<T, K>[]>((nodes, child) => {\n          const value = this._nodes.value.get(this._getExpansionKey(child));\n          if (value) {\n            nodes.push(value);\n          }\n\n          return nodes;\n        }, []),\n      ),\n    );\n  }\n\n  /** `keydown` event handler; this just passes the event to the `TreeKeyManager`. */\n  _sendKeydownToKeyManager(event: KeyboardEvent) {\n    this._keyManager.onKeydown(event);\n  }\n\n  /** Gets all nested descendants of a given node. */\n  private _getDescendants(dataNode: T): Observable<T[]> {\n    if (this.treeControl) {\n      return observableOf(this.treeControl.getDescendants(dataNode));\n    }\n    if (this.levelAccessor) {\n      const results = this._findChildrenByLevel(\n        this.levelAccessor,\n        this._flattenedNodes.value,\n        dataNode,\n        Infinity,\n      );\n      return observableOf(results);\n    }\n    if (this.childrenAccessor) {\n      return this._getAllChildrenRecursively(dataNode).pipe(\n        reduce((allChildren: T[], nextChildren) => {\n          allChildren.push(...nextChildren);\n          return allChildren;\n        }, []),\n      );\n    }\n    throw getTreeControlMissingError();\n  }\n\n  /**\n   * Gets all children and sub-children of the provided node.\n   *\n   * This will emit multiple times, in the order that the children will appear\n   * in the tree, and can be combined with a `reduce` operator.\n   */\n  private _getAllChildrenRecursively(dataNode: T): Observable<T[]> {\n    if (!this.childrenAccessor) {\n      return observableOf([]);\n    }\n\n    return coerceObservable(this.childrenAccessor(dataNode)).pipe(\n      take(1),\n      switchMap(children => {\n        // Here, we cache the parents of a particular child so that we can compute the levels.\n        for (const child of children) {\n          this._parents.set(this._getExpansionKey(child), dataNode);\n        }\n        return observableOf(...children).pipe(\n          concatMap(child => concat(observableOf([child]), this._getAllChildrenRecursively(child))),\n        );\n      }),\n    );\n  }\n\n  private _getExpansionKey(dataNode: T): K {\n    // In the case that a key accessor function was not provided by the\n    // tree user, we'll default to using the node object itself as the key.\n    //\n    // This cast is safe since:\n    // - if an expansionKey is provided, TS will infer the type of K to be\n    //   the return type.\n    // - if it's not, then K will be defaulted to T.\n    return this.expansionKey?.(dataNode) ?? (dataNode as unknown as K);\n  }\n\n  private _getAriaSet(node: T) {\n    const key = this._getExpansionKey(node);\n    const parent = this._parents.get(key);\n    const parentKey = parent ? this._getExpansionKey(parent) : null;\n    const set = this._ariaSets.get(parentKey);\n    return set ?? [node];\n  }\n\n  /**\n   * Finds the parent for the given node. If this is a root node, this\n   * returns null. If we're unable to determine the parent, for example,\n   * if we don't have cached node data, this returns undefined.\n   */\n  private _findParentForNode(node: T, index: number, cachedNodes: readonly T[]): T | null {\n    // In all cases, we have a mapping from node to level; all we need to do here is backtrack in\n    // our flattened list of nodes to determine the first node that's of a level lower than the\n    // provided node.\n    if (!cachedNodes.length) {\n      return null;\n    }\n    const currentLevel = this._levels.get(this._getExpansionKey(node)) ?? 0;\n    for (let parentIndex = index - 1; parentIndex >= 0; parentIndex--) {\n      const parentNode = cachedNodes[parentIndex];\n      const parentLevel = this._levels.get(this._getExpansionKey(parentNode)) ?? 0;\n\n      if (parentLevel < currentLevel) {\n        return parentNode;\n      }\n    }\n    return null;\n  }\n\n  /**\n   * Given a set of root nodes and the current node level, flattens any nested\n   * nodes into a single array.\n   *\n   * If any nodes are not expanded, then their children will not be added into the array.\n   * This will still traverse all nested children in order to build up our internal data\n   * models, but will not include them in the returned array.\n   */\n  private _flattenNestedNodesWithExpansion(nodes: readonly T[], level = 0): Observable<T[]> {\n    const childrenAccessor = this._getChildrenAccessor();\n    // If we're using a level accessor, we don't need to flatten anything.\n    if (!childrenAccessor) {\n      return observableOf([...nodes]);\n    }\n\n    return observableOf(...nodes).pipe(\n      concatMap(node => {\n        const parentKey = this._getExpansionKey(node);\n        if (!this._parents.has(parentKey)) {\n          this._parents.set(parentKey, null);\n        }\n        this._levels.set(parentKey, level);\n\n        const children = coerceObservable(childrenAccessor(node));\n        return concat(\n          observableOf([node]),\n          children.pipe(\n            take(1),\n            tap(childNodes => {\n              this._ariaSets.set(parentKey, [...(childNodes ?? [])]);\n              for (const child of childNodes ?? []) {\n                const childKey = this._getExpansionKey(child);\n                this._parents.set(childKey, node);\n                this._levels.set(childKey, level + 1);\n              }\n            }),\n            switchMap(childNodes => {\n              if (!childNodes) {\n                return observableOf([]);\n              }\n              return this._flattenNestedNodesWithExpansion(childNodes, level + 1).pipe(\n                map(nestedNodes => (this.isExpanded(node) ? nestedNodes : [])),\n              );\n            }),\n          ),\n        );\n      }),\n      reduce((results, children) => {\n        results.push(...children);\n        return results;\n      }, [] as T[]),\n    );\n  }\n\n  /**\n   * Converts children for certain tree configurations.\n   *\n   * This also computes parent, level, and group data.\n   */\n  private _computeRenderingData(\n    nodes: readonly T[],\n    nodeType: 'flat' | 'nested',\n  ): Observable<{\n    renderNodes: readonly T[];\n    flattenedNodes: readonly T[];\n  }> {\n    // The only situations where we have to convert children types is when\n    // they're mismatched; i.e. if the tree is using a childrenAccessor and the\n    // nodes are flat, or if the tree is using a levelAccessor and the nodes are\n    // nested.\n    if (this.childrenAccessor && nodeType === 'flat') {\n      // This flattens children into a single array.\n      this._ariaSets.set(null, [...nodes]);\n      return this._flattenNestedNodesWithExpansion(nodes).pipe(\n        map(flattenedNodes => ({\n          renderNodes: flattenedNodes,\n          flattenedNodes,\n        })),\n      );\n    } else if (this.levelAccessor && nodeType === 'nested') {\n      // In the nested case, we only look for root nodes. The CdkNestedNode\n      // itself will handle rendering each individual node's children.\n      const levelAccessor = this.levelAccessor;\n      return observableOf(nodes.filter(node => levelAccessor(node) === 0)).pipe(\n        map(rootNodes => ({\n          renderNodes: rootNodes,\n          flattenedNodes: nodes,\n        })),\n        tap(({flattenedNodes}) => {\n          this._calculateParents(flattenedNodes);\n        }),\n      );\n    } else if (nodeType === 'flat') {\n      // In the case of a TreeControl, we know that the node type matches up\n      // with the TreeControl, and so no conversions are necessary. Otherwise,\n      // we've already confirmed that the data model matches up with the\n      // desired node type here.\n      return observableOf({renderNodes: nodes, flattenedNodes: nodes}).pipe(\n        tap(({flattenedNodes}) => {\n          this._calculateParents(flattenedNodes);\n        }),\n      );\n    } else {\n      // For nested nodes, we still need to perform the node flattening in order\n      // to maintain our caches for various tree operations.\n      this._ariaSets.set(null, [...nodes]);\n      return this._flattenNestedNodesWithExpansion(nodes).pipe(\n        map(flattenedNodes => ({\n          renderNodes: nodes,\n          flattenedNodes,\n        })),\n      );\n    }\n  }\n\n  private _updateCachedData(flattenedNodes: readonly T[]) {\n    this._flattenedNodes.next(flattenedNodes);\n  }\n\n  private _updateKeyManagerItems(flattenedNodes: readonly T[]) {\n    this._keyManagerNodes.next(flattenedNodes);\n  }\n\n  /** Traverse the flattened node data and compute parents, levels, and group data. */\n  private _calculateParents(flattenedNodes: readonly T[]): void {\n    const levelAccessor = this._getLevelAccessor();\n    if (!levelAccessor) {\n      return;\n    }\n\n    this._parents.clear();\n    this._ariaSets.clear();\n\n    for (let index = 0; index < flattenedNodes.length; index++) {\n      const dataNode = flattenedNodes[index];\n      const key = this._getExpansionKey(dataNode);\n      this._levels.set(key, levelAccessor(dataNode));\n      const parent = this._findParentForNode(dataNode, index, flattenedNodes);\n      this._parents.set(key, parent);\n      const parentKey = parent ? this._getExpansionKey(parent) : null;\n\n      const group = this._ariaSets.get(parentKey) ?? [];\n      group.splice(index, 0, dataNode);\n      this._ariaSets.set(parentKey, group);\n    }\n  }\n}\n\n/**\n * Tree node for CdkTree. It contains the data in the tree node.\n */\n@Directive({\n  selector: 'cdk-tree-node',\n  exportAs: 'cdkTreeNode',\n  host: {\n    'class': 'cdk-tree-node',\n    '[attr.aria-expanded]': '_getAriaExpanded()',\n    '[attr.aria-level]': 'level + 1',\n    '[attr.aria-posinset]': '_getPositionInSet()',\n    '[attr.aria-setsize]': '_getSetSize()',\n    '[tabindex]': '_tabindex',\n    'role': 'treeitem',\n    '(click)': '_setActiveItem()',\n    '(focus)': '_focusItem()',\n  },\n  standalone: true,\n})\nexport class CdkTreeNode<T, K = T> implements OnDestroy, OnInit, TreeKeyManagerItem {\n  protected _tabindex: number | null = -1;\n\n  /**\n   * The role of the tree node.\n   *\n   * @deprecated This will be ignored; the tree will automatically determine the appropriate role\n   * for tree node. This input will be removed in a future version.\n   * @breaking-change 21.0.0\n   */\n  @Input() get role(): 'treeitem' | 'group' {\n    return 'treeitem';\n  }\n\n  set role(_role: 'treeitem' | 'group') {\n    // ignore any role setting, we handle this internally.\n  }\n\n  /**\n   * Whether or not this node is expandable.\n   *\n   * If not using `FlatTreeControl`, or if `isExpandable` is not provided to\n   * `NestedTreeControl`, this should be provided for correct node a11y.\n   */\n  @Input({transform: booleanAttribute})\n  get isExpandable() {\n    return this._isExpandable();\n  }\n  set isExpandable(isExpandable: boolean) {\n    this._inputIsExpandable = isExpandable;\n  }\n\n  @Input()\n  get isExpanded(): boolean {\n    return this._tree.isExpanded(this._data);\n  }\n  set isExpanded(isExpanded: boolean) {\n    if (isExpanded) {\n      this.expand();\n    } else {\n      this.collapse();\n    }\n  }\n\n  /**\n   * Whether or not this node is disabled. If it's disabled, then the user won't be able to focus\n   * or activate this node.\n   */\n  @Input({transform: booleanAttribute}) isDisabled: boolean;\n\n  /**\n   * The text used to locate this item during typeahead. If not specified, the `textContent` will\n   * will be used.\n   */\n  @Input('cdkTreeNodeTypeaheadLabel') typeaheadLabel: string | null;\n\n  getLabel(): string {\n    return this.typeaheadLabel || this._elementRef.nativeElement.textContent?.trim() || '';\n  }\n\n  /** This emits when the node has been programatically activated or activated by keyboard. */\n  @Output()\n  readonly activation: EventEmitter<T> = new EventEmitter<T>();\n\n  /** This emits when the node's expansion status has been changed. */\n  @Output()\n  readonly expandedChange: EventEmitter<boolean> = new EventEmitter<boolean>();\n\n  /**\n   * The most recently created `CdkTreeNode`. We save it in static variable so we can retrieve it\n   * in `CdkTree` and set the data to it.\n   */\n  static mostRecentTreeNode: CdkTreeNode<any> | null = null;\n\n  /** Subject that emits when the component has been destroyed. */\n  protected readonly _destroyed = new Subject<void>();\n\n  /** Emits when the node's data has changed. */\n  readonly _dataChanges = new Subject<void>();\n\n  private _inputIsExpandable: boolean = false;\n  /**\n   * Flag used to determine whether or not we should be focusing the actual element based on\n   * some user interaction (click or focus). On click, we don't forcibly focus the element\n   * since the click could trigger some other component that wants to grab its own focus\n   * (e.g. menu, dialog).\n   */\n  private _shouldFocus = true;\n  private _parentNodeAriaLevel: number;\n\n  /** The tree node's data. */\n  get data(): T {\n    return this._data;\n  }\n  set data(value: T) {\n    if (value !== this._data) {\n      this._data = value;\n      this._dataChanges.next();\n    }\n  }\n  protected _data: T;\n\n  /* If leaf node, return true to not assign aria-expanded attribute */\n  get isLeafNode(): boolean {\n    // If flat tree node data returns false for expandable property, it's a leaf node\n    if (\n      this._tree.treeControl?.isExpandable !== undefined &&\n      !this._tree.treeControl.isExpandable(this._data)\n    ) {\n      return true;\n\n      // If nested tree node data returns 0 descendants, it's a leaf node\n    } else if (\n      this._tree.treeControl?.isExpandable === undefined &&\n      this._tree.treeControl?.getDescendants(this._data).length === 0\n    ) {\n      return true;\n    }\n\n    return false;\n  }\n\n  get level(): number {\n    // If the tree has a levelAccessor, use it to get the level. Otherwise read the\n    // aria-level off the parent node and use it as the level for this node (note aria-level is\n    // 1-indexed, while this property is 0-indexed, so we don't need to increment).\n    return this._tree._getLevel(this._data) ?? this._parentNodeAriaLevel;\n  }\n\n  /** Determines if the tree node is expandable. */\n  _isExpandable(): boolean {\n    if (this._tree.treeControl) {\n      if (this.isLeafNode) {\n        return false;\n      }\n\n      // For compatibility with trees created using TreeControl before we added\n      // CdkTreeNode#isExpandable.\n      return true;\n    }\n    return this._inputIsExpandable;\n  }\n\n  /**\n   * Determines the value for `aria-expanded`.\n   *\n   * For non-expandable nodes, this is `null`.\n   */\n  _getAriaExpanded(): string | null {\n    if (!this._isExpandable()) {\n      return null;\n    }\n    return String(this.isExpanded);\n  }\n\n  /**\n   * Determines the size of this node's parent's child set.\n   *\n   * This is intended to be used for `aria-setsize`.\n   */\n  _getSetSize(): number {\n    return this._tree._getSetSize(this._data);\n  }\n\n  /**\n   * Determines the index (starting from 1) of this node in its parent's child set.\n   *\n   * This is intended to be used for `aria-posinset`.\n   */\n  _getPositionInSet(): number {\n    return this._tree._getPositionInSet(this._data);\n  }\n\n  private _changeDetectorRef = inject(ChangeDetectorRef);\n\n  constructor(\n    protected _elementRef: ElementRef<HTMLElement>,\n    protected _tree: CdkTree<T, K>,\n  ) {\n    CdkTreeNode.mostRecentTreeNode = this as CdkTreeNode<T, K>;\n  }\n\n  ngOnInit(): void {\n    this._parentNodeAriaLevel = getParentNodeAriaLevel(this._elementRef.nativeElement);\n    this._tree\n      ._getExpansionModel()\n      .changed.pipe(\n        map(() => this.isExpanded),\n        distinctUntilChanged(),\n      )\n      .subscribe(() => {\n        this._changeDetectorRef.markForCheck();\n      });\n    this._tree._setNodeTypeIfUnset('flat');\n    this._tree._registerNode(this);\n  }\n\n  ngOnDestroy() {\n    // If this is the last tree node being destroyed,\n    // clear out the reference to avoid leaking memory.\n    if (CdkTreeNode.mostRecentTreeNode === this) {\n      CdkTreeNode.mostRecentTreeNode = null;\n    }\n\n    this._dataChanges.complete();\n    this._destroyed.next();\n    this._destroyed.complete();\n  }\n\n  getParent(): CdkTreeNode<T, K> | null {\n    return this._tree._getNodeParent(this) ?? null;\n  }\n\n  getChildren(): CdkTreeNode<T, K>[] | Observable<CdkTreeNode<T, K>[]> {\n    return this._tree._getNodeChildren(this);\n  }\n\n  /** Focuses this data node. Implemented for TreeKeyManagerItem. */\n  focus(): void {\n    this._tabindex = 0;\n    if (this._shouldFocus) {\n      this._elementRef.nativeElement.focus();\n    }\n\n    this._changeDetectorRef.markForCheck();\n  }\n\n  /** Defocus this data node. */\n  unfocus(): void {\n    this._tabindex = -1;\n\n    this._changeDetectorRef.markForCheck();\n  }\n\n  /** Emits an activation event. Implemented for TreeKeyManagerItem. */\n  activate(): void {\n    if (this.isDisabled) {\n      return;\n    }\n    this.activation.next(this._data);\n  }\n\n  /** Collapses this data node. Implemented for TreeKeyManagerItem. */\n  collapse(): void {\n    if (this.isExpandable) {\n      this._tree.collapse(this._data);\n    }\n  }\n\n  /** Expands this data node. Implemented for TreeKeyManagerItem. */\n  expand(): void {\n    if (this.isExpandable) {\n      this._tree.expand(this._data);\n    }\n  }\n\n  _focusItem() {\n    if (this.isDisabled) {\n      return;\n    }\n    this._tree._keyManager.focusItem(this);\n  }\n\n  _setActiveItem() {\n    if (this.isDisabled) {\n      return;\n    }\n    this._shouldFocus = false;\n    this._tree._keyManager.focusItem(this);\n    this._shouldFocus = true;\n  }\n\n  _emitExpansionState(expanded: boolean) {\n    this.expandedChange.emit(expanded);\n  }\n}\n\nfunction getParentNodeAriaLevel(nodeElement: HTMLElement): number {\n  let parent = nodeElement.parentElement;\n  while (parent && !isNodeElement(parent)) {\n    parent = parent.parentElement;\n  }\n  if (!parent) {\n    if (typeof ngDevMode === 'undefined' || ngDevMode) {\n      throw Error('Incorrect tree structure containing detached node.');\n    } else {\n      return -1;\n    }\n  } else if (parent.classList.contains('cdk-nested-tree-node')) {\n    return numberAttribute(parent.getAttribute('aria-level')!);\n  } else {\n    // The ancestor element is the cdk-tree itself\n    return 0;\n  }\n}\n\nfunction isNodeElement(element: HTMLElement) {\n  const classList = element.classList;\n  return !!(classList?.contains('cdk-nested-tree-node') || classList?.contains('cdk-tree'));\n}\n"]}
|