@angular/cdk 19.0.0-next.1 → 19.0.0-next.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/a11y/index.d.ts +5 -1
- package/fesm2022/a11y.mjs +21 -11
- package/fesm2022/a11y.mjs.map +1 -1
- package/fesm2022/cdk.mjs +1 -1
- package/fesm2022/cdk.mjs.map +1 -1
- package/fesm2022/drag-drop.mjs +2 -1
- package/fesm2022/drag-drop.mjs.map +1 -1
- package/fesm2022/overlay.mjs +20 -1
- package/fesm2022/overlay.mjs.map +1 -1
- package/fesm2022/private.mjs +9 -7
- package/fesm2022/private.mjs.map +1 -1
- package/fesm2022/text-field.mjs +17 -2
- package/fesm2022/text-field.mjs.map +1 -1
- package/fesm2022/tree.mjs +13 -7
- package/fesm2022/tree.mjs.map +1 -1
- package/overlay/_index.scss +15 -13
- package/overlay/index.d.ts +5 -0
- package/overlay-prebuilt.css +1 -1
- package/package.json +1 -57
- package/private/index.d.ts +1 -0
- package/schematics/ng-add/index.js +1 -1
- package/schematics/ng-add/index.mjs +1 -1
- package/text-field/index.d.ts +1 -0
- package/tree/index.d.ts +3 -1
- package/esm2022/a11y/a11y-module.mjs +0 -31
- package/esm2022/a11y/a11y_public_index.mjs +0 -5
- package/esm2022/a11y/aria-describer/aria-describer.mjs +0 -232
- package/esm2022/a11y/aria-describer/aria-reference.mjs +0 -47
- package/esm2022/a11y/fake-event-detection.mjs +0 -29
- package/esm2022/a11y/focus-monitor/focus-monitor.mjs +0 -468
- package/esm2022/a11y/focus-trap/configurable-focus-trap-config.mjs +0 -9
- package/esm2022/a11y/focus-trap/configurable-focus-trap-factory.mjs +0 -54
- package/esm2022/a11y/focus-trap/configurable-focus-trap.mjs +0 -51
- package/esm2022/a11y/focus-trap/event-listener-inert-strategy.mjs +0 -61
- package/esm2022/a11y/focus-trap/focus-trap-inert-strategy.mjs +0 -11
- package/esm2022/a11y/focus-trap/focus-trap-manager.mjs +0 -53
- package/esm2022/a11y/focus-trap/focus-trap.mjs +0 -396
- package/esm2022/a11y/high-contrast-mode/high-contrast-mode-detector.mjs +0 -116
- package/esm2022/a11y/index.mjs +0 -9
- package/esm2022/a11y/input-modality/input-modality-detector.mjs +0 -176
- package/esm2022/a11y/interactivity-checker/interactivity-checker.mjs +0 -238
- package/esm2022/a11y/key-manager/activedescendant-key-manager.mjs +0 -20
- package/esm2022/a11y/key-manager/focus-key-manager.mjs +0 -29
- package/esm2022/a11y/key-manager/list-key-manager.mjs +0 -357
- package/esm2022/a11y/key-manager/noop-tree-key-manager.mjs +0 -94
- package/esm2022/a11y/key-manager/tree-key-manager-strategy.mjs +0 -9
- package/esm2022/a11y/key-manager/tree-key-manager.mjs +0 -345
- package/esm2022/a11y/key-manager/typeahead.mjs +0 -91
- package/esm2022/a11y/live-announcer/live-announcer-tokens.mjs +0 -19
- package/esm2022/a11y/live-announcer/live-announcer.mjs +0 -210
- package/esm2022/a11y/public-api.mjs +0 -30
- package/esm2022/accordion/accordion-item.mjs +0 -162
- package/esm2022/accordion/accordion-module.mjs +0 -24
- package/esm2022/accordion/accordion.mjs +0 -65
- package/esm2022/accordion/accordion_public_index.mjs +0 -5
- package/esm2022/accordion/index.mjs +0 -9
- package/esm2022/accordion/public-api.mjs +0 -11
- package/esm2022/bidi/bidi-module.mjs +0 -23
- package/esm2022/bidi/bidi_public_index.mjs +0 -5
- package/esm2022/bidi/dir-document-token.mjs +0 -33
- package/esm2022/bidi/dir.mjs +0 -70
- package/esm2022/bidi/directionality.mjs +0 -52
- package/esm2022/bidi/index.mjs +0 -9
- package/esm2022/bidi/public-api.mjs +0 -12
- package/esm2022/clipboard/clipboard-module.mjs +0 -23
- package/esm2022/clipboard/clipboard.mjs +0 -53
- package/esm2022/clipboard/clipboard_public_index.mjs +0 -5
- package/esm2022/clipboard/copy-to-clipboard.mjs +0 -100
- package/esm2022/clipboard/index.mjs +0 -9
- package/esm2022/clipboard/pending-copy.mjs +0 -71
- package/esm2022/clipboard/public-api.mjs +0 -12
- package/esm2022/coercion/array.mjs +0 -11
- package/esm2022/coercion/boolean-property.mjs +0 -12
- package/esm2022/coercion/css-pixel-value.mjs +0 -15
- package/esm2022/coercion/element.mjs +0 -16
- package/esm2022/coercion/index.mjs +0 -9
- package/esm2022/coercion/number-property.mjs +0 -24
- package/esm2022/coercion/private/index.mjs +0 -9
- package/esm2022/coercion/private/observable.mjs +0 -19
- package/esm2022/coercion/private/private_public_index.mjs +0 -5
- package/esm2022/coercion/public-api.mjs +0 -14
- package/esm2022/coercion/string-array.mjs +0 -38
- package/esm2022/collections/array-data-source.mjs +0 -21
- package/esm2022/collections/collection-viewer.mjs +0 -9
- package/esm2022/collections/collections_public_index.mjs +0 -5
- package/esm2022/collections/data-source.mjs +0 -19
- package/esm2022/collections/dispose-view-repeater-strategy.mjs +0 -48
- package/esm2022/collections/index.mjs +0 -9
- package/esm2022/collections/public-api.mjs +0 -17
- package/esm2022/collections/recycle-view-repeater-strategy.mjs +0 -129
- package/esm2022/collections/selection-model.mjs +0 -225
- package/esm2022/collections/tree-adapter.mjs +0 -9
- package/esm2022/collections/unique-selection-dispatcher.mjs +0 -55
- package/esm2022/collections/view-repeater.mjs +0 -26
- package/esm2022/dialog/dialog-config.mjs +0 -70
- package/esm2022/dialog/dialog-container.mjs +0 -308
- package/esm2022/dialog/dialog-injectors.mjs +0 -40
- package/esm2022/dialog/dialog-module.mjs +0 -40
- package/esm2022/dialog/dialog-ref.mjs +0 -85
- package/esm2022/dialog/dialog.mjs +0 -303
- package/esm2022/dialog/dialog_public_index.mjs +0 -5
- package/esm2022/dialog/index.mjs +0 -9
- package/esm2022/dialog/public-api.mjs +0 -14
- package/esm2022/drag-drop/directives/assertions.mjs +0 -18
- package/esm2022/drag-drop/directives/config.mjs +0 -14
- package/esm2022/drag-drop/directives/drag-handle.mjs +0 -68
- package/esm2022/drag-drop/directives/drag-placeholder.mjs +0 -43
- package/esm2022/drag-drop/directives/drag-preview.mjs +0 -48
- package/esm2022/drag-drop/directives/drag.mjs +0 -520
- package/esm2022/drag-drop/directives/drop-list-group.mjs +0 -47
- package/esm2022/drag-drop/directives/drop-list.mjs +0 -352
- package/esm2022/drag-drop/dom/clone-node.mjs +0 -65
- package/esm2022/drag-drop/dom/dom-rect.mjs +0 -64
- package/esm2022/drag-drop/dom/parent-position-tracker.mjs +0 -76
- package/esm2022/drag-drop/dom/root-node.mjs +0 -21
- package/esm2022/drag-drop/dom/styling.mjs +0 -89
- package/esm2022/drag-drop/dom/transition-duration.mjs +0 -36
- package/esm2022/drag-drop/drag-drop-module.mjs +0 -49
- package/esm2022/drag-drop/drag-drop-registry.mjs +0 -248
- package/esm2022/drag-drop/drag-drop.mjs +0 -57
- package/esm2022/drag-drop/drag-drop_public_index.mjs +0 -5
- package/esm2022/drag-drop/drag-events.mjs +0 -9
- package/esm2022/drag-drop/drag-parent.mjs +0 -16
- package/esm2022/drag-drop/drag-ref.mjs +0 -1097
- package/esm2022/drag-drop/drag-utils.mjs +0 -60
- package/esm2022/drag-drop/drop-list-ref.mjs +0 -672
- package/esm2022/drag-drop/index.mjs +0 -9
- package/esm2022/drag-drop/preview-ref.mjs +0 -125
- package/esm2022/drag-drop/public-api.mjs +0 -23
- package/esm2022/drag-drop/sorting/drop-list-sort-strategy.mjs +0 -9
- package/esm2022/drag-drop/sorting/mixed-sort-strategy.mjs +0 -246
- package/esm2022/drag-drop/sorting/single-axis-sort-strategy.mjs +0 -343
- package/esm2022/index.mjs +0 -9
- package/esm2022/keycodes/index.mjs +0 -9
- package/esm2022/keycodes/keycodes.mjs +0 -127
- package/esm2022/keycodes/keycodes_public_index.mjs +0 -5
- package/esm2022/keycodes/modifiers.mjs +0 -18
- package/esm2022/keycodes/public-api.mjs +0 -10
- package/esm2022/layout/breakpoints-observer.mjs +0 -105
- package/esm2022/layout/breakpoints.mjs +0 -29
- package/esm2022/layout/index.mjs +0 -9
- package/esm2022/layout/layout-module.mjs +0 -19
- package/esm2022/layout/layout_public_index.mjs +0 -5
- package/esm2022/layout/media-matcher.mjs +0 -94
- package/esm2022/layout/public-api.mjs +0 -12
- package/esm2022/listbox/index.mjs +0 -9
- package/esm2022/listbox/listbox-module.mjs +0 -24
- package/esm2022/listbox/listbox.mjs +0 -888
- package/esm2022/listbox/listbox_public_index.mjs +0 -5
- package/esm2022/listbox/public-api.mjs +0 -10
- package/esm2022/menu/context-menu-trigger.mjs +0 -221
- package/esm2022/menu/event-detection.mjs +0 -28
- package/esm2022/menu/index.mjs +0 -9
- package/esm2022/menu/menu-aim.mjs +0 -204
- package/esm2022/menu/menu-bar.mjs +0 -134
- package/esm2022/menu/menu-base.mjs +0 -192
- package/esm2022/menu/menu-errors.mjs +0 -22
- package/esm2022/menu/menu-group.mjs +0 -31
- package/esm2022/menu/menu-interface.mjs +0 -11
- package/esm2022/menu/menu-item-checkbox.mjs +0 -50
- package/esm2022/menu/menu-item-radio.mjs +0 -72
- package/esm2022/menu/menu-item-selectable.mjs +0 -36
- package/esm2022/menu/menu-item.mjs +0 -270
- package/esm2022/menu/menu-module.mjs +0 -60
- package/esm2022/menu/menu-stack.mjs +0 -163
- package/esm2022/menu/menu-trigger-base.mjs +0 -119
- package/esm2022/menu/menu-trigger.mjs +0 -313
- package/esm2022/menu/menu.mjs +0 -135
- package/esm2022/menu/menu_public_index.mjs +0 -5
- package/esm2022/menu/pointer-focus-tracker.mjs +0 -51
- package/esm2022/menu/public-api.mjs +0 -24
- package/esm2022/observers/index.mjs +0 -9
- package/esm2022/observers/observe-content.mjs +0 -216
- package/esm2022/observers/observers_public_index.mjs +0 -5
- package/esm2022/observers/private/index.mjs +0 -9
- package/esm2022/observers/private/private_public_index.mjs +0 -5
- package/esm2022/observers/private/shared-resize-observer.mjs +0 -125
- package/esm2022/observers/public-api.mjs +0 -9
- package/esm2022/overlay/dispatchers/base-overlay-dispatcher.mjs +0 -52
- package/esm2022/overlay/dispatchers/index.mjs +0 -10
- package/esm2022/overlay/dispatchers/overlay-keyboard-dispatcher.mjs +0 -81
- package/esm2022/overlay/dispatchers/overlay-outside-click-dispatcher.mjs +0 -151
- package/esm2022/overlay/fullscreen-overlay-container.mjs +0 -94
- package/esm2022/overlay/index.mjs +0 -9
- package/esm2022/overlay/overlay-config.mjs +0 -45
- package/esm2022/overlay/overlay-container.mjs +0 -81
- package/esm2022/overlay/overlay-directives.mjs +0 -432
- package/esm2022/overlay/overlay-module.mjs +0 -28
- package/esm2022/overlay/overlay-ref.mjs +0 -437
- package/esm2022/overlay/overlay.mjs +0 -126
- package/esm2022/overlay/overlay_public_index.mjs +0 -5
- package/esm2022/overlay/position/connected-position.mjs +0 -88
- package/esm2022/overlay/position/flexible-connected-position-strategy.mjs +0 -1002
- package/esm2022/overlay/position/global-position-strategy.mjs +0 -237
- package/esm2022/overlay/position/overlay-position-builder.mjs +0 -50
- package/esm2022/overlay/position/position-strategy.mjs +0 -9
- package/esm2022/overlay/position/scroll-clip.mjs +0 -40
- package/esm2022/overlay/public-api.mjs +0 -22
- package/esm2022/overlay/scroll/block-scroll-strategy.mjs +0 -80
- package/esm2022/overlay/scroll/close-scroll-strategy.mjs +0 -65
- package/esm2022/overlay/scroll/index.mjs +0 -14
- package/esm2022/overlay/scroll/noop-scroll-strategy.mjs +0 -17
- package/esm2022/overlay/scroll/reposition-scroll-strategy.mjs +0 -61
- package/esm2022/overlay/scroll/scroll-strategy-options.mjs +0 -55
- package/esm2022/overlay/scroll/scroll-strategy.mjs +0 -14
- package/esm2022/platform/features/input-types.mjs +0 -59
- package/esm2022/platform/features/passive-listeners.mjs +0 -36
- package/esm2022/platform/features/scrolling.mjs +0 -104
- package/esm2022/platform/features/shadow-dom.mjs +0 -54
- package/esm2022/platform/features/test-environment.mjs +0 -24
- package/esm2022/platform/index.mjs +0 -9
- package/esm2022/platform/platform-module.mjs +0 -19
- package/esm2022/platform/platform.mjs +0 -82
- package/esm2022/platform/platform_public_index.mjs +0 -5
- package/esm2022/platform/public-api.mjs +0 -15
- package/esm2022/portal/dom-portal-outlet.mjs +0 -158
- package/esm2022/portal/index.mjs +0 -9
- package/esm2022/portal/portal-directives.mjs +0 -251
- package/esm2022/portal/portal-errors.mjs +0 -51
- package/esm2022/portal/portal-injector.mjs +0 -28
- package/esm2022/portal/portal.mjs +0 -189
- package/esm2022/portal/portal_public_index.mjs +0 -5
- package/esm2022/portal/public-api.mjs +0 -12
- package/esm2022/private/index.mjs +0 -9
- package/esm2022/private/private_public_index.mjs +0 -5
- package/esm2022/private/public-api.mjs +0 -9
- package/esm2022/private/style-loader.mjs +0 -50
- package/esm2022/public-api.mjs +0 -9
- package/esm2022/scrolling/fixed-size-virtual-scroll.mjs +0 -218
- package/esm2022/scrolling/index.mjs +0 -9
- package/esm2022/scrolling/public-api.mjs +0 -20
- package/esm2022/scrolling/scroll-dispatcher.mjs +0 -163
- package/esm2022/scrolling/scrollable.mjs +0 -177
- package/esm2022/scrolling/scrolling-module.mjs +0 -69
- package/esm2022/scrolling/scrolling_public_index.mjs +0 -5
- package/esm2022/scrolling/viewport-ruler.mjs +0 -146
- package/esm2022/scrolling/virtual-for-of.mjs +0 -299
- package/esm2022/scrolling/virtual-scroll-repeater.mjs +0 -9
- package/esm2022/scrolling/virtual-scroll-strategy.mjs +0 -11
- package/esm2022/scrolling/virtual-scroll-viewport.mjs +0 -445
- package/esm2022/scrolling/virtual-scrollable-element.mjs +0 -42
- package/esm2022/scrolling/virtual-scrollable-window.mjs +0 -41
- package/esm2022/scrolling/virtual-scrollable.mjs +0 -40
- package/esm2022/stepper/index.mjs +0 -9
- package/esm2022/stepper/public-api.mjs +0 -13
- package/esm2022/stepper/step-header.mjs +0 -31
- package/esm2022/stepper/step-label.mjs +0 -24
- package/esm2022/stepper/stepper-button.mjs +0 -58
- package/esm2022/stepper/stepper-module.mjs +0 -41
- package/esm2022/stepper/stepper.mjs +0 -452
- package/esm2022/stepper/stepper_public_index.mjs +0 -5
- package/esm2022/table/can-stick.mjs +0 -46
- package/esm2022/table/cell.mjs +0 -245
- package/esm2022/table/coalesced-style-scheduler.mjs +0 -79
- package/esm2022/table/index.mjs +0 -9
- package/esm2022/table/public-api.mjs +0 -20
- package/esm2022/table/row.mjs +0 -317
- package/esm2022/table/sticky-position-listener.mjs +0 -11
- package/esm2022/table/sticky-styler.mjs +0 -347
- package/esm2022/table/table-errors.mjs +0 -67
- package/esm2022/table/table-module.mjs +0 -93
- package/esm2022/table/table.mjs +0 -1148
- package/esm2022/table/table_public_index.mjs +0 -5
- package/esm2022/table/text-column.mjs +0 -154
- package/esm2022/table/tokens.mjs +0 -16
- package/esm2022/testing/change-detection.mjs +0 -102
- package/esm2022/testing/component-harness.mjs +0 -328
- package/esm2022/testing/element-dimensions.mjs +0 -9
- package/esm2022/testing/harness-environment.mjs +0 -220
- package/esm2022/testing/index.mjs +0 -9
- package/esm2022/testing/public-api.mjs +0 -15
- package/esm2022/testing/selenium-webdriver/index.mjs +0 -9
- package/esm2022/testing/selenium-webdriver/public-api.mjs +0 -10
- package/esm2022/testing/selenium-webdriver/selenium-web-driver-element.mjs +0 -216
- package/esm2022/testing/selenium-webdriver/selenium-web-driver-harness-environment.mjs +0 -88
- package/esm2022/testing/selenium-webdriver/selenium-webdriver-keys.mjs +0 -64
- package/esm2022/testing/test-element-errors.mjs +0 -15
- package/esm2022/testing/test-element.mjs +0 -49
- package/esm2022/testing/testbed/fake-events/dispatch-events.mjs +0 -53
- package/esm2022/testing/testbed/fake-events/element-focus.mjs +0 -39
- package/esm2022/testing/testbed/fake-events/event-objects.mjs +0 -127
- package/esm2022/testing/testbed/fake-events/index.mjs +0 -14
- package/esm2022/testing/testbed/fake-events/type-in-element.mjs +0 -97
- package/esm2022/testing/testbed/index.mjs +0 -9
- package/esm2022/testing/testbed/public-api.mjs +0 -10
- package/esm2022/testing/testbed/task-state-zone-interceptor.mjs +0 -83
- package/esm2022/testing/testbed/testbed-harness-environment.mjs +0 -165
- package/esm2022/testing/testbed/unit-test-element.mjs +0 -258
- package/esm2022/testing/text-filtering.mjs +0 -21
- package/esm2022/text-field/autofill.mjs +0 -114
- package/esm2022/text-field/autosize.mjs +0 -309
- package/esm2022/text-field/index.mjs +0 -9
- package/esm2022/text-field/public-api.mjs +0 -11
- package/esm2022/text-field/text-field-module.mjs +0 -24
- package/esm2022/text-field/text-field_public_index.mjs +0 -5
- package/esm2022/tree/control/base-tree-control.mjs +0 -62
- package/esm2022/tree/control/flat-tree-control.mjs +0 -57
- package/esm2022/tree/control/nested-tree-control.mjs +0 -67
- package/esm2022/tree/control/tree-control.mjs +0 -2
- package/esm2022/tree/index.mjs +0 -9
- package/esm2022/tree/nested-node.mjs +0 -103
- package/esm2022/tree/node.mjs +0 -36
- package/esm2022/tree/outlet.mjs +0 -40
- package/esm2022/tree/padding.mjs +0 -129
- package/esm2022/tree/public-api.mjs +0 -20
- package/esm2022/tree/toggle.mjs +0 -51
- package/esm2022/tree/tree-errors.mjs +0 -44
- package/esm2022/tree/tree-module.mjs +0 -49
- package/esm2022/tree/tree.mjs +0 -1123
- package/esm2022/tree/tree_public_index.mjs +0 -5
- package/esm2022/version.mjs +0 -11
package/esm2022/tree/tree.mjs
DELETED
|
@@ -1,1123 +0,0 @@
|
|
|
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';
|
|
15
|
-
import { CdkTreeNodeDef, CdkTreeNodeOutletContext } from './node';
|
|
16
|
-
import { CdkTreeNodeOutlet } from './outlet';
|
|
17
|
-
import { getMultipleTreeControlsError, getTreeControlMissingError, getTreeMissingMatchingNodeDefError, getTreeMultipleDefaultNodeDefsError, getTreeNoValidDataSourceError, } from './tree-errors';
|
|
18
|
-
import * as i0 from "@angular/core";
|
|
19
|
-
import * as i1 from "@angular/cdk/bidi";
|
|
20
|
-
/**
|
|
21
|
-
* CDK tree component that connects with a data source to retrieve data of type `T` and renders
|
|
22
|
-
* dataNodes with hierarchy. Updates the dataNodes when new data is provided by the data source.
|
|
23
|
-
*/
|
|
24
|
-
export class CdkTree {
|
|
25
|
-
/**
|
|
26
|
-
* Provides a stream containing the latest data array to render. Influenced by the tree's
|
|
27
|
-
* stream of view window (what dataNodes are currently on screen).
|
|
28
|
-
* Data source can be an observable of data array, or a data array to render.
|
|
29
|
-
*/
|
|
30
|
-
get dataSource() {
|
|
31
|
-
return this._dataSource;
|
|
32
|
-
}
|
|
33
|
-
set dataSource(dataSource) {
|
|
34
|
-
if (this._dataSource !== dataSource) {
|
|
35
|
-
this._switchDataSource(dataSource);
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
constructor(_differs, _changeDetectorRef, _dir) {
|
|
39
|
-
this._differs = _differs;
|
|
40
|
-
this._changeDetectorRef = _changeDetectorRef;
|
|
41
|
-
this._dir = _dir;
|
|
42
|
-
/** Subject that emits when the component has been destroyed. */
|
|
43
|
-
this._onDestroy = new Subject();
|
|
44
|
-
/** Level of nodes */
|
|
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();
|
|
57
|
-
// TODO(tinayuangao): Setup a listener for scrolling, emit the calculated view to viewChange.
|
|
58
|
-
// Remove the MAX_VALUE in viewChange
|
|
59
|
-
/**
|
|
60
|
-
* Stream containing the latest information on what rows are being displayed on screen.
|
|
61
|
-
* Can be used by the data source to as a heuristic of what data should be provided.
|
|
62
|
-
*/
|
|
63
|
-
this.viewChange = new BehaviorSubject({
|
|
64
|
-
start: 0,
|
|
65
|
-
end: Number.MAX_VALUE,
|
|
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;
|
|
85
|
-
}
|
|
86
|
-
ngAfterContentInit() {
|
|
87
|
-
this._initializeKeyManager();
|
|
88
|
-
}
|
|
89
|
-
ngAfterContentChecked() {
|
|
90
|
-
this._updateDefaultNodeDefinition();
|
|
91
|
-
this._subscribeToDataChanges();
|
|
92
|
-
}
|
|
93
|
-
ngOnDestroy() {
|
|
94
|
-
this._nodeOutlet.viewContainer.clear();
|
|
95
|
-
this.viewChange.complete();
|
|
96
|
-
this._onDestroy.next();
|
|
97
|
-
this._onDestroy.complete();
|
|
98
|
-
if (this._dataSource && typeof this._dataSource.disconnect === 'function') {
|
|
99
|
-
this.dataSource.disconnect(this);
|
|
100
|
-
}
|
|
101
|
-
if (this._dataSubscription) {
|
|
102
|
-
this._dataSubscription.unsubscribe();
|
|
103
|
-
this._dataSubscription = null;
|
|
104
|
-
}
|
|
105
|
-
// In certain tests, the tree might be destroyed before this is initialized
|
|
106
|
-
// in `ngAfterContentInit`.
|
|
107
|
-
this._keyManager?.destroy();
|
|
108
|
-
}
|
|
109
|
-
ngOnInit() {
|
|
110
|
-
this._checkTreeControlUsage();
|
|
111
|
-
this._initializeDataDiffer();
|
|
112
|
-
}
|
|
113
|
-
ngAfterViewInit() {
|
|
114
|
-
this._viewInit = true;
|
|
115
|
-
}
|
|
116
|
-
_updateDefaultNodeDefinition() {
|
|
117
|
-
const defaultNodeDefs = this._nodeDefs.filter(def => !def.when);
|
|
118
|
-
if (defaultNodeDefs.length > 1 && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
|
119
|
-
throw getTreeMultipleDefaultNodeDefsError();
|
|
120
|
-
}
|
|
121
|
-
this._defaultNodeDef = defaultNodeDefs[0];
|
|
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);
|
|
132
|
-
}
|
|
133
|
-
}
|
|
134
|
-
/**
|
|
135
|
-
* Switch to the provided data source by resetting the data and unsubscribing from the current
|
|
136
|
-
* render change subscription if one exists. If the data source is null, interpret this by
|
|
137
|
-
* clearing the node outlet. Otherwise start listening for new data.
|
|
138
|
-
*/
|
|
139
|
-
_switchDataSource(dataSource) {
|
|
140
|
-
if (this._dataSource && typeof this._dataSource.disconnect === 'function') {
|
|
141
|
-
this.dataSource.disconnect(this);
|
|
142
|
-
}
|
|
143
|
-
if (this._dataSubscription) {
|
|
144
|
-
this._dataSubscription.unsubscribe();
|
|
145
|
-
this._dataSubscription = null;
|
|
146
|
-
}
|
|
147
|
-
// Remove the all dataNodes if there is now no data source
|
|
148
|
-
if (!dataSource) {
|
|
149
|
-
this._nodeOutlet.viewContainer.clear();
|
|
150
|
-
}
|
|
151
|
-
this._dataSource = dataSource;
|
|
152
|
-
if (this._nodeDefs) {
|
|
153
|
-
this._subscribeToDataChanges();
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
_getExpansionModel() {
|
|
157
|
-
if (!this.treeControl) {
|
|
158
|
-
this._expansionModel ??= new SelectionModel(true);
|
|
159
|
-
return this._expansionModel;
|
|
160
|
-
}
|
|
161
|
-
return this.treeControl.expansionModel;
|
|
162
|
-
}
|
|
163
|
-
/** Set up a subscription for the data provided by the data source. */
|
|
164
|
-
_subscribeToDataChanges() {
|
|
165
|
-
if (this._dataSubscription) {
|
|
166
|
-
return;
|
|
167
|
-
}
|
|
168
|
-
let dataStream;
|
|
169
|
-
if (isDataSource(this._dataSource)) {
|
|
170
|
-
dataStream = this._dataSource.connect(this);
|
|
171
|
-
}
|
|
172
|
-
else if (isObservable(this._dataSource)) {
|
|
173
|
-
dataStream = this._dataSource;
|
|
174
|
-
}
|
|
175
|
-
else if (Array.isArray(this._dataSource)) {
|
|
176
|
-
dataStream = observableOf(this._dataSource);
|
|
177
|
-
}
|
|
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;
|
|
214
|
-
}
|
|
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
|
-
}
|
|
276
|
-
}
|
|
277
|
-
}
|
|
278
|
-
/** Check for changes made in the data and render each change (node added/removed/moved). */
|
|
279
|
-
renderNodeChanges(data, dataDiffer = this._dataDiffer, viewContainer = this._nodeOutlet.viewContainer, parentData) {
|
|
280
|
-
const changes = dataDiffer.diff(data);
|
|
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) {
|
|
289
|
-
return;
|
|
290
|
-
}
|
|
291
|
-
changes?.forEachOperation((item, adjustedPreviousIndex, currentIndex) => {
|
|
292
|
-
if (item.previousIndex == null) {
|
|
293
|
-
this.insertNode(data[currentIndex], currentIndex, viewContainer, parentData);
|
|
294
|
-
}
|
|
295
|
-
else if (currentIndex == null) {
|
|
296
|
-
viewContainer.remove(adjustedPreviousIndex);
|
|
297
|
-
}
|
|
298
|
-
else {
|
|
299
|
-
const view = viewContainer.get(adjustedPreviousIndex);
|
|
300
|
-
viewContainer.move(view, currentIndex);
|
|
301
|
-
}
|
|
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
|
-
});
|
|
312
|
-
// TODO: change to `this._changeDetectorRef.markForCheck()`, or just switch this component to
|
|
313
|
-
// use signals.
|
|
314
|
-
this._changeDetectorRef.detectChanges();
|
|
315
|
-
}
|
|
316
|
-
/**
|
|
317
|
-
* Finds the matching node definition that should be used for this node data. If there is only
|
|
318
|
-
* one node definition, it is returned. Otherwise, find the node definition that has a when
|
|
319
|
-
* predicate that returns true with the data. If none return true, return the default node
|
|
320
|
-
* definition.
|
|
321
|
-
*/
|
|
322
|
-
_getNodeDef(data, i) {
|
|
323
|
-
if (this._nodeDefs.length === 1) {
|
|
324
|
-
return this._nodeDefs.first;
|
|
325
|
-
}
|
|
326
|
-
const nodeDef = this._nodeDefs.find(def => def.when && def.when(i, data)) || this._defaultNodeDef;
|
|
327
|
-
if (!nodeDef && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
|
328
|
-
throw getTreeMissingMatchingNodeDefError();
|
|
329
|
-
}
|
|
330
|
-
return nodeDef;
|
|
331
|
-
}
|
|
332
|
-
/**
|
|
333
|
-
* Create the embedded view for the data node template and place it in the correct index location
|
|
334
|
-
* within the data node view container.
|
|
335
|
-
*/
|
|
336
|
-
insertNode(nodeData, index, viewContainer, parentData) {
|
|
337
|
-
const levelAccessor = this._getLevelAccessor();
|
|
338
|
-
const node = this._getNodeDef(nodeData, index);
|
|
339
|
-
const key = this._getExpansionKey(nodeData);
|
|
340
|
-
// Node context that will be provided to created embedded view
|
|
341
|
-
const context = new CdkTreeNodeOutletContext(nodeData);
|
|
342
|
-
parentData ??= this._parents.get(key) ?? undefined;
|
|
343
|
-
// If the tree is flat tree, then use the `getLevel` function in flat tree control
|
|
344
|
-
// Otherwise, use the level of parent node.
|
|
345
|
-
if (levelAccessor) {
|
|
346
|
-
context.level = levelAccessor(nodeData);
|
|
347
|
-
}
|
|
348
|
-
else if (parentData !== undefined && this._levels.has(this._getExpansionKey(parentData))) {
|
|
349
|
-
context.level = this._levels.get(this._getExpansionKey(parentData)) + 1;
|
|
350
|
-
}
|
|
351
|
-
else {
|
|
352
|
-
context.level = 0;
|
|
353
|
-
}
|
|
354
|
-
this._levels.set(key, context.level);
|
|
355
|
-
// Use default tree nodeOutlet, or nested node's nodeOutlet
|
|
356
|
-
const container = viewContainer ? viewContainer : this._nodeOutlet.viewContainer;
|
|
357
|
-
container.createEmbeddedView(node.template, context, index);
|
|
358
|
-
// Set the data to just created `CdkTreeNode`.
|
|
359
|
-
// The `CdkTreeNode` created from `createEmbeddedView` will be saved in static variable
|
|
360
|
-
// `mostRecentTreeNode`. We get it from static variable and pass the node data to it.
|
|
361
|
-
if (CdkTreeNode.mostRecentTreeNode) {
|
|
362
|
-
CdkTreeNode.mostRecentTreeNode.data = nodeData;
|
|
363
|
-
}
|
|
364
|
-
}
|
|
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.2.0-next.2", 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.2.0-next.2", 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 }); }
|
|
786
|
-
}
|
|
787
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.0-next.2", ngImport: i0, type: CdkTree, decorators: [{
|
|
788
|
-
type: Component,
|
|
789
|
-
args: [{
|
|
790
|
-
selector: 'cdk-tree',
|
|
791
|
-
exportAs: 'cdkTree',
|
|
792
|
-
template: `<ng-container cdkTreeNodeOutlet></ng-container>`,
|
|
793
|
-
host: {
|
|
794
|
-
'class': 'cdk-tree',
|
|
795
|
-
'role': 'tree',
|
|
796
|
-
'(keydown)': '_sendKeydownToKeyManager($event)',
|
|
797
|
-
},
|
|
798
|
-
encapsulation: ViewEncapsulation.None,
|
|
799
|
-
// The "OnPush" status for the `CdkTree` component is effectively a noop, so we are removing it.
|
|
800
|
-
// The view for `CdkTree` consists entirely of templates declared in other views. As they are
|
|
801
|
-
// declared elsewhere, they are checked when their declaration points are checked.
|
|
802
|
-
// tslint:disable-next-line:validate-decorators
|
|
803
|
-
changeDetection: ChangeDetectionStrategy.Default,
|
|
804
|
-
standalone: true,
|
|
805
|
-
imports: [CdkTreeNodeOutlet],
|
|
806
|
-
}]
|
|
807
|
-
}], ctorParameters: () => [{ type: i0.IterableDiffers }, { type: i0.ChangeDetectorRef }, { type: i1.Directionality }], propDecorators: { dataSource: [{
|
|
808
|
-
type: Input
|
|
809
|
-
}], treeControl: [{
|
|
810
|
-
type: Input
|
|
811
|
-
}], levelAccessor: [{
|
|
812
|
-
type: Input
|
|
813
|
-
}], childrenAccessor: [{
|
|
814
|
-
type: Input
|
|
815
|
-
}], trackBy: [{
|
|
816
|
-
type: Input
|
|
817
|
-
}], expansionKey: [{
|
|
818
|
-
type: Input
|
|
819
|
-
}], _nodeOutlet: [{
|
|
820
|
-
type: ViewChild,
|
|
821
|
-
args: [CdkTreeNodeOutlet, { static: true }]
|
|
822
|
-
}], _nodeDefs: [{
|
|
823
|
-
type: ContentChildren,
|
|
824
|
-
args: [CdkTreeNodeDef, {
|
|
825
|
-
// We need to use `descendants: true`, because Ivy will no longer match
|
|
826
|
-
// indirect descendants if it's left as false.
|
|
827
|
-
descendants: true,
|
|
828
|
-
}]
|
|
829
|
-
}] } });
|
|
830
|
-
/**
|
|
831
|
-
* Tree node for CdkTree. It contains the data in the tree node.
|
|
832
|
-
*/
|
|
833
|
-
export class CdkTreeNode {
|
|
834
|
-
/**
|
|
835
|
-
* The role of the tree node.
|
|
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
|
|
840
|
-
*/
|
|
841
|
-
get role() {
|
|
842
|
-
return 'treeitem';
|
|
843
|
-
}
|
|
844
|
-
set role(_role) {
|
|
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
|
-
if ((this.data && !this._isExpandable) || !this._inputIsExpandable) {
|
|
859
|
-
return;
|
|
860
|
-
}
|
|
861
|
-
// If the node is being set to expandable, ensure that the status of the
|
|
862
|
-
// node is propagated
|
|
863
|
-
if (this._inputIsExpanded) {
|
|
864
|
-
this.expand();
|
|
865
|
-
}
|
|
866
|
-
else if (this._inputIsExpanded === false) {
|
|
867
|
-
this.collapse();
|
|
868
|
-
}
|
|
869
|
-
}
|
|
870
|
-
get isExpanded() {
|
|
871
|
-
return this._tree.isExpanded(this._data);
|
|
872
|
-
}
|
|
873
|
-
set isExpanded(isExpanded) {
|
|
874
|
-
this._inputIsExpanded = isExpanded;
|
|
875
|
-
if (isExpanded) {
|
|
876
|
-
this.expand();
|
|
877
|
-
}
|
|
878
|
-
else {
|
|
879
|
-
this.collapse();
|
|
880
|
-
}
|
|
881
|
-
}
|
|
882
|
-
getLabel() {
|
|
883
|
-
return this.typeaheadLabel || this._elementRef.nativeElement.textContent?.trim() || '';
|
|
884
|
-
}
|
|
885
|
-
/**
|
|
886
|
-
* The most recently created `CdkTreeNode`. We save it in static variable so we can retrieve it
|
|
887
|
-
* in `CdkTree` and set the data to it.
|
|
888
|
-
*/
|
|
889
|
-
static { this.mostRecentTreeNode = null; }
|
|
890
|
-
/** The tree node's data. */
|
|
891
|
-
get data() {
|
|
892
|
-
return this._data;
|
|
893
|
-
}
|
|
894
|
-
set data(value) {
|
|
895
|
-
if (value !== this._data) {
|
|
896
|
-
this._data = value;
|
|
897
|
-
this._dataChanges.next();
|
|
898
|
-
}
|
|
899
|
-
}
|
|
900
|
-
/* If leaf node, return true to not assign aria-expanded attribute */
|
|
901
|
-
get isLeafNode() {
|
|
902
|
-
// If flat tree node data returns false for expandable property, it's a leaf node
|
|
903
|
-
if (this._tree.treeControl?.isExpandable !== undefined &&
|
|
904
|
-
!this._tree.treeControl.isExpandable(this._data)) {
|
|
905
|
-
return true;
|
|
906
|
-
// If nested tree node data returns 0 descendants, it's a leaf node
|
|
907
|
-
}
|
|
908
|
-
else if (this._tree.treeControl?.isExpandable === undefined &&
|
|
909
|
-
this._tree.treeControl?.getDescendants(this._data).length === 0) {
|
|
910
|
-
return true;
|
|
911
|
-
}
|
|
912
|
-
return false;
|
|
913
|
-
}
|
|
914
|
-
get level() {
|
|
915
|
-
// If the tree has a levelAccessor, use it to get the level. Otherwise read the
|
|
916
|
-
// aria-level off the parent node and use it as the level for this node (note aria-level is
|
|
917
|
-
// 1-indexed, while this property is 0-indexed, so we don't need to increment).
|
|
918
|
-
return this._tree._getLevel(this._data) ?? this._parentNodeAriaLevel;
|
|
919
|
-
}
|
|
920
|
-
/** Determines if the tree node is expandable. */
|
|
921
|
-
_isExpandable() {
|
|
922
|
-
if (this._tree.treeControl) {
|
|
923
|
-
if (this.isLeafNode) {
|
|
924
|
-
return false;
|
|
925
|
-
}
|
|
926
|
-
// For compatibility with trees created using TreeControl before we added
|
|
927
|
-
// CdkTreeNode#isExpandable.
|
|
928
|
-
return true;
|
|
929
|
-
}
|
|
930
|
-
return this._inputIsExpandable;
|
|
931
|
-
}
|
|
932
|
-
/**
|
|
933
|
-
* Determines the value for `aria-expanded`.
|
|
934
|
-
*
|
|
935
|
-
* For non-expandable nodes, this is `null`.
|
|
936
|
-
*/
|
|
937
|
-
_getAriaExpanded() {
|
|
938
|
-
if (!this._isExpandable()) {
|
|
939
|
-
return null;
|
|
940
|
-
}
|
|
941
|
-
return String(this.isExpanded);
|
|
942
|
-
}
|
|
943
|
-
/**
|
|
944
|
-
* Determines the size of this node's parent's child set.
|
|
945
|
-
*
|
|
946
|
-
* This is intended to be used for `aria-setsize`.
|
|
947
|
-
*/
|
|
948
|
-
_getSetSize() {
|
|
949
|
-
return this._tree._getSetSize(this._data);
|
|
950
|
-
}
|
|
951
|
-
/**
|
|
952
|
-
* Determines the index (starting from 1) of this node in its parent's child set.
|
|
953
|
-
*
|
|
954
|
-
* This is intended to be used for `aria-posinset`.
|
|
955
|
-
*/
|
|
956
|
-
_getPositionInSet() {
|
|
957
|
-
return this._tree._getPositionInSet(this._data);
|
|
958
|
-
}
|
|
959
|
-
constructor(_elementRef, _tree) {
|
|
960
|
-
this._elementRef = _elementRef;
|
|
961
|
-
this._tree = _tree;
|
|
962
|
-
this._tabindex = -1;
|
|
963
|
-
/** This emits when the node has been programatically activated or activated by keyboard. */
|
|
964
|
-
this.activation = new EventEmitter();
|
|
965
|
-
/** This emits when the node's expansion status has been changed. */
|
|
966
|
-
this.expandedChange = new EventEmitter();
|
|
967
|
-
/** Subject that emits when the component has been destroyed. */
|
|
968
|
-
this._destroyed = new Subject();
|
|
969
|
-
/** Emits when the node's data has changed. */
|
|
970
|
-
this._dataChanges = new Subject();
|
|
971
|
-
this._inputIsExpandable = false;
|
|
972
|
-
this._inputIsExpanded = undefined;
|
|
973
|
-
/**
|
|
974
|
-
* Flag used to determine whether or not we should be focusing the actual element based on
|
|
975
|
-
* some user interaction (click or focus). On click, we don't forcibly focus the element
|
|
976
|
-
* since the click could trigger some other component that wants to grab its own focus
|
|
977
|
-
* (e.g. menu, dialog).
|
|
978
|
-
*/
|
|
979
|
-
this._shouldFocus = true;
|
|
980
|
-
this._changeDetectorRef = inject(ChangeDetectorRef);
|
|
981
|
-
CdkTreeNode.mostRecentTreeNode = this;
|
|
982
|
-
}
|
|
983
|
-
ngOnInit() {
|
|
984
|
-
this._parentNodeAriaLevel = getParentNodeAriaLevel(this._elementRef.nativeElement);
|
|
985
|
-
this._tree
|
|
986
|
-
._getExpansionModel()
|
|
987
|
-
.changed.pipe(map(() => this.isExpanded), distinctUntilChanged())
|
|
988
|
-
.subscribe(() => {
|
|
989
|
-
this._changeDetectorRef.markForCheck();
|
|
990
|
-
});
|
|
991
|
-
this._tree._setNodeTypeIfUnset('flat');
|
|
992
|
-
this._tree._registerNode(this);
|
|
993
|
-
}
|
|
994
|
-
ngOnDestroy() {
|
|
995
|
-
// If this is the last tree node being destroyed,
|
|
996
|
-
// clear out the reference to avoid leaking memory.
|
|
997
|
-
if (CdkTreeNode.mostRecentTreeNode === this) {
|
|
998
|
-
CdkTreeNode.mostRecentTreeNode = null;
|
|
999
|
-
}
|
|
1000
|
-
this._dataChanges.complete();
|
|
1001
|
-
this._destroyed.next();
|
|
1002
|
-
this._destroyed.complete();
|
|
1003
|
-
}
|
|
1004
|
-
getParent() {
|
|
1005
|
-
return this._tree._getNodeParent(this) ?? null;
|
|
1006
|
-
}
|
|
1007
|
-
getChildren() {
|
|
1008
|
-
return this._tree._getNodeChildren(this);
|
|
1009
|
-
}
|
|
1010
|
-
/** Focuses this data node. Implemented for TreeKeyManagerItem. */
|
|
1011
|
-
focus() {
|
|
1012
|
-
this._tabindex = 0;
|
|
1013
|
-
if (this._shouldFocus) {
|
|
1014
|
-
this._elementRef.nativeElement.focus();
|
|
1015
|
-
}
|
|
1016
|
-
this._changeDetectorRef.markForCheck();
|
|
1017
|
-
}
|
|
1018
|
-
/** Defocus this data node. */
|
|
1019
|
-
unfocus() {
|
|
1020
|
-
this._tabindex = -1;
|
|
1021
|
-
this._changeDetectorRef.markForCheck();
|
|
1022
|
-
}
|
|
1023
|
-
/** Emits an activation event. Implemented for TreeKeyManagerItem. */
|
|
1024
|
-
activate() {
|
|
1025
|
-
if (this.isDisabled) {
|
|
1026
|
-
return;
|
|
1027
|
-
}
|
|
1028
|
-
this.activation.next(this._data);
|
|
1029
|
-
}
|
|
1030
|
-
/** Collapses this data node. Implemented for TreeKeyManagerItem. */
|
|
1031
|
-
collapse() {
|
|
1032
|
-
if (this.isExpandable) {
|
|
1033
|
-
this._tree.collapse(this._data);
|
|
1034
|
-
}
|
|
1035
|
-
}
|
|
1036
|
-
/** Expands this data node. Implemented for TreeKeyManagerItem. */
|
|
1037
|
-
expand() {
|
|
1038
|
-
if (this.isExpandable) {
|
|
1039
|
-
this._tree.expand(this._data);
|
|
1040
|
-
}
|
|
1041
|
-
}
|
|
1042
|
-
_focusItem() {
|
|
1043
|
-
if (this.isDisabled) {
|
|
1044
|
-
return;
|
|
1045
|
-
}
|
|
1046
|
-
this._tree._keyManager.focusItem(this);
|
|
1047
|
-
}
|
|
1048
|
-
_setActiveItem() {
|
|
1049
|
-
if (this.isDisabled) {
|
|
1050
|
-
return;
|
|
1051
|
-
}
|
|
1052
|
-
this._shouldFocus = false;
|
|
1053
|
-
this._tree._keyManager.focusItem(this);
|
|
1054
|
-
this._shouldFocus = true;
|
|
1055
|
-
}
|
|
1056
|
-
_emitExpansionState(expanded) {
|
|
1057
|
-
this.expandedChange.emit(expanded);
|
|
1058
|
-
}
|
|
1059
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.0-next.2", ngImport: i0, type: CdkTreeNode, deps: [{ token: i0.ElementRef }, { token: CdkTree }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
1060
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "16.1.0", version: "18.2.0-next.2", 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 }); }
|
|
1061
|
-
}
|
|
1062
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.0-next.2", ngImport: i0, type: CdkTreeNode, decorators: [{
|
|
1063
|
-
type: Directive,
|
|
1064
|
-
args: [{
|
|
1065
|
-
selector: 'cdk-tree-node',
|
|
1066
|
-
exportAs: 'cdkTreeNode',
|
|
1067
|
-
host: {
|
|
1068
|
-
'class': 'cdk-tree-node',
|
|
1069
|
-
'[attr.aria-expanded]': '_getAriaExpanded()',
|
|
1070
|
-
'[attr.aria-level]': 'level + 1',
|
|
1071
|
-
'[attr.aria-posinset]': '_getPositionInSet()',
|
|
1072
|
-
'[attr.aria-setsize]': '_getSetSize()',
|
|
1073
|
-
'[tabindex]': '_tabindex',
|
|
1074
|
-
'role': 'treeitem',
|
|
1075
|
-
'(click)': '_setActiveItem()',
|
|
1076
|
-
'(focus)': '_focusItem()',
|
|
1077
|
-
},
|
|
1078
|
-
standalone: true,
|
|
1079
|
-
}]
|
|
1080
|
-
}], ctorParameters: () => [{ type: i0.ElementRef }, { type: CdkTree }], propDecorators: { role: [{
|
|
1081
|
-
type: Input
|
|
1082
|
-
}], isExpandable: [{
|
|
1083
|
-
type: Input,
|
|
1084
|
-
args: [{ transform: booleanAttribute }]
|
|
1085
|
-
}], isExpanded: [{
|
|
1086
|
-
type: Input
|
|
1087
|
-
}], isDisabled: [{
|
|
1088
|
-
type: Input,
|
|
1089
|
-
args: [{ transform: booleanAttribute }]
|
|
1090
|
-
}], typeaheadLabel: [{
|
|
1091
|
-
type: Input,
|
|
1092
|
-
args: ['cdkTreeNodeTypeaheadLabel']
|
|
1093
|
-
}], activation: [{
|
|
1094
|
-
type: Output
|
|
1095
|
-
}], expandedChange: [{
|
|
1096
|
-
type: Output
|
|
1097
|
-
}] } });
|
|
1098
|
-
function getParentNodeAriaLevel(nodeElement) {
|
|
1099
|
-
let parent = nodeElement.parentElement;
|
|
1100
|
-
while (parent && !isNodeElement(parent)) {
|
|
1101
|
-
parent = parent.parentElement;
|
|
1102
|
-
}
|
|
1103
|
-
if (!parent) {
|
|
1104
|
-
if (typeof ngDevMode === 'undefined' || ngDevMode) {
|
|
1105
|
-
throw Error('Incorrect tree structure containing detached node.');
|
|
1106
|
-
}
|
|
1107
|
-
else {
|
|
1108
|
-
return -1;
|
|
1109
|
-
}
|
|
1110
|
-
}
|
|
1111
|
-
else if (parent.classList.contains('cdk-nested-tree-node')) {
|
|
1112
|
-
return numberAttribute(parent.getAttribute('aria-level'));
|
|
1113
|
-
}
|
|
1114
|
-
else {
|
|
1115
|
-
// The ancestor element is the cdk-tree itself
|
|
1116
|
-
return 0;
|
|
1117
|
-
}
|
|
1118
|
-
}
|
|
1119
|
-
function isNodeElement(element) {
|
|
1120
|
-
const classList = element.classList;
|
|
1121
|
-
return !!(classList?.contains('cdk-nested-tree-node') || classList?.contains('cdk-tree'));
|
|
1122
|
-
}
|
|
1123
|
-
//# 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    if ((this.data && !this._isExpandable) || !this._inputIsExpandable) {\n      return;\n    }\n    // If the node is being set to expandable, ensure that the status of the\n    // node is propagated\n    if (this._inputIsExpanded) {\n      this.expand();\n    } else if (this._inputIsExpanded === false) {\n      this.collapse();\n    }\n  }\n\n  @Input()\n  get isExpanded(): boolean {\n    return this._tree.isExpanded(this._data);\n  }\n  set isExpanded(isExpanded: boolean) {\n    this._inputIsExpanded = isExpanded;\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  private _inputIsExpanded: boolean | undefined = undefined;\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"]}
|