@@ -6,6 +6,7 @@ import { getIn } from 'formik';
6
6
import pluralize from '../../utils/pluralize' ;
7
7
import isEqual from '../../utils/isEqual' ;
8
8
import Text from '../../Text/web' ;
9
+ import View from '../../View/web' ;
9
10
import Space from '../../Space/web' ;
10
11
import Checkbox from '../../Checkbox/web' ;
11
12
import Trigger from './Trigger' ;
@@ -24,12 +25,14 @@ class Select extends React.Component {
24
25
}
25
26
26
27
componentDidMount ( ) {
27
- const { name, defaultSelected, multiple } = this . props ;
28
+ const { name, defaultSelected, multiple, options } = this . props ;
28
29
const { formik } = this . context ;
29
30
30
31
if ( formik && name ) {
31
32
if ( defaultSelected == null || defaultSelected === '' ) {
32
33
formik . setFieldValue ( name , multiple ? [ ] : '' ) ;
34
+ } else if ( defaultSelected === 'selectAll' ) {
35
+ formik . setFieldValue ( name , options ) ;
33
36
} else {
34
37
formik . setFieldValue ( name , defaultSelected ) ;
35
38
}
@@ -44,10 +47,16 @@ class Select extends React.Component {
44
47
const formikValues = getIn ( formik . values , nextProps . name ) ;
45
48
if ( formikValues == null || formikValues === '' ) {
46
49
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
+ }
47
58
} 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 ) ) ;
51
60
}
52
61
this . setState ( ( prevState ) => {
53
62
if ( ! isEqual ( prevState . selectedOptions , newSelectedOptions ) ) {
@@ -59,36 +68,55 @@ class Select extends React.Component {
59
68
}
60
69
61
70
onChange = ( selectedValues ) => {
62
- const { name, onChange } = this . props ;
71
+ const { name, onChange, multiple } = this . props ;
63
72
const { formik } = this . context ;
73
+ if ( multiple ) {
74
+ // eslint-disable-next-line
75
+ selectedValues = selectedValues . filter ( ( value ) => value !== 'selectAll' ) ;
76
+ }
64
77
if ( formik && name ) {
65
78
formik . setFieldValue ( name , selectedValues ) ;
66
79
formik . setFieldTouched ( name , true ) ;
67
80
}
68
81
onChange ( selectedValues ) ;
69
82
}
70
83
71
- onSelect = ( selectedOption , stateAndHelpers ) => {
84
+ onSelect = ( selectedOption ) => {
72
85
const { selectedOptions } = this . state ;
73
- const { multiple } = this . props ;
86
+ const { multiple, options } = this . props ;
74
87
75
88
if ( multiple ) {
76
89
let newSelectedOptions = [ ] ;
77
90
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 ( ) ;
81
105
} else {
82
- // multiple: add option
83
106
newSelectedOptions = [
84
107
...selectedOptions ,
85
108
selectedOption ,
86
109
] ;
110
+ if ( selectedOptions . length === options . length ) {
111
+ newSelectedOptions = [ {
112
+ label : 'Select all' ,
113
+ value : 'selectAll' ,
114
+ } ] . concat ( newSelectedOptions ) ;
115
+ }
87
116
}
88
117
this . setState ( {
89
118
selectedOptions : newSelectedOptions ,
90
119
} , ( ) => {
91
- stateAndHelpers . openMenu ( ) ;
92
120
this . onChange ( this . getOptionsValue ( newSelectedOptions ) ) ;
93
121
} ) ;
94
122
} else {
@@ -106,6 +134,8 @@ class Select extends React.Component {
106
134
107
135
if ( defaultSelected == null || defaultSelected === '' ) {
108
136
defaultSelectedOptions = [ ] ;
137
+ } else if ( defaultSelected === 'selectAll' ) {
138
+ defaultSelectedOptions = this . makeOptions ( ) ;
109
139
} else if ( defaultSelected != null ) {
110
140
defaultSelectedOptions = Array . isArray ( defaultSelected )
111
141
? defaultSelectedOptions . concat ( defaultSelected . map ( this . remakeOption ) )
@@ -126,9 +156,11 @@ class Select extends React.Component {
126
156
return placeholder || '' ;
127
157
}
128
158
if ( multiple ) {
159
+ // eslint-disable-next-line
160
+ selectedOptions = selectedOptions . filter ( ( option ) => option . value !== 'selectAll' ) ;
129
161
return `${ selectedOptions . length } ${ pluralize ( selectedOptions . length , label ) } ` ;
130
162
}
131
- return selectedOptions [ 0 ] . label ;
163
+ return selectedOptions [ 0 ] ? selectedOptions [ 0 ] . label : ( placeholder || '' ) ;
132
164
}
133
165
134
166
getOptionsValue = ( options ) =>
@@ -141,20 +173,43 @@ class Select extends React.Component {
141
173
. map ( ( { value } ) => value )
142
174
. includes ( option . value ) ;
143
175
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
+ }
148
190
149
191
remakeOption = ( value ) => {
150
- const { options } = this . props ;
151
- const fullOptions = options . map ( this . makeOption ) ;
192
+ const fullOptions = this . makeOptions ( ) ;
152
193
return fullOptions . find ( ( option ) => isEqual ( option . value , value ) ) ;
153
194
} ;
154
195
155
196
itemToString = ( option ) =>
156
197
String ( option . value )
157
198
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
+
158
213
render ( ) {
159
214
const {
160
215
selectedOptions,
@@ -185,10 +240,11 @@ class Select extends React.Component {
185
240
error = error && error . replace ( name , label || name ) ;
186
241
}
187
242
188
- options = options . map ( this . makeOption ) ;
243
+ options = this . makeOptions ( ) ;
189
244
190
245
return (
191
246
< Downshift
247
+ stateReducer = { this . stateReducer }
192
248
selectedItem = { selectedOptions }
193
249
onSelect = { this . onSelect }
194
250
itemToString = { this . itemToString }
@@ -258,12 +314,16 @@ class Select extends React.Component {
258
314
>
259
315
{
260
316
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 >
267
327
) : (
268
328
< Text truncate >
269
329
{ `${ options [ index ] . label } ` }
0 commit comments