UNPKG

@blueprintjs/select

Version:

Components related to selecting items from a list

184 lines 13 kB
/* * 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. */ import { __assign, __extends, __rest } from "tslib"; import classNames from "classnames"; import * as React from "react"; import { AbstractPureComponent, Button, Classes as CoreClasses, DISPLAYNAME_PREFIX, InputGroup, Popover, PopupKind, refHandler, setRef, Utils, } from "@blueprintjs/core"; import { Cross, Search } from "@blueprintjs/icons"; import { Classes } from "../../common"; import { QueryList } from "../query-list/queryList"; /** * Select component. * * @see https://blueprintjs.com/docs/#select/select */ var Select = /** @class */ (function (_super) { __extends(Select, _super); function Select() { var _a; var _this = _super.apply(this, arguments) || this; _this.state = { isOpen: false }; _this.inputElement = null; _this.queryList = null; _this.handleInputRef = refHandler(_this, "inputElement", (_a = _this.props.inputProps) === null || _a === void 0 ? void 0 : _a.inputRef); _this.handleQueryListRef = function (ref) { return (_this.queryList = ref); }; _this.listboxId = Utils.uniqueId("listbox"); _this.renderQueryList = function (listProps) { // not using defaultProps cuz they're hard to type with generics (can't use <T> on static members) var _a = _this.props, _b = _a.filterable, filterable = _b === void 0 ? true : _b, _c = _a.disabled, disabled = _c === void 0 ? false : _c, _d = _a.inputProps, inputProps = _d === void 0 ? {} : _d, _e = _a.placeholder, placeholder = _e === void 0 ? "Filter..." : _e, _f = _a.popoverContentProps, popoverContentProps = _f === void 0 ? {} : _f, _g = _a.popoverProps, popoverProps = _g === void 0 ? {} : _g, popoverRef = _a.popoverRef; var input = (React.createElement(InputGroup, __assign({ "aria-autocomplete": "list", leftIcon: React.createElement(Search, null), placeholder: placeholder, rightElement: _this.maybeRenderClearButton(listProps.query) }, inputProps, { inputRef: _this.handleInputRef, onChange: listProps.handleQueryChange, value: listProps.query }))); var handleKeyDown = listProps.handleKeyDown, handleKeyUp = listProps.handleKeyUp; // N.B. no need to set `fill` since that is unused with the `renderTarget` API return (React.createElement(Popover, __assign({ autoFocus: false, enforceFocus: false, isOpen: _this.state.isOpen, disabled: disabled, placement: popoverProps.position || popoverProps.placement ? undefined : "bottom-start" }, popoverProps, { className: classNames(listProps.className, popoverProps.className), content: React.createElement("div", __assign({}, popoverContentProps, { onKeyDown: handleKeyDown, onKeyUp: handleKeyUp }), filterable ? input : undefined, listProps.itemList), onClosing: _this.handlePopoverClosing, onInteraction: _this.handlePopoverInteraction, onOpened: _this.handlePopoverOpened, onOpening: _this.handlePopoverOpening, popoverClassName: classNames(Classes.SELECT_POPOVER, popoverProps.popoverClassName), popupKind: 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 = function (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 return function (_a) { var _b; var _isOpen = _a.isOpen, ref = _a.ref, targetProps = __rest(_a, ["isOpen", "ref"]); var _c = _this.props, disabled = _c.disabled, _d = _c.popoverProps, popoverProps = _d === void 0 ? {} : _d, popoverTargetProps = _c.popoverTargetProps; var handleKeyDown = listProps.handleKeyDown, handleKeyUp = listProps.handleKeyUp; var _e = popoverProps.targetTagName, targetTagName = _e === void 0 ? "div" : _e; return React.createElement(targetTagName, __assign(__assign(__assign({ "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: classNames(targetProps.className, popoverTargetProps === null || popoverTargetProps === void 0 ? void 0 : popoverTargetProps.className, (_b = {}, _b[CoreClasses.FILL] = _this.props.fill, _b)), // 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: ref, role: "combobox" }), _this.props.children); }; }; _this.withPopoverTargetPropsHandler = function (eventType, handler) { switch (eventType) { case "keydown": return function (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 function (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 = function (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 (Utils.isKeyboardClick(event)) { _this.setState({ isOpen: true }); } }; _this.handleItemSelect = function (item, event) { var _a, _b; var target = event === null || event === void 0 ? void 0 : event.target; var menuItem = target === null || target === void 0 ? void 0 : target.closest(".".concat(CoreClasses.MENU_ITEM)); var menuItemDismiss = menuItem === null || menuItem === void 0 ? void 0 : menuItem.matches(".".concat(CoreClasses.POPOVER_DISMISS)); var 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 = function (isOpen, event) { var _a, _b; _this.setState({ isOpen: 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 = function (node) { var _a, _b, _c; // save currently focused element before popover steals focus, so we can restore it when closing. _this.previousFocusedElement = (_a = 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 = function (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(function () { var _a; var _b = _this.props.inputProps, inputProps = _b === void 0 ? {} : _b; // 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 = function (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(function () { 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 = function () { return _this.queryList && _this.queryList.setQuery("", true); }; return _this; } /** @deprecated no longer necessary now that the TypeScript parser supports type arguments on JSX element tags */ Select.ofType = function () { return Select; }; Select.prototype.render = function () { // omit props specific to this component, spread the rest. var _a = this.props, filterable = _a.filterable, inputProps = _a.inputProps, menuProps = _a.menuProps, popoverProps = _a.popoverProps, restProps = __rest(_a, ["filterable", "inputProps", "menuProps", "popoverProps"]); return (React.createElement(QueryList, __assign({}, restProps, { menuProps: __assign(__assign({ "aria-label": "selectable options" }, menuProps), { id: this.listboxId }), onItemSelect: this.handleItemSelect, ref: this.handleQueryListRef, renderer: this.renderQueryList }))); }; Select.prototype.componentDidUpdate = function (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)) { setRef((_c = prevProps.inputProps) === null || _c === void 0 ? void 0 : _c.inputRef, null); this.handleInputRef = refHandler(this, "inputElement", (_d = this.props.inputProps) === null || _d === void 0 ? void 0 : _d.inputRef); 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(); } }; Select.prototype.maybeRenderClearButton = function (query) { return query.length > 0 ? (React.createElement(Button, { "aria-label": "Clear filter query", icon: React.createElement(Cross, null), minimal: true, onClick: this.resetQuery, title: "Clear filter query" })) : undefined; }; Select.displayName = "".concat(DISPLAYNAME_PREFIX, ".Select"); return Select; }(AbstractPureComponent)); export { Select }; //# sourceMappingURL=select.js.map