@elastic/eui
Version:
Elastic UI Component Library
279 lines (275 loc) • 14.3 kB
JavaScript
var _excluded = ["stylesMemoizer", "className", "id", "name", "placeholder", "isInvalid", "disabled", "fullWidth", "isLoading", "inputRef", "incremental", "compressed", "onSearch", "isClearable", "append", "prepend"];
function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
function _extends() { return _extends = Object.assign ? Object.assign.bind() : function (n) { for (var e = 1; e < arguments.length; e++) { var t = arguments[e]; for (var r in t) ({}).hasOwnProperty.call(t, r) && (n[r] = t[r]); } return n; }, _extends.apply(null, arguments); }
function _objectWithoutProperties(e, t) { if (null == e) return {}; var o, r, i = _objectWithoutPropertiesLoose(e, t); if (Object.getOwnPropertySymbols) { var n = Object.getOwnPropertySymbols(e); for (r = 0; r < n.length; r++) o = n[r], t.indexOf(o) >= 0 || {}.propertyIsEnumerable.call(e, o) && (i[o] = e[o]); } return i; }
function _objectWithoutPropertiesLoose(r, e) { if (null == r) return {}; var t = {}; for (var n in r) if ({}.hasOwnProperty.call(r, n)) { if (e.indexOf(n) >= 0) continue; t[n] = r[n]; } return t; }
function _classCallCheck(a, n) { if (!(a instanceof n)) throw new TypeError("Cannot call a class as a function"); }
function _defineProperties(e, r) { for (var t = 0; t < r.length; t++) { var o = r[t]; o.enumerable = o.enumerable || !1, o.configurable = !0, "value" in o && (o.writable = !0), Object.defineProperty(e, _toPropertyKey(o.key), o); } }
function _createClass(e, r, t) { return r && _defineProperties(e.prototype, r), t && _defineProperties(e, t), Object.defineProperty(e, "prototype", { writable: !1 }), e; }
function _callSuper(t, o, e) { return o = _getPrototypeOf(o), _possibleConstructorReturn(t, _isNativeReflectConstruct() ? Reflect.construct(o, e || [], _getPrototypeOf(t).constructor) : o.apply(t, e)); }
function _possibleConstructorReturn(t, e) { if (e && ("object" == _typeof(e) || "function" == typeof e)) return e; if (void 0 !== e) throw new TypeError("Derived constructors may only return object or undefined"); return _assertThisInitialized(t); }
function _assertThisInitialized(e) { if (void 0 === e) throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); return e; }
function _isNativeReflectConstruct() { try { var t = !Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); } catch (t) {} return (_isNativeReflectConstruct = function _isNativeReflectConstruct() { return !!t; })(); }
function _getPrototypeOf(t) { return _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf.bind() : function (t) { return t.__proto__ || Object.getPrototypeOf(t); }, _getPrototypeOf(t); }
function _inherits(t, e) { if ("function" != typeof e && null !== e) throw new TypeError("Super expression must either be null or a function"); t.prototype = Object.create(e && e.prototype, { constructor: { value: t, writable: !0, configurable: !0 } }), Object.defineProperty(t, "prototype", { writable: !1 }), e && _setPrototypeOf(t, e); }
function _setPrototypeOf(t, e) { return _setPrototypeOf = Object.setPrototypeOf ? Object.setPrototypeOf.bind() : function (t, e) { return t.__proto__ = e, t; }, _setPrototypeOf(t, e); }
function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == _typeof(i) ? i : i + ""; }
function _toPrimitive(t, r) { if ("object" != _typeof(t) || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != _typeof(i)) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0 and the Server Side Public License, v 1; you may not use this file except
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/
import React, { Component } from 'react';
import PropTypes from "prop-types";
import classNames from 'classnames';
import { keys, withEuiStylesMemoizer } from '../../../services';
import { Browser } from '../../../services/browser';
import { EuiI18n } from '../../i18n';
import { EuiFormControlLayout } from '../form_control_layout';
import { EuiValidatableControl } from '../validatable_control';
import { FormContext } from '../eui_form_context';
import { euiFieldSearchStyles } from './field_search.styles';
import { jsx as ___EmotionJSX } from "@emotion/react";
var isSearchSupported = false;
export var EuiFieldSearchClass = /*#__PURE__*/function (_Component) {
function EuiFieldSearchClass() {
var _this;
_classCallCheck(this, EuiFieldSearchClass);
for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
args[_key] = arguments[_key];
}
_this = _callSuper(this, EuiFieldSearchClass, [].concat(args));
_defineProperty(_this, "state", {
value: _this.props.value || String(_this.props.defaultValue || '')
});
_defineProperty(_this, "inputElement", null);
_defineProperty(_this, "cleanups", []);
_defineProperty(_this, "onClear", function () {
// clear the field's value
// 1. React doesn't listen for `change` events, instead it maps `input` events to `change`
// 2. React only fires the mapped `change` event if the element's value has changed
// 3. An input's value is, in addition to other methods, tracked by intercepting element.value = '...'
//
// So we have to go below the element's value setter to avoid React intercepting it,
// only then will React treat the value as different and fire its `change` event
//
// https://stackoverflow.com/questions/23892547/what-is-the-best-way-to-trigger-onchange-event-in-react-js
var nativeInputValue = Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, 'value');
var nativeInputValueSetter = nativeInputValue ? nativeInputValue.set : undefined;
if (nativeInputValueSetter) {
nativeInputValueSetter.call(_this.inputElement, '');
}
// dispatch input event
var event = new Event('input', {
bubbles: true,
cancelable: false
});
if (_this.inputElement) {
_this.inputElement.dispatchEvent(event);
// set focus on the search field
_this.inputElement.focus();
_this.inputElement.dispatchEvent(new Event('change'));
}
_this.setState({
value: ''
});
var _this$props = _this.props,
incremental = _this$props.incremental,
onSearch = _this$props.onSearch;
if (onSearch && incremental) {
onSearch('');
}
});
_defineProperty(_this, "setRef", function (inputElement) {
_this.inputElement = inputElement;
if (_this.props.inputRef) {
_this.props.inputRef(inputElement);
}
});
_defineProperty(_this, "onKeyUp", function (event, incremental, onSearch) {
_this.setState({
value: event.target.value
});
if (_this.props.onKeyUp) {
_this.props.onKeyUp(event);
if (event.defaultPrevented) {
return;
}
}
if (onSearch && (event.key !== keys.ENTER && incremental || event.key === keys.ENTER && !isSearchSupported)) {
onSearch(event.target.value);
}
});
return _this;
}
_inherits(EuiFieldSearchClass, _Component);
return _createClass(EuiFieldSearchClass, [{
key: "componentDidMount",
value: function componentDidMount() {
var _this2 = this;
if (!this.inputElement) return;
isSearchSupported = Browser.isEventSupported('search', this.inputElement);
if (isSearchSupported) {
var onSearch = function onSearch(event) {
if (_this2.props.onSearch) {
if (!event || !event.target || event.defaultPrevented) return;
_this2.props.onSearch(event.target.value);
}
};
this.inputElement.addEventListener('search', onSearch);
this.cleanups.push(function () {
if (!_this2.inputElement) return;
_this2.inputElement.removeEventListener('search', onSearch);
});
}
var onChange = function onChange(event) {
if (event.target && event.target.value !== _this2.state.value) {
_this2.setState({
value: event.target.value
});
if (_this2.props.onSearch) {
_this2.props.onSearch(event.target.value);
}
}
};
this.inputElement.addEventListener('change', onChange);
}
}, {
key: "componentWillUnmount",
value: function componentWillUnmount() {
this.cleanups.forEach(function (cleanup) {
return cleanup();
});
}
}, {
key: "render",
value: function render() {
var _this3 = this;
var _ref = this.context,
defaultFullWidth = _ref.defaultFullWidth;
var _this$props2 = this.props,
stylesMemoizer = _this$props2.stylesMemoizer,
className = _this$props2.className,
id = _this$props2.id,
name = _this$props2.name,
placeholder = _this$props2.placeholder,
isInvalid = _this$props2.isInvalid,
disabled = _this$props2.disabled,
_this$props2$fullWidt = _this$props2.fullWidth,
fullWidth = _this$props2$fullWidt === void 0 ? defaultFullWidth : _this$props2$fullWidt,
isLoading = _this$props2.isLoading,
inputRef = _this$props2.inputRef,
incremental = _this$props2.incremental,
compressed = _this$props2.compressed,
onSearch = _this$props2.onSearch,
_isClearable = _this$props2.isClearable,
append = _this$props2.append,
prepend = _this$props2.prepend,
rest = _objectWithoutProperties(_this$props2, _excluded);
var value = this.props.value;
if (typeof this.props.value !== 'string') value = this.state.value;
// Set actual value of isClearable if value exists as well
var isClearable = Boolean(_isClearable && value && !rest.readOnly && !disabled);
var classes = classNames('euiFieldSearch', {
'euiFieldSearch-isLoading': isLoading,
'euiFieldSearch-isClearable': isClearable,
'euiFieldSearch-isInvalid': isInvalid
}, className);
var styles = stylesMemoizer(euiFieldSearchStyles);
var cssStyles = [styles.euiFieldSearch, compressed ? styles.compressed : styles.uncompressed, fullWidth ? styles.fullWidth : styles.formWidth, (prepend || append) && styles.inGroup];
return ___EmotionJSX(EuiI18n, {
token: "euiFieldSearch.clearSearchButtonLabel",
default: "Clear search input"
}, function (clearSearchButtonLabel) {
return ___EmotionJSX(EuiFormControlLayout, {
icon: "magnify",
fullWidth: fullWidth,
isLoading: isLoading,
isInvalid: isInvalid,
isDisabled: disabled,
clear: isClearable ? {
onClick: _this3.onClear,
'aria-label': clearSearchButtonLabel,
'data-test-subj': 'clearSearchButton'
} : undefined,
compressed: compressed,
append: append,
prepend: prepend
}, ___EmotionJSX(EuiValidatableControl, {
isInvalid: isInvalid
}, ___EmotionJSX("input", _extends({
type: "search",
id: id,
name: name,
placeholder: placeholder,
className: classes,
css: cssStyles,
onKeyUp: function onKeyUp(e) {
return _this3.onKeyUp(e, incremental, onSearch);
},
disabled: disabled,
ref: _this3.setRef
}, rest))));
});
}
}]);
}(Component);
_defineProperty(EuiFieldSearchClass, "contextType", FormContext);
_defineProperty(EuiFieldSearchClass, "defaultProps", {
isLoading: false,
incremental: false,
compressed: false,
isClearable: true
});
EuiFieldSearchClass.propTypes = {
name: PropTypes.string,
id: PropTypes.string,
placeholder: PropTypes.string,
value: PropTypes.string,
isInvalid: PropTypes.bool,
/**
* Expand to fill 100% of the parent.
* Defaults to `fullWidth` prop of `<EuiForm>`.
* @default false
*/
fullWidth: PropTypes.bool,
isLoading: PropTypes.bool,
/**
* Called when the user presses [Enter] OR on change if the incremental prop is `true`.
* If you don't need the on[Enter] functionality, prefer using onChange
*/
onSearch: PropTypes.func,
/**
* When `true` the search will be executed (that is, the `onSearch` will be called) as the
* user types.
*/
incremental: PropTypes.bool,
/**
* when `true` creates a shorter height input
*/
compressed: PropTypes.bool,
inputRef: PropTypes.func,
/**
* Shows a button that quickly clears any input
*/
isClearable: PropTypes.bool,
/**
* Creates an input group with element(s) coming before input
* `string` | `ReactElement` or an array of these
*/
prepend: PropTypes.oneOfType([PropTypes.oneOfType([PropTypes.string.isRequired, PropTypes.element.isRequired]).isRequired, PropTypes.arrayOf(PropTypes.oneOfType([PropTypes.string.isRequired, PropTypes.element.isRequired]).isRequired).isRequired]),
/**
* Creates an input group with element(s) coming after input.
* `string` | `ReactElement` or an array of these
*/
append: PropTypes.oneOfType([PropTypes.oneOfType([PropTypes.string.isRequired, PropTypes.element.isRequired]).isRequired, PropTypes.arrayOf(PropTypes.oneOfType([PropTypes.string.isRequired, PropTypes.element.isRequired]).isRequired).isRequired]),
className: PropTypes.string,
"aria-label": PropTypes.string,
"data-test-subj": PropTypes.string,
css: PropTypes.any
};
export var EuiFieldSearch = withEuiStylesMemoizer(EuiFieldSearchClass);