aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/console/components/console.jsx8
-rw-r--r--src/console/components/console/completion.jsx55
-rw-r--r--src/console/reducers/index.js70
-rw-r--r--test/console/components/console/completion.test.jsx138
-rw-r--r--test/console/reducers/console.test.js38
5 files changed, 232 insertions, 77 deletions
diff --git a/src/console/components/console.jsx b/src/console/components/console.jsx
index 23c93e3..7994f78 100644
--- a/src/console/components/console.jsx
+++ b/src/console/components/console.jsx
@@ -6,6 +6,8 @@ import Completion from './console/completion';
import Message from './console/message';
import * as consoleActions from '../../console/actions/console';
+const COMPLETION_MAX_ITEMS = 33;
+
class ConsoleComponent extends Component {
onBlur() {
if (this.props.mode === 'command' || this.props.mode === 'find') {
@@ -105,7 +107,11 @@ class ConsoleComponent extends Component {
case 'command':
case 'find':
return <div className='vimvixen-console-command-wrapper'>
- <Completion />
+ <Completion
+ size={COMPLETION_MAX_ITEMS}
+ completions={this.props.completions}
+ select={this.props.select}
+ />
<Input
ref={(c) => { this.input = c; }}
mode={this.props.mode}
diff --git a/src/console/components/console/completion.jsx b/src/console/components/console/completion.jsx
index c60543b..d836cec 100644
--- a/src/console/components/console/completion.jsx
+++ b/src/console/components/console/completion.jsx
@@ -1,5 +1,4 @@
import { Component, h } from 'preact';
-import { connect } from 'preact-redux';
const CompletionTitle = (props) => {
return <li className='vimvixen-console-completion-title' >{props.title}</li>;
@@ -25,25 +24,60 @@ const CompletionItem = (props) => {
class CompletionComponent extends Component {
+ constructor() {
+ super();
+ this.state = { viewOffset: 0, select: -1 };
+ }
+
+ static getDerivedStateFromProps(nextProps, prevState) {
+ if (prevState.select === nextProps.select) {
+ return null;
+ }
+
+ let viewSelect = (() => {
+ let index = 0;
+ for (let i = 0; i < nextProps.completions.length; ++i) {
+ ++index;
+ let g = nextProps.completions[i];
+ if (nextProps.select + i + 1 < index + g.items.length) {
+ return nextProps.select + i + 1;
+ }
+ index += g.items.length;
+ }
+ })();
+
+ let viewOffset = 0;
+ if (nextProps.select < 0) {
+ viewOffset = 0;
+ } else if (prevState.select < nextProps.select) {
+ viewOffset = Math.max(prevState.viewOffset,
+ viewSelect - nextProps.size + 1);
+ } else if (prevState.select > nextProps.select) {
+ viewOffset = Math.min(prevState.viewOffset, viewSelect);
+ }
+ return { viewOffset, select: nextProps.select };
+ }
+
render() {
let eles = [];
- for (let i = 0; i < this.props.completions.length; ++i) {
- let group = this.props.completions[i];
+ let index = 0;
+
+ for (let group of this.props.completions) {
eles.push(<CompletionTitle title={ group.name }/>);
- for (let j = 0; j < group.items.length; ++j) {
- let item = group.items[j];
- let selected =
- i === this.props.groupSelection &&
- j === this.props.itemSelection;
+ for (let item of group.items) {
eles.push(<CompletionItem
icon={item.icon}
caption={item.caption}
url={item.url}
- highlight={selected}
+ highlight={index === this.props.select}
/ >);
+ ++index;
}
}
+ let viewOffset = this.state.viewOffset;
+ eles = eles.slice(viewOffset, viewOffset + this.props.size);
+
return (
<ul className='vimvixen-console-completion'>
{ eles }
@@ -52,5 +86,4 @@ class CompletionComponent extends Component {
}
}
-const mapStateToProps = state => state;
-export default connect(mapStateToProps)(CompletionComponent);
+export default CompletionComponent;
diff --git a/src/console/reducers/index.js b/src/console/reducers/index.js
index 7dcad17..614a72f 100644
--- a/src/console/reducers/index.js
+++ b/src/console/reducers/index.js
@@ -6,52 +6,43 @@ const defaultState = {
consoleText: '',
completionSource: '',
completions: [],
- groupSelection: -1,
- itemSelection: -1,
+ select: -1,
+ viewIndex: 0,
};
const nextSelection = (state) => {
if (state.completions.length === 0) {
- return [-1, -1];
+ return -1;
}
- if (state.groupSelection < 0) {
- return [0, 0];
+ if (state.select < 0) {
+ return 0;
}
- let group = state.completions[state.groupSelection];
- if (state.groupSelection + 1 >= state.completions.length &&
- state.itemSelection + 1 >= group.items.length) {
- return [-1, -1];
+ let length = state.completions
+ .map(g => g.items.length)
+ .reduce((x, y) => x + y);
+ if (state.select + 1 < length) {
+ return state.select + 1;
}
- if (state.itemSelection + 1 >= group.items.length) {
- return [state.groupSelection + 1, 0];
- }
- return [state.groupSelection, state.itemSelection + 1];
+ return -1;
};
const prevSelection = (state) => {
- if (state.groupSelection < 0) {
- return [
- state.completions.length - 1,
- state.completions[state.completions.length - 1].items.length - 1
- ];
- }
- if (state.groupSelection === 0 && state.itemSelection === 0) {
- return [-1, -1];
- } else if (state.itemSelection === 0) {
- return [
- state.groupSelection - 1,
- state.completions[state.groupSelection - 1].items.length - 1
- ];
+ let length = state.completions
+ .map(g => g.items.length)
+ .reduce((x, y) => x + y);
+ if (state.select < 0) {
+ return length - 1;
}
- return [state.groupSelection, state.itemSelection - 1];
+ return state.select - 1;
};
-const nextConsoleText = (completions, group, item, defaults) => {
- if (group < 0 || item < 0) {
+const nextConsoleText = (completions, select, defaults) => {
+ if (select < 0) {
return defaults;
}
- return completions[group].items[item].content;
+ let items = completions.map(g => g.items).reduce((g1, g2) => g1.concat(g2));
+ return items[select].content;
};
// eslint-disable-next-line max-lines-per-function
@@ -90,25 +81,20 @@ export default function reducer(state = defaultState, action = {}) {
return { ...state,
completions: action.completions,
completionSource: action.completionSource,
- groupSelection: -1,
- itemSelection: -1, };
+ select: -1 };
case actions.CONSOLE_COMPLETION_NEXT: {
- let next = nextSelection(state);
+ let select = nextSelection(state);
return { ...state,
- groupSelection: next[0],
- itemSelection: next[1],
+ select: select,
consoleText: nextConsoleText(
- state.completions, next[0], next[1],
- state.completionSource), };
+ state.completions, select, state.completionSource) };
}
case actions.CONSOLE_COMPLETION_PREV: {
- let next = prevSelection(state);
+ let select = prevSelection(state);
return { ...state,
- groupSelection: next[0],
- itemSelection: next[1],
+ select: select,
consoleText: nextConsoleText(
- state.completions, next[0], next[1],
- state.completionSource), };
+ state.completions, select, state.completionSource) };
}
default:
return state;
diff --git a/test/console/components/console/completion.test.jsx b/test/console/components/console/completion.test.jsx
new file mode 100644
index 0000000..0b48fe2
--- /dev/null
+++ b/test/console/components/console/completion.test.jsx
@@ -0,0 +1,138 @@
+import { h, render } from 'preact';
+import Completion from 'console/components/console/completion'
+
+describe("console/components/console/completion", () => {
+ let completions = [{
+ name: "Fruit",
+ items: [{ caption: "apple" }, { caption: "banana" }, { caption: "cherry" }],
+ }, {
+ name: "Element",
+ items: [{ caption: "argon" }, { caption: "boron" }, { caption: "carbon" }],
+ }];
+
+ beforeEach(() => {
+ document.body.innerHTML = '';
+ });
+
+ it('renders Completion component', () => {
+ let ul = render(<Completion
+ completions={completions}
+ size={30}
+ />, document.body);
+
+ expect(ul.children).to.have.lengthOf(8);
+ expect(ul.children[0].textContent).to.equal('Fruit');
+ expect(ul.children[1].textContent).to.equal('apple');
+ expect(ul.children[2].textContent).to.equal('banana');
+ expect(ul.children[3].textContent).to.equal('cherry');
+ expect(ul.children[4].textContent).to.equal('Element');
+ expect(ul.children[5].textContent).to.equal('argon');
+ expect(ul.children[6].textContent).to.equal('boron');
+ expect(ul.children[7].textContent).to.equal('carbon');
+ });
+
+ it('highlight current item', () => {
+ let ul = render(<Completion
+ completions={completions}
+ size={30}
+ select={3}
+ />, document.body);
+ expect(ul.children[5].className.split(' ')).to.include('vimvixen-completion-selected');
+ });
+
+ it('does not highlight any items', () => {
+ let ul = render(<Completion
+ completions={completions}
+ size={30}
+ select={-1}
+ />, document.body);
+ for (let li of ul.children) {
+ expect(li.className.split(' ')).not.to.include('vimvixen-completion-selected');
+ }
+ });
+
+
+ it('limits completion items', () => {
+ let ul = render(<Completion
+ completions={completions}
+ size={3}
+ select={-1}
+ />, document.body);
+ expect(Array.from(ul.children).map(e => e.textContent)).to.deep.equal(['Fruit', 'apple', 'banana']);
+
+ ul = render(<Completion
+ completions={completions}
+ size={3} select={0}
+ />, document.body, ul);
+
+ expect(Array.from(ul.children).map(e => e.textContent)).to.deep.equal(['Fruit', 'apple', 'banana']);
+ expect(ul.children[1].className.split(' ')).to.include('vimvixen-completion-selected');
+ })
+
+ it('scrolls up to down with select', () => {
+ let ul = render(<Completion
+ completions={completions}
+ size={3}
+ select={1}
+ />, document.body);
+
+ expect(Array.from(ul.children).map(e => e.textContent)).to.deep.equal(['Fruit', 'apple', 'banana']);
+ expect(ul.children[2].className.split(' ')).to.include('vimvixen-completion-selected');
+
+ ul = render(<Completion
+ completions={completions}
+ size={3}
+ select={2}
+ />, document.body, ul);
+
+ expect(Array.from(ul.children).map(e => e.textContent)).to.deep.equal(['apple', 'banana', 'cherry']);
+ expect(ul.children[2].className.split(' ')).to.include('vimvixen-completion-selected');
+
+ ul = render(<Completion
+ completions={completions}
+ size={3}
+ select={3}
+ />, document.body, ul);
+
+ expect(Array.from(ul.children).map(e => e.textContent)).to.deep.equal(['cherry', 'Element', 'argon']);
+ expect(ul.children[2].className.split(' ')).to.include('vimvixen-completion-selected');
+ });
+
+ it('scrolls up to down with select', () => {
+ let ul = render(<Completion
+ completions={completions}
+ size={3}
+ select={5}
+ />, document.body);
+
+ expect(Array.from(ul.children).map(e => e.textContent)).to.deep.equal(['argon', 'boron', 'carbon']);
+ expect(ul.children[2].className.split(' ')).to.include('vimvixen-completion-selected');
+
+ ul = render(<Completion
+ completions={completions}
+ size={3}
+ select={4}
+ />, document.body, ul);
+
+ expect(Array.from(ul.children).map(e => e.textContent)).to.deep.equal(['argon', 'boron', 'carbon']);
+ expect(ul.children[1].className.split(' ')).to.include('vimvixen-completion-selected');
+
+ ul = render(<Completion
+ completions={completions}
+ size={3}
+ select={3}
+ />, document.body, ul);
+
+ expect(Array.from(ul.children).map(e => e.textContent)).to.deep.equal(['argon', 'boron', 'carbon']);
+ expect(ul.children[0].className.split(' ')).to.include('vimvixen-completion-selected');
+
+ ul = render(<Completion
+ completions={completions}
+ size={3}
+ select={2}
+ />, document.body, ul);
+
+ expect(Array.from(ul.children).map(e => e.textContent)).to.deep.equal(['cherry', 'Element', 'argon']);
+ expect(ul.children[0].className.split(' ')).to.include('vimvixen-completion-selected');
+ });
+});
diff --git a/test/console/reducers/console.test.js b/test/console/reducers/console.test.js
index db40088..d5a38cf 100644
--- a/test/console/reducers/console.test.js
+++ b/test/console/reducers/console.test.js
@@ -8,8 +8,7 @@ describe("console reducer", () => {
expect(state).to.have.property('messageText', '');
expect(state).to.have.property('consoleText', '');
expect(state).to.have.deep.property('completions', []);
- expect(state).to.have.property('groupSelection', -1);
- expect(state).to.have.property('itemSelection', -1);
+ expect(state).to.have.property('select', -1);
});
it('return next state for CONSOLE_HIDE', () => {
@@ -60,8 +59,7 @@ describe("console reducer", () => {
it ('return next state for CONSOLE_SET_COMPLETIONS', () => {
let state = {
- groupSelection: 0,
- itemSelection: 0,
+ select: 0,
completions: [],
}
let action = {
@@ -76,15 +74,13 @@ describe("console reducer", () => {
}
state = reducer(state, action);
expect(state).to.have.property('completions', action.completions);
- expect(state).to.have.property('groupSelection', -1);
- expect(state).to.have.property('itemSelection', -1);
+ expect(state).to.have.property('select', -1);
});
it ('return next state for CONSOLE_COMPLETION_NEXT', () => {
let action = { type: actions.CONSOLE_COMPLETION_NEXT };
let state = {
- groupSelection: -1,
- itemSelection: -1,
+ select: -1,
completions: [{
name: 'Apple',
items: [1, 2]
@@ -95,24 +91,22 @@ describe("console reducer", () => {
};
state = reducer(state, action);
- expect(state).to.have.property('groupSelection', 0);
- expect(state).to.have.property('itemSelection', 0);
+ expect(state).to.have.property('select', 0);
state = reducer(state, action);
- expect(state).to.have.property('groupSelection', 0);
- expect(state).to.have.property('itemSelection', 1);
+ expect(state).to.have.property('select', 1);
state = reducer(state, action);
+ expect(state).to.have.property('select', 2);
+
state = reducer(state, action);
- expect(state).to.have.property('groupSelection', -1);
- expect(state).to.have.property('itemSelection', -1);
+ expect(state).to.have.property('select', -1);
});
it ('return next state for CONSOLE_COMPLETION_PREV', () => {
let action = { type: actions.CONSOLE_COMPLETION_PREV };
let state = {
- groupSelection: -1,
- itemSelection: -1,
+ select: -1,
completions: [{
name: 'Apple',
items: [1, 2]
@@ -123,17 +117,15 @@ describe("console reducer", () => {
};
state = reducer(state, action);
- expect(state).to.have.property('groupSelection', 1);
- expect(state).to.have.property('itemSelection', 0);
+ expect(state).to.have.property('select', 2);
state = reducer(state, action);
- expect(state).to.have.property('groupSelection', 0);
- expect(state).to.have.property('itemSelection', 1);
+ expect(state).to.have.property('select', 1);
state = reducer(state, action);
+ expect(state).to.have.property('select', 0);
+
state = reducer(state, action);
- expect(state).to.have.property('groupSelection', -1);
- expect(state).to.have.property('itemSelection', -1);
+ expect(state).to.have.property('select', -1);
});
-
});