1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import React from 'react';
18 } from '@mui/material';
19 import withStyles from '@mui/styles/withStyles';
20 import { CustomStyleRulesCallback } from 'common/custom-theme';
21 import { PopperProps } from '@mui/material/Popper';
22 import { WithStyles } from '@mui/styles';
23 import { noop } from 'lodash';
25 export interface AutocompleteProps<Item, Suggestion> {
30 suggestions?: Suggestion[];
34 onChange: (event: React.ChangeEvent<HTMLInputElement>) => void;
35 onBlur?: (event: React.FocusEvent<HTMLInputElement>) => void;
36 onFocus?: (event: React.FocusEvent<HTMLInputElement>) => void;
37 onCreate?: () => void;
38 onDelete?: (item: Item, index: number) => void;
39 onSelect?: (suggestion: Suggestion) => void;
40 renderChipValue?: (item: Item) => string;
41 renderChipTooltip?: (item: Item) => string;
42 renderSuggestion?: (suggestion: Suggestion) => React.ReactNode;
45 export interface AutocompleteState {
46 suggestionsOpen: boolean;
47 selectedSuggestionIndex: number;
50 export class Autocomplete<Value, Suggestion> extends React.Component<AutocompleteProps<Value, Suggestion>, AutocompleteState> {
53 suggestionsOpen: false,
54 selectedSuggestionIndex: 0,
57 containerRef = React.createRef<HTMLDivElement>();
58 inputRef = React.createRef<HTMLInputElement>();
62 <FormControl variant="standard" fullWidth error={this.props.error}>
65 {this.renderHelperText()}
66 {this.renderSuggestions()}
72 const { label } = this.props;
73 return label && <InputLabel>{label}</InputLabel>;
78 disabled={this.props.disabled}
79 autoFocus={this.props.autofocus}
80 inputRef={this.inputRef}
81 value={this.props.value}
82 startAdornment={this.renderChips()}
83 onFocus={this.handleFocus}
84 onBlur={this.handleBlur}
85 onChange={this.props.onChange}
86 onKeyPress={this.handleKeyPress}
87 onKeyDown={this.handleNavigationKeyPress}
92 return <FormHelperText>{this.props.helperText}</FormHelperText>;
96 const { suggestions = [] } = this.props;
99 open={this.isSuggestionBoxOpen()}
100 anchorEl={this.inputRef.current}
101 key={suggestions.length}>
102 <Paper onMouseDown={this.preventBlur}>
103 <List dense style={{ width: this.getSuggestionsWidth() }}>
105 (suggestion, index) =>
109 onClick={this.handleSelect(suggestion)}
110 selected={index === this.state.selectedSuggestionIndex}>
111 {this.renderSuggestion(suggestion)}
120 isSuggestionBoxOpen() {
121 const { suggestions = [] } = this.props;
122 return this.state.suggestionsOpen && suggestions.length > 0;
125 handleFocus = (event: React.FocusEvent<HTMLInputElement>) => {
126 const { onFocus = noop } = this.props;
127 this.setState({ suggestionsOpen: true });
131 handleBlur = (event: React.FocusEvent<HTMLInputElement>) => {
133 const { onBlur = noop } = this.props;
134 this.setState({ suggestionsOpen: false });
139 handleKeyPress = (event: React.KeyboardEvent<HTMLInputElement>) => {
140 const { onCreate = noop, onSelect = noop, suggestions = [] } = this.props;
141 const { selectedSuggestionIndex } = this.state;
142 if (event.key === 'Enter') {
143 if (this.isSuggestionBoxOpen() && selectedSuggestionIndex < suggestions.length) {
144 // prevent form submissions when selecting a suggestion
145 event.preventDefault();
146 onSelect(suggestions[selectedSuggestionIndex]);
147 } else if (this.props.value.length > 0) {
153 handleNavigationKeyPress = ({ key }: React.KeyboardEvent<HTMLInputElement>) => {
154 if (key === 'ArrowUp') {
155 this.updateSelectedSuggestionIndex(-1);
156 } else if (key === 'ArrowDown') {
157 this.updateSelectedSuggestionIndex(1);
161 updateSelectedSuggestionIndex(value: -1 | 1) {
162 const { suggestions = [] } = this.props;
163 this.setState(({ selectedSuggestionIndex }) => ({
164 selectedSuggestionIndex: (selectedSuggestionIndex + value) % suggestions.length
169 const { items, onDelete } = this.props;
172 * If input startAdornment prop is not undefined, input's label will stay above the input.
173 * If there is not items, we want the label to go back to placeholder position.
174 * That why we return without a value instead of returning a result of a _map_ which is an empty array.
176 if (items.length === 0) {
182 const tooltip = this.props.renderChipTooltip ? this.props.renderChipTooltip(item) : '';
183 if (tooltip && tooltip.length) {
184 return <span key={index}>
185 <Tooltip title={tooltip}>
187 label={this.renderChipValue(item)}
189 onDelete={onDelete && !this.props.disabled ? (() => onDelete(item, index)) : undefined} />
192 return <span key={index}><Chip
193 label={this.renderChipValue(item)}
194 onDelete={onDelete && !this.props.disabled ? (() => onDelete(item, index)) : undefined} /></span>
200 renderChipValue(value: Value) {
201 const { renderChipValue } = this.props;
202 return renderChipValue ? renderChipValue(value) : JSON.stringify(value);
205 preventBlur = (event: React.MouseEvent<HTMLElement>) => {
206 event.preventDefault();
209 handleClickAway = (event: React.MouseEvent<HTMLElement>) => {
210 if (event.target !== this.inputRef.current) {
211 this.setState({ suggestionsOpen: false });
215 handleSelect(suggestion: Suggestion) {
217 const { onSelect = noop } = this.props;
218 const { current } = this.inputRef;
222 onSelect(suggestion);
226 renderSuggestion(suggestion: Suggestion) {
227 const { renderSuggestion } = this.props;
228 return renderSuggestion
229 ? renderSuggestion(suggestion)
230 : <ListItemText>{JSON.stringify(suggestion)}</ListItemText>;
233 getSuggestionsWidth() {
234 return this.containerRef.current ? this.containerRef.current.offsetWidth : 'auto';
238 type ChipClasses = 'root';
240 const chipStyles: CustomStyleRulesCallback<ChipClasses> = theme => ({
242 marginRight: theme.spacing(0.25),
243 height: theme.spacing(3),
247 const Chip = withStyles(chipStyles)(MuiChip);
249 type PopperClasses = 'root';
251 const popperStyles: CustomStyleRulesCallback<ChipClasses> = theme => ({
253 zIndex: theme.zIndex.modal,
257 const Popper = withStyles(popperStyles)(
258 ({ classes, ...props }: PopperProps & WithStyles<PopperClasses>) =>
259 <MuiPopper {...props} className={classes.root} />
262 type InputClasses = 'root';
264 const inputStyles: CustomStyleRulesCallback<InputClasses> = () => ({
275 const Input = withStyles(inputStyles)(MuiInput);
277 const Paper = withStyles({