diff --git a/CHANGELOG.md b/CHANGELOG.md
index 8b401a4b7f8..0112479698e 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,5 +1,8 @@
## [`master`](https://github.com/elastic/eui/tree/master)
+- `EuiComboBox` is now decorated with `data-test-subj` selectors for the search input (`comboxBoxSearchInput`), toggle button (`comboBoxToggleListButton`), and clear button (`comboBoxClearButton`) ([#918](https://github.com/elastic/eui/pull/918))
+- `EuiComboBox` now gives focus to the search input when the user clicks the clear button, to prevent focus from defaulting to the body ([#918](https://github.com/elastic/eui/pull/918))
+
**Bug fixes**
- Added `role="dialog"` to `EuiFlyout` to improve screen reader accessibility ([#916](https://github.com/elastic/eui/pull/916))
diff --git a/src/components/combo_box/__snapshots__/combo_box.test.js.snap b/src/components/combo_box/__snapshots__/combo_box.test.js.snap
index 0d50dc5a6ad..61d5b2c3973 100644
--- a/src/components/combo_box/__snapshots__/combo_box.test.js.snap
+++ b/src/components/combo_box/__snapshots__/combo_box.test.js.snap
@@ -3,8 +3,69 @@
exports[`EuiComboBox is rendered 1`] = `
+`;
+
+exports[`props isClearable=false disallows user from clearing input when no options are selected 1`] = `
+
@@ -14,7 +75,6 @@ exports[`EuiComboBox is rendered 1`] = `
inputRef={[Function]}
isListOpen={false}
onChange={[Function]}
- onClear={[Function]}
onClick={[Function]}
onCloseListClick={[Function]}
onFocus={[Function]}
@@ -30,11 +90,9 @@ exports[`EuiComboBox is rendered 1`] = `
`;
-exports[`props isClearable is false with selectedOptions 1`] = `
+exports[`props isClearable=false disallows user from clearing input when options are selected 1`] = `
@@ -68,40 +126,9 @@ exports[`props isClearable is false with selectedOptions 1`] = `
`;
-exports[`props isClearable is false without selectedOptions 1`] = `
+exports[`props isDisabled is rendered 1`] = `
-
-
-`;
-
-exports[`props isDisabled 1`] = `
-
@@ -133,11 +160,9 @@ exports[`props isDisabled 1`] = `
`;
-exports[`props options 1`] = `
+exports[`props options are rendered 1`] = `
@@ -163,11 +188,9 @@ exports[`props options 1`] = `
`;
-exports[`props selectedOptions 1`] = `
+exports[`props selectedOptions are rendered 1`] = `
@@ -202,11 +225,9 @@ exports[`props selectedOptions 1`] = `
`;
-exports[`props singleSelection 1`] = `
+exports[`props singleSelection is rendered 1`] = `
diff --git a/src/components/combo_box/combo_box.js b/src/components/combo_box/combo_box.js
index 94bc1509a5c..4de12691c19 100644
--- a/src/components/combo_box/combo_box.js
+++ b/src/components/combo_box/combo_box.js
@@ -401,6 +401,9 @@ export class EuiComboBox extends Component {
clearSelectedOptions = () => {
this.props.onChange([]);
+ // Clicking the clear button will also cause it to disappear. This would result in focus
+ // shifting unexpectedly to the body element so we set it to the input which is more reasonable,
+ this.searchInput.focus();
}
onComboBoxClick = () => {
diff --git a/src/components/combo_box/combo_box.test.js b/src/components/combo_box/combo_box.test.js
index f1a0c3cc5d4..63797c23bd7 100644
--- a/src/components/combo_box/combo_box.test.js
+++ b/src/components/combo_box/combo_box.test.js
@@ -1,6 +1,7 @@
import React from 'react';
-import { shallow } from 'enzyme';
-import { requiredProps } from '../../test';
+import { shallow, render, mount } from 'enzyme';
+import sinon from 'sinon';
+import { requiredProps, findTestSubject } from '../../test';
import { EuiComboBox } from './combo_box';
@@ -29,7 +30,7 @@ const options = [{
describe('EuiComboBox', () => {
test('is rendered', () => {
- const component = shallow(
+ const component = render(
);
@@ -38,49 +39,45 @@ describe('EuiComboBox', () => {
});
describe('props', () => {
- test('options', () => {
+ test('options are rendered', () => {
+ // NOTE: It's tough to test this because the dropdown containing the options opens up in
+ // a portal.
const component = shallow(
-
+
);
expect(component).toMatchSnapshot();
});
- test('selectedOptions', () => {
+ test('selectedOptions are rendered', () => {
const component = shallow(
);
expect(component).toMatchSnapshot();
});
- describe('isClearable is false', () => {
- test('without selectedOptions', () => {
+ describe('isClearable=false disallows user from clearing input', () => {
+ test('when no options are selected', () => {
const component = shallow(
);
expect(component).toMatchSnapshot();
});
- test('with selectedOptions', () => {
+ test('when options are selected', () => {
const component = shallow(
);
@@ -88,29 +85,59 @@ describe('props', () => {
});
});
- test('singleSelection', () => {
+ test('singleSelection is rendered', () => {
const component = shallow(
);
expect(component).toMatchSnapshot();
});
- test('isDisabled', () => {
+ test('isDisabled is rendered', () => {
const component = shallow(
);
expect(component).toMatchSnapshot();
});
});
+
+describe('behavior', () => {
+ describe('clear button', () => {
+ test('calls onChange callback with empty array', () => {
+ const onChangeHandler = sinon.spy();
+ const component = mount(
+
+ );
+
+ findTestSubject(component, 'comboBoxClearButton').simulate('click');
+ sinon.assert.calledOnce(onChangeHandler);
+ sinon.assert.calledWith(onChangeHandler, []);
+ });
+
+ test('focuses the input', () => {
+ const component = mount(
+ {}}
+ />
+ );
+
+ findTestSubject(component, 'comboBoxClearButton').simulate('click');
+ expect(findTestSubject(component, 'comboBoxSearchInput').getDOMNode()).toBe(document.activeElement);
+ });
+ });
+});
diff --git a/src/components/combo_box/combo_box_input/combo_box_input.js b/src/components/combo_box/combo_box_input/combo_box_input.js
index 80612d61843..335bb752cc4 100644
--- a/src/components/combo_box/combo_box_input/combo_box_input.js
+++ b/src/components/combo_box/combo_box_input/combo_box_input.js
@@ -152,6 +152,7 @@ export class EuiComboBoxInput extends Component {
if (!isDisabled && onClear && hasSelectedOptions) {
clickProps.clear = {
onClick: onClear,
+ 'data-test-subj': 'comboBoxClearButton',
};
}
@@ -161,7 +162,8 @@ export class EuiComboBoxInput extends Component {
onClick: isListOpen && !isDisabled ? onCloseListClick : onOpenListClick,
ref: toggleButtonRef,
'aria-label': isListOpen ? 'Close list of options' : 'Open list of options',
- disabled: isDisabled
+ disabled: isDisabled,
+ 'data-test-subj': 'comboBoxToggleListButton',
};
return (
@@ -188,6 +190,7 @@ export class EuiComboBoxInput extends Component {
ref={autoSizeInputRef}
inputRef={inputRef}
disabled={isDisabled}
+ data-test-subj="comboBoxSearchInput"
/>
{removeOptionMessage}