Skip to content

Commit 407c869

Browse files
authored
feat: Add "Select all" control for Multiselect (#3336)
1 parent bb0af35 commit 407c869

28 files changed

+857
-179
lines changed

pages/multiselect/constants.ts

+75
Original file line numberDiff line numberDiff line change
@@ -33,3 +33,78 @@ export const getInlineAriaLabel = (selectedOptions: MultiselectProps.Options) =>
3333

3434
return label + ' selected';
3535
};
36+
37+
export const groupedOptions = [
38+
{
39+
label: 'First category',
40+
options: [
41+
{
42+
value: 'option1',
43+
label: 'option1',
44+
},
45+
{
46+
value: 'option2',
47+
label: 'option2',
48+
description: 'option2',
49+
tags: ['2-CPU', '2Gb RAM', 'Stuff', 'More stuff', 'A lot'],
50+
disabled: true,
51+
},
52+
{
53+
value: 'option3',
54+
label: 'option3',
55+
description: 'option3',
56+
tags: ['2-CPU', '2Gb RAM'],
57+
},
58+
],
59+
},
60+
{
61+
label: 'Second category',
62+
options: [
63+
{
64+
value: 'option4',
65+
label: 'option4',
66+
description: 'option4',
67+
tags: ['2-CPU', '2Gb RAM'],
68+
},
69+
{
70+
value: 'option5',
71+
label: 'option5',
72+
description: 'option5',
73+
tags: ['2-CPU', '2Gb RAM', 'Stuff', 'More stuff', 'A lot'],
74+
disabled: true,
75+
},
76+
{
77+
value: 'option6',
78+
label: 'option6',
79+
description: 'option6',
80+
tags: ['2-CPU', '2Gb RAM'],
81+
},
82+
],
83+
},
84+
{
85+
label: 'Third category',
86+
options: [
87+
{
88+
value: 'option7',
89+
label: 'option7',
90+
description: 'option7',
91+
tags: ['2-CPU', '2Gb RAM'],
92+
disabled: true,
93+
},
94+
{
95+
value: 'option8',
96+
label: 'option8',
97+
description: 'option8',
98+
tags: ['2-CPU', '2Gb RAM', 'Stuff', 'More stuff', 'A lot'],
99+
disabled: true,
100+
},
101+
{
102+
value: 'option9',
103+
label: 'option9',
104+
description: 'option9',
105+
tags: ['2-CPU', '2Gb RAM'],
106+
disabled: true,
107+
},
108+
],
109+
},
110+
];

pages/multiselect/multiselect.test.page.tsx

+3-78
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ import * as React from 'react';
55
import Box from '~components/box';
66
import Multiselect, { MultiselectProps } from '~components/multiselect';
77

8-
import { deselectAriaLabel, getInlineAriaLabel, i18nStrings } from './constants';
8+
import { deselectAriaLabel, getInlineAriaLabel, groupedOptions, i18nStrings } from './constants';
99

