diff --git a/packages/rekit-studio/.eslintrc b/packages/rekit-studio/.eslintrc index 6aa0b18e..41afa62d 100644 --- a/packages/rekit-studio/.eslintrc +++ b/packages/rekit-studio/.eslintrc @@ -35,6 +35,8 @@ "new-cap": 0, "class-methods-use-this": 0, "react/jsx-filename-extension": 0, + "jsx-a11y/no-noninteractive-element-interactions": 0, + "no-return-assign": 0, "react/prefer-stateless-function": 0, "react/forbid-prop-types": 0, "react/no-danger": 0, diff --git a/packages/rekit-studio/package.json b/packages/rekit-studio/package.json index 23544e53..801c63d4 100644 --- a/packages/rekit-studio/package.json +++ b/packages/rekit-studio/package.json @@ -133,8 +133,7 @@ "express": "^4.15.2", "express-history-api-fallback": "^2.0.0", "file-loader": "^1.1.6", - "fuse.js": "^3.2.0", - "fuzzy": "^0.1.3", + "fuzzysort": "^1.1.1", "highlight.js": "^9.11.0", "immutability-helper": "^2.6.4", "isomorphic-fetch": "^2.2.1", diff --git a/packages/rekit-studio/src/features/home/QuickOpen.js b/packages/rekit-studio/src/features/home/QuickOpen.js index 1d08e792..e9e80631 100644 --- a/packages/rekit-studio/src/features/home/QuickOpen.js +++ b/packages/rekit-studio/src/features/home/QuickOpen.js @@ -3,9 +3,9 @@ import PropTypes from 'prop-types'; import { bindActionCreators } from 'redux'; import { connect } from 'react-redux'; import { Icon } from 'antd'; -// import Fuse from 'fuse.js'; -import fuzzy from 'fuzzy'; -import { getElementData } from './helpers'; +import fuzzysort from 'fuzzysort'; +import scrollIntoView from 'dom-scroll-into-view'; +import history from '../../common/history'; import * as actions from './redux/actions'; export class QuickOpen extends Component { @@ -15,6 +15,7 @@ export class QuickOpen extends Component { }; state = { + selectedIndex: 0, search: '', results: [], visible: false, @@ -27,8 +28,27 @@ export class QuickOpen extends Component { document.body.removeEventListener('keydown', this.handleKeydown); } + highlightMatch(item) { + const indexes = item.indexes.reduce((prev, i) => { + prev[i] = true; + return prev; + }, {}); + + const posLength = item.obj.position.length + 1; + const highlightedName = item.obj.name + .split('') + .map((c, i) => (indexes[posLength + i] ? `${c}` : c)) + .join(''); + + const highlightedPos = item.obj.position + .split('') + .map((c, i) => (indexes[i] ? `${c}` : c)) + .join(''); + return `${highlightedName}${highlightedPos}`; + } + handleKeydown = evt => { - if (evt.key === 'p' && evt.metaKey) { + if (evt.key === 'p' && (evt.metaKey || evt.ctrlKey)) { evt.preventDefault(); evt.stopPropagation(); const newState = { @@ -37,6 +57,7 @@ export class QuickOpen extends Component { if (newState.visible) { newState.search = ''; newState.results = []; + newState.selectedIndex = 0; } this.setState(newState, () => { if (this.inputNode) this.inputNode.focus(); @@ -54,7 +75,6 @@ export class QuickOpen extends Component { }; handleInputChange = evt => { - console.log('searching: ', evt.target.value); this.setState({ search: evt.target.value }); if (evt.target.value.indexOf('|') >= 0) { this.setState({ results: [] }); @@ -64,44 +84,60 @@ export class QuickOpen extends Component { const { elementById } = this.props.home; const list = Object.keys(elementById) .map(k => { - const item = { ...elementById[k] }; - if (/component|action/.test(item.type)) { + const item = { + ...elementById[k], + }; + if (/component|action|misc/.test(item.type)) { item.position = item.file .split('/') .slice(0, -1) .join('/'); + item.toSearch = `${item.position}/${item.name}`; } return item; }) - .filter(e => /component|action/.test(e.type)) - .map(e => `${e.position}/|${e.name}`); - - const options = { pre: '', post: '' }; - const results = fuzzy.filter(evt.target.value, list, options).map(s => { - const arr = s.string.split('|'); - const name = arr.pop(); - return `${name}${arr.join('')}`; - }); - this.setState({ results }); + .filter(e => /component|action|misc/.test(e.type)); + + const results = fuzzysort.go(evt.target.value, list, { key: 'toSearch' }); + this.setState({ results, selectedIndex: 0 }); }; + handleInputKeyDown = evt => { + const scrollToSelected = () => { + scrollIntoView(this.resultsNode.querySelectorAll('li')[this.state.selectedIndex], this.resultsNode, { + onlyScrollIfNeeded: true, + }); + }; + switch (evt.key) { + case 'Enter': + this.handleItemClick(this.state.selectedIndex); + evt.preventDefault(); + break; + case 'ArrowUp': + evt.preventDefault(); + this.setState({ + selectedIndex: this.state.selectedIndex < 1 ? this.state.results.length - 1 : this.state.selectedIndex - 1, + }, scrollToSelected); + + break; + case 'ArrowDown': + evt.preventDefault(); + this.setState({ + selectedIndex: this.state.selectedIndex < this.state.results.length - 1 ? this.state.selectedIndex + 1 : 0, + }, scrollToSelected); + break; + default: + break; + } + }; + + handleItemClick = index => { + this.setState({ visible: false }); + const item = this.state.results[index]; + history.push(`/element/${encodeURIComponent(item.obj.file)}/code`); + }; render() { if (!this.state.visible) return null; - const { elementById } = this.props.home; - - const result = Object.keys(elementById) - .map(k => { - const item = { ...elementById[k] }; - if (/component|action/.test(item.type)) { - item.position = item.file - .split('/') - .slice(0, -1) - .join('/'); - } - return item; - }) - .filter(e => /component|action|misc|file/.test(e.type)) - .slice(0, 15); const iconTypes = { component: 'appstore-o', @@ -117,16 +153,21 @@ export class QuickOpen extends Component { onBlur={this.handleInputBlur} placeholder="Type something to search..." value={this.state.search} + onKeyDown={this.handleInputKeyDown} onChange={this.handleInputChange} /> -