react-dropdown-select
Advanced tools
Comparing version 1.3.2 to 1.3.3
@@ -29,2 +29,3 @@ "use strict"; | ||
dropdownGap: parentProps.dropdownGap, | ||
dropdownHeight: parentProps.dropdownHeight, | ||
className: "react-dropdown-select-dropdown" | ||
@@ -60,4 +61,7 @@ }, parentProps.dropdownRenderer ? parentProps.dropdownRenderer(parentProps, parentState, parentMethods) : _react.default.createElement(_react.default.Fragment, null, parentMethods.searchResults().length === 0 ? _react.default.createElement(_NoData.default, { | ||
return selectBounds.width; | ||
}, "px;padding:0;display:flex;flex-direction:column;background:#fff;border-radius:2px;box-shadow:0 0 10px 0 #0000003b;max-height:300px;overflow:auto;:focus{outline:none;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb21wb25lbnRzL0Ryb3Bkb3duLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQStDMkIiLCJmaWxlIjoiLi4vLi4vc3JjL2NvbXBvbmVudHMvRHJvcGRvd24uanMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuXG5pbXBvcnQgTm9EYXRhIGZyb20gJy4vTm9EYXRhJztcbmltcG9ydCBJdGVtIGZyb20gJy4vSXRlbSc7XG5cbmNvbnN0IERyb3Bkb3duID0gKHsgcGFyZW50UHJvcHMsIHBhcmVudFN0YXRlLCBwYXJlbnRNZXRob2RzIH0pID0+IChcbiAgPERyb3BEb3duXG4gICAgdGFiSW5kZXg9XCItMVwiXG4gICAgYXJpYS1leHBhbmRlZD1cInRydWVcIlxuICAgIHJvbGU9XCJsaXN0XCJcbiAgICBvcGVuT25Ub3A9e3BhcmVudFByb3BzLm9wZW5PblRvcH1cbiAgICBzZWxlY3RCb3VuZHM9e3BhcmVudFN0YXRlLnNlbGVjdEJvdW5kc31cbiAgICBkcm9wZG93bkdhcD17cGFyZW50UHJvcHMuZHJvcGRvd25HYXB9XG4gICAgY2xhc3NOYW1lPVwicmVhY3QtZHJvcGRvd24tc2VsZWN0LWRyb3Bkb3duXCI+XG4gICAge3BhcmVudFByb3BzLmRyb3Bkb3duUmVuZGVyZXIgPyAoXG4gICAgICBwYXJlbnRQcm9wcy5kcm9wZG93blJlbmRlcmVyKHBhcmVudFByb3BzLCBwYXJlbnRTdGF0ZSwgcGFyZW50TWV0aG9kcylcbiAgICApIDogKFxuICAgICAgPFJlYWN0LkZyYWdtZW50PlxuICAgICAgICB7cGFyZW50TWV0aG9kcy5zZWFyY2hSZXN1bHRzKCkubGVuZ3RoID09PSAwID8gKFxuICAgICAgICAgIDxOb0RhdGFcbiAgICAgICAgICAgIGNsYXNzTmFtZT1cInJlYWN0LWRyb3Bkb3duLXNlbGVjdC1uby1kYXRhXCJcbiAgICAgICAgICAgIHBhcmVudFN0YXRlPXtwYXJlbnRTdGF0ZX1cbiAgICAgICAgICAgIHBhcmVudFByb3BzPXtwYXJlbnRQcm9wc31cbiAgICAgICAgICAgIHBhcmVudE1ldGhvZHM9e3BhcmVudE1ldGhvZHN9XG4gICAgICAgICAgLz5cbiAgICAgICAgKSA6IChcbiAgICAgICAgICBwYXJlbnRNZXRob2RzXG4gICAgICAgICAgICAuc2VhcmNoUmVzdWx0cygpXG4gICAgICAgICAgICAubWFwKChpdGVtLCBpdGVtSW5kZXgpID0+IChcbiAgICAgICAgICAgICAgPEl0ZW1cbiAgICAgICAgICAgICAgICBrZXk9e2l0ZW1bcGFyZW50UHJvcHMudmFsdWVGaWVsZF19XG4gICAgICAgICAgICAgICAgaXRlbT17aXRlbX1cbiAgICAgICAgICAgICAgICBpdGVtSW5kZXg9e2l0ZW1JbmRleH1cbiAgICAgICAgICAgICAgICBwYXJlbnRTdGF0ZT17cGFyZW50U3RhdGV9XG4gICAgICAgICAgICAgICAgcGFyZW50UHJvcHM9e3BhcmVudFByb3BzfVxuICAgICAgICAgICAgICAgIHBhcmVudE1ldGhvZHM9e3BhcmVudE1ldGhvZHN9XG4gICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICApKVxuICAgICAgICApfVxuICAgICAgPC9SZWFjdC5GcmFnbWVudD5cbiAgICApfVxuICA8L0Ryb3BEb3duPlxuKTtcblxuRHJvcGRvd24ucHJvcFR5cGVzID0ge307XG5cbmNvbnN0IERyb3BEb3duID0gc3R5bGVkLmRpdmBcbiAgcG9zaXRpb246IGFic29sdXRlO1xuICAkeyh7IHNlbGVjdEJvdW5kcywgZHJvcGRvd25HYXAsIG9wZW5PblRvcH0pID0+IG9wZW5PblRvcCBcbiAgPyBgYm90dG9tOiAke3NlbGVjdEJvdW5kcy5oZWlnaHQgKyAyICsgZHJvcGRvd25HYXB9cHhgIFxuICA6IGB0b3A6ICR7c2VsZWN0Qm91bmRzLmhlaWdodCArIDIgKyBkcm9wZG93bkdhcH1weGB9O1xuICBsZWZ0OiAwO1xuICBib3JkZXI6IDFweCBzb2xpZCAjY2NjO1xuICB3aWR0aDogJHsoeyBzZWxlY3RCb3VuZHMgfSkgPT4gc2VsZWN0Qm91bmRzLndpZHRofXB4O1xuICBwYWRkaW5nOiAwO1xuICBkaXNwbGF5OiBmbGV4O1xuICBmbGV4LWRpcmVjdGlvbjogY29sdW1uO1xuICBiYWNrZ3JvdW5kOiAjZmZmO1xuICBib3JkZXItcmFkaXVzOiAycHg7XG4gIGJveC1zaGFkb3c6IDAgMCAxMHB4IDAgIzAwMDAwMDNiO1xuICBtYXgtaGVpZ2h0OiAzMDBweDtcbiAgb3ZlcmZsb3c6IGF1dG87XG4gIFxuICA6Zm9jdXMge1xuICAgIG91dGxpbmU6IG5vbmU7XG4gIH1cbn1cbmA7XG5cbmV4cG9ydCBkZWZhdWx0IERyb3Bkb3duO1xuIl19 */")); | ||
}, "px;padding:0;display:flex;flex-direction:column;background:#fff;border-radius:2px;box-shadow:0 0 10px 0 #0000003b;max-height:", function (_ref4) { | ||
var dropdownHeight = _ref4.dropdownHeight; | ||
return dropdownHeight; | ||
}, ";overflow:auto;:focus{outline:none;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb21wb25lbnRzL0Ryb3Bkb3duLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQWdEMkIiLCJmaWxlIjoiLi4vLi4vc3JjL2NvbXBvbmVudHMvRHJvcGRvd24uanMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuXG5pbXBvcnQgTm9EYXRhIGZyb20gJy4vTm9EYXRhJztcbmltcG9ydCBJdGVtIGZyb20gJy4vSXRlbSc7XG5cbmNvbnN0IERyb3Bkb3duID0gKHsgcGFyZW50UHJvcHMsIHBhcmVudFN0YXRlLCBwYXJlbnRNZXRob2RzIH0pID0+IChcbiAgPERyb3BEb3duXG4gICAgdGFiSW5kZXg9XCItMVwiXG4gICAgYXJpYS1leHBhbmRlZD1cInRydWVcIlxuICAgIHJvbGU9XCJsaXN0XCJcbiAgICBvcGVuT25Ub3A9e3BhcmVudFByb3BzLm9wZW5PblRvcH1cbiAgICBzZWxlY3RCb3VuZHM9e3BhcmVudFN0YXRlLnNlbGVjdEJvdW5kc31cbiAgICBkcm9wZG93bkdhcD17cGFyZW50UHJvcHMuZHJvcGRvd25HYXB9XG4gICAgZHJvcGRvd25IZWlnaHQ9e3BhcmVudFByb3BzLmRyb3Bkb3duSGVpZ2h0fVxuICAgIGNsYXNzTmFtZT1cInJlYWN0LWRyb3Bkb3duLXNlbGVjdC1kcm9wZG93blwiPlxuICAgIHtwYXJlbnRQcm9wcy5kcm9wZG93blJlbmRlcmVyID8gKFxuICAgICAgcGFyZW50UHJvcHMuZHJvcGRvd25SZW5kZXJlcihwYXJlbnRQcm9wcywgcGFyZW50U3RhdGUsIHBhcmVudE1ldGhvZHMpXG4gICAgKSA6IChcbiAgICAgIDxSZWFjdC5GcmFnbWVudD5cbiAgICAgICAge3BhcmVudE1ldGhvZHMuc2VhcmNoUmVzdWx0cygpLmxlbmd0aCA9PT0gMCA/IChcbiAgICAgICAgICA8Tm9EYXRhXG4gICAgICAgICAgICBjbGFzc05hbWU9XCJyZWFjdC1kcm9wZG93bi1zZWxlY3Qtbm8tZGF0YVwiXG4gICAgICAgICAgICBwYXJlbnRTdGF0ZT17cGFyZW50U3RhdGV9XG4gICAgICAgICAgICBwYXJlbnRQcm9wcz17cGFyZW50UHJvcHN9XG4gICAgICAgICAgICBwYXJlbnRNZXRob2RzPXtwYXJlbnRNZXRob2RzfVxuICAgICAgICAgIC8+XG4gICAgICAgICkgOiAoXG4gICAgICAgICAgcGFyZW50TWV0aG9kc1xuICAgICAgICAgICAgLnNlYXJjaFJlc3VsdHMoKVxuICAgICAgICAgICAgLm1hcCgoaXRlbSwgaXRlbUluZGV4KSA9PiAoXG4gICAgICAgICAgICAgIDxJdGVtXG4gICAgICAgICAgICAgICAga2V5PXtpdGVtW3BhcmVudFByb3BzLnZhbHVlRmllbGRdfVxuICAgICAgICAgICAgICAgIGl0ZW09e2l0ZW19XG4gICAgICAgICAgICAgICAgaXRlbUluZGV4PXtpdGVtSW5kZXh9XG4gICAgICAgICAgICAgICAgcGFyZW50U3RhdGU9e3BhcmVudFN0YXRlfVxuICAgICAgICAgICAgICAgIHBhcmVudFByb3BzPXtwYXJlbnRQcm9wc31cbiAgICAgICAgICAgICAgICBwYXJlbnRNZXRob2RzPXtwYXJlbnRNZXRob2RzfVxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgKSlcbiAgICAgICAgKX1cbiAgICAgIDwvUmVhY3QuRnJhZ21lbnQ+XG4gICAgKX1cbiAgPC9Ecm9wRG93bj5cbik7XG5cbkRyb3Bkb3duLnByb3BUeXBlcyA9IHt9O1xuXG5jb25zdCBEcm9wRG93biA9IHN0eWxlZC5kaXZgXG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgJHsoeyBzZWxlY3RCb3VuZHMsIGRyb3Bkb3duR2FwLCBvcGVuT25Ub3B9KSA9PiBvcGVuT25Ub3AgXG4gID8gYGJvdHRvbTogJHtzZWxlY3RCb3VuZHMuaGVpZ2h0ICsgMiArIGRyb3Bkb3duR2FwfXB4YCBcbiAgOiBgdG9wOiAke3NlbGVjdEJvdW5kcy5oZWlnaHQgKyAyICsgZHJvcGRvd25HYXB9cHhgfTtcbiAgbGVmdDogMDtcbiAgYm9yZGVyOiAxcHggc29saWQgI2NjYztcbiAgd2lkdGg6ICR7KHsgc2VsZWN0Qm91bmRzIH0pID0+IHNlbGVjdEJvdW5kcy53aWR0aH1weDtcbiAgcGFkZGluZzogMDtcbiAgZGlzcGxheTogZmxleDtcbiAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbiAgYmFja2dyb3VuZDogI2ZmZjtcbiAgYm9yZGVyLXJhZGl1czogMnB4O1xuICBib3gtc2hhZG93OiAwIDAgMTBweCAwICMwMDAwMDAzYjtcbiAgbWF4LWhlaWdodDogJHsoeyBkcm9wZG93bkhlaWdodCB9KSA9PiBkcm9wZG93bkhlaWdodH07XG4gIG92ZXJmbG93OiBhdXRvO1xuICBcbiAgOmZvY3VzIHtcbiAgICBvdXRsaW5lOiBub25lO1xuICB9XG59XG5gO1xuXG5leHBvcnQgZGVmYXVsdCBEcm9wZG93bjtcbiJdfQ== */")); | ||
var _default = Dropdown; | ||
exports.default = _default; |
@@ -467,2 +467,3 @@ "use strict"; | ||
openOnTop: false, | ||
dropdownHeight: '300px', | ||
onDropdownOpen: function onDropdownOpen() { | ||
@@ -493,4 +494,4 @@ return undefined; | ||
return color; | ||
}, "3c;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/index.js"],"names":[],"mappings":"AAyXsC","file":"../src/index.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport styled from '@emotion/styled';\nimport ClickOutHandler from 'react-onclickout';\n\nimport Content from './components/Content';\nimport Dropdown from './components/Dropdown';\nimport Loading from './components/Loading';\nimport Clear from './components/Clear';\nimport Separator from './components/Separator';\nimport DropdownHandle from './components/DropdownHandle';\n\nconst debounce = (fn, delay = 0) => {\n  let timerId;\n\n  return (...args) => {\n    if (timerId) {\n      clearTimeout(timerId);\n    }\n    timerId = setTimeout(() => {\n      fn(...args);\n      timerId = null;\n    }, delay);\n  };\n};\n\nexport class Select extends React.Component {\n  static propTypes = {\n    onChange: PropTypes.func.isRequired,\n    onDropdownClose: PropTypes.func,\n    onDropdownOpen: PropTypes.func,\n    onClearAll: PropTypes.func,\n    onSelectAll: PropTypes.func,\n    values: PropTypes.array,\n    options: PropTypes.array.isRequired,\n    keepOpen: PropTypes.bool,\n    dropdownGap: PropTypes.number,\n    multi: PropTypes.bool,\n    placeholder: PropTypes.string,\n    addPlaceholder: PropTypes.string,\n    disabled: PropTypes.bool,\n    className: PropTypes.string,\n    loading: PropTypes.bool,\n    clearable: PropTypes.bool,\n    separator: PropTypes.bool,\n    dropdownHandle: PropTypes.bool,\n    searchBy: PropTypes.string,\n    closeOnScroll: PropTypes.bool,\n    openOnTop: PropTypes.bool,\n    style: PropTypes.object,\n    contentRenderer: PropTypes.func,\n    dropdownRenderer: PropTypes.func,\n    itemRenderer: PropTypes.func,\n    noDataRenderer: PropTypes.func,\n    optionRenderer: PropTypes.func,\n    inputRenderer: PropTypes.func,\n    loadingRenderer: PropTypes.func,\n    clearRenderer: PropTypes.func,\n    separatorRenderer: PropTypes.func,\n    dropdownHandleRenderer: PropTypes.func\n  };\n\n  constructor(props) {\n    super(props);\n\n    this.state = {\n      dropdown: false,\n      values: props.values,\n      search: '',\n      selectBounds: {}\n    };\n\n    this.methods = {\n      removeItem: this.removeItem,\n      dropDown: this.dropDown,\n      addItem: this.addItem,\n      setSearch: this.setSearch,\n      getInputSize: this.getInputSize,\n      toggleSelectAll: this.toggleSelectAll,\n      clearAll: this.clearAll,\n      selectAll: this.selectAll,\n      searchResults: this.searchResults,\n      getSelectRef: this.getSelectRef,\n      isSelected: this.isSelected,\n      getSelectBounds: this.getSelectBounds,\n      areAllSelected: this.areAllSelected,\n      handleKeyDown: this.handleKeyDown,\n      activeCursorItem: this.activeCursorItem\n    };\n\n    this.select = React.createRef();\n  }\n\n  componentDidMount() {\n    window.addEventListener('resize', debounce(this.updateSelectBounds));\n    window.addEventListener('scroll', debounce(this.onScroll));\n\n    this.props.onChange(this.state.values);\n\n    this.dropDown('close');\n\n    if (this.select) {\n      this.updateSelectBounds();\n    }\n  }\n\n  componentDidUpdate(prevProps, prevState) {\n    if (prevState.values !== this.state.values) {\n      this.props.onChange(this.state.values);\n      this.updateSelectBounds();\n    }\n\n    if (prevState.values !== this.state.values && this.props.closeOnSelect) {\n      this.dropDown('close');\n    }\n\n    if (prevProps.multi !== this.props.multi) {\n      this.updateSelectBounds();\n    }\n\n    if (prevState.dropdown && prevState.dropdown !== this.state.dropdown) {\n      this.onDropdownClose();\n    }\n\n    if (!prevState.dropdown && prevState.dropdown !== this.state.dropdown) {\n      this.props.onDropdownOpen();\n    }\n  }\n\n  componentWillUnmount() {\n    window.removeEventListener(\n      'resize',\n      debounce(this.updateSelectBounds, this.props.debounceDelay)\n    );\n    window.removeEventListener('scroll', debounce(this.onScroll, this.props.debounceDelay));\n  }\n\n  onDropdownClose = () => {\n    this.setState({ cursor: null });\n    this.props.onDropdownClose();\n  };\n\n  onScroll = () => {\n    if (this.props.closeOnScroll) {\n      this.dropDown('close');\n    }\n\n    this.updateSelectBounds();\n  };\n\n  updateSelectBounds = () =>\n    this.select.current &&\n    this.setState({\n      selectBounds: this.select.current.getBoundingClientRect()\n    });\n\n  getSelectBounds = () => this.state.selectBounds;\n\n  dropDown = (action = 'toggle') => {\n    if (this.props.keepOpen) {\n      return this.setState({ dropdown: true });\n    }\n\n    if (action === 'close') {\n      this.select.current.blur();\n      return this.setState({ dropdown: false, search: '' });\n    }\n\n    if (action === 'open') {\n      return this.setState({ dropdown: true });\n    }\n\n    if (action === 'toggle') {\n      this.select.current.focus();\n      return this.setState({ dropdown: !this.state.dropdown });\n    }\n\n    return false;\n  };\n\n  getSelectRef = () => this.select.current;\n\n  addItem = (item) => {\n    if (this.props.multi) {\n      if (this.state.values.indexOf(item) !== -1) {\n        return this.removeItem(null, item, false);\n      }\n\n      this.setState({\n        values: [...this.state.values, item]\n      });\n    } else {\n      this.setState({\n        values: [item],\n        dropdown: false,\n        search: ''\n      });\n    }\n\n    return true;\n  };\n\n  removeItem = (event, item, close = false) => {\n    if (event && close) {\n      event.preventDefault();\n      event.stopPropagation();\n      this.dropDown('close');\n    }\n\n    this.setState({\n      values: this.state.values.filter(\n        (values) => values[this.props.valueField] !== item[this.props.valueField]\n      )\n    });\n  };\n\n  setSearch = (event) => {\n    this.setState({\n      cursor: null\n    });\n\n    this.setState({\n      search: event.target.value\n    });\n  };\n\n  getInputSize = () => {\n    if (this.state.search) {\n      return this.state.search.length;\n    }\n\n    if (this.state.values.length > 0) {\n      return this.props.addPlaceholder.length;\n    }\n\n    return this.props.placeholder.length;\n  };\n\n  toggleSelectAll = () => {\n    return this.setState({\n      values: this.state.values.length === 0 ? this.selectAll() : this.clearAll()\n    });\n  };\n\n  clearAll = () => {\n    this.props.onClearAll();\n    this.setState({\n      values: []\n    });\n  };\n\n  selectAll = () => {\n    this.props.onSelectAll();\n    return this.setState({\n      values: this.props.options.filter((option) => !option.disabled)\n    });\n  };\n\n  isSelected = (option) => this.state.values.indexOf(option) !== -1;\n\n  areAllSelected = () =>\n    this.state.values.length === this.props.options.filter((option) => !option.disabled).length;\n\n  searchResults = () => {\n    const regexp = new RegExp(this.state.search, 'i');\n\n    return this.props.options.filter((item) =>\n      regexp.test(item[this.props.searchBy] || item[[this.props.labelField]])\n    );\n  };\n\n  activeCursorItem = (activeCursorItem) =>\n    this.setState({\n      activeCursorItem\n    });\n\n  handleKeyDown = (event) => {\n    const { cursor } = this.state;\n\n    if (event.key === 'Escape') {\n      this.dropDown('close');\n    }\n\n    if (event.key === 'Enter') {\n      !this.state.activeCursorItem.disabled && this.addItem(this.state.activeCursorItem);\n    }\n\n    if (event.key === 'ArrowUp' && cursor > 0) {\n      this.setState((prevState) => ({\n        cursor: prevState.cursor - 1\n      }));\n      event.preventDefault();\n    } else if (event.key === 'ArrowDown' && cursor < this.searchResults().length + 1) {\n      this.setState((prevState) => ({\n        cursor: prevState.cursor + 1\n      }));\n      event.preventDefault();\n    }\n  };\n\n  render() {\n    return (\n      <ClickOutHandler onClickOut={() => this.dropDown('close')}>\n        <ReactDropdownSelect\n          onKeyDown={this.handleKeyDown}\n          tabIndex=\"0\"\n          style={this.props.style}\n          ref={this.select}\n          disabled={this.props.disabled}\n          className={this.props.className}\n          color={this.props.color}>\n          <Content parentProps={this.props} parentState={this.state} parentMethods={this.methods} />\n\n          {this.props.loading && <Loading parentProps={this.props} />}\n\n          {this.props.clearable && (\n            <Clear parentProps={this.props} parentState={this.state} parentMethods={this.methods} />\n          )}\n\n          {this.props.separator && (\n            <Separator\n              parentProps={this.props}\n              parentState={this.state}\n              parentMethods={this.methods}\n            />\n          )}\n\n          {this.props.dropdownHandle && (\n            <DropdownHandle\n              onCLick={() => this.select.current.focus()}\n              parentProps={this.props}\n              parentState={this.state}\n              parentMethods={this.methods}\n            />\n          )}\n\n          {this.state.dropdown && (\n            <Dropdown\n              parentProps={this.props}\n              parentState={this.state}\n              parentMethods={this.methods}\n            />\n          )}\n        </ReactDropdownSelect>\n      </ClickOutHandler>\n    );\n  }\n}\n\nSelect.defaultProps = {\n  addPlaceholder: '+',\n  placeholder: 'Select...',\n  values: [],\n  options: [],\n  multi: false,\n  disabled: false,\n  searchBy: 'label',\n  clearable: true,\n  dropdownHandle: true,\n  separator: true,\n  keepOpen: undefined,\n  noDataLabel: 'No data',\n  dropdownGap: 5,\n  closeOnScroll: false,\n  debounceDelay: 0,\n  labelField: 'label',\n  valueField: 'value',\n  color: '#0074D9',\n  keepSelectedInList: true,\n  closeOnSelect: false,\n  openOnTop: false,\n  onDropdownOpen: () => undefined,\n  onDropdownClose: () => undefined,\n  onClearAll: () => undefined,\n  onSelectAll: () => undefined\n};\n\nconst ReactDropdownSelect = styled.div`\n  position: relative;\n  display: flex;\n  border: 1px solid #ccc;\n  width: 100%;\n  border-radius: 2px;\n  padding: 2px 5px;\n  flex-direction: row;\n  align-items: center;\n  min-height: 36px;\n  ${({ disabled }) =>\n    disabled ? 'cursor: not-allowed;pointer-events: none;opacity: 0.3;' : 'pointer-events: all;'}\n\n  :hover, \n  :focus-within {\n    border-color: ${({ color }) => color};\n  }\n\n  :focus {\n    outline: 0;\n    box-shadow: 0 0 0 3px ${({ color }) => color}3c;\n  }\n`;\n\nexport default Select;\n"]} */")); | ||
}, "3c;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/index.js"],"names":[],"mappings":"AA0XsC","file":"../src/index.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport styled from '@emotion/styled';\nimport ClickOutHandler from 'react-onclickout';\n\nimport Content from './components/Content';\nimport Dropdown from './components/Dropdown';\nimport Loading from './components/Loading';\nimport Clear from './components/Clear';\nimport Separator from './components/Separator';\nimport DropdownHandle from './components/DropdownHandle';\n\nconst debounce = (fn, delay = 0) => {\n  let timerId;\n\n  return (...args) => {\n    if (timerId) {\n      clearTimeout(timerId);\n    }\n    timerId = setTimeout(() => {\n      fn(...args);\n      timerId = null;\n    }, delay);\n  };\n};\n\nexport class Select extends React.Component {\n  static propTypes = {\n    onChange: PropTypes.func.isRequired,\n    onDropdownClose: PropTypes.func,\n    onDropdownOpen: PropTypes.func,\n    onClearAll: PropTypes.func,\n    onSelectAll: PropTypes.func,\n    values: PropTypes.array,\n    options: PropTypes.array.isRequired,\n    keepOpen: PropTypes.bool,\n    dropdownGap: PropTypes.number,\n    multi: PropTypes.bool,\n    placeholder: PropTypes.string,\n    addPlaceholder: PropTypes.string,\n    disabled: PropTypes.bool,\n    className: PropTypes.string,\n    loading: PropTypes.bool,\n    clearable: PropTypes.bool,\n    separator: PropTypes.bool,\n    dropdownHandle: PropTypes.bool,\n    searchBy: PropTypes.string,\n    closeOnScroll: PropTypes.bool,\n    openOnTop: PropTypes.bool,\n    style: PropTypes.object,\n    contentRenderer: PropTypes.func,\n    dropdownRenderer: PropTypes.func,\n    itemRenderer: PropTypes.func,\n    noDataRenderer: PropTypes.func,\n    optionRenderer: PropTypes.func,\n    inputRenderer: PropTypes.func,\n    loadingRenderer: PropTypes.func,\n    clearRenderer: PropTypes.func,\n    separatorRenderer: PropTypes.func,\n    dropdownHandleRenderer: PropTypes.func\n  };\n\n  constructor(props) {\n    super(props);\n\n    this.state = {\n      dropdown: false,\n      values: props.values,\n      search: '',\n      selectBounds: {}\n    };\n\n    this.methods = {\n      removeItem: this.removeItem,\n      dropDown: this.dropDown,\n      addItem: this.addItem,\n      setSearch: this.setSearch,\n      getInputSize: this.getInputSize,\n      toggleSelectAll: this.toggleSelectAll,\n      clearAll: this.clearAll,\n      selectAll: this.selectAll,\n      searchResults: this.searchResults,\n      getSelectRef: this.getSelectRef,\n      isSelected: this.isSelected,\n      getSelectBounds: this.getSelectBounds,\n      areAllSelected: this.areAllSelected,\n      handleKeyDown: this.handleKeyDown,\n      activeCursorItem: this.activeCursorItem\n    };\n\n    this.select = React.createRef();\n  }\n\n  componentDidMount() {\n    window.addEventListener('resize', debounce(this.updateSelectBounds));\n    window.addEventListener('scroll', debounce(this.onScroll));\n\n    this.props.onChange(this.state.values);\n\n    this.dropDown('close');\n\n    if (this.select) {\n      this.updateSelectBounds();\n    }\n  }\n\n  componentDidUpdate(prevProps, prevState) {\n    if (prevState.values !== this.state.values) {\n      this.props.onChange(this.state.values);\n      this.updateSelectBounds();\n    }\n\n    if (prevState.values !== this.state.values && this.props.closeOnSelect) {\n      this.dropDown('close');\n    }\n\n    if (prevProps.multi !== this.props.multi) {\n      this.updateSelectBounds();\n    }\n\n    if (prevState.dropdown && prevState.dropdown !== this.state.dropdown) {\n      this.onDropdownClose();\n    }\n\n    if (!prevState.dropdown && prevState.dropdown !== this.state.dropdown) {\n      this.props.onDropdownOpen();\n    }\n  }\n\n  componentWillUnmount() {\n    window.removeEventListener(\n      'resize',\n      debounce(this.updateSelectBounds, this.props.debounceDelay)\n    );\n    window.removeEventListener('scroll', debounce(this.onScroll, this.props.debounceDelay));\n  }\n\n  onDropdownClose = () => {\n    this.setState({ cursor: null });\n    this.props.onDropdownClose();\n  };\n\n  onScroll = () => {\n    if (this.props.closeOnScroll) {\n      this.dropDown('close');\n    }\n\n    this.updateSelectBounds();\n  };\n\n  updateSelectBounds = () =>\n    this.select.current &&\n    this.setState({\n      selectBounds: this.select.current.getBoundingClientRect()\n    });\n\n  getSelectBounds = () => this.state.selectBounds;\n\n  dropDown = (action = 'toggle') => {\n    if (this.props.keepOpen) {\n      return this.setState({ dropdown: true });\n    }\n\n    if (action === 'close') {\n      this.select.current.blur();\n      return this.setState({ dropdown: false, search: '' });\n    }\n\n    if (action === 'open') {\n      return this.setState({ dropdown: true });\n    }\n\n    if (action === 'toggle') {\n      this.select.current.focus();\n      return this.setState({ dropdown: !this.state.dropdown });\n    }\n\n    return false;\n  };\n\n  getSelectRef = () => this.select.current;\n\n  addItem = (item) => {\n    if (this.props.multi) {\n      if (this.state.values.indexOf(item) !== -1) {\n        return this.removeItem(null, item, false);\n      }\n\n      this.setState({\n        values: [...this.state.values, item]\n      });\n    } else {\n      this.setState({\n        values: [item],\n        dropdown: false,\n        search: ''\n      });\n    }\n\n    return true;\n  };\n\n  removeItem = (event, item, close = false) => {\n    if (event && close) {\n      event.preventDefault();\n      event.stopPropagation();\n      this.dropDown('close');\n    }\n\n    this.setState({\n      values: this.state.values.filter(\n        (values) => values[this.props.valueField] !== item[this.props.valueField]\n      )\n    });\n  };\n\n  setSearch = (event) => {\n    this.setState({\n      cursor: null\n    });\n\n    this.setState({\n      search: event.target.value\n    });\n  };\n\n  getInputSize = () => {\n    if (this.state.search) {\n      return this.state.search.length;\n    }\n\n    if (this.state.values.length > 0) {\n      return this.props.addPlaceholder.length;\n    }\n\n    return this.props.placeholder.length;\n  };\n\n  toggleSelectAll = () => {\n    return this.setState({\n      values: this.state.values.length === 0 ? this.selectAll() : this.clearAll()\n    });\n  };\n\n  clearAll = () => {\n    this.props.onClearAll();\n    this.setState({\n      values: []\n    });\n  };\n\n  selectAll = () => {\n    this.props.onSelectAll();\n    return this.setState({\n      values: this.props.options.filter((option) => !option.disabled)\n    });\n  };\n\n  isSelected = (option) => this.state.values.indexOf(option) !== -1;\n\n  areAllSelected = () =>\n    this.state.values.length === this.props.options.filter((option) => !option.disabled).length;\n\n  searchResults = () => {\n    const regexp = new RegExp(this.state.search, 'i');\n\n    return this.props.options.filter((item) =>\n      regexp.test(item[this.props.searchBy] || item[[this.props.labelField]])\n    );\n  };\n\n  activeCursorItem = (activeCursorItem) =>\n    this.setState({\n      activeCursorItem\n    });\n\n  handleKeyDown = (event) => {\n    const { cursor } = this.state;\n\n    if (event.key === 'Escape') {\n      this.dropDown('close');\n    }\n\n    if (event.key === 'Enter') {\n      !this.state.activeCursorItem.disabled && this.addItem(this.state.activeCursorItem);\n    }\n\n    if (event.key === 'ArrowUp' && cursor > 0) {\n      this.setState((prevState) => ({\n        cursor: prevState.cursor - 1\n      }));\n      event.preventDefault();\n    } else if (event.key === 'ArrowDown' && cursor < this.searchResults().length + 1) {\n      this.setState((prevState) => ({\n        cursor: prevState.cursor + 1\n      }));\n      event.preventDefault();\n    }\n  };\n\n  render() {\n    return (\n      <ClickOutHandler onClickOut={() => this.dropDown('close')}>\n        <ReactDropdownSelect\n          onKeyDown={this.handleKeyDown}\n          tabIndex=\"0\"\n          style={this.props.style}\n          ref={this.select}\n          disabled={this.props.disabled}\n          className={this.props.className}\n          color={this.props.color}>\n          <Content parentProps={this.props} parentState={this.state} parentMethods={this.methods} />\n\n          {this.props.loading && <Loading parentProps={this.props} />}\n\n          {this.props.clearable && (\n            <Clear parentProps={this.props} parentState={this.state} parentMethods={this.methods} />\n          )}\n\n          {this.props.separator && (\n            <Separator\n              parentProps={this.props}\n              parentState={this.state}\n              parentMethods={this.methods}\n            />\n          )}\n\n          {this.props.dropdownHandle && (\n            <DropdownHandle\n              onCLick={() => this.select.current.focus()}\n              parentProps={this.props}\n              parentState={this.state}\n              parentMethods={this.methods}\n            />\n          )}\n\n          {this.state.dropdown && (\n            <Dropdown\n              parentProps={this.props}\n              parentState={this.state}\n              parentMethods={this.methods}\n            />\n          )}\n        </ReactDropdownSelect>\n      </ClickOutHandler>\n    );\n  }\n}\n\nSelect.defaultProps = {\n  addPlaceholder: '+',\n  placeholder: 'Select...',\n  values: [],\n  options: [],\n  multi: false,\n  disabled: false,\n  searchBy: 'label',\n  clearable: true,\n  dropdownHandle: true,\n  separator: true,\n  keepOpen: undefined,\n  noDataLabel: 'No data',\n  dropdownGap: 5,\n  closeOnScroll: false,\n  debounceDelay: 0,\n  labelField: 'label',\n  valueField: 'value',\n  color: '#0074D9',\n  keepSelectedInList: true,\n  closeOnSelect: false,\n  openOnTop: false,\n  dropdownHeight: '300px',\n  onDropdownOpen: () => undefined,\n  onDropdownClose: () => undefined,\n  onClearAll: () => undefined,\n  onSelectAll: () => undefined\n};\n\nconst ReactDropdownSelect = styled.div`\n  position: relative;\n  display: flex;\n  border: 1px solid #ccc;\n  width: 100%;\n  border-radius: 2px;\n  padding: 2px 5px;\n  flex-direction: row;\n  align-items: center;\n  min-height: 36px;\n  ${({ disabled }) =>\n    disabled ? 'cursor: not-allowed;pointer-events: none;opacity: 0.3;' : 'pointer-events: all;'}\n\n  :hover, \n  :focus-within {\n    border-color: ${({ color }) => color};\n  }\n\n  :focus {\n    outline: 0;\n    box-shadow: 0 0 0 3px ${({ color }) => color}3c;\n  }\n`;\n\nexport default Select;\n"]} */")); | ||
var _default = Select; | ||
exports.default = _default; |
{ | ||
"name": "react-dropdown-select", | ||
"version": "1.3.2", | ||
"version": "1.3.3", | ||
"description": "Customizable dropdown select for react", | ||
@@ -5,0 +5,0 @@ "main": "dist/index.js", |
# react-dropdown-select | ||
Customisable dropdown select for react | ||
@@ -15,11 +16,8 @@ | ||
```import Select from "react-dropdown-select";``` | ||
`import Select from "react-dropdown-select";` | ||
and use as: | ||
```jsx | ||
<Select | ||
options={options} | ||
onChange={(values) => this.setValues(values)} | ||
/> | ||
<Select options={options} onChange={(values) => this.setValues(values)} /> | ||
``` | ||
@@ -34,44 +32,46 @@ | ||
### Preview | ||
> https://sanusart.github.io/react-dropdown-select | ||
| Props | Type | Default | Description | | ||
|------------------------|--------|-------------|---------------------------------------------------------------------------------------------| | ||
| **Component props** |||| | ||
| values | array | [] | Selected values | | ||
| options | array | [] | Available options, (option with key `disabled: true` will be disabled) | | ||
| keepOpen | bool | false | If true, dropdown will always stay open (good for debugging) | | ||
| dropdownGap | number | 5 | Gap between select element and dropdown | | ||
| multi | bool | false | If true - will act as multi-select, if false - only one option will be selected at the time | | ||
| placeholder | string | "Select..." | Placeholder shown where there are no selected values | | ||
| addPlaceholder | string | "+" | Secondary placeholder on search field if any value selected | | ||
| disabled | bool | false | Disable select and all interactions | | ||
| style | object | {} | style object to pass to sselect | | ||
| className | string | | CSS class attribute to pass to sselect | | ||
| loading | bool | false | loading indicator | | ||
| clearable | bool | true | Clear all indicator | | ||
| separator | bool | true | Separator line between close all and dropdown handle | | ||
| dropdownHandle | bool | true | dropdown handle to open/close dropdown | | ||
| searchBy | string | label | search by object property in values | | ||
| labelField | string | "label" | field in data to use for label | | ||
| valueField | string | "value" | field in data to use for value | | ||
| color | string | "#0074D9" | base color to use in component, also can be overwritten via CSS | | ||
| closeOnScroll | bool | false | If true, scrolling the page will close the dropdown | | ||
| closeOnSelect | bool | false | If true, selecting option will close the dropdown | | ||
| openOnTop | bool | false | If true, dropdown will open on top of the select | | ||
| keepSelectedInList | bool | true | If false, selected item will not appear in a list | | ||
| **Callback props** |||| | ||
| onChange | func | | On values change callback, returns array of values objects | | ||
| onDropdownClose | func | | fires upon dropdown close | | ||
| onDropdownOpen | func | | fires upon dropdown open | | ||
| onClearAll | func | | fires upon clearing all values (via custom renderers) | | ||
| onSelectAll | func | | fires upon selecting all values (via custom renderers) | | ||
| [contentRenderer](https://sanusart.github.io/react-dropdown-select/prop/content-renderer) | func | | Overrides internal content component (the contents of the select component) | | ||
| [itemRenderer](https://sanusart.github.io/react-dropdown-select/prop/item-renderer) | func | | Overrides internal item in a dropdown | | ||
| [noDataRenderer](https://sanusart.github.io/react-dropdown-select/prop/no-data-renderer) | func | | Overrides internal "no data" (shown where search has no results) | | ||
| [optionRenderer](https://sanusart.github.io/react-dropdown-select/prop/option-renderer) | func | | Overrides internal option (the pillow with an "x") on the select content | | ||
| [inputRenderer](https://sanusart.github.io/react-dropdown-select/prop/input-renderer) | func | | Overrides internal input text | | ||
| [loadingRenderer](https://sanusart.github.io/react-dropdown-select/prop/loading-renderer) | func | | Overrides internal loading | | ||
| [clearRenderer](https://sanusart.github.io/react-dropdown-select/prop/clear-renderer) | func | | Overrides internal clear button | | ||
| [separatorRenderer](https://sanusart.github.io/react-dropdown-select/prop/separator-renderer) | func | | Overrides internal separator | | ||
| [dropdownRenderer](https://sanusart.github.io/react-dropdown-select/prop/dropdown-renderer) | func | | Overrides internal dropdown component | | ||
> https://sanusart.github.io/react-dropdown-select/demo | ||
| Props | Type | Default | Description | | ||
| -------------------------------------------------------------------------------------------------------- | ------ | ----------- | ------------------------------------------------------------------------------------------- | | ||
| **Component props** | | | | | ||
| values | array | [] | Selected values | | ||
| options | array | [] | Available options, (option with key `disabled: true` will be disabled) | | ||
| keepOpen | bool | false | If true, dropdown will always stay open (good for debugging) | | ||
| dropdownGap | number | 5 | Gap between select element and dropdown | | ||
| multi | bool | false | If true - will act as multi-select, if false - only one option will be selected at the time | | ||
| placeholder | string | "Select..." | Placeholder shown where there are no selected values | | ||
| addPlaceholder | string | "+" | Secondary placeholder on search field if any value selected | | ||
| disabled | bool | false | Disable select and all interactions | | ||
| style | object | {} | style object to pass to sselect | | ||
| className | string | | CSS class attribute to pass to sselect | | ||
| loading | bool | false | loading indicator | | ||
| clearable | bool | true | Clear all indicator | | ||
| separator | bool | true | Separator line between close all and dropdown handle | | ||
| dropdownHandle | bool | true | dropdown handle to open/close dropdown | | ||
| dropdownHeight | string | "300px" | min-height of a dropdown | | ||
| searchBy | string | label | search by object property in values | | ||
| labelField | string | "label" | field in data to use for label | | ||
| valueField | string | "value" | field in data to use for value | | ||
| color | string | "#0074D9" | base color to use in component, also can be overwritten via CSS | | ||
| closeOnScroll | bool | false | If true, scrolling the page will close the dropdown | | ||
| closeOnSelect | bool | false | If true, selecting option will close the dropdown | | ||
| [openOnTop](https://sanusart.github.io/react-dropdown-select/prop/open-on-top) | bool | false | If true, dropdown will open on top of the select | | ||
| keepSelectedInList | bool | true | If false, selected item will not appear in a list | | ||
| **Callback props** | | | | | ||
| onChange | func | | On values change callback, returns array of values objects | | ||
| onDropdownClose | func | | fires upon dropdown close | | ||
| onDropdownOpen | func | | fires upon dropdown open | | ||
| onClearAll | func | | fires upon clearing all values (via custom renderers) | | ||
| onSelectAll | func | | fires upon selecting all values (via custom renderers) | | ||
| [contentRenderer](https://sanusart.github.io/react-dropdown-select/prop/content-renderer) | func | | Overrides internal content component (the contents of the select component) | | ||
| [itemRenderer](https://sanusart.github.io/react-dropdown-select/prop/item-renderer) | func | | Overrides internal item in a dropdown | | ||
| [noDataRenderer](https://sanusart.github.io/react-dropdown-select/prop/no-data-renderer) | func | | Overrides internal "no data" (shown where search has no results) | | ||
| [optionRenderer](https://sanusart.github.io/react-dropdown-select/prop/option-renderer) | func | | Overrides internal option (the pillow with an "x") on the select content | | ||
| [inputRenderer](https://sanusart.github.io/react-dropdown-select/prop/input-renderer) | func | | Overrides internal input text | | ||
| [loadingRenderer](https://sanusart.github.io/react-dropdown-select/prop/loading-renderer) | func | | Overrides internal loading | | ||
| [clearRenderer](https://sanusart.github.io/react-dropdown-select/prop/clear-renderer) | func | | Overrides internal clear button | | ||
| [separatorRenderer](https://sanusart.github.io/react-dropdown-select/prop/separator-renderer) | func | | Overrides internal separator | | ||
| [dropdownRenderer](https://sanusart.github.io/react-dropdown-select/prop/dropdown-renderer) | func | | Overrides internal dropdown component | | ||
| [dropdownHandleRenderer](https://sanusart.github.io/react-dropdown-select/prop/dropdown-handle-renderer) | func | | Overrides internal dropdown handle | |
Long strings
Supply chain riskContains long string literals, which may be a sign of obfuscated or packed code.
Found 1 instance in 1 package
Long strings
Supply chain riskContains long string literals, which may be a sign of obfuscated or packed code.
Found 1 instance in 1 package
85899
966