UNPKG

matrix-react-sdk

Version:
404 lines (335 loc) 44.7 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireWildcard(require("react")); var _propTypes = _interopRequireDefault(require("prop-types")); var _classnames = _interopRequireDefault(require("classnames")); var _AccessibleButton = _interopRequireDefault(require("./AccessibleButton")); var _languageHandler = require("../../../languageHandler"); var _Keyboard = require("../../../Keyboard"); var _replaceableComponent = require("../../../utils/replaceableComponent"); var _dec, _class, _temp; class MenuOption extends _react.default.Component { constructor(props) { super(props); this._onMouseEnter = this._onMouseEnter.bind(this); this._onClick = this._onClick.bind(this); } _onMouseEnter() { this.props.onMouseEnter(this.props.dropdownKey); } _onClick(e) { e.preventDefault(); e.stopPropagation(); this.props.onClick(this.props.dropdownKey); } render() { const optClasses = (0, _classnames.default)({ mx_Dropdown_option: true, mx_Dropdown_option_highlight: this.props.highlighted }); return /*#__PURE__*/_react.default.createElement("div", { id: this.props.id, className: optClasses, onClick: this._onClick, onMouseEnter: this._onMouseEnter, role: "option", "aria-selected": this.props.highlighted, ref: this.props.inputRef }, this.props.children); } } (0, _defineProperty2.default)(MenuOption, "defaultProps", { disabled: false }); MenuOption.propTypes = { children: _propTypes.default.oneOfType([_propTypes.default.arrayOf(_propTypes.default.node), _propTypes.default.node]), highlighted: _propTypes.default.bool, dropdownKey: _propTypes.default.string, onClick: _propTypes.default.func.isRequired, onMouseEnter: _propTypes.default.func.isRequired, inputRef: _propTypes.default.any }; /* * Reusable dropdown select control, akin to react-select, * but somewhat simpler as react-select is 79KB of minified * javascript. * * TODO: Port NetworkDropdown to use this. */ let Dropdown = (_dec = (0, _replaceableComponent.replaceableComponent)("views.elements.Dropdown"), _dec(_class = (_temp = class Dropdown extends _react.default.Component { constructor(props) { super(props); (0, _defineProperty2.default)(this, "_onInputKeyDown", e => { let handled = true; // These keys don't generate keypress events and so needs to be on keyup switch (e.key) { case _Keyboard.Key.ENTER: this.props.onOptionChange(this.state.highlightedOption); // fallthrough case _Keyboard.Key.ESCAPE: this._close(); break; case _Keyboard.Key.ARROW_DOWN: this.setState({ highlightedOption: this._nextOption(this.state.highlightedOption) }); break; case _Keyboard.Key.ARROW_UP: this.setState({ highlightedOption: this._prevOption(this.state.highlightedOption) }); break; default: handled = false; } if (handled) { e.preventDefault(); e.stopPropagation(); } }); this.dropdownRootElement = null; this.ignoreEvent = null; this._onInputClick = this._onInputClick.bind(this); this._onRootClick = this._onRootClick.bind(this); this._onDocumentClick = this._onDocumentClick.bind(this); this._onMenuOptionClick = this._onMenuOptionClick.bind(this); this._onInputChange = this._onInputChange.bind(this); this._collectRoot = this._collectRoot.bind(this); this._collectInputTextBox = this._collectInputTextBox.bind(this); this._setHighlightedOption = this._setHighlightedOption.bind(this); this.inputTextBox = null; this._reindexChildren(this.props.children); const firstChild = _react.default.Children.toArray(props.children)[0]; this.state = { // True if the menu is dropped-down expanded: false, // The key of the highlighted option // (the option that would become selected if you pressed enter) highlightedOption: firstChild ? firstChild.key : null, // the current search query searchQuery: '' }; } // TODO: [REACT-WARNING] Replace component with real class, use constructor for refs UNSAFE_componentWillMount() { // eslint-disable-line camelcase this._button = /*#__PURE__*/(0, _react.createRef)(); // Listen for all clicks on the document so we can close the // menu when the user clicks somewhere else document.addEventListener('click', this._onDocumentClick, false); } componentWillUnmount() { document.removeEventListener('click', this._onDocumentClick, false); } // TODO: [REACT-WARNING] Replace with appropriate lifecycle event UNSAFE_componentWillReceiveProps(nextProps) { // eslint-disable-line camelcase if (!nextProps.children || nextProps.children.length === 0) { return; } this._reindexChildren(nextProps.children); const firstChild = nextProps.children[0]; this.setState({ highlightedOption: firstChild ? firstChild.key : null }); } _reindexChildren(children) { this.childrenByKey = {}; _react.default.Children.forEach(children, child => { this.childrenByKey[child.key] = child; }); } _onDocumentClick(ev) { // Close the dropdown if the user clicks anywhere that isn't // within our root element if (ev !== this.ignoreEvent) { this.setState({ expanded: false }); } } _onRootClick(ev) { // This captures any clicks that happen within our elements, // such that we can then ignore them when they're seen by the // click listener on the document handler, ie. not close the // dropdown immediately after opening it. // NB. We can't just stopPropagation() because then the event // doesn't reach the React onClick(). this.ignoreEvent = ev; } _onInputClick(ev) { if (this.props.disabled) return; if (!this.state.expanded) { this.setState({ expanded: true }); ev.preventDefault(); } } _close() { this.setState({ expanded: false }); // their focus was on the input, its getting unmounted, move it to the button if (this._button.current) { this._button.current.focus(); } } _onMenuOptionClick(dropdownKey) { this._close(); this.props.onOptionChange(dropdownKey); } _onInputChange(e) { this.setState({ searchQuery: e.target.value }); if (this.props.onSearchChange) { this.props.onSearchChange(e.target.value); } } _collectRoot(e) { if (this.dropdownRootElement) { this.dropdownRootElement.removeEventListener('click', this._onRootClick, false); } if (e) { e.addEventListener('click', this._onRootClick, false); } this.dropdownRootElement = e; } _collectInputTextBox(e) { this.inputTextBox = e; if (e) e.focus(); } _setHighlightedOption(optionKey) { this.setState({ highlightedOption: optionKey }); } _nextOption(optionKey) { const keys = Object.keys(this.childrenByKey); const index = keys.indexOf(optionKey); return keys[(index + 1) % keys.length]; } _prevOption(optionKey) { const keys = Object.keys(this.childrenByKey); const index = keys.indexOf(optionKey); return keys[(index - 1) % keys.length]; } _scrollIntoView(node) { if (node) { node.scrollIntoView({ block: "nearest", behavior: "auto" }); } } _getMenuOptions() { const options = _react.default.Children.map(this.props.children, child => { const highlighted = this.state.highlightedOption === child.key; return /*#__PURE__*/_react.default.createElement(MenuOption, { id: `${this.props.id}__${child.key}`, key: child.key, dropdownKey: child.key, highlighted: highlighted, onMouseEnter: this._setHighlightedOption, onClick: this._onMenuOptionClick, inputRef: highlighted ? this._scrollIntoView : undefined }, child); }); if (options.length === 0) { return [/*#__PURE__*/_react.default.createElement("div", { key: "0", className: "mx_Dropdown_option", role: "option" }, (0, _languageHandler._t)("No results"))]; } return options; } render() { let currentValue; const menuStyle = {}; if (this.props.menuWidth) menuStyle.width = this.props.menuWidth; let menu; if (this.state.expanded) { if (this.props.searchEnabled) { currentValue = /*#__PURE__*/_react.default.createElement("input", { type: "text", className: "mx_Dropdown_option", ref: this._collectInputTextBox, onKeyDown: this._onInputKeyDown, onChange: this._onInputChange, value: this.state.searchQuery, role: "combobox", "aria-autocomplete": "list", "aria-activedescendant": `${this.props.id}__${this.state.highlightedOption}`, "aria-owns": `${this.props.id}_listbox`, "aria-disabled": this.props.disabled, "aria-label": this.props.label }); } menu = /*#__PURE__*/_react.default.createElement("div", { className: "mx_Dropdown_menu", style: menuStyle, role: "listbox", id: `${this.props.id}_listbox` }, this._getMenuOptions()); } if (!currentValue) { const selectedChild = this.props.getShortOption ? this.props.getShortOption(this.props.value) : this.childrenByKey[this.props.value]; currentValue = /*#__PURE__*/_react.default.createElement("div", { className: "mx_Dropdown_option", id: `${this.props.id}_value` }, selectedChild); } const dropdownClasses = { mx_Dropdown: true, mx_Dropdown_disabled: this.props.disabled }; if (this.props.className) { dropdownClasses[this.props.className] = true; } // Note the menu sits inside the AccessibleButton div so it's anchored // to the input, but overflows below it. The root contains both. return /*#__PURE__*/_react.default.createElement("div", { className: (0, _classnames.default)(dropdownClasses), ref: this._collectRoot }, /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, { className: "mx_Dropdown_input mx_no_textinput", onClick: this._onInputClick, "aria-haspopup": "listbox", "aria-expanded": this.state.expanded, disabled: this.props.disabled, inputRef: this._button, "aria-label": this.props.label, "aria-describedby": `${this.props.id}_value` }, currentValue, /*#__PURE__*/_react.default.createElement("span", { className: "mx_Dropdown_arrow" }), menu)); } }, _temp)) || _class); exports.default = Dropdown; Dropdown.propTypes = { id: _propTypes.default.string.isRequired, // The width that the dropdown should be. If specified, // the dropped-down part of the menu will be set to this // width. menuWidth: _propTypes.default.number, // Called when the selected option changes onOptionChange: _propTypes.default.func.isRequired, // Called when the value of the search field changes onSearchChange: _propTypes.default.func, searchEnabled: _propTypes.default.bool, // Function that, given the key of an option, returns // a node representing that option to be displayed in the // box itself as the currently-selected option (ie. as // opposed to in the actual dropped-down part). If // unspecified, the appropriate child element is used as // in the dropped-down menu. getShortOption: _propTypes.default.func, value: _propTypes.default.string, // negative for consistency with HTML disabled: _propTypes.default.bool, // ARIA label label: _propTypes.default.string.isRequired }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/elements/Dropdown.js"],"names":["MenuOption","React","Component","constructor","props","_onMouseEnter","bind","_onClick","onMouseEnter","dropdownKey","e","preventDefault","stopPropagation","onClick","render","optClasses","mx_Dropdown_option","mx_Dropdown_option_highlight","highlighted","id","inputRef","children","disabled","propTypes","PropTypes","oneOfType","arrayOf","node","bool","string","func","isRequired","any","Dropdown","handled","key","Key","ENTER","onOptionChange","state","highlightedOption","ESCAPE","_close","ARROW_DOWN","setState","_nextOption","ARROW_UP","_prevOption","dropdownRootElement","ignoreEvent","_onInputClick","_onRootClick","_onDocumentClick","_onMenuOptionClick","_onInputChange","_collectRoot","_collectInputTextBox","_setHighlightedOption","inputTextBox","_reindexChildren","firstChild","Children","toArray","expanded","searchQuery","UNSAFE_componentWillMount","_button","document","addEventListener","componentWillUnmount","removeEventListener","UNSAFE_componentWillReceiveProps","nextProps","length","childrenByKey","forEach","child","ev","current","focus","target","value","onSearchChange","optionKey","keys","Object","index","indexOf","_scrollIntoView","scrollIntoView","block","behavior","_getMenuOptions","options","map","undefined","currentValue","menuStyle","menuWidth","width","menu","searchEnabled","_onInputKeyDown","label","selectedChild","getShortOption","dropdownClasses","mx_Dropdown","mx_Dropdown_disabled","className","number"],"mappings":"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","sourcesContent":["/*\nCopyright 2017 Vector Creations Ltd\nCopyright 2019 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2019 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport React, {createRef} from 'react';\nimport PropTypes from 'prop-types';\nimport classnames from 'classnames';\nimport AccessibleButton from './AccessibleButton';\nimport { _t } from '../../../languageHandler';\nimport {Key} from \"../../../Keyboard\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\n\nclass MenuOption extends React.Component {\n    constructor(props) {\n        super(props);\n        this._onMouseEnter = this._onMouseEnter.bind(this);\n        this._onClick = this._onClick.bind(this);\n    }\n\n    static defaultProps = {\n        disabled: false,\n    };\n\n    _onMouseEnter() {\n        this.props.onMouseEnter(this.props.dropdownKey);\n    }\n\n    _onClick(e) {\n        e.preventDefault();\n        e.stopPropagation();\n        this.props.onClick(this.props.dropdownKey);\n    }\n\n    render() {\n        const optClasses = classnames({\n            mx_Dropdown_option: true,\n            mx_Dropdown_option_highlight: this.props.highlighted,\n        });\n\n        return <div\n            id={this.props.id}\n            className={optClasses}\n            onClick={this._onClick}\n            onMouseEnter={this._onMouseEnter}\n            role=\"option\"\n            aria-selected={this.props.highlighted}\n            ref={this.props.inputRef}\n        >\n            { this.props.children }\n        </div>;\n    }\n}\n\nMenuOption.propTypes = {\n    children: PropTypes.oneOfType([\n      PropTypes.arrayOf(PropTypes.node),\n      PropTypes.node,\n    ]),\n    highlighted: PropTypes.bool,\n    dropdownKey: PropTypes.string,\n    onClick: PropTypes.func.isRequired,\n    onMouseEnter: PropTypes.func.isRequired,\n    inputRef: PropTypes.any,\n};\n\n/*\n * Reusable dropdown select control, akin to react-select,\n * but somewhat simpler as react-select is 79KB of minified\n * javascript.\n *\n * TODO: Port NetworkDropdown to use this.\n */\n@replaceableComponent(\"views.elements.Dropdown\")\nexport default class Dropdown extends React.Component {\n    constructor(props) {\n        super(props);\n\n        this.dropdownRootElement = null;\n        this.ignoreEvent = null;\n\n        this._onInputClick = this._onInputClick.bind(this);\n        this._onRootClick = this._onRootClick.bind(this);\n        this._onDocumentClick = this._onDocumentClick.bind(this);\n        this._onMenuOptionClick = this._onMenuOptionClick.bind(this);\n        this._onInputChange = this._onInputChange.bind(this);\n        this._collectRoot = this._collectRoot.bind(this);\n        this._collectInputTextBox = this._collectInputTextBox.bind(this);\n        this._setHighlightedOption = this._setHighlightedOption.bind(this);\n\n        this.inputTextBox = null;\n\n        this._reindexChildren(this.props.children);\n\n        const firstChild = React.Children.toArray(props.children)[0];\n\n        this.state = {\n            // True if the menu is dropped-down\n            expanded: false,\n            // The key of the highlighted option\n            // (the option that would become selected if you pressed enter)\n            highlightedOption: firstChild ? firstChild.key : null,\n            // the current search query\n            searchQuery: '',\n        };\n    }\n\n    // TODO: [REACT-WARNING] Replace component with real class, use constructor for refs\n    UNSAFE_componentWillMount() { // eslint-disable-line camelcase\n        this._button = createRef();\n        // Listen for all clicks on the document so we can close the\n        // menu when the user clicks somewhere else\n        document.addEventListener('click', this._onDocumentClick, false);\n    }\n\n    componentWillUnmount() {\n        document.removeEventListener('click', this._onDocumentClick, false);\n    }\n\n    // TODO: [REACT-WARNING] Replace with appropriate lifecycle event\n    UNSAFE_componentWillReceiveProps(nextProps) { // eslint-disable-line camelcase\n        if (!nextProps.children || nextProps.children.length === 0) {\n            return;\n        }\n        this._reindexChildren(nextProps.children);\n        const firstChild = nextProps.children[0];\n        this.setState({\n            highlightedOption: firstChild ? firstChild.key : null,\n        });\n    }\n\n    _reindexChildren(children) {\n        this.childrenByKey = {};\n        React.Children.forEach(children, (child) => {\n            this.childrenByKey[child.key] = child;\n        });\n    }\n\n    _onDocumentClick(ev) {\n        // Close the dropdown if the user clicks anywhere that isn't\n        // within our root element\n        if (ev !== this.ignoreEvent) {\n            this.setState({\n                expanded: false,\n            });\n        }\n    }\n\n    _onRootClick(ev) {\n        // This captures any clicks that happen within our elements,\n        // such that we can then ignore them when they're seen by the\n        // click listener on the document handler, ie. not close the\n        // dropdown immediately after opening it.\n        // NB. We can't just stopPropagation() because then the event\n        // doesn't reach the React onClick().\n        this.ignoreEvent = ev;\n    }\n\n    _onInputClick(ev) {\n        if (this.props.disabled) return;\n\n        if (!this.state.expanded) {\n            this.setState({\n                expanded: true,\n            });\n            ev.preventDefault();\n        }\n    }\n\n    _close() {\n        this.setState({\n            expanded: false,\n        });\n        // their focus was on the input, its getting unmounted, move it to the button\n        if (this._button.current) {\n            this._button.current.focus();\n        }\n    }\n\n    _onMenuOptionClick(dropdownKey) {\n        this._close();\n        this.props.onOptionChange(dropdownKey);\n    }\n\n    _onInputKeyDown = (e) => {\n        let handled = true;\n\n        // These keys don't generate keypress events and so needs to be on keyup\n        switch (e.key) {\n            case Key.ENTER:\n                this.props.onOptionChange(this.state.highlightedOption);\n                // fallthrough\n            case Key.ESCAPE:\n                this._close();\n                break;\n            case Key.ARROW_DOWN:\n                this.setState({\n                    highlightedOption: this._nextOption(this.state.highlightedOption),\n                });\n                break;\n            case Key.ARROW_UP:\n                this.setState({\n                    highlightedOption: this._prevOption(this.state.highlightedOption),\n                });\n                break;\n            default:\n                handled = false;\n        }\n\n        if (handled) {\n            e.preventDefault();\n            e.stopPropagation();\n        }\n    }\n\n    _onInputChange(e) {\n        this.setState({\n            searchQuery: e.target.value,\n        });\n        if (this.props.onSearchChange) {\n            this.props.onSearchChange(e.target.value);\n        }\n    }\n\n    _collectRoot(e) {\n        if (this.dropdownRootElement) {\n            this.dropdownRootElement.removeEventListener(\n                'click', this._onRootClick, false,\n            );\n        }\n        if (e) {\n            e.addEventListener('click', this._onRootClick, false);\n        }\n        this.dropdownRootElement = e;\n    }\n\n    _collectInputTextBox(e) {\n        this.inputTextBox = e;\n        if (e) e.focus();\n    }\n\n    _setHighlightedOption(optionKey) {\n        this.setState({\n            highlightedOption: optionKey,\n        });\n    }\n\n    _nextOption(optionKey) {\n        const keys = Object.keys(this.childrenByKey);\n        const index = keys.indexOf(optionKey);\n        return keys[(index + 1) % keys.length];\n    }\n\n    _prevOption(optionKey) {\n        const keys = Object.keys(this.childrenByKey);\n        const index = keys.indexOf(optionKey);\n        return keys[(index - 1) % keys.length];\n    }\n\n    _scrollIntoView(node) {\n        if (node) {\n            node.scrollIntoView({\n                block: \"nearest\",\n                behavior: \"auto\",\n            });\n        }\n    }\n\n    _getMenuOptions() {\n        const options = React.Children.map(this.props.children, (child) => {\n            const highlighted = this.state.highlightedOption === child.key;\n            return (\n                <MenuOption\n                    id={`${this.props.id}__${child.key}`}\n                    key={child.key}\n                    dropdownKey={child.key}\n                    highlighted={highlighted}\n                    onMouseEnter={this._setHighlightedOption}\n                    onClick={this._onMenuOptionClick}\n                    inputRef={highlighted ? this._scrollIntoView : undefined}\n                >\n                    { child }\n                </MenuOption>\n            );\n        });\n        if (options.length === 0) {\n            return [<div key=\"0\" className=\"mx_Dropdown_option\" role=\"option\">\n                { _t(\"No results\") }\n            </div>];\n        }\n        return options;\n    }\n\n    render() {\n        let currentValue;\n\n        const menuStyle = {};\n        if (this.props.menuWidth) menuStyle.width = this.props.menuWidth;\n\n        let menu;\n        if (this.state.expanded) {\n            if (this.props.searchEnabled) {\n                currentValue = (\n                    <input\n                        type=\"text\"\n                        className=\"mx_Dropdown_option\"\n                        ref={this._collectInputTextBox}\n                        onKeyDown={this._onInputKeyDown}\n                        onChange={this._onInputChange}\n                        value={this.state.searchQuery}\n                        role=\"combobox\"\n                        aria-autocomplete=\"list\"\n                        aria-activedescendant={`${this.props.id}__${this.state.highlightedOption}`}\n                        aria-owns={`${this.props.id}_listbox`}\n                        aria-disabled={this.props.disabled}\n                        aria-label={this.props.label}\n                    />\n                );\n            }\n            menu = (\n                <div className=\"mx_Dropdown_menu\" style={menuStyle} role=\"listbox\" id={`${this.props.id}_listbox`}>\n                    { this._getMenuOptions() }\n                </div>\n            );\n        }\n\n        if (!currentValue) {\n            const selectedChild = this.props.getShortOption ?\n                this.props.getShortOption(this.props.value) :\n                this.childrenByKey[this.props.value];\n            currentValue = <div className=\"mx_Dropdown_option\" id={`${this.props.id}_value`}>\n                { selectedChild }\n            </div>;\n        }\n\n        const dropdownClasses = {\n            mx_Dropdown: true,\n            mx_Dropdown_disabled: this.props.disabled,\n        };\n        if (this.props.className) {\n            dropdownClasses[this.props.className] = true;\n        }\n\n        // Note the menu sits inside the AccessibleButton div so it's anchored\n        // to the input, but overflows below it. The root contains both.\n        return <div className={classnames(dropdownClasses)} ref={this._collectRoot}>\n            <AccessibleButton\n                className=\"mx_Dropdown_input mx_no_textinput\"\n                onClick={this._onInputClick}\n                aria-haspopup=\"listbox\"\n                aria-expanded={this.state.expanded}\n                disabled={this.props.disabled}\n                inputRef={this._button}\n                aria-label={this.props.label}\n                aria-describedby={`${this.props.id}_value`}\n            >\n                { currentValue }\n                <span className=\"mx_Dropdown_arrow\" />\n                { menu }\n            </AccessibleButton>\n        </div>;\n    }\n}\n\nDropdown.propTypes = {\n    id: PropTypes.string.isRequired,\n    // The width that the dropdown should be. If specified,\n    // the dropped-down part of the menu will be set to this\n    // width.\n    menuWidth: PropTypes.number,\n    // Called when the selected option changes\n    onOptionChange: PropTypes.func.isRequired,\n    // Called when the value of the search field changes\n    onSearchChange: PropTypes.func,\n    searchEnabled: PropTypes.bool,\n    // Function that, given the key of an option, returns\n    // a node representing that option to be displayed in the\n    // box itself as the currently-selected option (ie. as\n    // opposed to in the actual dropped-down part). If\n    // unspecified, the appropriate child element is used as\n    // in the dropped-down menu.\n    getShortOption: PropTypes.func,\n    value: PropTypes.string,\n    // negative for consistency with HTML\n    disabled: PropTypes.bool,\n    // ARIA label\n    label: PropTypes.string.isRequired,\n};\n"]}