Skip to content

Commit 06dd309

Browse files
committed
Remove multiselectable part from tabs ui pattern
1 parent f54cc85 commit 06dd309

File tree

5 files changed

+38
-85
lines changed

5 files changed

+38
-85
lines changed

src/cdk-experimental/tabs/tabs.ts

-4
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,6 @@ export class CdkTabs {
5353
'[attr.tabindex]': 'pattern.tabindex()',
5454
'[attr.aria-disabled]': 'pattern.disabled()',
5555
'[attr.aria-orientation]': 'pattern.orientation()',
56-
'[attr.aria-multiselectable]': 'pattern.multiselectable()',
5756
'[attr.aria-activedescendant]': 'pattern.activedescendant()',
5857
'(keydown)': 'pattern.onKeydown($event)',
5958
'(mousedown)': 'pattern.onPointerdown($event)',
@@ -77,9 +76,6 @@ export class CdkTablist {
7776
/** Whether the tablist is vertically or horizontally oriented. */
7877
orientation = input<'vertical' | 'horizontal'>('horizontal');
7978

80-
/** Whether multiple items in the list can be selected at once. */
81-
multiselectable = input(false, {transform: booleanAttribute});
82-
8379
/** Whether focus should wrap when navigating. */
8480
wrap = input(true, {transform: booleanAttribute});
8581

src/cdk-experimental/ui-patterns/tabs/tablist.ts

+15-56
Original file line numberDiff line numberDiff line change
@@ -6,14 +6,13 @@
66
* found in the LICENSE file at https://angular.dev/license
77
*/
88

9-
import {ModifierKey as Modifier} from '../behaviors/event-manager/event-manager';
109
import {KeyboardEventManager} from '../behaviors/event-manager/keyboard-event-manager';
1110
import {PointerEventManager} from '../behaviors/event-manager/pointer-event-manager';
1211
import {TabPattern} from './tab';
1312
import {ListSelection, ListSelectionInputs} from '../behaviors/list-selection/list-selection';
1413
import {ListNavigation, ListNavigationInputs} from '../behaviors/list-navigation/list-navigation';
1514
import {ListFocus, ListFocusInputs} from '../behaviors/list-focus/list-focus';
16-
import {computed, Signal} from '@angular/core';
15+
import {computed, signal, Signal} from '@angular/core';
1716

1817
/** The selection operations that the tablist can perform. */
1918
interface SelectOptions {
@@ -28,7 +27,7 @@ interface SelectOptions {
2827

2928
/** The required inputs for the tablist. */
3029
export type TablistInputs = ListNavigationInputs<TabPattern> &
31-
ListSelectionInputs<TabPattern> &
30+
Omit<ListSelectionInputs<TabPattern>, 'multiselectable'> &
3231
ListFocusInputs<TabPattern> & {
3332
disabled: Signal<boolean>;
3433
};
@@ -56,12 +55,6 @@ export class TablistPattern {
5655
/** The id of the current active tab. */
5756
activedescendant = computed(() => this.focusManager.getActiveDescendant());
5857

59-
/** Whether multiple tabs in the tablist can be selected at once. */
60-
multiselectable: Signal<boolean>;
61-
62-
/** The number of tabs in the tablist. */
63-
setsize = computed(() => this.navigation.inputs.items().length);
64-
6558
followFocus = computed(() => this.inputs.selectionMode() === 'follow');
6659

6760
/** The key used to navigate to the previous tab in the tablist. */
@@ -84,50 +77,20 @@ export class TablistPattern {
8477
keydown = computed(() => {
8578
const manager = new KeyboardEventManager();
8679

87-
if (!this.followFocus()) {
88-
manager
89-
.on(this.prevKey, () => this.prev())
90-
.on(this.nextKey, () => this.next())
91-
.on('Home', () => this.first())
92-
.on('End', () => this.last());
93-
}
94-
9580
if (this.followFocus()) {
9681
manager
9782
.on(this.prevKey, () => this.prev({selectOne: true}))
9883
.on(this.nextKey, () => this.next({selectOne: true}))
9984
.on('Home', () => this.first({selectOne: true}))
10085
.on('End', () => this.last({selectOne: true}));
101-
}
102-
103-
if (this.inputs.multiselectable()) {
104-
manager
105-
.on(Modifier.Shift, ' ', () => this._updateSelection({selectFromAnchor: true}))
106-
.on(Modifier.Shift, this.prevKey, () => this.prev({toggle: true}))
107-
.on(Modifier.Shift, this.nextKey, () => this.next({toggle: true}))
108-
.on(Modifier.Ctrl | Modifier.Shift, 'Home', () => this.first({selectFromActive: true}))
109-
.on(Modifier.Ctrl | Modifier.Shift, 'End', () => this.last({selectFromActive: true}))
110-
.on(Modifier.Ctrl, 'A', () => this._updateSelection({selectAll: true}));
111-
}
112-
113-
if (!this.followFocus() && this.inputs.multiselectable()) {
114-
manager
115-
.on(' ', () => this._updateSelection({toggle: true}))
116-
.on('Enter', () => this._updateSelection({toggle: true}));
117-
}
118-
119-
if (!this.followFocus() && !this.inputs.multiselectable()) {
120-
manager
121-
.on(' ', () => this._updateSelection({toggleOne: true}))
122-
.on('Enter', () => this._updateSelection({toggleOne: true}));
123-
}
124-
125-
if (this.inputs.multiselectable() && this.followFocus()) {
86+
} else {
12687
manager
127-
.on(Modifier.Ctrl, this.prevKey, () => this.prev())
128-
.on(Modifier.Ctrl, this.nextKey, () => this.next())
129-
.on(Modifier.Ctrl, 'Home', () => this.first()) // TODO: Not in spec but prob should be.
130-
.on(Modifier.Ctrl, 'End', () => this.last()); // TODO: Not in spec but prob should be.
88+
.on(this.prevKey, () => this.prev())
89+
.on(this.nextKey, () => this.next())
90+
.on('Home', () => this.first())
91+
.on('End', () => this.last())
92+
.on(' ', () => this._updateSelection({selectOne: true}))
93+
.on('Enter', () => this._updateSelection({selectOne: true}));
13194
}
13295

13396
return manager;
@@ -136,25 +99,21 @@ export class TablistPattern {
13699
/** The pointerdown event manager for the tablist. */
137100
pointerdown = computed(() => {
138101
const manager = new PointerEventManager();
139-
140-
if (this.inputs.multiselectable()) {
141-
manager
142-
.on(e => this.goto(e, {toggle: true}))
143-
.on(Modifier.Shift, e => this.goto(e, {selectFromActive: true}));
144-
} else {
145-
manager.on(e => this.goto(e, {toggleOne: true}));
146-
}
102+
manager.on(e => this.goto(e, {selectOne: true}));
147103

148104
return manager;
149105
});
150106

151107
constructor(readonly inputs: TablistInputs) {
152108
this.disabled = inputs.disabled;
153109
this.orientation = inputs.orientation;
154-
this.multiselectable = inputs.multiselectable;
155110

156111
this.navigation = new ListNavigation(inputs);
157-
this.selection = new ListSelection({...inputs, navigation: this.navigation});
112+
this.selection = new ListSelection({
113+
...inputs,
114+
navigation: this.navigation,
115+
multiselectable: signal(false),
116+
});
158117
this.focusManager = new ListFocus({...inputs, navigation: this.navigation});
159118
}
160119

Original file line numberDiff line numberDiff line change
@@ -1,20 +1,20 @@
1-
.example-listbox-controls {
1+
.example-tablist-controls {
22
display: flex;
33
align-items: center;
44
gap: 16px;
55
padding-bottom: 16px;
66
}
77

8-
.tabs {
8+
.example-tabs {
99
border: 1px solid var(--mat-sys-outline);
1010
min-height: 30vh;
1111
}
1212

13-
.tabs:has(ul[aria-orientation='vertical']) {
13+
.example-tabs:has(ul[aria-orientation='vertical']) {
1414
display: flex;
1515
}
1616

17-
ul {
17+
.example-tablist {
1818
gap: 8px;
1919
margin: 0;
2020
padding: 8px;
@@ -28,22 +28,22 @@ ul {
2828
overflow: scroll;
2929
}
3030

31-
ul[aria-orientation='vertical'] {
31+
.example-tablist[aria-orientation='vertical'] {
3232
flex-direction: column;
3333
border-bottom: initial;
3434
border-right: 1px solid var(--mat-sys-outline);
3535
justify-content: start;
3636
}
3737

38-
ul[aria-orientation='horizontal'] li::before {
38+
.example-tablist[aria-orientation='horizontal'] li::before {
3939
display: none;
4040
}
4141

42-
ul[aria-orientation='horizontal'] li[aria-selected='true']::before {
42+
.example-tablist[aria-orientation='horizontal'] li[aria-selected='true']::before {
4343
display: block;
4444
}
4545

46-
li {
46+
.example-tab {
4747
gap: 16px;
4848
padding: 16px;
4949
display: flex;
@@ -52,25 +52,25 @@ li {
5252
border-radius: var(--mat-sys-corner-extra-small);
5353
}
5454

55-
li:hover,
56-
li[tabindex='0'] {
55+
.example-tab:hover,
56+
.example-tab[tabindex='0'] {
5757
outline: 1px solid var(--mat-sys-outline);
5858
background: var(--mat-sys-surface-container);
5959
}
6060

61-
li:focus-within {
61+
.example-tab:focus-within {
6262
outline: 2px solid var(--mat-sys-primary);
6363
background: var(--mat-sys-surface-container);
6464
}
6565

66-
li[aria-selected='true'] {
66+
.example-tab[aria-selected='true'] {
6767
background-color: var(--mat-sys-secondary-container);
6868
}
6969

70-
[role='tabpanel'] {
70+
.example-tabpanel {
7171
margin: 8px;
7272
}
7373

74-
[role='tabpanel'][aria-hidden='true'] {
74+
.example-tabpanel[aria-hidden='true'] {
7575
display: none;
7676
}

src/components-examples/cdk-experimental/tabs/cdk-tabs/cdk-tabs-example.html

+9-10
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
1-
<div class="example-listbox-controls">
1+
<div class="example-tablist-controls">
22
<mat-checkbox [formControl]="wrap">Wrap</mat-checkbox>
3-
<mat-checkbox [formControl]="multi">Multi</mat-checkbox>
43
<mat-checkbox [formControl]="disabled">Disabled</mat-checkbox>
54
<mat-checkbox [formControl]="skipDisabled">Skip Disabled</mat-checkbox>
65

@@ -30,24 +29,24 @@
3029
</div>
3130

3231
<!-- #docregion tabs -->
33-
<div cdkTabs class="tabs">
32+
<div cdkTabs class="example-tabs">
3433
<ul
3534
cdkTablist
35+
class="example-tablist"
3636
[wrap]="wrap.value"
3737
[disabled]="disabled.value"
38-
[multiselectable]="multi.value"
3938
[skipDisabled]="skipDisabled.value"
4039
[orientation]="orientation"
4140
[focusMode]="focusMode"
4241
[selectionMode]="selectionMode"
4342
>
44-
<li cdkTab>tab 1</li>
45-
<li cdkTab>tab 2</li>
46-
<li cdkTab>tab 3</li>
43+
<li cdkTab class="example-tab">tab 1</li>
44+
<li cdkTab class="example-tab">tab 2</li>
45+
<li cdkTab class="example-tab">tab 3</li>
4746
</ul>
4847

49-
<div cdkTabpanel>Tabpanel 1</div>
50-
<div cdkTabpanel>Tabpanel 2</div>
51-
<div cdkTabpanel>Tabpanel 3</div>
48+
<div cdkTabpanel class="example-tabpanel">Tabpanel 1</div>
49+
<div cdkTabpanel class="example-tabpanel">Tabpanel 2</div>
50+
<div cdkTabpanel class="example-tabpanel">Tabpanel 3</div>
5251
</div>
5352
<!-- #enddocregion tabs -->

src/components-examples/cdk-experimental/tabs/cdk-tabs/cdk-tabs-example.ts

-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,6 @@ export class CdkTabsExample {
2828
selectionMode: 'explicit' | 'follow' = 'follow';
2929

3030
wrap = new FormControl(true, {nonNullable: true});
31-
multi = new FormControl(false, {nonNullable: true});
3231
disabled = new FormControl(false, {nonNullable: true});
3332
skipDisabled = new FormControl(true, {nonNullable: true});
3433
}

0 commit comments

Comments
 (0)