1010
const _selectedOptions1 = [
1111
{
@@ -36,83 +36,8 @@ const _selectedOptions2 = [
3636
disabled: true,
3737
},
3838
];
39-
const options1 = [
40-
{
41-
label: 'First category',
42-
options: [
43-
{
44-
value: 'option1',
45-
label: 'option1',
46-
},
47-
{
48-
value: 'option2',
49-
label: 'option2',
50-
description: 'option2',
51-
tags: ['2-CPU', '2Gb RAM', 'Stuff', 'More stuff', 'A lot'],
52-
disabled: true,
53-
},
54-
{
55-
value: 'option3',
56-
label: 'option3',
57-
description: 'option3',
58-
tags: ['2-CPU', '2Gb RAM'],
59-
},
60-
],
61-
},
62-
{
63-
label: 'Second category',
64-
options: [
65-
{
66-
value: 'option4',
67-
label: 'option4',
68-
description: 'option4',
69-
tags: ['2-CPU', '2Gb RAM'],
70-
},
71-
{
72-
value: 'option5',
73-
label: 'option5',
74-
description: 'option5',
75-
tags: ['2-CPU', '2Gb RAM', 'Stuff', 'More stuff', 'A lot'],
76-
disabled: true,
77-
},
78-
{
79-
value: 'option6',
80-
label: 'option6',
81-
description: 'option6',
82-
tags: ['2-CPU', '2Gb RAM'],
83-
},
84-
],
85-
},
86-
];
87-
const options2 = [
88-
...options1,
89-
{
90-
label: 'Third category',
91-
options: [
92-
{
93-
value: 'option7',
94-
label: 'option7',
95-
description: 'option7',
96-
tags: ['2-CPU', '2Gb RAM'],
97-
disabled: true,
98-
},
99-
{
100-
value: 'option8',
101-
label: 'option8',
102-
description: 'option8',
103-
tags: ['2-CPU', '2Gb RAM', 'Stuff', 'More stuff', 'A lot'],
104-
disabled: true,
105-
},
106-
{
107-
value: 'option9',
108-
label: 'option9',
109-
description: 'option9',
110-
tags: ['2-CPU', '2Gb RAM'],
111-
disabled: true,
112-
},
113-
],
114-
},
115-
];
39+
const options1 = groupedOptions.slice(0, 2); // First 2 option groups
40+
const options2 = groupedOptions; // All 3 option groups
11641

11742
export default function MultiselectPage() {
11843
const [selectedOptions1, setSelectedOptions1] = React.useState<MultiselectProps.Options>(_selectedOptions1);

pages/multiselect/select-all.page.tsx

+172
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,172 @@
1+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2+
// SPDX-License-Identifier: Apache-2.0
3+
import React, { useContext, useState } from 'react';
4+
5+
import Box from '~components/box';
6+
import { OptionGroup } from '~components/internal/components/option/interfaces';
7+
import Multiselect, { MultiselectProps } from '~components/multiselect';
8+
import SpaceBetween from '~components/space-between';
9+
10+
import AppContext, { AppContextType } from '../app/app-context';
11+
import ScreenshotArea from '../utils/screenshot-area';
12+
import { deselectAriaLabel, getInlineAriaLabel, groupedOptions, i18nStrings } from './constants';
13+
14+
type DemoContext = React.Context<
15+
AppContextType<{
16+
closeAfter?: boolean;
17+
expandToViewport?: boolean;
18+
inlineTokens?: boolean;
19+
manyOptions?: boolean;
20+
virtualScroll?: boolean;
21+
withDisabledOptions?: boolean;
22+
withFiltering?: boolean;
23+
withGroups?: boolean;
24+
}>
25+
>;
26+
27+
// Mix of grouped and top-level options
28+
const groupedOptionsWithDisabledOptions: MultiselectProps.Options = [
29+
...groupedOptions.slice(0, 2), // First 2 groups
30+
...groupedOptions[2].options, // children of the 2rd group
31+
];
32+
33+
const initialSelectedOptions = [
34+
(groupedOptionsWithDisabledOptions[0] as OptionGroup).options[2],
35+
(groupedOptionsWithDisabledOptions[1] as OptionGroup).options[0],
36+
];
37+
38+
export default function MultiselectPage() {
39+
const { urlParams, setUrlParams } = useContext(AppContext as DemoContext);
40+
const [selectedOptions, setSelectedOptions] = useState<MultiselectProps.Options>(initialSelectedOptions);
41+
42+
const groupedOptions: MultiselectProps.Options = urlParams.withDisabledOptions
43+
? groupedOptionsWithDisabledOptions
44+
: groupedOptionsWithDisabledOptions.map(option => ({
45+
...option,
46+
disabled: false,
47+
options:
48+
'options' in option && option.options.length
49+
? option.options.map(childOption => ({ ...childOption, disabled: false }))
50+
: undefined,
51+
}));
52+
53+
const baseOptions: MultiselectProps.Options = urlParams.withGroups
54+
? groupedOptions
55+
: groupedOptions.reduce(
56+
(previousValue: MultiselectProps.Options, currentValue: MultiselectProps.Option) =>
57+
'options' in currentValue && (currentValue as OptionGroup).options?.length
58+
? [...previousValue, ...(currentValue as OptionGroup).options]
59+
: [...previousValue, currentValue],
60+
[]
61+
);
62+
63+
const options = urlParams.manyOptions
64+
? [
65+
...baseOptions,
66+
...Array(100)
67+
.fill(undefined)
68+
.map((_, index) => ({
69+
value: `option${index + baseOptions.length + 1}`,
70+
label: `option${index + baseOptions.length + 1}`,
71+
description: `option${index + baseOptions.length + 1}`,
72+
tags: ['2-CPU', '2Gb RAM'],
73+
})),
74+
]
75+
: baseOptions;
76+
77+
return (
78+
<article>
79+
<h1>Multiselect with &quot;Select all&quot;</h1>
80+
<Box padding={{ horizontal: 'l' }}>
81+
<SpaceBetween size="xxl">
82+
<SpaceBetween direction="horizontal" size="l">
83+
<label>
84+
<input
85+
type="checkbox"
86+
checked={!!urlParams.withGroups}
87+
onChange={e => setUrlParams({ withGroups: e.target.checked })}
88+
/>{' '}
89+
With groups
90+
</label>
91+
<label>
92+
<input
93+
type="checkbox"
94+
checked={!!urlParams.withDisabledOptions}
95+
onChange={e => setUrlParams({ withDisabledOptions: e.target.checked })}
96+
/>{' '}
97+
With disabled options
98+
</label>
99+
<label>
100+
<input
101+
type="checkbox"
102+
checked={!!urlParams.withFiltering}
103+
onChange={e => setUrlParams({ withFiltering: e.target.checked })}
104+
/>{' '}
105+
With filtering
106+
</label>
107+
<label>
108+
<input
109+
type="checkbox"
110+
checked={!!urlParams.expandToViewport}
111+
onChange={e => setUrlParams({ expandToViewport: e.target.checked })}
112+
/>{' '}
113+
Expand to viewport
114+
</label>
115+
<label>
116+
<input
117+
type="checkbox"
118+
checked={!!urlParams.inlineTokens}
119+
onChange={e => setUrlParams({ inlineTokens: e.target.checked })}
120+
/>{' '}
121+
Inline tokens
122+
</label>
123+
<label>
124+
<input
125+
type="checkbox"
126+
checked={!!urlParams.closeAfter}
127+
onChange={e => setUrlParams({ closeAfter: e.target.checked })}
128+
/>{' '}
129+
Close after selection
130+
</label>
131+
<label>
132+
<input
133+
type="checkbox"
134+
checked={!!urlParams.virtualScroll}
135+
onChange={e => setUrlParams({ virtualScroll: e.target.checked })}
136+
/>{' '}
137+
Virtual scroll
138+
</label>
139+
<label>
140+
<input
141+
type="checkbox"
142+
checked={!!urlParams.manyOptions}
143+
onChange={e => setUrlParams({ manyOptions: e.target.checked })}
144+
/>{' '}
145+
Many options
146+
</label>
147+
</SpaceBetween>
148+
149+
<ScreenshotArea>
150+
<Multiselect
151+
selectedOptions={selectedOptions}
152+
deselectAriaLabel={deselectAriaLabel}
153+
filteringType={urlParams.withFiltering ? 'auto' : 'none'}
154+
options={options}
155+
i18nStrings={{ ...i18nStrings, selectAllText: 'Select all' }}
156+
enableSelectAll={true}
157+
placeholder={'Choose options'}
158+
onChange={event => {
159+
setSelectedOptions(event.detail.selectedOptions);
160+
}}
161+
ariaLabel={urlParams.inlineTokens ? getInlineAriaLabel(selectedOptions) : undefined}
162+
inlineTokens={urlParams.inlineTokens}
163+
expandToViewport={urlParams.expandToViewport}
164+
keepOpen={!urlParams.closeAfter}
165+
virtualScroll={urlParams.virtualScroll}
166+
/>
167+
</ScreenshotArea>
168+
</SpaceBetween>
169+
</Box>
170+
</article>
171+
);
172+
}

src/__tests__/snapshot-tests/__snapshots__/documenter.test.ts.snap

+16-2
Original file line numberDiff line numberDiff line change
@@ -11761,6 +11761,12 @@ To use it correctly, define an ID for the element you want to use as label and s
1176111761
"optional": true,
1176211762
"type": "boolean",
1176311763
},
11764+
{
11765+
"description": "Enables users to select and deselect all options with a special extra checkbox which is displayed at the start of the dropdown.",
11766+
"name": "enableSelectAll",
11767+
"optional": true,
11768+
"type": "boolean",
11769+
},
1176411770
{
1176511771
"description": "Provides a text alternative for the error icon in the error message.",
1176611772
"i18nTag": true,
@@ -11859,11 +11865,19 @@ Only use this if the selected options are displayed elsewhere on the page.",
1185911865
},
1186011866
{
1186111867
"description": "An object containing all the localized strings required by the component.
11862-
Note that the string for \`tokenLimitShowMore\` should not contain the number of hidden tokens
11863-
because it will be added by the component automatically.",
11868+
* \`selectAllText\` (string) - Specifies the text to be displayed next to the checkbox that selects or deselects all options.
11869+
* \`tokenLimitShowFewer\` (string) - Specifies the text to be displayed in the "Show fewer" button for the token group control.
11870+
* \`tokenLimitShowMore\` (string) - Specifies the text to be displayed in the "Show more" button for the token group control. This string should not contain the number of hidden tokens
11871+
because this will be added by the component automatically.",
11872+
"i18nTag": true,
1186411873
"inlineType": {
1186511874
"name": "MultiselectProps.I18nStrings",
1186611875
"properties": [
11876+
{
11877+
"name": "selectAllText",
11878+
"optional": true,
11879+
"type": "string",
11880+
},
1186711881
{
1186811882
"name": "tokenLimitShowFewer",
1186911883
"optional": true,

src/__tests__/snapshot-tests/__snapshots__/test-utils-selectors.test.tsx.snap

+1
Original file line numberDiff line numberDiff line change
@@ -383,6 +383,7 @@ exports[`test-utils selectors 1`] = `
383383
"awsui_root_1t44z",
384384
"awsui_root_qwoo0",
385385
"awsui_root_vrgzu",
386+
"awsui_select-all_15o6u",
386387
"awsui_selectable-item_15o6u",
387388
"awsui_selected_15o6u",
388389
"awsui_tag_1p2cx",

src/internal/components/option/interfaces.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ export interface OptionGroup extends BaseOption {
3434
}
3535

3636
export interface DropdownOption {
37-
type?: string;
37+
type?: 'child' | 'parent' | 'select-all' | 'use-entered';
3838
disabled?: boolean;
3939
disabledReason?: string;
4040
option: OptionDefinition | OptionGroup;

0 commit comments

Comments
 (0)