From 7f6953e59ce18cc41e36d77634f5a2817cf79412 Mon Sep 17 00:00:00 2001 From: Dave Snider Date: Thu, 17 May 2018 16:24:32 -0700 Subject: [PATCH 01/11] style up focus for combo buttons --- .../form_control_layout/_form_control_layout.scss | 14 +++++++++++++- .../form/form_control_layout/_mixins.scss | 5 +++++ 2 files changed, 18 insertions(+), 1 deletion(-) diff --git a/src/components/form/form_control_layout/_form_control_layout.scss b/src/components/form/form_control_layout/_form_control_layout.scss index 7e197063cea..f5d241acac2 100644 --- a/src/components/form/form_control_layout/_form_control_layout.scss +++ b/src/components/form/form_control_layout/_form_control_layout.scss @@ -27,15 +27,27 @@ .euiFormControlLayout__iconButton { pointer-events: all; + height: $euiSize; + width: $euiSize; @include size($euiSize); .euiIcon { vertical-align: baseline; } + svg { + margin-top: -$euiSizeXS; + } + + &:focus { + background: $euiColorPrimary; + border-radius: 100%; + color: $euiColorGhost; + } + @at-root { .euiFormControlLayout--compressed#{&} { - top: $euiFormControlPadding--compressed - 1px; + top: $euiFormControlPadding--compressed; } } } diff --git a/src/components/form/form_control_layout/_mixins.scss b/src/components/form/form_control_layout/_mixins.scss index 05865c366fd..30828beea91 100644 --- a/src/components/form/form_control_layout/_mixins.scss +++ b/src/components/form/form_control_layout/_mixins.scss @@ -5,6 +5,11 @@ background-color: transparentize($euiColorMediumShade, .5); border-radius: $clearSize; line-height: $clearSize; + transition: background-color $euiAnimSpeedExtraFast $euiAnimSlightResistance; + + &:focus, &:hover { + background-color: $euiColorPrimary; + } #{$iconClass} { @include size($euiSizeS); From e0c81c9b8cedee25ee5246f99a08fe68c250a66e Mon Sep 17 00:00:00 2001 From: CJ Cenizal Date: Tue, 22 May 2018 18:53:33 -0700 Subject: [PATCH 02/11] Fix bug where clicking the combo box caret would focus on the button. - Set focus on searchInput when you click the caret to open the combo box. - Once open, clicking it again is a no-op. --- .../__snapshots__/combo_box.test.js.snap | 21 +++++++------------ src/components/combo_box/combo_box.js | 7 +++++-- .../combo_box_input/combo_box_input.js | 8 +++---- 3 files changed, 15 insertions(+), 21 deletions(-) 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 5285fbbb094..9d79a33edc7 100644 --- a/src/components/combo_box/__snapshots__/combo_box.test.js.snap +++ b/src/components/combo_box/__snapshots__/combo_box.test.js.snap @@ -16,9 +16,8 @@ exports[`EuiComboBox is rendered 1`] = ` onChange={[Function]} onClear={[Function]} onClick={[Function]} - onClose={[Function]} onFocus={[Function]} - onOpen={[Function]} + onOpenListClick={[Function]} onRemoveOption={[Function]} searchValue="" selectedOptions={Array []} @@ -44,9 +43,8 @@ exports[`props isClearable is false with selectedOptions 1`] = ` isListOpen={false} onChange={[Function]} onClick={[Function]} - onClose={[Function]} onFocus={[Function]} - onOpen={[Function]} + onOpenListClick={[Function]} onRemoveOption={[Function]} searchValue="" selectedOptions={ @@ -81,9 +79,8 @@ exports[`props isClearable is false without selectedOptions 1`] = ` isListOpen={false} onChange={[Function]} onClick={[Function]} - onClose={[Function]} onFocus={[Function]} - onOpen={[Function]} + onOpenListClick={[Function]} onRemoveOption={[Function]} searchValue="" selectedOptions={Array []} @@ -110,9 +107,8 @@ exports[`props isDisabled 1`] = ` isListOpen={false} onChange={[Function]} onClick={[Function]} - onClose={[Function]} onFocus={[Function]} - onOpen={[Function]} + onOpenListClick={[Function]} onRemoveOption={[Function]} searchValue="" selectedOptions={ @@ -145,9 +141,8 @@ exports[`props options 1`] = ` onChange={[Function]} onClear={[Function]} onClick={[Function]} - onClose={[Function]} onFocus={[Function]} - onOpen={[Function]} + onOpenListClick={[Function]} onRemoveOption={[Function]} searchValue="" selectedOptions={Array []} @@ -174,9 +169,8 @@ exports[`props selectedOptions 1`] = ` onChange={[Function]} onClear={[Function]} onClick={[Function]} - onClose={[Function]} onFocus={[Function]} - onOpen={[Function]} + onOpenListClick={[Function]} onRemoveOption={[Function]} searchValue="" selectedOptions={ @@ -212,9 +206,8 @@ exports[`props singleSelection 1`] = ` onChange={[Function]} onClear={[Function]} onClick={[Function]} - onClose={[Function]} onFocus={[Function]} - onOpen={[Function]} + onOpenListClick={[Function]} onRemoveOption={[Function]} searchValue="" selectedOptions={ diff --git a/src/components/combo_box/combo_box.js b/src/components/combo_box/combo_box.js index 7d4813b5189..3fd85cfa6bc 100644 --- a/src/components/combo_box/combo_box.js +++ b/src/components/combo_box/combo_box.js @@ -417,6 +417,10 @@ export class EuiComboBox extends Component { } }; + onOpenListClick = () => { + this.searchInput.focus(); + }; + onSearchChange = (searchValue) => { if (this.props.onSearchChange) { this.props.onSearchChange(searchValue); @@ -586,8 +590,7 @@ export class EuiComboBox extends Component { onClear={isClearable && !isDisabled ? this.clearSelectedOptions : undefined} hasSelectedOptions={selectedOptions.length > 0} isListOpen={isListOpen} - onOpen={this.openList} - onClose={this.closeList} + onOpenListClick={this.onOpenListClick} singleSelection={singleSelection} isDisabled={isDisabled} /> 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 33ff228776e..0f181f51e19 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 @@ -26,8 +26,7 @@ export class EuiComboBoxInput extends Component { onClear: PropTypes.func, hasSelectedOptions: PropTypes.bool.isRequired, isListOpen: PropTypes.bool.isRequired, - onOpen: PropTypes.func.isRequired, - onClose: PropTypes.func.isRequired, + onOpenListClick: PropTypes.func.isRequired, singleSelection: PropTypes.bool, isDisabled: PropTypes.bool, } @@ -86,8 +85,7 @@ export class EuiComboBoxInput extends Component { onClear, hasSelectedOptions, isListOpen, - onOpen, - onClose, + onOpenListClick, singleSelection, isDisabled, } = this.props; @@ -150,7 +148,7 @@ export class EuiComboBoxInput extends Component { if (!isDisabled) { clickProps.onClear = hasSelectedOptions ? onClear : undefined; - clickProps.onIconClick = isListOpen ? onClose : onOpen; + clickProps.onIconClick = isListOpen ? undefined : onOpenListClick; } return ( From a5e37c8d4525043e7d894a8b7e87cdf452a3a16c Mon Sep 17 00:00:00 2001 From: CJ Cenizal Date: Tue, 22 May 2018 19:39:04 -0700 Subject: [PATCH 03/11] Fix EuiComboBox focus trap. Make clear button keyboard-accessible. - Redesign EuiFormControlLayout props to use icon and clear configuration objects. --- CHANGELOG.md | 2 + .../__snapshots__/combo_box.test.js.snap | 7 ++ src/components/combo_box/combo_box.js | 50 +++++--- .../combo_box_input/combo_box_input.js | 20 ++- .../__snapshots__/date_picker.test.js.snap | 1 - .../form/field_number/field_number.js | 10 +- .../form/field_number/field_number.test.js | 8 +- src/components/form/field_text/field_text.js | 10 +- .../form/field_text/field_text.test.js | 8 +- .../form_control_layout.test.js.snap | 68 ++++++++++- .../form_control_layout.js | 64 +++++++--- .../form_control_layout.test.js | 115 +++++++++++++++--- .../form/form_control_layout/index.js | 1 + .../select/__snapshots__/select.test.js.snap | 21 ++-- src/components/form/select/select.js | 8 +- 15 files changed, 311 insertions(+), 82 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index f0ce27e6e10..675e7784c0c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,9 +4,11 @@ - `EuiSearchBar` no longer has an `onParse` callback, and now passes an object to `onChange` with the shape `{ query, queryText, error }` ([#863](https://github.com/elastic/eui/pull/863)) - `EuiInMemoryTable`'s `search.onChange` callback now passes an object with `{ query, queryText, error }` instead of only the query ([#863](https://github.com/elastic/eui/pull/863)) +- `EuiFormControlLayout` no longer has `onClear`, `iconSide`, or `onIconClick` props. Instead of `onClear` it now accepts a `clear` object of the shape `{ onClick }`. Instead of the icon props, it now accepts a single `icon` prop which be either a string or an object of the shape `{ type, side, onClick }`. ([#866](https://github.com/elastic/eui/pull/866)) **Bug fixes** +- `EuiComboBox` is no longer a focus trap and the clear button is now keyboard-accessible ([#866](https://github.com/elastic/eui/pull/866)) - `EuiButton`, `EuiButtonEmpty`, and `EuiButtonIcon` now look and behave disabled when `isDisabled={true}` ([#862](https://github.com/elastic/eui/pull/862)) - `EuiGlobalToastList` no longer triggers `Uncaught TypeError: _this.callback is not a function` ([#865](https://github.com/elastic/eui/pull/865)) - `EuiGlobalToastList` checks to see if it has dismissed a toast before re-dismissing it ([#868](https://github.com/elastic/eui/pull/868)) 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 9d79a33edc7..c4dfbe24c95 100644 --- a/src/components/combo_box/__snapshots__/combo_box.test.js.snap +++ b/src/components/combo_box/__snapshots__/combo_box.test.js.snap @@ -10,6 +10,7 @@ exports[`EuiComboBox is rendered 1`] = ` > { const tabbableItems = tabbable(document); - const comboBoxIndex = tabbableItems.indexOf(this.searchInput); - // Wrap to last tabbable if tabbing backwards. - if (amount < 0) { - if (comboBoxIndex === 0) { - tabbableItems[tabbableItems.length - 1].focus(); - return; + if (document.activeElement === this.searchInput) { + const searchInputIndex = tabbableItems.indexOf(this.searchInput); + + // Wrap to last tabbable if tabbing backwards. + if (amount < 0) { + if (searchInputIndex === 0) { + tabbableItems[tabbableItems.length - 1].focus(); + return; + } } + + // Otherwise tab to the next adjacent item. + tabbableItems[searchInputIndex + amount].focus(); + return; } - // Wrap to first tabbable if tabbing forwards. - if (amount > 0) { - if (comboBoxIndex === tabbableItems.length - 1) { - tabbableItems[0].focus(); - return; + if (document.activeElement === this.clearButton) { + const clearButtonIndex = tabbableItems.indexOf(this.clearButton); + + // Wrap to first tabbable if tabbing forwards. + if (amount > 0) { + if (clearButtonIndex === tabbableItems.length - 1) { + tabbableItems[0].focus(); + return; + } } - } - tabbableItems[comboBoxIndex + amount].focus(); + // Otherwise tab to the next adjacent item. + tabbableItems[clearButtonIndex + amount].focus(); + } }; incrementActiveOptionIndex = throttle(amount => { @@ -290,14 +302,10 @@ export class EuiComboBox extends Component { }; onFocus = () => { - document.addEventListener('click', this.onDocumentFocusChange); - document.addEventListener('focusin', this.onDocumentFocusChange); this.openList(); } onBlur = () => { - document.removeEventListener('click', this.onDocumentFocusChange); - document.removeEventListener('focusin', this.onDocumentFocusChange); this.closeList(); } @@ -309,6 +317,7 @@ export class EuiComboBox extends Component { || this.optionsList === event.target || this.optionsList && this.optionsList.contains(event.target) ) { + this.onFocus(); return; } @@ -454,8 +463,14 @@ export class EuiComboBox extends Component { this.options[index] = node; }; + clearButtonRef = node => { + this.clearButton = node; + }; + componentDidMount() { this._isMounted = true; + document.addEventListener('click', this.onDocumentFocusChange); + document.addEventListener('focusin', this.onDocumentFocusChange); // TODO: This will need to be called once the actual stylesheet loads. setTimeout(() => { @@ -593,6 +608,7 @@ export class EuiComboBox extends Component { onOpenListClick={this.onOpenListClick} singleSelection={singleSelection} isDisabled={isDisabled} + clearButtonRef={this.clearButtonRef} /> {optionsList} 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 0f181f51e19..d26fd2e1aea 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 @@ -29,6 +29,7 @@ export class EuiComboBoxInput extends Component { onOpenListClick: PropTypes.func.isRequired, singleSelection: PropTypes.bool, isDisabled: PropTypes.bool, + clearButtonRef: PropTypes.func, } constructor(props) { @@ -88,6 +89,7 @@ export class EuiComboBoxInput extends Component { onOpenListClick, singleSelection, isDisabled, + clearButtonRef, } = this.props; const pills = selectedOptions.map((option) => { @@ -147,14 +149,24 @@ export class EuiComboBoxInput extends Component { const clickProps = {}; if (!isDisabled) { - clickProps.onClear = hasSelectedOptions ? onClear : undefined; - clickProps.onIconClick = isListOpen ? undefined : onOpenListClick; + clickProps.clear = { + onClick: hasSelectedOptions ? onClear : undefined, + ref: clearButtonRef, + }; } + const icon = { + type: 'arrowDown', + side: 'right', + onClick: isListOpen && !isDisabled ? undefined : onOpenListClick, + // We want to remove this from the tab order because you can open the combo box by tabbing + // to it already. + tabIndex: '-1', + }; + return (
diff --git a/src/components/form/field_number/field_number.js b/src/components/form/field_number/field_number.js index a75471e455a..eba2f0885b9 100644 --- a/src/components/form/field_number/field_number.js +++ b/src/components/form/field_number/field_number.js @@ -3,6 +3,7 @@ import PropTypes from 'prop-types'; import classNames from 'classnames'; import { + ICON_SIDES, EuiFormControlLayout, } from '../form_control_layout'; @@ -81,7 +82,14 @@ EuiFieldNumber.propTypes = { max: PropTypes.number, step: PropTypes.number, value: numberOrEmptyString, - icon: PropTypes.string, + icon: PropTypes.oneOfType([ + PropTypes.string, + PropTypes.shape({ + type: PropTypes.string, + side: PropTypes.oneOf(ICON_SIDES), + onClick: PropTypes.func, + }), + ]), isInvalid: PropTypes.bool, fullWidth: PropTypes.bool, isLoading: PropTypes.bool, diff --git a/src/components/form/field_number/field_number.test.js b/src/components/form/field_number/field_number.test.js index 41b16cb32a5..15158c1d8b1 100644 --- a/src/components/form/field_number/field_number.test.js +++ b/src/components/form/field_number/field_number.test.js @@ -4,7 +4,13 @@ import { requiredProps } from '../../../test/required_props'; import { EuiFieldNumber } from './field_number'; -jest.mock('../form_control_layout', () => ({ EuiFormControlLayout: 'eui-form-control-layout' })); +jest.mock('../form_control_layout', () => { + const formControlLayout = require.requireActual('../form_control_layout') + return { + ...formControlLayout, + EuiFormControlLayout: 'eui-form-control-layout', + } +}); jest.mock('../validatable_control', () => ({ EuiValidatableControl: 'eui-validatable-control' })); describe('EuiFieldNumber', () => { diff --git a/src/components/form/field_text/field_text.js b/src/components/form/field_text/field_text.js index 7fb91cd72f9..361ebfc87e6 100644 --- a/src/components/form/field_text/field_text.js +++ b/src/components/form/field_text/field_text.js @@ -3,6 +3,7 @@ import PropTypes from 'prop-types'; import classNames from 'classnames'; import { + ICON_SIDES, EuiFormControlLayout, } from '../form_control_layout'; @@ -61,7 +62,14 @@ EuiFieldText.propTypes = { id: PropTypes.string, placeholder: PropTypes.string, value: PropTypes.string, - icon: PropTypes.string, + icon: PropTypes.oneOfType([ + PropTypes.string, + PropTypes.shape({ + type: PropTypes.string, + side: PropTypes.oneOf(ICON_SIDES), + onClick: PropTypes.func, + }), + ]), isInvalid: PropTypes.bool, inputRef: PropTypes.func, fullWidth: PropTypes.bool, diff --git a/src/components/form/field_text/field_text.test.js b/src/components/form/field_text/field_text.test.js index d0b7367c851..1f04f7db058 100644 --- a/src/components/form/field_text/field_text.test.js +++ b/src/components/form/field_text/field_text.test.js @@ -4,7 +4,13 @@ import { requiredProps } from '../../../test/required_props'; import { EuiFieldText } from './field_text'; -jest.mock('../form_control_layout', () => ({ EuiFormControlLayout: 'eui-form-control-layout' })); +jest.mock('../form_control_layout', () => { + const formControlLayout = require.requireActual('../form_control_layout') + return { + ...formControlLayout, + EuiFormControlLayout: 'eui-form-control-layout', + } +}); jest.mock('../validatable_control', () => ({ EuiValidatableControl: 'eui-validatable-control' })); describe('EuiFieldText', () => { diff --git a/src/components/form/form_control_layout/__snapshots__/form_control_layout.test.js.snap b/src/components/form/form_control_layout/__snapshots__/form_control_layout.test.js.snap index a010abb1aaa..f6a6ac04bcb 100644 --- a/src/components/form/form_control_layout/__snapshots__/form_control_layout.test.js.snap +++ b/src/components/form/form_control_layout/__snapshots__/form_control_layout.test.js.snap @@ -2,19 +2,52 @@ exports[`EuiFormControlLayout is rendered 1`] = `
`; +exports[`EuiFormControlLayout props clear onClick is rendered 1`] = ` +
+ +
+`; + exports[`EuiFormControlLayout props fullWidth is rendered 1`] = `
`; -exports[`EuiFormControlLayout props icon is rendered 1`] = ` +exports[`EuiFormControlLayout props icon is rendered as a string 1`] = `
@@ -40,7 +73,34 @@ exports[`EuiFormControlLayout props icon is rendered 1`] = `
`; -exports[`EuiFormControlLayout props iconSide left is rendered 1`] = ` +exports[`EuiFormControlLayout props icon is rendered as an object 1`] = ` +
+ +
+`; + +exports[`EuiFormControlLayout props icon side left is rendered 1`] = `
@@ -66,7 +126,7 @@ exports[`EuiFormControlLayout props iconSide left is rendered 1`] = `
`; -exports[`EuiFormControlLayout props iconSide right is rendered 1`] = ` +exports[`EuiFormControlLayout props icon side right is rendered 1`] = `
diff --git a/src/components/form/form_control_layout/form_control_layout.js b/src/components/form/form_control_layout/form_control_layout.js index 88fef6766df..6146efb3005 100644 --- a/src/components/form/form_control_layout/form_control_layout.js +++ b/src/components/form/form_control_layout/form_control_layout.js @@ -15,13 +15,12 @@ export const ICON_SIDES = Object.keys(iconSideToClassNameMap); export const EuiFormControlLayout = ({ children, icon, + clear, fullWidth, - onClear, - iconSide, isLoading, - onIconClick, compressed, - className + className, + ...rest }) => { const classes = classNames( @@ -42,11 +41,26 @@ export const EuiFormControlLayout = ({ let optionalIcon; if (icon) { + // Normalize the icon to an object if it's a string. + const iconProps = typeof icon === 'string' ? { + type: icon, + } : icon; + + const { + className: iconClassName, + type: iconType, + side: iconSide = 'left', + onClick: onIconClick, + ref: iconRef, + ...iconRest + } = iconProps + const iconClasses = classNames( 'euiFormControlLayout__icon', iconSideToClassNameMap[iconSide], + iconClassName, { - 'euiFormControlLayout__iconButton': onIconClick + 'euiFormControlLayout__iconButton': onIconClick, }, ); @@ -55,9 +69,11 @@ export const EuiFormControlLayout = ({ ) @@ -66,18 +82,28 @@ export const EuiFormControlLayout = ({
`; @@ -53,7 +57,7 @@ exports[`EuiFormControlLayout props icon is rendered as a string 1`] = ` >
`; @@ -157,7 +165,11 @@ exports[`EuiFormControlLayout props isLoading is rendered 1`] = ` class="euiFormControlLayout" >
+ class="euiFormControlLayout__icons" + > +
+
`; diff --git a/src/components/form/form_control_layout/_form_control_layout.scss b/src/components/form/form_control_layout/_form_control_layout.scss index f5d241acac2..cb0b11f81d4 100644 --- a/src/components/form/form_control_layout/_form_control_layout.scss +++ b/src/components/form/form_control_layout/_form_control_layout.scss @@ -13,10 +13,13 @@ } .euiFormControlLayout__icon { + pointer-events: none; + } + + .euiFormControlLayout__icon--left { position: absolute; top: $euiFormControlPadding; left: $euiFormControlPadding; - pointer-events: none; @at-root { .euiFormControlLayout--compressed#{&} { @@ -25,7 +28,7 @@ } } - .euiFormControlLayout__iconButton { + .euiFormControlLayout__icon--button { pointer-events: all; height: $euiSize; width: $euiSize; @@ -44,61 +47,22 @@ border-radius: 100%; color: $euiColorGhost; } - - @at-root { - .euiFormControlLayout--compressed#{&} { - top: $euiFormControlPadding--compressed; - } - } } - .euiFormControlLayout__clear { + .euiFormControlLayout__icons { position: absolute; - top: $euiFormControlPadding; - right: $euiFormControlPadding; - - @include euiFormControlLayout__clear('.euiFormControlLayout__clearIcon'); - - @at-root { - .euiFormControlLayout--compressed#{&} { - top: $euiFormControlPadding--compressed + 2px; - } - } - - // Loading spinner needs adjustment if clear also exists - ~ .euiFormControlLayout__loading { - right: $euiFormControlPadding*3; - } - } - - .euiFormControlLayout__icon--right { - left: auto; right: $euiFormControlPadding; + top: 0; + bottom: 0; + display: flex; + align-items: center; - // Loading spinner needs adjustment if icon also exists like selects. - ~ .euiFormControlLayout__loading { - right: $euiFormControlPadding*3; - } - - // Same with clear buttons - ~ .euiFormControlLayout__clear { - right: $euiFormControlPadding*3; - - ~ .euiFormControlLayout__loading { - right: $euiFormControlPadding*5; - } + > * + * { + margin-left: $euiFormControlPadding; } } - .euiFormControlLayout__loading { - position: absolute; - top: $euiFormControlPadding; - right: $euiFormControlPadding; - - @at-root { - .euiFormControlLayout--compressed#{&} { - top: $euiFormControlPadding--compressed + 1px; - } - } + .euiFormControlLayout__clear { + @include euiFormControlLayout__clear('.euiFormControlLayout__clearIcon'); } } diff --git a/src/components/form/form_control_layout/form_control_layout.js b/src/components/form/form_control_layout/form_control_layout.js index 6146efb3005..eb0d7f1ac51 100644 --- a/src/components/form/form_control_layout/form_control_layout.js +++ b/src/components/form/form_control_layout/form_control_layout.js @@ -1,4 +1,4 @@ -import React from 'react'; +import React, { Fragment, Component } from 'react'; import PropTypes from 'prop-types'; import classNames from 'classnames'; @@ -6,122 +6,163 @@ import { EuiIcon } from '../../icon'; import { EuiLoadingSpinner } from '../../loading'; const iconSideToClassNameMap = { - left: '', - right: 'euiFormControlLayout__icon--right', + left: 'euiFormControlLayout__icon--left', + right: '', }; export const ICON_SIDES = Object.keys(iconSideToClassNameMap); -export const EuiFormControlLayout = ({ - children, - icon, - clear, - fullWidth, - isLoading, - compressed, - className, - ...rest -}) => { - - const classes = classNames( - 'euiFormControlLayout', - { - 'euiFormControlLayout--fullWidth': fullWidth, - 'euiFormControlLayout--compressed': compressed, - }, - className - ); - - let optionalLoader; - if (isLoading) { - optionalLoader = ( - - ); - } - - let optionalIcon; - if (icon) { - // Normalize the icon to an object if it's a string. - const iconProps = typeof icon === 'string' ? { - type: icon, - } : icon; - +export class EuiFormControlLayout extends Component { + render() { const { - className: iconClassName, - type: iconType, - side: iconSide = 'left', - onClick: onIconClick, - ref: iconRef, - ...iconRest - } = iconProps - - const iconClasses = classNames( - 'euiFormControlLayout__icon', - iconSideToClassNameMap[iconSide], - iconClassName, + children, + icon, // eslint-disable-line no-unused-vars + clear, // eslint-disable-line no-unused-vars + fullWidth, + isLoading, // eslint-disable-line no-unused-vars + compressed, + className, + ...rest + } = this.props; + + const classes = classNames( + 'euiFormControlLayout', { - 'euiFormControlLayout__iconButton': onIconClick, + 'euiFormControlLayout--fullWidth': fullWidth, + 'euiFormControlLayout--compressed': compressed, }, + className ); - if (onIconClick) { - optionalIcon = ( + return ( +
+ {children} + {this.renderIcons()} +
+ ); + } + + renderIcons() { + const { + isLoading, + icon, + clear, + } = this.props; + + let optionalLoader; + if (isLoading) { + optionalLoader = ( + + ); + } + + let optionalIconSide; + let optionalIcon; + if (icon) { + // Normalize the icon to an object if it's a string. + const iconProps = typeof icon === 'string' ? { + type: icon, + } : icon; + + const { + className: iconClassName, + type: iconType, + side: iconSide = 'left', + onClick: onIconClick, + ref: iconRef, + ...iconRest + } = iconProps + + optionalIconSide = iconSide + + const iconClasses = classNames( + 'euiFormControlLayout__icon', + iconSideToClassNameMap[iconSide], + iconClassName, + { + 'euiFormControlLayout__icon--button': onIconClick, + }, + ); + + if (onIconClick) { + optionalIcon = ( + + ) + } else { + optionalIcon = ( +