UNPKG

@blueprintjs/select

Version:

Components related to selecting items from a list

190 lines 12.1 kB
"use strict"; /* * Copyright 2022 Palantir Technologies, Inc. All rights reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ Object.defineProperty(exports, "__esModule", { value: true }); exports.Select = void 0; const tslib_1 = require("tslib"); const classnames_1 = tslib_1.__importDefault(require("classnames")); const React = tslib_1.__importStar(require("react")); const core_1 = require("@blueprintjs/core"); const icons_1 = require("@blueprintjs/icons"); const common_1 = require("../../common"); const queryList_1 = require("../query-list/queryList"); /** * Select component. * * @see https://blueprintjs.com/docs/#select/select */ class Select extends core_1.AbstractPureComponent { constructor() { var _a; super(...arguments); this.state = { isOpen: false }; this.inputElement = null; this.queryList = null; this.handleInputRef = (0, core_1.refHandler)(this, "inputElement", (_a = this.props.inputProps) === null || _a === void 0 ? void 0 : _a.inputRef); this.handleQueryListRef = (ref) => (this.queryList = ref); this.listboxId = core_1.Utils.uniqueId("listbox"); this.renderQueryList = (listProps) => { // not using defaultProps cuz they're hard to type with generics (can't use <T> on static members) const { filterable = true, disabled = false, inputProps = {}, placeholder = "Filter...", popoverContentProps = {}, popoverProps = {}, popoverRef, } = this.props; const input = (React.createElement(core_1.InputGroup, { "aria-autocomplete": "list", leftIcon: React.createElement(icons_1.Search, null), placeholder: placeholder, rightElement: this.maybeRenderClearButton(listProps.query), ...inputProps, inputRef: this.handleInputRef, onChange: listProps.handleQueryChange, value: listProps.query })); const { handleKeyDown, handleKeyUp } = listProps; // N.B. no need to set `fill` since that is unused with the `renderTarget` API return (React.createElement(core_1.Popover, { autoFocus: false, enforceFocus: false, isOpen: this.state.isOpen, disabled: disabled, placement: popoverProps.position || popoverProps.placement ? undefined : "bottom-start", ...popoverProps, className: (0, classnames_1.default)(listProps.className, popoverProps.className), content: React.createElement("div", { ...popoverContentProps, onKeyDown: handleKeyDown, onKeyUp: handleKeyUp }, filterable ? input : undefined, listProps.itemList), onClosing: this.handlePopoverClosing, onInteraction: this.handlePopoverInteraction, onOpened: this.handlePopoverOpened, onOpening: this.handlePopoverOpening, popoverClassName: (0, classnames_1.default)(common_1.Classes.SELECT_POPOVER, popoverProps.popoverClassName), popupKind: core_1.PopupKind.LISTBOX, ref: popoverRef, renderTarget: this.getPopoverTargetRenderer(listProps, this.state.isOpen) })); }; // We use the renderTarget API to flatten the rendered DOM and make it easier to implement features like // the "fill" prop. Note that we must take `isOpen` as an argument to force this render function to be called // again after that state changes. this.getPopoverTargetRenderer = (listProps, isOpen) => // N.B. pull out `isOpen` so that it's not forwarded to the DOM, but remember not to use it directly // since it may be stale (`renderTarget` is not re-invoked on this.state changes). // eslint-disable-next-line react/display-name ({ isOpen: _isOpen, ref, ...targetProps }) => { const { disabled, popoverProps = {}, popoverTargetProps } = this.props; const { handleKeyDown, handleKeyUp } = listProps; const { targetTagName = "div" } = popoverProps; return React.createElement(targetTagName, { "aria-controls": this.listboxId, ...popoverTargetProps, ...targetProps, "aria-disabled": disabled, "aria-expanded": isOpen, // Note that we must set FILL here in addition to children to get the wrapper element to full width className: (0, classnames_1.default)(targetProps.className, popoverTargetProps === null || popoverTargetProps === void 0 ? void 0 : popoverTargetProps.className, { [core_1.Classes.FILL]: this.props.fill, }), // Normally, Popover would also need to attach its own `onKeyDown` handler via `targetProps`, // but in our case we fully manage that interaction and listen for key events to open/close // the popover, so we elide it from the DOM. onKeyDown: this.withPopoverTargetPropsHandler("keydown", isOpen ? handleKeyDown : this.handleTargetKeyDown), onKeyUp: this.withPopoverTargetPropsHandler("keyup", isOpen ? handleKeyUp : undefined), ref, role: "combobox", }, this.props.children); }; this.withPopoverTargetPropsHandler = (eventType, handler) => { switch (eventType) { case "keydown": return event => { var _a, _b; handler === null || handler === void 0 ? void 0 : handler(event); (_b = (_a = this.props.popoverTargetProps) === null || _a === void 0 ? void 0 : _a.onKeyDown) === null || _b === void 0 ? void 0 : _b.call(_a, event); }; case "keyup": return event => { var _a, _b; handler === null || handler === void 0 ? void 0 : handler(event); (_b = (_a = this.props.popoverTargetProps) === null || _a === void 0 ? void 0 : _a.onKeyUp) === null || _b === void 0 ? void 0 : _b.call(_a, event); }; } }; /** * Target wrapper element "keydown" handler while the popover is closed. */ this.handleTargetKeyDown = (event) => { // open popover when arrow key pressed on target while closed if (event.key === "ArrowUp" || event.key === "ArrowDown") { event.preventDefault(); this.setState({ isOpen: true }); } else if (core_1.Utils.isKeyboardClick(event)) { this.setState({ isOpen: true }); } }; this.handleItemSelect = (item, event) => { var _a, _b; const target = event === null || event === void 0 ? void 0 : event.target; const menuItem = target === null || target === void 0 ? void 0 : target.closest(`.${core_1.Classes.MENU_ITEM}`); const menuItemDismiss = menuItem === null || menuItem === void 0 ? void 0 : menuItem.matches(`.${core_1.Classes.POPOVER_DISMISS}`); const shouldDismiss = menuItemDismiss !== null && menuItemDismiss !== void 0 ? menuItemDismiss : true; this.setState({ isOpen: !shouldDismiss }); (_b = (_a = this.props).onItemSelect) === null || _b === void 0 ? void 0 : _b.call(_a, item, event); }; this.handlePopoverInteraction = (isOpen, event) => { var _a, _b; this.setState({ isOpen }); (_b = (_a = this.props.popoverProps) === null || _a === void 0 ? void 0 : _a.onInteraction) === null || _b === void 0 ? void 0 : _b.call(_a, isOpen, event); }; this.handlePopoverOpening = (node) => { var _a, _b, _c; // save currently focused element before popover steals focus, so we can restore it when closing. this.previousFocusedElement = (_a = core_1.Utils.getActiveElement(this.inputElement)) !== null && _a !== void 0 ? _a : undefined; if (this.props.resetOnClose) { this.resetQuery(); } (_c = (_b = this.props.popoverProps) === null || _b === void 0 ? void 0 : _b.onOpening) === null || _c === void 0 ? void 0 : _c.call(_b, node); }; this.handlePopoverOpened = (node) => { var _a, _b; // scroll active item into view after popover transition completes and all dimensions are stable. if (this.queryList != null) { this.queryList.scrollActiveItemIntoView(); } this.requestAnimationFrame(() => { var _a; const { inputProps = {} } = this.props; // autofocus is enabled by default if (inputProps.autoFocus !== false) { (_a = this.inputElement) === null || _a === void 0 ? void 0 : _a.focus(); } }); (_b = (_a = this.props.popoverProps) === null || _a === void 0 ? void 0 : _a.onOpened) === null || _b === void 0 ? void 0 : _b.call(_a, node); }; this.handlePopoverClosing = (node) => { var _a, _b; // restore focus to saved element. // timeout allows popover to begin closing and remove focus handlers beforehand. /* istanbul ignore next */ this.requestAnimationFrame(() => { if (this.previousFocusedElement !== undefined) { this.previousFocusedElement.focus(); this.previousFocusedElement = undefined; } }); (_b = (_a = this.props.popoverProps) === null || _a === void 0 ? void 0 : _a.onClosing) === null || _b === void 0 ? void 0 : _b.call(_a, node); }; this.resetQuery = () => this.queryList && this.queryList.setQuery("", true); } /** @deprecated no longer necessary now that the TypeScript parser supports type arguments on JSX element tags */ static ofType() { return Select; } render() { // omit props specific to this component, spread the rest. const { filterable, inputProps, menuProps, popoverProps, ...restProps } = this.props; return (React.createElement(queryList_1.QueryList, { ...restProps, menuProps: { "aria-label": "selectable options", ...menuProps, id: this.listboxId }, onItemSelect: this.handleItemSelect, ref: this.handleQueryListRef, renderer: this.renderQueryList })); } componentDidUpdate(prevProps, prevState) { var _a, _b, _c, _d, _e; if (((_a = prevProps.inputProps) === null || _a === void 0 ? void 0 : _a.inputRef) !== ((_b = this.props.inputProps) === null || _b === void 0 ? void 0 : _b.inputRef)) { (0, core_1.setRef)((_c = prevProps.inputProps) === null || _c === void 0 ? void 0 : _c.inputRef, null); this.handleInputRef = (0, core_1.refHandler)(this, "inputElement", (_d = this.props.inputProps) === null || _d === void 0 ? void 0 : _d.inputRef); (0, core_1.setRef)((_e = this.props.inputProps) === null || _e === void 0 ? void 0 : _e.inputRef, this.inputElement); } if (this.state.isOpen && !prevState.isOpen && this.queryList != null) { this.queryList.scrollActiveItemIntoView(); } } maybeRenderClearButton(query) { return query.length > 0 ? (React.createElement(core_1.Button, { "aria-label": "Clear filter query", icon: React.createElement(icons_1.Cross, null), onClick: this.resetQuery, title: "Clear filter query", variant: "minimal" })) : undefined; } } exports.Select = Select; Select.displayName = `${core_1.DISPLAYNAME_PREFIX}.Select`; //# sourceMappingURL=select.js.map