// Copyright 2015, 2016 Ethcore (UK) Ltd. // This file is part of Parity. // Parity is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // Parity is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // You should have received a copy of the GNU General Public License // along with Parity. If not, see . import React, { Component, PropTypes } from 'react'; import keycode from 'keycode'; import { MenuItem, AutoComplete as MUIAutoComplete } from 'material-ui'; import { PopoverAnimationVertical } from 'material-ui/Popover'; export default class AutoComplete extends Component { static propTypes = { onChange: PropTypes.func.isRequired, onUpdateInput: PropTypes.func, disabled: PropTypes.bool, label: PropTypes.string, hint: PropTypes.string, error: PropTypes.string, value: PropTypes.string, className: PropTypes.string, filter: PropTypes.func, renderItem: PropTypes.func, entry: PropTypes.object, entries: PropTypes.oneOfType([ PropTypes.array, PropTypes.object ]) } state = { lastChangedValue: undefined, entry: null, open: false, fakeBlur: false } render () { const { disabled, error, hint, label, value, className, filter, onUpdateInput } = this.props; const { open } = this.state; return ( ); } getDataSource () { const { renderItem, entries } = this.props; const entriesArray = (entries instanceof Array) ? entries : Object.values(entries); if (renderItem && typeof renderItem === 'function') { return entriesArray.map(entry => renderItem(entry)); } return entriesArray.map(entry => ({ text: entry, value: ( ) })); } onKeyDown = (event) => { const { muiAutocomplete } = this.refs; switch (keycode(event)) { case 'down': const { menu } = muiAutocomplete.refs; menu.handleKeyDown(event); this.setState({ fakeBlur: true }); break; case 'enter': case 'tab': event.preventDefault(); event.stopPropagation(); event.which = 'useless'; const e = new CustomEvent('down'); e.which = 40; muiAutocomplete.handleKeyDown(e); break; } } onChange = (item, idx) => { if (idx === -1) { return; } const { entries } = this.props; const entriesArray = (entries instanceof Array) ? entries : Object.values(entries); const entry = entriesArray[idx]; this.handleOnChange(entry); this.setState({ entry, open: false }); } onBlur = (event) => { const { onUpdateInput } = this.props; // TODO: Handle blur gracefully where we use onUpdateInput (currently replaces // input where text is allowed with the last selected value from the dropdown) if (!onUpdateInput) { window.setTimeout(() => { const { entry, fakeBlur } = this.state; if (fakeBlur) { this.setState({ fakeBlur: false }); return; } this.handleOnChange(entry); }, 200); } } onFocus = () => { const { entry } = this.props; this.setState({ entry, open: true }, () => { this.handleOnChange(null, true); }); } handleOnChange = (value, empty) => { const { lastChangedValue } = this.state; if (value !== lastChangedValue) { this.setState({ lastChangedValue: value }); this.props.onChange(value, empty); } } }