Skip to content

Commit 23938f9

Browse files
dhruvduttwhilelucky
authored andcommitted
feat(Select/web): support "Select All" (#193)
1 parent 2db6ea9 commit 23938f9

File tree

1 file changed

+86
-26
lines changed

1 file changed

+86
-26
lines changed

src/Select/web/Select.js

+86-26
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import { getIn } from 'formik';
66
import pluralize from '../../utils/pluralize';
77
import isEqual from '../../utils/isEqual';
88
import Text from '../../Text/web';
9+
import View from '../../View/web';
910
import Space from '../../Space/web';
1011
import Checkbox from '../../Checkbox/web';
1112
import Trigger from './Trigger';
@@ -24,12 +25,14 @@ class Select extends React.Component {
2425
}
2526

2627
componentDidMount() {
27-
const { name, defaultSelected, multiple } = this.props;
28+
const { name, defaultSelected, multiple, options } = this.props;
2829
const { formik } = this.context;
2930

3031
if (formik && name) {
3132
if (defaultSelected == null || defaultSelected === '') {
3233
formik.setFieldValue(name, multiple ? [] : '');
34+
} else if (defaultSelected === 'selectAll') {
35+
formik.setFieldValue(name, options);
3336
} else {
3437
formik.setFieldValue(name, defaultSelected);
3538
}
@@ -44,10 +47,16 @@ class Select extends React.Component {
4447
const formikValues = getIn(formik.values, nextProps.name);
4548
if (formikValues == null || formikValues === '') {
4649
newSelectedOptions = [];
50+
} else if (Array.isArray(formikValues)) {
51+
newSelectedOptions = newSelectedOptions.concat(formikValues.map(this.remakeOption));
52+
if (newSelectedOptions.length === nextProps.options.length) {
53+
newSelectedOptions = [{
54+
label: 'Select all',
55+
value: 'selectAll',
56+
}].concat(newSelectedOptions);
57+
}
4758
} else {
48-
newSelectedOptions = Array.isArray(formikValues)
49-
? newSelectedOptions.concat(formikValues.map(this.remakeOption))
50-
: newSelectedOptions.concat(this.remakeOption(formikValues));
59+
newSelectedOptions = newSelectedOptions.concat(this.remakeOption(formikValues));
5160
}
5261
this.setState((prevState) => {
5362
if (!isEqual(prevState.selectedOptions, newSelectedOptions)) {
@@ -59,36 +68,55 @@ class Select extends React.Component {
5968
}
6069

6170
onChange = (selectedValues) => {
62-
const { name, onChange } = this.props;
71+
const { name, onChange, multiple } = this.props;
6372
const { formik } = this.context;
73+
if (multiple) {
74+
// eslint-disable-next-line
75+
selectedValues = selectedValues.filter((value) => value !== 'selectAll');
76+
}
6477
if (formik && name) {
6578
formik.setFieldValue(name, selectedValues);
6679
formik.setFieldTouched(name, true);
6780
}
6881
onChange(selectedValues);
6982
}
7083

71-
onSelect = (selectedOption, stateAndHelpers) => {
84+
onSelect = (selectedOption) => {
7285
const { selectedOptions } = this.state;
73-
const { multiple } = this.props;
86+
const { multiple, options } = this.props;
7487

7588
if (multiple) {
7689
let newSelectedOptions = [];
7790
if (this.isOptionSelected(selectedOptions, selectedOption)) {
78-
// multiple: remove option
79-
newSelectedOptions = selectedOptions
80-
.filter((option) => !isEqual(option.value, selectedOption.value));
91+
if (selectedOption.value === 'selectAll') {
92+
newSelectedOptions = [];
93+
} else {
94+
// multiple: remove option
95+
newSelectedOptions = selectedOptions
96+
.filter((option) => !isEqual(option.value, selectedOption.value));
97+
98+
const newSelectedOptionsCount = newSelectedOptions.filter((option) => option.value !== 'selectAll').length;
99+
if (newSelectedOptionsCount !== options.length) {
100+
newSelectedOptions = newSelectedOptions.filter((option) => option.value !== 'selectAll');
101+
}
102+
}
103+
} else if (selectedOption.value === 'selectAll') {
104+
newSelectedOptions = this.makeOptions();
81105
} else {
82-
// multiple: add option
83106
newSelectedOptions = [
84107
...selectedOptions,
85108
selectedOption,
86109
];
110+
if (selectedOptions.length === options.length) {
111+
newSelectedOptions = [{
112+
label: 'Select all',
113+
value: 'selectAll',
114+
}].concat(newSelectedOptions);
115+
}
87116
}
88117
this.setState({
89118
selectedOptions: newSelectedOptions,
90119
}, () => {
91-
stateAndHelpers.openMenu();
92120
this.onChange(this.getOptionsValue(newSelectedOptions));
93121
});
94122
} else {
@@ -106,6 +134,8 @@ class Select extends React.Component {
106134

107135
if (defaultSelected == null || defaultSelected === '') {
108136
defaultSelectedOptions = [];
137+
} else if (defaultSelected === 'selectAll') {
138+
defaultSelectedOptions = this.makeOptions();
109139
} else if (defaultSelected != null) {
110140
defaultSelectedOptions = Array.isArray(defaultSelected)
111141
? defaultSelectedOptions.concat(defaultSelected.map(this.remakeOption))
@@ -126,9 +156,11 @@ class Select extends React.Component {
126156
return placeholder || '';
127157
}
128158
if (multiple) {
159+
// eslint-disable-next-line
160+
selectedOptions = selectedOptions.filter((option) => option.value !== 'selectAll');
129161
return `${selectedOptions.length} ${pluralize(selectedOptions.length, label)}`;
130162
}
131-
return selectedOptions[0].label;
163+
return selectedOptions[0] ? selectedOptions[0].label : (placeholder || '');
132164
}
133165

134166
getOptionsValue = (options) =>
@@ -141,20 +173,43 @@ class Select extends React.Component {
141173
.map(({ value }) => value)
142174
.includes(option.value);
143175

144-
makeOption = (option) => ({
145-
label: (option && option.label) || option,
146-
value: (option && option.value) || option,
147-
});
176+
makeOptions = () => {
177+
const { multiple } = this.props;
178+
let { options } = this.props;
179+
if (multiple) {
180+
options = [{
181+
label: 'Select all',
182+
value: 'selectAll',
183+
}].concat(options);
184+
}
185+
return options.map((option) => ({
186+
label: (option && option.label) || option,
187+
value: (option && option.value) || option,
188+
}));
189+
}
148190

149191
remakeOption = (value) => {
150-
const { options } = this.props;
151-
const fullOptions = options.map(this.makeOption);
192+
const fullOptions = this.makeOptions();
152193
return fullOptions.find((option) => isEqual(option.value, value));
153194
};
154195

155196
itemToString = (option) =>
156197
String(option.value)
157198

199+
stateReducer = (state, changes) => {
200+
switch (changes.type) {
201+
case Downshift.stateChangeTypes.keyDownEnter:
202+
case Downshift.stateChangeTypes.clickItem:
203+
return {
204+
...changes,
205+
highlightedIndex: state.highlightedIndex,
206+
isOpen: this.props.multiple,
207+
};
208+
default:
209+
return changes;
210+
}
211+
}
212+
158213
render() {
159214
const {
160215
selectedOptions,
@@ -185,10 +240,11 @@ class Select extends React.Component {
185240
error = error && error.replace(name, label || name);
186241
}
187242

188-
options = options.map(this.makeOption);
243+
options = this.makeOptions();
189244

190245
return (
191246
<Downshift
247+
stateReducer={this.stateReducer}
192248
selectedItem={selectedOptions}
193249
onSelect={this.onSelect}
194250
itemToString={this.itemToString}
@@ -258,12 +314,16 @@ class Select extends React.Component {
258314
>
259315
{
260316
multiple ? (
261-
<Space padding={[0]}>
262-
<Checkbox
263-
label={<Text truncate>{options[index].label}</Text>}
264-
checked={this.isOptionSelected(dsSelectedOptions, options[index])}
265-
/>
266-
</Space>
317+
<View style={{ pointerEvents: 'none' }}>
318+
<Space padding={[0]}>
319+
<Checkbox
320+
label={<Text truncate>{options[index].label}</Text>}
321+
checked={
322+
this.isOptionSelected(dsSelectedOptions, options[index])
323+
}
324+
/>
325+
</Space>
326+
</View>
267327
) : (
268328
<Text truncate>
269329
{`${options[index].label}`}

0 commit comments

Comments
 (0)