mirror of
https://github.com/danog/react-datalist-input.git
synced 2024-12-11 17:09:39 +01:00
198 lines
7.1 KiB
JavaScript
198 lines
7.1 KiB
JavaScript
import React from 'react';
|
|
import PropTypes from 'prop-types';
|
|
|
|
import './DataListInput.css';
|
|
|
|
class DataListInput extends React.Component {
|
|
|
|
constructor(props) {
|
|
super(props);
|
|
|
|
this.state = {
|
|
/* last valid item that was selected from the drop down menu */
|
|
lastValidItem: undefined,
|
|
/* current input text */
|
|
currentInput: "",
|
|
/* current set of matching items */
|
|
matchingItems: [],
|
|
/* visibility property of the drop down menu */
|
|
visible: false,
|
|
/* index of the currently focused item in the drop down menu */
|
|
focusIndex: 0,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* gets called when someone starts to write in the input field
|
|
* @param event
|
|
*/
|
|
onHandleInput = (event) => {
|
|
const currentInput = event.target.value;
|
|
const matchingItems = this.props.items.filter((item) => {
|
|
if (typeof(this.props.match) === typeof(Function))
|
|
return this.props.match(currentInput, item);
|
|
return this.match(currentInput, item);
|
|
});
|
|
this.setState({
|
|
currentInput: currentInput,
|
|
matchingItems: matchingItems,
|
|
focusIndex: 0,
|
|
visible: true,
|
|
});
|
|
};
|
|
|
|
/**
|
|
* default function for matching the current input value (needle) and the values of the items array
|
|
* @param currentInput
|
|
* @param item
|
|
* @returns {boolean}
|
|
*/
|
|
match = (currentInput, item) => {
|
|
return item.label.substr(0, currentInput.length).toUpperCase() === currentInput.toUpperCase();
|
|
};
|
|
|
|
/**
|
|
* function for getting the index of the currentValue inside a value of the values array
|
|
* @param currentInput
|
|
* @param item
|
|
* @returns {number}
|
|
*/
|
|
indexOfMatch = (currentInput, item) => {
|
|
return item.label.toUpperCase().indexOf(currentInput.toUpperCase());
|
|
};
|
|
|
|
/**
|
|
* handle key events
|
|
* @param event
|
|
*/
|
|
onHandleKeydown = (event) => {
|
|
// only do something if drop-down div is visible
|
|
if (!this.state.visible) return;
|
|
let currentFocusIndex = this.state.focusIndex;
|
|
if (event.keyCode === 40 || event.keyCode === 9) {
|
|
// If the arrow DOWN key or tab is pressed increase the currentFocus variable:
|
|
currentFocusIndex += 1;
|
|
if (currentFocusIndex >= this.state.matchingItems.length) currentFocusIndex = 0;
|
|
this.setState({
|
|
focusIndex: currentFocusIndex,
|
|
});
|
|
// prevent tab to jump to the next input field if drop down is still open
|
|
event.preventDefault();
|
|
} else if (event.keyCode === 38) {
|
|
// If the arrow UP key is pressed, decrease the currentFocus variable:
|
|
currentFocusIndex -= 1;
|
|
if (currentFocusIndex <= -1) currentFocusIndex = this.state.matchingItems.length - 1;
|
|
this.setState({
|
|
focusIndex: currentFocusIndex,
|
|
});
|
|
} else if (event.keyCode === 13) {
|
|
// Enter pressed, similar to onClickItem
|
|
if (this.state.focusIndex > -1) {
|
|
// Simulate a click on the "active" item:
|
|
const selectedItem = this.state.matchingItems[currentFocusIndex];
|
|
this.onSelect(selectedItem);
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* onClickItem gets called when onClick happens on one of the list elements
|
|
* @param event
|
|
*/
|
|
onClickItem = (event) => {
|
|
// update the input value and close the dropdown again
|
|
const selectedKey = event.currentTarget.children[1].value;
|
|
const selectedItem = this.state.matchingItems.find(item => item.key === selectedKey);
|
|
this.onSelect(selectedItem);
|
|
};
|
|
|
|
/**
|
|
* onSelect is called onClickItem and onEnter upon an option of the drop down menu
|
|
* does nothing if the key has not changed since the last onSelect event
|
|
* @param selectedItem
|
|
*/
|
|
onSelect = (selectedItem) => {
|
|
if (this.state.lastValidItem !== undefined && selectedItem.key === this.state.lastValidItem.key){
|
|
// do not trigger the callback function
|
|
// but still change state to fit new selection
|
|
this.setState({
|
|
currentInput: selectedItem.label,
|
|
visible: false,
|
|
focusIndex: -1,
|
|
});
|
|
return;
|
|
}
|
|
// change state to fit new selection
|
|
this.setState({
|
|
currentInput: selectedItem.label,
|
|
lastValidItem: selectedItem,
|
|
visible: false,
|
|
focusIndex: -1,
|
|
});
|
|
// callback function onSelect
|
|
this.props.onSelect(selectedItem);
|
|
};
|
|
|
|
renderItems = ( items, focusIndex, activeItemClassName, itemClassName) => (
|
|
<div className="datalist-items">
|
|
{items.map((item, i) => {
|
|
const isActive = focusIndex === i;
|
|
const itemActiveClasses = isActive ? `datalist-active-item ${activeItemClassName}` : ''
|
|
const itemClasses = `${itemClassName} ${itemActiveClasses};`
|
|
return (
|
|
<div onClick={this.onClickItem}
|
|
className={itemClasses}
|
|
key={item.key}>
|
|
{item.label.substr(0, this.indexOfMatch(currentInput, item))}
|
|
<strong>{item.label.substr(this.indexOfMatch(currentInput, item), currentInput.length)}</strong>
|
|
{item.label.substr(this.indexOfMatch(currentInput, item) + currentInput.length)}
|
|
<input type='hidden' value={item.key}/>
|
|
</div>
|
|
)
|
|
})}
|
|
</div>
|
|
);
|
|
|
|
renderInputField = ( placeholder, currentInput, inputClassName ) => (
|
|
<input onKeyDown={this.onHandleKeydown} onInput={this.onHandleInput} type="text"
|
|
className={ `autocomplete-input ${inputClassName}` }
|
|
placeholder={placeholder} value={currentInput}/>
|
|
)
|
|
|
|
render() {
|
|
const { currentInput, matchingItems, focusIndex, visible } = this.state;
|
|
const { placeholder, inputClassName, activeItemClassName, itemClassName, requiredInputLength } = this.props;
|
|
const reachedRequiredLength = currentInput.length >= requiredInputLength;
|
|
return (
|
|
<div className="datalist-input">
|
|
{ this.renderInputField( placeholder, currentInput, inputClassName ) }
|
|
{ reachedRequiredLength && visible &&
|
|
this.renderItems( matchingItems, focusIndex, activeItemClassName, itemClassName )
|
|
}
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
|
|
DataListInput.propTypes = {
|
|
items: PropTypes.array.isRequired,
|
|
placeholder: PropTypes.string,
|
|
onSelect: PropTypes.func.isRequired,
|
|
match: PropTypes.func,
|
|
inputClassName: PropTypes.string,
|
|
itemClassName: PropTypes.string,
|
|
activeItemClassName: PropTypes.string,
|
|
requiredInputLength: PropTypes.number,
|
|
};
|
|
|
|
DataListInput.defaultProps = {
|
|
placeholder: '',
|
|
match: undefined,
|
|
inputClassName: '',
|
|
itemClassName: '',
|
|
activeItemClassName: '',
|
|
requiredInputLength: 1,
|
|
};
|
|
|
|
export default DataListInput;
|