UNPKG

matrix-react-sdk

Version:
443 lines (375 loc) 52 kB
"use strict"; var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = exports.SpaceItem = void 0; var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends")); var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireDefault(require("react")); var _classnames = _interopRequireDefault(require("classnames")); var _RoomAvatar = _interopRequireDefault(require("../avatars/RoomAvatar")); var _SpaceStore = _interopRequireDefault(require("../../../stores/SpaceStore")); var _SpaceTreeLevelLayoutStore = _interopRequireDefault(require("../../../stores/SpaceTreeLevelLayoutStore")); var _NotificationBadge = _interopRequireDefault(require("../rooms/NotificationBadge")); var _RovingAccessibleButton = require("../../../accessibility/roving/RovingAccessibleButton"); var _RovingAccessibleTooltipButton = require("../../../accessibility/roving/RovingAccessibleTooltipButton"); var _IconizedContextMenu = _interopRequireWildcard(require("../context_menus/IconizedContextMenu")); var _languageHandler = require("../../../languageHandler"); var _ContextMenuTooltipButton = require("../../../accessibility/context_menu/ContextMenuTooltipButton"); var _ContextMenu = require("../../structures/ContextMenu"); var _space = require("../../../utils/space"); var _MatrixClientContext = _interopRequireDefault(require("../../../contexts/MatrixClientContext")); var _AccessibleButton = _interopRequireDefault(require("../elements/AccessibleButton")); var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher")); var _actions = require("../../../dispatcher/actions"); var _RoomViewStore = _interopRequireDefault(require("../../../stores/RoomViewStore")); var _RightPanelStorePhases = require("../../../stores/RightPanelStorePhases"); var _event = require("matrix-js-sdk/src/@types/event"); var _StaticNotificationState = require("../../../stores/notifications/StaticNotificationState"); var _NotificationColor = require("../../../stores/notifications/NotificationColor"); /* 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. */ class SpaceItem extends _react.default.PureComponent /*:: <IItemProps, IItemState>*/ { constructor(props) { super(props); (0, _defineProperty2.default)(this, "onContextMenu", (ev /*: React.MouseEvent*/ ) => { if (this.props.space.getMyMembership() !== "join") return; ev.preventDefault(); ev.stopPropagation(); this.setState({ contextMenuPosition: { right: ev.clientX, top: ev.clientY, height: 0 } }); }); (0, _defineProperty2.default)(this, "onClick", (ev /*: React.MouseEvent*/ ) => { ev.preventDefault(); ev.stopPropagation(); _SpaceStore.default.instance.setActiveSpace(this.props.space); }); (0, _defineProperty2.default)(this, "onMenuOpenClick", (ev /*: React.MouseEvent*/ ) => { ev.preventDefault(); ev.stopPropagation(); const target = ev.target; this.setState({ contextMenuPosition: target.getBoundingClientRect() }); }); (0, _defineProperty2.default)(this, "onMenuClose", () => { this.setState({ contextMenuPosition: null }); }); (0, _defineProperty2.default)(this, "onInviteClick", (ev /*: ButtonEvent*/ ) => { ev.preventDefault(); ev.stopPropagation(); (0, _space.showSpaceInvite)(this.props.space); this.setState({ contextMenuPosition: null }); // also close the menu }); (0, _defineProperty2.default)(this, "onSettingsClick", (ev /*: ButtonEvent*/ ) => { ev.preventDefault(); ev.stopPropagation(); (0, _space.showSpaceSettings)(this.context, this.props.space); this.setState({ contextMenuPosition: null }); // also close the menu }); (0, _defineProperty2.default)(this, "onLeaveClick", (ev /*: ButtonEvent*/ ) => { ev.preventDefault(); ev.stopPropagation(); _dispatcher.default.dispatch({ action: "leave_room", room_id: this.props.space.roomId }); this.setState({ contextMenuPosition: null }); // also close the menu }); (0, _defineProperty2.default)(this, "onNewRoomClick", (ev /*: ButtonEvent*/ ) => { ev.preventDefault(); ev.stopPropagation(); (0, _space.showCreateNewRoom)(this.context, this.props.space); this.setState({ contextMenuPosition: null }); // also close the menu }); (0, _defineProperty2.default)(this, "onAddExistingRoomClick", (ev /*: ButtonEvent*/ ) => { ev.preventDefault(); ev.stopPropagation(); (0, _space.showAddExistingRooms)(this.context, this.props.space); this.setState({ contextMenuPosition: null }); // also close the menu }); (0, _defineProperty2.default)(this, "onMembersClick", (ev /*: ButtonEvent*/ ) => { ev.preventDefault(); ev.stopPropagation(); if (!_RoomViewStore.default.getRoomId()) { _dispatcher.default.dispatch({ action: "view_room", room_id: this.props.space.roomId }, true); } _dispatcher.default.dispatch({ action: _actions.Action.SetRightPanelPhase, phase: _RightPanelStorePhases.RightPanelPhases.SpaceMemberList, refireParams: { space: this.props.space } }); this.setState({ contextMenuPosition: null }); // also close the menu }); (0, _defineProperty2.default)(this, "onExploreRoomsClick", (ev /*: ButtonEvent*/ ) => { ev.preventDefault(); ev.stopPropagation(); _dispatcher.default.dispatch({ action: "view_room", room_id: this.props.space.roomId }); this.setState({ contextMenuPosition: null }); // also close the menu }); const collapsed = _SpaceTreeLevelLayoutStore.default.instance.getSpaceCollapsedState(props.space.roomId, this.props.parents, !props.isNested // default to collapsed for root items ); this.state = { collapsed: collapsed, contextMenuPosition: null }; } toggleCollapse(evt) { if (this.props.onExpand && this.state.collapsed) { this.props.onExpand(); } const newCollapsedState = !this.state.collapsed; _SpaceTreeLevelLayoutStore.default.instance.setSpaceCollapsedState(this.props.space.roomId, this.props.parents, newCollapsedState); this.setState({ collapsed: newCollapsedState }); // don't bubble up so encapsulating button for space // doesn't get triggered evt.stopPropagation(); } renderContextMenu() /*: React.ReactElement*/ { if (this.props.space.getMyMembership() !== "join") return null; let contextMenu = null; if (this.state.contextMenuPosition) { const userId = this.context.getUserId(); let inviteOption; if (this.props.space.getJoinRule() === "public" || this.props.space.canInvite(userId)) { inviteOption = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, { className: "mx_SpacePanel_contextMenu_inviteButton", iconClassName: "mx_SpacePanel_iconInvite", label: (0, _languageHandler._t)("Invite people"), onClick: this.onInviteClick }); } let settingsOption; let leaveSection; if ((0, _space.shouldShowSpaceSettings)(this.context, this.props.space)) { settingsOption = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, { iconClassName: "mx_SpacePanel_iconSettings", label: (0, _languageHandler._t)("Settings"), onClick: this.onSettingsClick }); } else { leaveSection = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOptionList, { red: true, first: true }, /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, { iconClassName: "mx_SpacePanel_iconLeave", label: (0, _languageHandler._t)("Leave space"), onClick: this.onLeaveClick })); } const canAddRooms = this.props.space.currentState.maySendStateEvent(_event.EventType.SpaceChild, userId); let newRoomSection; if (this.props.space.currentState.maySendStateEvent(_event.EventType.SpaceChild, userId)) { newRoomSection = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOptionList, { first: true }, /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, { iconClassName: "mx_SpacePanel_iconPlus", label: (0, _languageHandler._t)("Create new room"), onClick: this.onNewRoomClick }), /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, { iconClassName: "mx_SpacePanel_iconHash", label: (0, _languageHandler._t)("Add existing room"), onClick: this.onAddExistingRoomClick })); } contextMenu = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.default, (0, _extends2.default)({}, (0, _ContextMenu.toRightOf)(this.state.contextMenuPosition, 0), { onFinished: this.onMenuClose, className: "mx_SpacePanel_contextMenu", compact: true }), /*#__PURE__*/_react.default.createElement("div", { className: "mx_SpacePanel_contextMenu_header" }, this.props.space.name), /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOptionList, { first: true }, inviteOption, /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, { iconClassName: "mx_SpacePanel_iconMembers", label: (0, _languageHandler._t)("Members"), onClick: this.onMembersClick }), settingsOption, /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, { iconClassName: "mx_SpacePanel_iconExplore", label: canAddRooms ? (0, _languageHandler._t)("Manage & explore rooms") : (0, _languageHandler._t)("Explore rooms"), onClick: this.onExploreRoomsClick })), newRoomSection, leaveSection); } return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_ContextMenuTooltipButton.ContextMenuTooltipButton, { className: "mx_SpaceButton_menuButton", onClick: this.onMenuOpenClick, title: (0, _languageHandler._t)("Space options"), isExpanded: !!this.state.contextMenuPosition }), contextMenu); } render() { const { space, activeSpaces, isNested } = this.props; const forceCollapsed = this.props.isPanelCollapsed; const isNarrow = this.props.isPanelCollapsed; const collapsed = this.state.collapsed || forceCollapsed; const childSpaces = _SpaceStore.default.instance.getChildSpaces(space.roomId).filter(s => !this.props.parents?.has(s.roomId)); const isActive = activeSpaces.includes(space); const itemClasses = (0, _classnames.default)({ "mx_SpaceItem": true, "mx_SpaceItem_narrow": isNarrow, "collapsed": collapsed, "hasSubSpaces": childSpaces && childSpaces.length }); const isInvite = space.getMyMembership() === "invite"; const classes = (0, _classnames.default)("mx_SpaceButton", { mx_SpaceButton_active: isActive, mx_SpaceButton_hasMenuOpen: !!this.state.contextMenuPosition, mx_SpaceButton_narrow: isNarrow, mx_SpaceButton_invite: isInvite }); const notificationState = isInvite ? _StaticNotificationState.StaticNotificationState.forSymbol("!", _NotificationColor.NotificationColor.Red) : _SpaceStore.default.instance.getNotificationState(space.roomId); let childItems; if (childSpaces && !collapsed) { childItems = /*#__PURE__*/_react.default.createElement(SpaceTreeLevel, { spaces: childSpaces, activeSpaces: activeSpaces, isNested: true, parents: new Set(this.props.parents).add(this.props.space.roomId) }); } let notifBadge; if (notificationState) { notifBadge = /*#__PURE__*/_react.default.createElement("div", { className: "mx_SpacePanel_badgeContainer" }, /*#__PURE__*/_react.default.createElement(_NotificationBadge.default, { forceCount: false, notification: notificationState })); } const avatarSize = isNested ? 24 : 32; const toggleCollapseButton = childSpaces && childSpaces.length ? /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, { className: "mx_SpaceButton_toggleCollapse", onClick: evt => this.toggleCollapse(evt) }) : null; let button; if (isNarrow) { button = /*#__PURE__*/_react.default.createElement(_RovingAccessibleTooltipButton.RovingAccessibleTooltipButton, { className: classes, title: space.name, onClick: this.onClick, onContextMenu: this.onContextMenu, forceHide: !!this.state.contextMenuPosition, role: "treeitem" }, toggleCollapseButton, /*#__PURE__*/_react.default.createElement("div", { className: "mx_SpaceButton_selectionWrapper" }, /*#__PURE__*/_react.default.createElement(_RoomAvatar.default, { width: avatarSize, height: avatarSize, room: space }), notifBadge, this.renderContextMenu())); } else { button = /*#__PURE__*/_react.default.createElement(_RovingAccessibleButton.RovingAccessibleButton, { className: classes, onClick: this.onClick, onContextMenu: this.onContextMenu, role: "treeitem" }, toggleCollapseButton, /*#__PURE__*/_react.default.createElement("div", { className: "mx_SpaceButton_selectionWrapper" }, /*#__PURE__*/_react.default.createElement(_RoomAvatar.default, { width: avatarSize, height: avatarSize, room: space }), /*#__PURE__*/_react.default.createElement("span", { className: "mx_SpaceButton_name" }, space.name), notifBadge, this.renderContextMenu())); } return /*#__PURE__*/_react.default.createElement("li", { className: itemClasses }, button, childItems); } } exports.SpaceItem = SpaceItem; (0, _defineProperty2.default)(SpaceItem, "contextType", _MatrixClientContext.default); const SpaceTreeLevel /*: React.FC<ITreeLevelProps>*/ = ({ spaces, activeSpaces, isNested, parents }) => { return /*#__PURE__*/_react.default.createElement("ul", { className: "mx_SpaceTreeLevel" }, spaces.map(s => { return /*#__PURE__*/_react.default.createElement(SpaceItem, { key: s.roomId, activeSpaces: activeSpaces, space: s, isNested: isNested, parents: parents }); })); }; var _default = SpaceTreeLevel; exports.default = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/spaces/SpaceTreeLevel.tsx"],"names":["SpaceItem","React","PureComponent","constructor","props","ev","space","getMyMembership","preventDefault","stopPropagation","setState","contextMenuPosition","right","clientX","top","clientY","height","SpaceStore","instance","setActiveSpace","target","getBoundingClientRect","context","defaultDispatcher","dispatch","action","room_id","roomId","RoomViewStore","getRoomId","Action","SetRightPanelPhase","phase","RightPanelPhases","SpaceMemberList","refireParams","collapsed","SpaceTreeLevelLayoutStore","getSpaceCollapsedState","parents","isNested","state","toggleCollapse","evt","onExpand","newCollapsedState","setSpaceCollapsedState","renderContextMenu","contextMenu","userId","getUserId","inviteOption","getJoinRule","canInvite","onInviteClick","settingsOption","leaveSection","onSettingsClick","onLeaveClick","canAddRooms","currentState","maySendStateEvent","EventType","SpaceChild","newRoomSection","onNewRoomClick","onAddExistingRoomClick","onMenuClose","name","onMembersClick","onExploreRoomsClick","onMenuOpenClick","render","activeSpaces","forceCollapsed","isPanelCollapsed","isNarrow","childSpaces","getChildSpaces","filter","s","has","isActive","includes","itemClasses","length","isInvite","classes","mx_SpaceButton_active","mx_SpaceButton_hasMenuOpen","mx_SpaceButton_narrow","mx_SpaceButton_invite","notificationState","StaticNotificationState","forSymbol","NotificationColor","Red","getNotificationState","childItems","Set","add","notifBadge","avatarSize","toggleCollapseButton","button","onClick","onContextMenu","MatrixClientContext","SpaceTreeLevel","spaces","map"],"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 from \"react\";\nimport classNames from \"classnames\";\nimport {Room} from \"matrix-js-sdk/src/models/room\";\n\nimport RoomAvatar from \"../avatars/RoomAvatar\";\nimport SpaceStore from \"../../../stores/SpaceStore\";\nimport SpaceTreeLevelLayoutStore from \"../../../stores/SpaceTreeLevelLayoutStore\";\nimport NotificationBadge from \"../rooms/NotificationBadge\";\nimport {RovingAccessibleButton} from \"../../../accessibility/roving/RovingAccessibleButton\";\nimport {RovingAccessibleTooltipButton} from \"../../../accessibility/roving/RovingAccessibleTooltipButton\";\nimport IconizedContextMenu, {\n    IconizedContextMenuOption,\n    IconizedContextMenuOptionList,\n} from \"../context_menus/IconizedContextMenu\";\nimport {_t} from \"../../../languageHandler\";\nimport {ContextMenuTooltipButton} from \"../../../accessibility/context_menu/ContextMenuTooltipButton\";\nimport {toRightOf} from \"../../structures/ContextMenu\";\nimport {\n    shouldShowSpaceSettings,\n    showAddExistingRooms,\n    showCreateNewRoom,\n    showSpaceInvite,\n    showSpaceSettings,\n} from \"../../../utils/space\";\nimport MatrixClientContext from \"../../../contexts/MatrixClientContext\";\nimport AccessibleButton, {ButtonEvent} from \"../elements/AccessibleButton\";\nimport defaultDispatcher from \"../../../dispatcher/dispatcher\";\nimport {Action} from \"../../../dispatcher/actions\";\nimport RoomViewStore from \"../../../stores/RoomViewStore\";\nimport {SetRightPanelPhasePayload} from \"../../../dispatcher/payloads/SetRightPanelPhasePayload\";\nimport {RightPanelPhases} from \"../../../stores/RightPanelStorePhases\";\nimport {EventType} from \"matrix-js-sdk/src/@types/event\";\nimport {StaticNotificationState} from \"../../../stores/notifications/StaticNotificationState\";\nimport {NotificationColor} from \"../../../stores/notifications/NotificationColor\";\n\ninterface IItemProps {\n    space?: Room;\n    activeSpaces: Room[];\n    isNested?: boolean;\n    isPanelCollapsed?: boolean;\n    onExpand?: Function;\n    parents?: Set<string>;\n}\n\ninterface IItemState {\n    collapsed: boolean;\n    contextMenuPosition: Pick<DOMRect, \"right\" | \"top\" | \"height\">;\n}\n\nexport class SpaceItem extends React.PureComponent<IItemProps, IItemState> {\n    static contextType = MatrixClientContext;\n\n    constructor(props) {\n        super(props);\n\n        const collapsed = SpaceTreeLevelLayoutStore.instance.getSpaceCollapsedState(\n            props.space.roomId,\n            this.props.parents,\n            !props.isNested, // default to collapsed for root items\n        );\n\n        this.state = {\n            collapsed: collapsed,\n            contextMenuPosition: null,\n        };\n    }\n\n    private toggleCollapse(evt) {\n        if (this.props.onExpand && this.state.collapsed) {\n            this.props.onExpand();\n        }\n        const newCollapsedState = !this.state.collapsed;\n\n        SpaceTreeLevelLayoutStore.instance.setSpaceCollapsedState(\n            this.props.space.roomId,\n            this.props.parents,\n            newCollapsedState,\n        );\n        this.setState({collapsed: newCollapsedState});\n        // don't bubble up so encapsulating button for space\n        // doesn't get triggered\n        evt.stopPropagation();\n    }\n\n    private onContextMenu = (ev: React.MouseEvent) => {\n        if (this.props.space.getMyMembership() !== \"join\") return;\n        ev.preventDefault();\n        ev.stopPropagation();\n        this.setState({\n            contextMenuPosition: {\n                right: ev.clientX,\n                top: ev.clientY,\n                height: 0,\n            },\n        });\n    }\n\n    private onClick = (ev: React.MouseEvent) => {\n        ev.preventDefault();\n        ev.stopPropagation();\n        SpaceStore.instance.setActiveSpace(this.props.space);\n    };\n\n    private onMenuOpenClick = (ev: React.MouseEvent) => {\n        ev.preventDefault();\n        ev.stopPropagation();\n        const target = ev.target as HTMLButtonElement;\n        this.setState({contextMenuPosition: target.getBoundingClientRect()});\n    };\n\n    private onMenuClose = () => {\n        this.setState({contextMenuPosition: null});\n    };\n\n    private onInviteClick = (ev: ButtonEvent) => {\n        ev.preventDefault();\n        ev.stopPropagation();\n\n        showSpaceInvite(this.props.space);\n        this.setState({contextMenuPosition: null}); // also close the menu\n    };\n\n    private onSettingsClick = (ev: ButtonEvent) => {\n        ev.preventDefault();\n        ev.stopPropagation();\n\n        showSpaceSettings(this.context, this.props.space);\n        this.setState({contextMenuPosition: null}); // also close the menu\n    };\n\n    private onLeaveClick = (ev: ButtonEvent) => {\n        ev.preventDefault();\n        ev.stopPropagation();\n\n        defaultDispatcher.dispatch({\n            action: \"leave_room\",\n            room_id: this.props.space.roomId,\n        });\n        this.setState({contextMenuPosition: null}); // also close the menu\n    };\n\n    private onNewRoomClick = (ev: ButtonEvent) => {\n        ev.preventDefault();\n        ev.stopPropagation();\n\n        showCreateNewRoom(this.context, this.props.space);\n        this.setState({contextMenuPosition: null}); // also close the menu\n    };\n\n    private onAddExistingRoomClick = (ev: ButtonEvent) => {\n        ev.preventDefault();\n        ev.stopPropagation();\n\n        showAddExistingRooms(this.context, this.props.space);\n        this.setState({contextMenuPosition: null}); // also close the menu\n    };\n\n    private onMembersClick = (ev: ButtonEvent) => {\n        ev.preventDefault();\n        ev.stopPropagation();\n\n        if (!RoomViewStore.getRoomId()) {\n            defaultDispatcher.dispatch({\n                action: \"view_room\",\n                room_id: this.props.space.roomId,\n            }, true);\n        }\n\n        defaultDispatcher.dispatch<SetRightPanelPhasePayload>({\n            action: Action.SetRightPanelPhase,\n            phase: RightPanelPhases.SpaceMemberList,\n            refireParams: { space: this.props.space },\n        });\n        this.setState({contextMenuPosition: null}); // also close the menu\n    };\n\n    private onExploreRoomsClick = (ev: ButtonEvent) => {\n        ev.preventDefault();\n        ev.stopPropagation();\n\n        defaultDispatcher.dispatch({\n            action: \"view_room\",\n            room_id: this.props.space.roomId,\n        });\n        this.setState({contextMenuPosition: null}); // also close the menu\n    };\n\n    private renderContextMenu(): React.ReactElement {\n        if (this.props.space.getMyMembership() !== \"join\") return null;\n\n        let contextMenu = null;\n        if (this.state.contextMenuPosition) {\n            const userId = this.context.getUserId();\n\n            let inviteOption;\n            if (this.props.space.getJoinRule() === \"public\" || this.props.space.canInvite(userId)) {\n                inviteOption = (\n                    <IconizedContextMenuOption\n                        className=\"mx_SpacePanel_contextMenu_inviteButton\"\n                        iconClassName=\"mx_SpacePanel_iconInvite\"\n                        label={_t(\"Invite people\")}\n                        onClick={this.onInviteClick}\n                    />\n                );\n            }\n\n            let settingsOption;\n            let leaveSection;\n            if (shouldShowSpaceSettings(this.context, this.props.space)) {\n                settingsOption = (\n                    <IconizedContextMenuOption\n                        iconClassName=\"mx_SpacePanel_iconSettings\"\n                        label={_t(\"Settings\")}\n                        onClick={this.onSettingsClick}\n                    />\n                );\n            } else {\n                leaveSection = <IconizedContextMenuOptionList red first>\n                    <IconizedContextMenuOption\n                        iconClassName=\"mx_SpacePanel_iconLeave\"\n                        label={_t(\"Leave space\")}\n                        onClick={this.onLeaveClick}\n                    />\n                </IconizedContextMenuOptionList>;\n            }\n\n            const canAddRooms = this.props.space.currentState.maySendStateEvent(EventType.SpaceChild, userId);\n\n            let newRoomSection;\n            if (this.props.space.currentState.maySendStateEvent(EventType.SpaceChild, userId)) {\n                newRoomSection = <IconizedContextMenuOptionList first>\n                    <IconizedContextMenuOption\n                        iconClassName=\"mx_SpacePanel_iconPlus\"\n                        label={_t(\"Create new room\")}\n                        onClick={this.onNewRoomClick}\n                    />\n                    <IconizedContextMenuOption\n                        iconClassName=\"mx_SpacePanel_iconHash\"\n                        label={_t(\"Add existing room\")}\n                        onClick={this.onAddExistingRoomClick}\n                    />\n                </IconizedContextMenuOptionList>;\n            }\n\n            contextMenu = <IconizedContextMenu\n                {...toRightOf(this.state.contextMenuPosition, 0)}\n                onFinished={this.onMenuClose}\n                className=\"mx_SpacePanel_contextMenu\"\n                compact\n            >\n                <div className=\"mx_SpacePanel_contextMenu_header\">\n                    { this.props.space.name }\n                </div>\n                <IconizedContextMenuOptionList first>\n                    { inviteOption }\n                    <IconizedContextMenuOption\n                        iconClassName=\"mx_SpacePanel_iconMembers\"\n                        label={_t(\"Members\")}\n                        onClick={this.onMembersClick}\n                    />\n                    { settingsOption }\n                    <IconizedContextMenuOption\n                        iconClassName=\"mx_SpacePanel_iconExplore\"\n                        label={canAddRooms ? _t(\"Manage & explore rooms\") : _t(\"Explore rooms\")}\n                        onClick={this.onExploreRoomsClick}\n                    />\n                </IconizedContextMenuOptionList>\n                { newRoomSection }\n                { leaveSection }\n            </IconizedContextMenu>;\n        }\n\n        return (\n            <React.Fragment>\n                <ContextMenuTooltipButton\n                    className=\"mx_SpaceButton_menuButton\"\n                    onClick={this.onMenuOpenClick}\n                    title={_t(\"Space options\")}\n                    isExpanded={!!this.state.contextMenuPosition}\n                />\n                { contextMenu }\n            </React.Fragment>\n        );\n    }\n\n    render() {\n        const {space, activeSpaces, isNested} = this.props;\n\n        const forceCollapsed = this.props.isPanelCollapsed;\n        const isNarrow = this.props.isPanelCollapsed;\n        const collapsed = this.state.collapsed || forceCollapsed;\n\n        const childSpaces = SpaceStore.instance.getChildSpaces(space.roomId)\n            .filter(s => !this.props.parents?.has(s.roomId));\n        const isActive = activeSpaces.includes(space);\n        const itemClasses = classNames({\n            \"mx_SpaceItem\": true,\n            \"mx_SpaceItem_narrow\": isNarrow,\n            \"collapsed\": collapsed,\n            \"hasSubSpaces\": childSpaces && childSpaces.length,\n        });\n\n        const isInvite = space.getMyMembership() === \"invite\";\n        const classes = classNames(\"mx_SpaceButton\", {\n            mx_SpaceButton_active: isActive,\n            mx_SpaceButton_hasMenuOpen: !!this.state.contextMenuPosition,\n            mx_SpaceButton_narrow: isNarrow,\n            mx_SpaceButton_invite: isInvite,\n        });\n        const notificationState = isInvite\n            ? StaticNotificationState.forSymbol(\"!\", NotificationColor.Red)\n            : SpaceStore.instance.getNotificationState(space.roomId);\n\n        let childItems;\n        if (childSpaces && !collapsed) {\n            childItems = <SpaceTreeLevel\n                spaces={childSpaces}\n                activeSpaces={activeSpaces}\n                isNested={true}\n                parents={new Set(this.props.parents).add(this.props.space.roomId)}\n            />;\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        const avatarSize = isNested ? 24 : 32;\n\n        const toggleCollapseButton = childSpaces && childSpaces.length ?\n            <AccessibleButton\n                className=\"mx_SpaceButton_toggleCollapse\"\n                onClick={evt => this.toggleCollapse(evt)}\n            /> : null;\n\n        let button;\n        if (isNarrow) {\n            button = (\n                <RovingAccessibleTooltipButton\n                    className={classes}\n                    title={space.name}\n                    onClick={this.onClick}\n                    onContextMenu={this.onContextMenu}\n                    forceHide={!!this.state.contextMenuPosition}\n                    role=\"treeitem\"\n                >\n                    { toggleCollapseButton }\n                    <div className=\"mx_SpaceButton_selectionWrapper\">\n                        <RoomAvatar width={avatarSize} height={avatarSize} room={space} />\n                        { notifBadge }\n                        { this.renderContextMenu() }\n                    </div>\n                </RovingAccessibleTooltipButton>\n            );\n        } else {\n            button = (\n                <RovingAcc