adev/src/content/guide/aria/tree.md
A tree displays hierarchical data where items can expand to reveal children or collapse to hide them. Users navigate with arrow keys, expand and collapse nodes, and optionally select items for navigation or data selection scenarios.
<docs-code-multifile preview hideCode path="adev/src/content/examples/aria/tree/src/single-select/basic/app/app.ts"> <docs-code header="TS" path="adev/src/content/examples/aria/tree/src/single-select/basic/app/app.ts"/> <docs-code header="HTML" path="adev/src/content/examples/aria/tree/src/single-select/basic/app/app.html"/> <docs-code header="CSS" path="adev/src/content/examples/aria/tree/src/single-select/basic/app/app.css"/> </docs-code-multifile>Trees work well for displaying hierarchical data where users need to navigate through nested structures.
Use trees when:
Avoid trees when:
Use a tree for navigation where clicking items triggers actions rather than selecting them.
<docs-tab-group> <docs-tab label="Basic"> <docs-code-multifile preview hideCode path="adev/src/content/examples/aria/tree/src/nav/basic/app/app.ts"> <docs-code header="TS" path="adev/src/content/examples/aria/tree/src/nav/basic/app/app.ts"/> <docs-code header="HTML" path="adev/src/content/examples/aria/tree/src/nav/basic/app/app.html"/> <docs-code header="CSS" path="adev/src/content/examples/aria/tree/src/nav/basic/app/app.css"/> </docs-code-multifile> </docs-tab> </docs-tab-group>Set [nav]="true" to enable navigation mode. This uses aria-current to indicate the current page instead of selection.
Enable single selection for scenarios where users choose one item from the tree.
<docs-tab-group> <docs-tab label="Basic"> <docs-code-multifile preview hideCode path="adev/src/content/examples/aria/tree/src/single-select/basic/app/app.ts"> <docs-code header="TS" path="adev/src/content/examples/aria/tree/src/single-select/basic/app/app.ts"/> <docs-code header="HTML" path="adev/src/content/examples/aria/tree/src/single-select/basic/app/app.html"/> <docs-code header="CSS" path="adev/src/content/examples/aria/tree/src/single-select/basic/app/app.css"/> </docs-code-multifile> </docs-tab> <docs-tab label="Retro"> <docs-code-multifile preview hideCode path="adev/src/content/examples/aria/tree/src/single-select/retro/app/app.ts"> <docs-code header="TS" path="adev/src/content/examples/aria/tree/src/single-select/retro/app/app.ts"/> <docs-code header="HTML" path="adev/src/content/examples/aria/tree/src/single-select/retro/app/app.html"/> <docs-code header="CSS" path="adev/src/content/examples/aria/tree/src/single-select/retro/app/app.css"/> </docs-code-multifile> </docs-tab> </docs-tab-group>Leave [multi]="false" (the default) for single selection. Users press Space to select the focused item.
Allow users to select multiple items from the tree.
<docs-tab-group> <docs-tab label="Basic"> <docs-code-multifile preview hideCode path="adev/src/content/examples/aria/tree/src/multi-select/basic/app/app.ts"> <docs-code header="TS" path="adev/src/content/examples/aria/tree/src/multi-select/basic/app/app.ts"/> <docs-code header="HTML" path="adev/src/content/examples/aria/tree/src/multi-select/basic/app/app.html"/> <docs-code header="CSS" path="adev/src/content/examples/aria/tree/src/multi-select/basic/app/app.css"/> </docs-code-multifile> </docs-tab> <docs-tab label="Retro"> <docs-code-multifile preview hideCode path="adev/src/content/examples/aria/tree/src/multi-select/retro/app/app.ts"> <docs-code header="TS" path="adev/src/content/examples/aria/tree/src/multi-select/retro/app/app.ts"/> <docs-code header="HTML" path="adev/src/content/examples/aria/tree/src/multi-select/retro/app/app.html"/> <docs-code header="CSS" path="adev/src/content/examples/aria/tree/src/multi-select/retro/app/app.css"/> </docs-code-multifile> </docs-tab> </docs-tab-group>Set [multi]="true" on the tree. Users select items individually with Space or select ranges with Shift+Arrow keys.
When selection follows focus, the focused item is automatically selected. This simplifies interaction for navigation scenarios.
<docs-tab-group> <docs-tab label="Basic"> <docs-code-multifile preview hideCode path="adev/src/content/examples/aria/tree/src/single-select-follow-focus/basic/app/app.ts"> <docs-code header="TS" path="adev/src/content/examples/aria/tree/src/single-select-follow-focus/basic/app/app.ts"/> <docs-code header="HTML" path="adev/src/content/examples/aria/tree/src/single-select-follow-focus/basic/app/app.html"/> <docs-code header="CSS" path="adev/src/content/examples/aria/tree/src/single-select-follow-focus/basic/app/app.css"/> </docs-code-multifile> </docs-tab> <docs-tab label="Retro"> <docs-code-multifile preview hideCode path="adev/src/content/examples/aria/tree/src/single-select-follow-focus/retro/app/app.ts"> <docs-code header="TS" path="adev/src/content/examples/aria/tree/src/single-select-follow-focus/retro/app/app.ts"/> <docs-code header="HTML" path="adev/src/content/examples/aria/tree/src/single-select-follow-focus/retro/app/app.html"/> <docs-code header="CSS" path="adev/src/content/examples/aria/tree/src/single-select-follow-focus/retro/app/app.css"/> </docs-code-multifile> </docs-tab> </docs-tab-group>Set [selectionMode]="'follow'" on the tree. Selection automatically updates as users navigate with arrow keys.
Disable specific tree nodes to prevent interaction. Control whether disabled items can receive focus.
<docs-tab-group> <docs-tab label="Basic"> <docs-code-multifile preview hideCode path="adev/src/content/examples/aria/tree/src/disabled-focusable/basic/app/app.ts"> <docs-code header="TS" path="adev/src/content/examples/aria/tree/src/disabled-focusable/basic/app/app.ts"/> <docs-code header="HTML" path="adev/src/content/examples/aria/tree/src/disabled-focusable/basic/app/app.html"/> <docs-code header="CSS" path="adev/src/content/examples/aria/tree/src/disabled-focusable/basic/app/app.css"/> </docs-code-multifile> </docs-tab> <docs-tab label="Retro"> <docs-code-multifile preview hideCode path="adev/src/content/examples/aria/tree/src/disabled-focusable/retro/app/app.ts"> <docs-code header="TS" path="adev/src/content/examples/aria/tree/src/disabled-focusable/retro/app/app.ts"/> <docs-code header="HTML" path="adev/src/content/examples/aria/tree/src/disabled-focusable/retro/app/app.html"/> <docs-code header="CSS" path="adev/src/content/examples/aria/tree/src/disabled-focusable/retro/app/app.css"/> </docs-code-multifile> </docs-tab> </docs-tab-group>When [softDisabled]="true" on the tree, disabled items can receive focus but cannot be activated or selected. When [softDisabled]="false", disabled items are skipped during keyboard navigation.
Angular Aria provides component harnesses for testing tree components. Here is an example of how to use the harnesses in a component test:
import {ComponentFixture, TestBed} from '@angular/core/testing';
import {HarnessLoader} from '@angular/cdk/testing';
import {TestbedHarnessEnvironment} from '@angular/cdk/testing/testbed';
import {TreeHarness} from '@angular/aria/tree/testing';
import {MyTreeComponent} from './my-tree'; // Your component
describe('MyTreeComponent', () => {
let fixture: ComponentFixture<MyTreeComponent>;
let loader: HarnessLoader;
beforeEach(async () => {
TestBed.configureTestingModule({
imports: [MyTreeComponent],
});
fixture = TestBed.createComponent(MyTreeComponent);
await fixture.whenStable();
loader = TestbedHarnessEnvironment.loader(fixture);
});
it('should navigate and expand tree items', async () => {
const tree = await loader.getHarness(TreeHarness);
// Get top-level structure representation
expect(await tree.getTreeStructure()).toEqual({
children: [{text: 'public'}, {text: 'src'}, {text: 'package.json'}],
});
// Get all items (currently visible)
const items = await tree.getItems();
expect(items.length).toBe(3);
// Expand the first item ('public')
expect(await items[0].isExpanded()).toBe(false);
await items[0].click();
expect(await items[0].isExpanded()).toBe(true);
// Verifying tree structure updates after expansion
expect(await tree.getTreeStructure()).toEqual({
children: [
{
text: 'public',
children: [{text: 'index.html'}, {text: 'styles.css'}],
},
{text: 'src'},
{text: 'package.json'},
],
});
});
});
The container directive that manages hierarchical navigation and selection.
| Property | Type | Default | Description |
|---|---|---|---|
disabled | boolean | false | Disables the entire tree |
softDisabled | boolean | true | When true, disabled items are focusable but not interactive |
multi | boolean | false | Whether multiple items can be selected |
selectionMode | 'explicit' | 'follow' | 'explicit' | Whether selection requires explicit action or follows focus |
nav | boolean | false | Whether the tree is in navigation mode (uses aria-current) |
wrap | boolean | true | Whether keyboard navigation wraps from last to first item |
focusMode | 'roving' | 'activedescendant' | 'roving' | Focus strategy used by the tree |
values | any[] | [] | Selected item values (supports two-way binding) |
| Method | Parameters | Description |
|---|---|---|
expandAll | none | Expands all tree nodes |
collapseAll | none | Collapses all tree nodes |
selectAll | none | Selects all items (only in multi-select mode) |
clearSelection | none | Clears all selection |
An individual node in the tree that can contain child nodes.
| Property | Type | Default | Description |
|---|---|---|---|
parent | Tree | TreeItemGroup | — | Required. The parent Tree root or TreeItemGroup. |
value | any | — | Required. Unique value for this tree item |
disabled | boolean | false | Disables this item |
expanded | boolean | false | Whether the node is expanded (supports two-way binding) |
| Property | Type | Description |
|---|---|---|
selected | Signal<boolean> | Whether the item is selected |
active | Signal<boolean> | Whether the item currently has focus |
hasChildren | Signal<boolean> | Whether the item has child nodes |
| Method | Parameters | Description |
|---|---|---|
expand | none | Expands this node |
collapse | none | Collapses this node |
toggle | none | Toggles the expansion state |
The structural directive applied to an ng-template that holds the children nodes of an expandable tree item.
| Property | Type | Default | Description |
|---|---|---|---|
ownedBy | TreeItem | — | Required. The reference of the parent ngTreeItem. |
<ul ngTree #tree="ngTree">
<li ngTreeItem [parent]="tree" value="parent" #parentItem="ngTreeItem">
Parent Item
<ul role="group">
<ng-template ngTreeItemGroup [ownedBy]="parentItem" #group="ngTreeItemGroup">
<li ngTreeItem [parent]="group" value="child1">Child 1</li>
<li ngTreeItem [parent]="group" value="child2">Child 2</li>
</ng-template>
</ul>
</li>
</ul>