1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import React from 'react';
11 FormControl, InputLabel, StyleRulesCallback, withStyles, RootRef, ListItemText, ListItem, List, FormHelperText, Tooltip, Typography
12 } from '@material-ui/core';
13 import { PopperProps } from '@material-ui/core/Popper';
14 import { WithStyles } from '@material-ui/core/styles';
15 import { noop } from 'lodash';
16 import { isGroup } from 'common/isGroup';
17 import { sortByKey } from 'common/objects';
19 export interface AutocompleteProps<Item, Suggestion> {
24 suggestions?: Suggestion[];
28 onChange: (event: React.ChangeEvent<HTMLInputElement>) => void;
29 onBlur?: (event: React.FocusEvent<HTMLInputElement>) => void;
30 onFocus?: (event: React.FocusEvent<HTMLInputElement>) => void;
31 onCreate?: () => void;
32 onDelete?: (item: Item, index: number) => void;
33 onSelect?: (suggestion: Suggestion) => void;
34 renderChipValue?: (item: Item) => string;
35 renderChipTooltip?: (item: Item) => string;
36 renderSuggestion?: (suggestion: Suggestion) => React.ReactNode;
37 category?: AutocompleteCat;
40 type AutocompleteClasses = 'sharingList' | 'emptyList' | 'listSubHeader' | 'numFound';
42 const autocompleteStyles: StyleRulesCallback<AutocompleteClasses> = theme => ({
46 scrollbarColor: 'rgba(0, 0, 0, 0.3) rgba(0, 0, 0, 0)',
47 '&::-webkit-scrollbar': {
50 '&::-webkit-scrollbar-thumb': {
51 backgroundColor: 'rgba(0, 0, 0, 0.3)',
54 '&::-webkit-scrollbar-track': {
55 backgroundColor: 'rgba(0, 0, 0, 0)',
65 alignItems: 'flex-end',
66 justifyContent: 'space-between',
74 export enum AutocompleteCat {
78 export interface AutocompleteState {
79 suggestionsOpen: boolean;
80 selectedSuggestionIndex: number;
83 export const Autocomplete = withStyles(autocompleteStyles)(
84 class Autocomplete<Value, Suggestion> extends React.Component<AutocompleteProps<Value, Suggestion> & WithStyles<AutocompleteClasses>, AutocompleteState> {
87 suggestionsOpen: false,
88 selectedSuggestionIndex: 0,
91 containerRef = React.createRef<HTMLDivElement>();
92 inputRef = React.createRef<HTMLInputElement>();
96 <RootRef rootRef={this.containerRef}>
97 <FormControl fullWidth error={this.props.error}>
100 {this.renderHelperText()}
101 {this.props.category === AutocompleteCat.SHARING ? this.renderSharingSuggestions() : this.renderSuggestions()}
108 const { label } = this.props;
109 return label && <InputLabel>{label}</InputLabel>;
114 disabled={this.props.disabled}
115 autoFocus={this.props.autofocus}
116 inputRef={this.inputRef}
117 value={this.props.value}
118 startAdornment={this.renderChips()}
119 onFocus={this.handleFocus}
120 onBlur={this.handleBlur}
121 onChange={this.props.onChange}
122 onKeyPress={this.handleKeyPress}
123 onKeyDown={this.handleNavigationKeyPress}
128 return <FormHelperText>{this.props.helperText}</FormHelperText>;
131 renderSuggestions() {
132 const { suggestions = [] } = this.props;
135 open={this.isSuggestionBoxOpen()}
136 anchorEl={this.inputRef.current}
137 key={suggestions.length}>
138 <Paper onMouseDown={this.preventBlur}>
139 <List dense style={{ width: this.getSuggestionsWidth() }}>
141 (suggestion, index) =>
145 onClick={this.handleSelect(suggestion)}
146 selected={index === this.state.selectedSuggestionIndex}>
147 {this.renderSuggestion(suggestion)}
156 renderSharingSuggestions() {
157 const { suggestions = [], classes } = this.props;
158 const users = sortByKey<Suggestion>(suggestions.filter(item => !isGroup(item)), 'fullName');
159 const groups = sortByKey<Suggestion>(suggestions.filter(item => isGroup(item)), 'name');
163 open={this.isSuggestionBoxOpen()}
164 anchorEl={this.inputRef.current}
165 key={suggestions.length}>
166 <Paper onMouseDown={this.preventBlur}>
167 <div className={classes.listSubHeader}>
168 Groups {<span className={classes.numFound}>{groups.length} {groups.length === 1 ? 'match' : 'matches'} found</span>}
170 <List dense className={classes.sharingList} style={{width: this.getSuggestionsWidth()}}>
172 (suggestion, index) =>
175 id={`groups-${index}`}
176 key={`groups-${index}`}
177 onClick={this.handleSelect(suggestion)}>
178 {this.renderSharingSuggestion(suggestion)}
182 <div className={classes.listSubHeader}>
183 Users {<span className={classes.numFound}>{users.length} {users.length === 1 ? 'match' : 'matches'} found</span>}
185 <List dense className={classes.sharingList} style={{width: this.getSuggestionsWidth()}}>
187 (suggestion, index) =>
190 id={`users-${index}`}
191 key={`users-${index}`}
192 onClick={this.handleSelect(suggestion)}>
193 {this.renderSharingSuggestion(suggestion)}
202 isSuggestionBoxOpen() {
203 const { suggestions = [] } = this.props;
204 return this.state.suggestionsOpen && suggestions.length > 0;
207 handleFocus = (event: React.FocusEvent<HTMLInputElement>) => {
208 const { onFocus = noop } = this.props;
209 this.setState({ suggestionsOpen: true });
213 handleBlur = (event: React.FocusEvent<HTMLInputElement>) => {
215 const { onBlur = noop } = this.props;
216 this.setState({ suggestionsOpen: false });
221 handleKeyPress = (event: React.KeyboardEvent<HTMLInputElement>) => {
222 const { onCreate = noop, onSelect = noop, suggestions = [] } = this.props;
223 const { selectedSuggestionIndex } = this.state;
224 if (event.key === 'Enter') {
225 if (this.isSuggestionBoxOpen() && selectedSuggestionIndex < suggestions.length) {
226 // prevent form submissions when selecting a suggestion
227 event.preventDefault();
228 onSelect(suggestions[selectedSuggestionIndex]);
229 } else if (this.props.value.length > 0) {
235 handleNavigationKeyPress = ({ key }: React.KeyboardEvent<HTMLInputElement>) => {
236 if (key === 'ArrowUp') {
237 this.updateSelectedSuggestionIndex(-1);
238 } else if (key === 'ArrowDown') {
239 this.updateSelectedSuggestionIndex(1);
243 updateSelectedSuggestionIndex(value: -1 | 1) {
244 const { suggestions = [] } = this.props;
245 this.setState(({ selectedSuggestionIndex }) => ({
246 selectedSuggestionIndex: (selectedSuggestionIndex + value) % suggestions.length
251 const { items, onDelete } = this.props;
254 * If input startAdornment prop is not undefined, input's label will stay above the input.
255 * If there is not items, we want the label to go back to placeholder position.
256 * That why we return without a value instead of returning a result of a _map_ which is an empty array.
258 if (items.length === 0) {
264 const tooltip = this.props.renderChipTooltip ? this.props.renderChipTooltip(item) : '';
265 if (tooltip && tooltip.length) {
266 return <span key={index}>
267 <Tooltip title={tooltip}>
269 label={this.renderChipValue(item)}
271 onDelete={onDelete && !this.props.disabled ? (() => onDelete(item, index)) : undefined} />
274 return <span key={index}><Chip
275 label={this.renderChipValue(item)}
276 onDelete={onDelete && !this.props.disabled ? (() => onDelete(item, index)) : undefined} /></span>
282 renderChipValue(value: Value) {
283 const { renderChipValue } = this.props;
284 return renderChipValue ? renderChipValue(value) : JSON.stringify(value);
287 preventBlur = (event: React.MouseEvent<HTMLElement>) => {
288 event.preventDefault();
291 handleClickAway = (event: React.MouseEvent<HTMLElement>) => {
292 if (event.target !== this.inputRef.current) {
293 this.setState({ suggestionsOpen: false });
297 handleSelect(suggestion: Suggestion) {
299 const { onSelect = noop } = this.props;
300 const { current } = this.inputRef;
304 onSelect(suggestion);
308 renderSuggestion(suggestion: Suggestion) {
309 const { renderSuggestion } = this.props;
310 return renderSuggestion
311 ? renderSuggestion(suggestion)
312 : <ListItemText>{JSON.stringify(suggestion)}</ListItemText>;
315 renderSharingSuggestion(suggestion: Suggestion) {
316 if (isGroup(suggestion)) {
317 return <ListItemText>
318 <Typography noWrap data-cy="sharing-suggestion">
319 {(suggestion as any).name}
322 return <ListItemText>
323 <Typography data-cy="sharing-suggestion">
324 {`${(suggestion as any).fullName} (${(suggestion as any).username})`}
329 getSuggestionsWidth() {
330 return this.containerRef.current ? this.containerRef.current.offsetWidth : 'auto';
334 type ChipClasses = 'root';
336 const chipStyles: StyleRulesCallback<ChipClasses> = theme => ({
338 marginRight: theme.spacing.unit / 4,
339 height: theme.spacing.unit * 3,
343 const Chip = withStyles(chipStyles)(MuiChip);
345 type PopperClasses = 'root';
347 const popperStyles: StyleRulesCallback<PopperClasses> = theme => ({
349 zIndex: theme.zIndex.modal,
353 const Popper = withStyles(popperStyles)(
354 ({ classes, ...props }: PopperProps & WithStyles<PopperClasses>) =>
355 <MuiPopper {...props} className={classes.root} />
358 type InputClasses = 'root';
360 const inputStyles: StyleRulesCallback<InputClasses> = () => ({
371 const Input = withStyles(inputStyles)(MuiInput);
373 const Paper = withStyles({