UNPKG

matrix-react-sdk

Version:
291 lines (244 loc) 32.2 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 _react = _interopRequireWildcard(require("react")); var _classnames = _interopRequireDefault(require("classnames")); var _languageHandler = require("../../../languageHandler"); var _RoomAvatar = _interopRequireDefault(require("../avatars/RoomAvatar")); var _ContextMenu = require("../../structures/ContextMenu"); var _SpaceCreateMenu = _interopRequireDefault(require("./SpaceCreateMenu")); var _SpaceTreeLevel = require("./SpaceTreeLevel"); var _AccessibleTooltipButton = _interopRequireDefault(require("../elements/AccessibleTooltipButton")); var _useEventEmitter = require("../../../hooks/useEventEmitter"); var _SpaceStore = _interopRequireWildcard(require("../../../stores/SpaceStore")); var _AutoHideScrollbar = _interopRequireDefault(require("../../structures/AutoHideScrollbar")); var _NotificationBadge = _interopRequireDefault(require("../rooms/NotificationBadge")); var _RovingTabIndex = require("../../../accessibility/RovingTabIndex"); var _Keyboard = require("../../../Keyboard"); var _RoomNotificationStateStore = require("../../../stores/notifications/RoomNotificationStateStore"); /* Copyright 2021 The Matrix.org Foundation C.I.C. 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. */ const SpaceButton /*: React.FC<IButtonProps>*/ = ({ space, className, selected, onClick, tooltip, notificationState, isNarrow, children }) => { const classes = (0, _classnames.default)("mx_SpaceButton", className, { mx_SpaceButton_active: selected, mx_SpaceButton_narrow: isNarrow }); let avatar = /*#__PURE__*/_react.default.createElement("div", { className: "mx_SpaceButton_avatarPlaceholder" }, /*#__PURE__*/_react.default.createElement("div", { className: "mx_SpaceButton_icon" })); if (space) { avatar = /*#__PURE__*/_react.default.createElement(_RoomAvatar.default, { width: 32, height: 32, room: space }); } let notifBadge; if (notificationState) { notifBadge = /*#__PURE__*/_react.default.createElement("div", { className: "mx_SpacePanel_badgeContainer" }, /*#__PURE__*/_react.default.createElement(_NotificationBadge.default, { forceCount: false, notification: notificationState })); } let button; if (isNarrow) { button = /*#__PURE__*/_react.default.createElement(_RovingTabIndex.RovingAccessibleTooltipButton, { className: classes, title: tooltip, onClick: onClick, role: "treeitem" }, /*#__PURE__*/_react.default.createElement("div", { className: "mx_SpaceButton_selectionWrapper" }, avatar, notifBadge, children)); } else { button = /*#__PURE__*/_react.default.createElement(_RovingTabIndex.RovingAccessibleButton, { className: classes, onClick: onClick, role: "treeitem" }, /*#__PURE__*/_react.default.createElement("div", { className: "mx_SpaceButton_selectionWrapper" }, avatar, /*#__PURE__*/_react.default.createElement("span", { className: "mx_SpaceButton_name" }, tooltip), notifBadge, children)); } return /*#__PURE__*/_react.default.createElement("li", { className: (0, _classnames.default)({ "mx_SpaceItem": true, "collapsed": isNarrow }) }, button); }; const useSpaces = () => /*: [Room[], Room[], Room | null]*/ { const [invites, setInvites] = (0, _react.useState)(_SpaceStore.default.instance.invitedSpaces); (0, _useEventEmitter.useEventEmitter)(_SpaceStore.default.instance, _SpaceStore.UPDATE_INVITED_SPACES, setInvites); const [spaces, setSpaces] = (0, _react.useState)(_SpaceStore.default.instance.spacePanelSpaces); (0, _useEventEmitter.useEventEmitter)(_SpaceStore.default.instance, _SpaceStore.UPDATE_TOP_LEVEL_SPACES, setSpaces); const [activeSpace, setActiveSpace] = (0, _react.useState)(_SpaceStore.default.instance.activeSpace); (0, _useEventEmitter.useEventEmitter)(_SpaceStore.default.instance, _SpaceStore.UPDATE_SELECTED_SPACE, setActiveSpace); return [invites, spaces, activeSpace]; }; const SpacePanel = () => { // We don't need the handle as we position the menu in a constant location // eslint-disable-next-line @typescript-eslint/no-unused-vars const [menuDisplayed, handle, openMenu, closeMenu] = (0, _ContextMenu.useContextMenu)(); const [invites, spaces, activeSpace] = useSpaces(); const [isPanelCollapsed, setPanelCollapsed] = (0, _react.useState)(true); const newClasses = (0, _classnames.default)("mx_SpaceButton_new", { mx_SpaceButton_newCancel: menuDisplayed }); let contextMenu = null; if (menuDisplayed) { contextMenu = /*#__PURE__*/_react.default.createElement(_SpaceCreateMenu.default, { onFinished: closeMenu }); } const onKeyDown = (ev /*: React.KeyboardEvent*/ ) => { let handled = true; switch (ev.key) { case _Keyboard.Key.ARROW_UP: onMoveFocus(ev.target, true); break; case _Keyboard.Key.ARROW_DOWN: onMoveFocus(ev.target, false); break; default: handled = false; } if (handled) { // consume all other keys in context menu ev.stopPropagation(); ev.preventDefault(); } }; const onMoveFocus = (element /*: Element*/ , up /*: boolean*/ ) => { let descending = false; // are we currently descending or ascending through the DOM tree? let classes /*: DOMTokenList*/ ; do { const child = up ? element.lastElementChild : element.firstElementChild; const sibling = up ? element.previousElementSibling : element.nextElementSibling; if (descending) { if (child) { element = child; } else if (sibling) { element = sibling; } else { descending = false; element = element.parentElement; } } else { if (sibling) { element = sibling; descending = true; } else { element = element.parentElement; } } if (element) { if (element.classList.contains("mx_ContextualMenu")) { // we hit the top element = up ? element.lastElementChild : element.firstElementChild; descending = true; } classes = element.classList; } } while (element && !classes.contains("mx_SpaceButton")); if (element) { element.focus(); } }; const activeSpaces = activeSpace ? [activeSpace] : []; const expandCollapseButtonTitle = isPanelCollapsed ? (0, _languageHandler._t)("Expand space panel") : (0, _languageHandler._t)("Collapse space panel"); // TODO drag and drop for re-arranging order return /*#__PURE__*/_react.default.createElement(_RovingTabIndex.RovingTabIndexProvider, { handleHomeEnd: true, onKeyDown: onKeyDown }, ({ onKeyDownHandler }) => /*#__PURE__*/_react.default.createElement("ul", { className: (0, _classnames.default)("mx_SpacePanel", { collapsed: isPanelCollapsed }), onKeyDown: onKeyDownHandler }, /*#__PURE__*/_react.default.createElement(_AutoHideScrollbar.default, { className: "mx_SpacePanel_spaceTreeWrapper" }, /*#__PURE__*/_react.default.createElement("div", { className: "mx_SpaceTreeLevel" }, /*#__PURE__*/_react.default.createElement(SpaceButton, { className: "mx_SpaceButton_home", onClick: () => _SpaceStore.default.instance.setActiveSpace(null), selected: !activeSpace, tooltip: (0, _languageHandler._t)("All rooms"), notificationState: _RoomNotificationStateStore.RoomNotificationStateStore.instance.globalState, isNarrow: isPanelCollapsed }), invites.map(s => /*#__PURE__*/_react.default.createElement(_SpaceTreeLevel.SpaceItem, { key: s.roomId, space: s, activeSpaces: activeSpaces, isPanelCollapsed: isPanelCollapsed, onExpand: () => setPanelCollapsed(false) })), spaces.map(s => /*#__PURE__*/_react.default.createElement(_SpaceTreeLevel.SpaceItem, { key: s.roomId, space: s, activeSpaces: activeSpaces, isPanelCollapsed: isPanelCollapsed, onExpand: () => setPanelCollapsed(false) }))), /*#__PURE__*/_react.default.createElement(SpaceButton, { className: newClasses, tooltip: menuDisplayed ? (0, _languageHandler._t)("Cancel") : (0, _languageHandler._t)("Create a space"), onClick: menuDisplayed ? closeMenu : () => { openMenu(); if (!isPanelCollapsed) setPanelCollapsed(true); }, isNarrow: isPanelCollapsed })), /*#__PURE__*/_react.default.createElement(_AccessibleTooltipButton.default, { className: (0, _classnames.default)("mx_SpacePanel_toggleCollapse", { expanded: !isPanelCollapsed }), onClick: () => { setPanelCollapsed(!isPanelCollapsed); if (menuDisplayed) closeMenu(); }, title: expandCollapseButtonTitle }), contextMenu)); }; var _default = SpacePanel; exports.default = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/spaces/SpacePanel.tsx"],"names":["SpaceButton","space","className","selected","onClick","tooltip","notificationState","isNarrow","children","classes","mx_SpaceButton_active","mx_SpaceButton_narrow","avatar","notifBadge","button","useSpaces","invites","setInvites","SpaceStore","instance","invitedSpaces","UPDATE_INVITED_SPACES","spaces","setSpaces","spacePanelSpaces","UPDATE_TOP_LEVEL_SPACES","activeSpace","setActiveSpace","UPDATE_SELECTED_SPACE","SpacePanel","menuDisplayed","handle","openMenu","closeMenu","isPanelCollapsed","setPanelCollapsed","newClasses","mx_SpaceButton_newCancel","contextMenu","onKeyDown","ev","handled","key","Key","ARROW_UP","onMoveFocus","target","ARROW_DOWN","stopPropagation","preventDefault","element","up","descending","child","lastElementChild","firstElementChild","sibling","previousElementSibling","nextElementSibling","parentElement","classList","contains","focus","activeSpaces","expandCollapseButtonTitle","onKeyDownHandler","collapsed","RoomNotificationStateStore","globalState","map","s","roomId","expanded"],"mappings":"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","sourcesContent":["/*\nCopyright 2021 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, {useState} from \"react\";\nimport classNames from \"classnames\";\nimport {Room} from \"matrix-js-sdk/src/models/room\";\n\nimport {_t} from \"../../../languageHandler\";\nimport RoomAvatar from \"../avatars/RoomAvatar\";\nimport {useContextMenu} from \"../../structures/ContextMenu\";\nimport SpaceCreateMenu from \"./SpaceCreateMenu\";\nimport {SpaceItem} from \"./SpaceTreeLevel\";\nimport AccessibleTooltipButton from \"../elements/AccessibleTooltipButton\";\nimport {useEventEmitter} from \"../../../hooks/useEventEmitter\";\nimport SpaceStore, {\n    UPDATE_INVITED_SPACES,\n    UPDATE_SELECTED_SPACE,\n    UPDATE_TOP_LEVEL_SPACES,\n} from \"../../../stores/SpaceStore\";\nimport AutoHideScrollbar from \"../../structures/AutoHideScrollbar\";\nimport NotificationBadge from \"../rooms/NotificationBadge\";\nimport {\n    RovingAccessibleButton,\n    RovingAccessibleTooltipButton,\n    RovingTabIndexProvider,\n} from \"../../../accessibility/RovingTabIndex\";\nimport {Key} from \"../../../Keyboard\";\nimport {RoomNotificationStateStore} from \"../../../stores/notifications/RoomNotificationStateStore\";\nimport {NotificationState} from \"../../../stores/notifications/NotificationState\";\n\ninterface IButtonProps {\n    space?: Room;\n    className?: string;\n    selected?: boolean;\n    tooltip?: string;\n    notificationState?: NotificationState;\n    isNarrow?: boolean;\n    onClick(): void;\n}\n\nconst SpaceButton: React.FC<IButtonProps> = ({\n    space,\n    className,\n    selected,\n    onClick,\n    tooltip,\n    notificationState,\n    isNarrow,\n    children,\n}) => {\n    const classes = classNames(\"mx_SpaceButton\", className, {\n        mx_SpaceButton_active: selected,\n        mx_SpaceButton_narrow: isNarrow,\n    });\n\n    let avatar = <div className=\"mx_SpaceButton_avatarPlaceholder\"><div className=\"mx_SpaceButton_icon\" /></div>;\n    if (space) {\n        avatar = <RoomAvatar width={32} height={32} room={space} />;\n    }\n\n    let notifBadge;\n    if (notificationState) {\n        notifBadge = <div className=\"mx_SpacePanel_badgeContainer\">\n            <NotificationBadge forceCount={false} notification={notificationState} />\n        </div>;\n    }\n\n    let button;\n    if (isNarrow) {\n        button = (\n            <RovingAccessibleTooltipButton className={classes} title={tooltip} onClick={onClick} role=\"treeitem\">\n                <div className=\"mx_SpaceButton_selectionWrapper\">\n                    { avatar }\n                    { notifBadge }\n                    { children }\n                </div>\n            </RovingAccessibleTooltipButton>\n        );\n    } else {\n        button = (\n            <RovingAccessibleButton className={classes} onClick={onClick} role=\"treeitem\">\n                <div className=\"mx_SpaceButton_selectionWrapper\">\n                    { avatar }\n                    <span className=\"mx_SpaceButton_name\">{ tooltip }</span>\n                    { notifBadge }\n                    { children }\n                </div>\n            </RovingAccessibleButton>\n        );\n    }\n\n    return <li className={classNames({\n        \"mx_SpaceItem\": true,\n        \"collapsed\": isNarrow,\n    })}>\n        { button }\n    </li>;\n}\n\nconst useSpaces = (): [Room[], Room[], Room | null] => {\n    const [invites, setInvites] = useState<Room[]>(SpaceStore.instance.invitedSpaces);\n    useEventEmitter(SpaceStore.instance, UPDATE_INVITED_SPACES, setInvites);\n    const [spaces, setSpaces] = useState<Room[]>(SpaceStore.instance.spacePanelSpaces);\n    useEventEmitter(SpaceStore.instance, UPDATE_TOP_LEVEL_SPACES, setSpaces);\n    const [activeSpace, setActiveSpace] = useState<Room>(SpaceStore.instance.activeSpace);\n    useEventEmitter(SpaceStore.instance, UPDATE_SELECTED_SPACE, setActiveSpace);\n    return [invites, spaces, activeSpace];\n};\n\nconst SpacePanel = () => {\n    // We don't need the handle as we position the menu in a constant location\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    const [menuDisplayed, handle, openMenu, closeMenu] = useContextMenu<void>();\n    const [invites, spaces, activeSpace] = useSpaces();\n    const [isPanelCollapsed, setPanelCollapsed] = useState(true);\n\n    const newClasses = classNames(\"mx_SpaceButton_new\", {\n        mx_SpaceButton_newCancel: menuDisplayed,\n    });\n\n    let contextMenu = null;\n    if (menuDisplayed) {\n        contextMenu = <SpaceCreateMenu onFinished={closeMenu} />;\n    }\n\n    const onKeyDown = (ev: React.KeyboardEvent) => {\n        let handled = true;\n\n        switch (ev.key) {\n            case Key.ARROW_UP:\n                onMoveFocus(ev.target as Element, true);\n                break;\n            case Key.ARROW_DOWN:\n                onMoveFocus(ev.target as Element, false);\n                break;\n            default:\n                handled = false;\n        }\n\n        if (handled) {\n            // consume all other keys in context menu\n            ev.stopPropagation();\n            ev.preventDefault();\n        }\n    };\n\n    const onMoveFocus = (element: Element, up: boolean) => {\n        let descending = false; // are we currently descending or ascending through the DOM tree?\n        let classes: DOMTokenList;\n\n        do {\n            const child = up ? element.lastElementChild : element.firstElementChild;\n            const sibling = up ? element.previousElementSibling : element.nextElementSibling;\n\n            if (descending) {\n                if (child) {\n                    element = child;\n                } else if (sibling) {\n                    element = sibling;\n                } else {\n                    descending = false;\n                    element = element.parentElement;\n                }\n            } else {\n                if (sibling) {\n                    element = sibling;\n                    descending = true;\n                } else {\n                    element = element.parentElement;\n                }\n            }\n\n            if (element) {\n                if (element.classList.contains(\"mx_ContextualMenu\")) { // we hit the top\n                    element = up ? element.lastElementChild : element.firstElementChild;\n                    descending = true;\n                }\n                classes = element.classList;\n            }\n        } while (element && !classes.contains(\"mx_SpaceButton\"));\n\n        if (element) {\n            (element as HTMLElement).focus();\n        }\n    };\n\n    const activeSpaces = activeSpace ? [activeSpace] : [];\n    const expandCollapseButtonTitle = isPanelCollapsed ? _t(\"Expand space panel\") : _t(\"Collapse space panel\");\n    // TODO drag and drop for re-arranging order\n    return <RovingTabIndexProvider handleHomeEnd={true} onKeyDown={onKeyDown}>\n        {({onKeyDownHandler}) => (\n            <ul\n                className={classNames(\"mx_SpacePanel\", { collapsed: isPanelCollapsed })}\n                onKeyDown={onKeyDownHandler}\n            >\n                <AutoHideScrollbar className=\"mx_SpacePanel_spaceTreeWrapper\">\n                    <div className=\"mx_SpaceTreeLevel\">\n                        <SpaceButton\n                            className=\"mx_SpaceButton_home\"\n                            onClick={() => SpaceStore.instance.setActiveSpace(null)}\n                            selected={!activeSpace}\n                            tooltip={_t(\"All rooms\")}\n                            notificationState={RoomNotificationStateStore.instance.globalState}\n                            isNarrow={isPanelCollapsed}\n                        />\n                        { invites.map(s => <SpaceItem\n                            key={s.roomId}\n                            space={s}\n                            activeSpaces={activeSpaces}\n                            isPanelCollapsed={isPanelCollapsed}\n                            onExpand={() => setPanelCollapsed(false)}\n                        />) }\n                        { spaces.map(s => <SpaceItem\n                            key={s.roomId}\n                            space={s}\n                            activeSpaces={activeSpaces}\n                            isPanelCollapsed={isPanelCollapsed}\n                            onExpand={() => setPanelCollapsed(false)}\n                        />) }\n                    </div>\n                    <SpaceButton\n                        className={newClasses}\n                        tooltip={menuDisplayed ? _t(\"Cancel\") : _t(\"Create a space\")}\n                        onClick={menuDisplayed ? closeMenu : () => {\n                            openMenu();\n                            if (!isPanelCollapsed) setPanelCollapsed(true);\n                        }}\n                        isNarrow={isPanelCollapsed}\n                    />\n                </AutoHideScrollbar>\n                <AccessibleTooltipButton\n                    className={classNames(\"mx_SpacePanel_toggleCollapse\", {expanded: !isPanelCollapsed})}\n                    onClick={() => {\n                        setPanelCollapsed(!isPanelCollapsed);\n                        if (menuDisplayed) closeMenu();\n                    }}\n                    title={expandCollapseButtonTitle}\n                />\n                { contextMenu }\n            </ul>\n        )}\n    </RovingTabIndexProvider>\n};\n\nexport default SpacePanel;\n"]}