UNPKG

@elastic/eui

Version:

Elastic UI Component Library

279 lines (275 loc) 14.3 kB
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);