matrix-react-sdk
Version:
SDK for matrix.org using React
209 lines (207 loc) • 39.8 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = exports.ThreadPanelHeaderFilterOptionItem = exports.ThreadPanelHeader = exports.ThreadFilterType = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireWildcard(require("react"));
var _matrix = require("matrix-js-sdk/src/matrix");
var _compoundWeb = require("@vector-im/compound-web");
var _logger = require("matrix-js-sdk/src/logger");
var _threads = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/threads"));
var _checkAll = require("../../../res/img/element-icons/check-all.svg");
var _BaseCard = _interopRequireDefault(require("../views/right_panel/BaseCard"));
var _MatrixClientContext = _interopRequireWildcard(require("../../contexts/MatrixClientContext"));
var _languageHandler = require("../../languageHandler");
var _ContextMenuButton = require("../../accessibility/context_menu/ContextMenuButton");
var _ContextMenu = _interopRequireWildcard(require("./ContextMenu"));
var _RoomContext = _interopRequireWildcard(require("../../contexts/RoomContext"));
var _TimelinePanel = _interopRequireDefault(require("./TimelinePanel"));
var _Layout = require("../../settings/enums/Layout");
var _Measured = _interopRequireDefault(require("../views/elements/Measured"));
var _PosthogTrackers = _interopRequireDefault(require("../../PosthogTrackers"));
var _Spinner = _interopRequireDefault(require("../views/elements/Spinner"));
var _notifications = require("../../utils/notifications");
var _EmptyState = _interopRequireDefault(require("../views/right_panel/EmptyState"));
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); }
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2.default)(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } /*
Copyright 2024 New Vector Ltd.
Copyright 2021-2023 The Matrix.org Foundation C.I.C.
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
let ThreadFilterType = exports.ThreadFilterType = /*#__PURE__*/function (ThreadFilterType) {
ThreadFilterType[ThreadFilterType["My"] = 0] = "My";
ThreadFilterType[ThreadFilterType["All"] = 1] = "All";
return ThreadFilterType;
}({});
const ThreadPanelHeaderFilterOptionItem = ({
label,
description,
onClick,
isSelected
}) => {
return /*#__PURE__*/_react.default.createElement(_ContextMenu.MenuItemRadio, {
active: isSelected,
className: "mx_ThreadPanel_Header_FilterOptionItem",
onClick: onClick
}, /*#__PURE__*/_react.default.createElement("span", null, label), /*#__PURE__*/_react.default.createElement("span", null, description));
};
exports.ThreadPanelHeaderFilterOptionItem = ThreadPanelHeaderFilterOptionItem;
const ThreadPanelHeader = ({
filterOption,
setFilterOption
}) => {
const mxClient = (0, _MatrixClientContext.useMatrixClientContext)();
const roomContext = (0, _RoomContext.useRoomContext)();
const [menuDisplayed, button, openMenu, closeMenu] = (0, _ContextMenu.useContextMenu)();
const options = [{
label: (0, _languageHandler._t)("threads|all_threads"),
description: (0, _languageHandler._t)("threads|all_threads_description"),
key: ThreadFilterType.All
}, {
label: (0, _languageHandler._t)("threads|my_threads"),
description: (0, _languageHandler._t)("threads|my_threads_description"),
key: ThreadFilterType.My
}];
const value = options.find(option => option.key === filterOption);
const contextMenuOptions = options.map(opt => /*#__PURE__*/_react.default.createElement(ThreadPanelHeaderFilterOptionItem, {
key: opt.key,
label: opt.label,
description: opt.description,
onClick: () => {
setFilterOption(opt.key);
closeMenu();
},
isSelected: opt === value
}));
const contextMenu = menuDisplayed ? /*#__PURE__*/_react.default.createElement(_ContextMenu.default, {
top: 108,
right: 33,
onFinished: closeMenu,
chevronFace: _ContextMenu.ChevronFace.Top,
wrapperClassName: "mx_BaseCard_header_title"
}, contextMenuOptions) : null;
const onMarkAllThreadsReadClick = _react.default.useCallback(e => {
_PosthogTrackers.default.trackInteraction("WebThreadsMarkAllReadButton", e);
if (!roomContext.room) {
_logger.logger.error("No room in context to mark all threads read");
return;
}
// This actually clears all room notifications by sending an unthreaded read receipt.
// We'd have to loop over all unread threads (pagninating back to find any we don't
// know about yet) and send threaded receipts for all of them... or implement a
// specific API for it. In practice, the user will have to be viewing the room to
// see this button, so will have marked the room itself read anyway.
(0, _notifications.clearRoomNotification)(roomContext.room, mxClient).catch(e => {
_logger.logger.error("Failed to mark all threads read", e);
});
}, [roomContext.room, mxClient]);
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_BaseCard_header_title"
}, /*#__PURE__*/_react.default.createElement(_compoundWeb.Tooltip, {
label: (0, _languageHandler._t)("threads|mark_all_read")
}, /*#__PURE__*/_react.default.createElement(_compoundWeb.IconButton, {
onClick: onMarkAllThreadsReadClick,
size: "24px"
}, /*#__PURE__*/_react.default.createElement(_checkAll.Icon, null))), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ThreadPanel_vertical_separator"
}), /*#__PURE__*/_react.default.createElement(_ContextMenuButton.ContextMenuButton, {
className: "mx_ThreadPanel_dropdown",
ref: button,
isExpanded: menuDisplayed,
onClick: ev => {
openMenu();
_PosthogTrackers.default.trackInteraction("WebRightPanelThreadPanelFilterDropdown", ev);
}
}, `${(0, _languageHandler._t)("threads|show_thread_filter")} ${value?.label}`), contextMenu);
};
exports.ThreadPanelHeader = ThreadPanelHeader;
const ThreadPanel = ({
roomId,
onClose,
permalinkCreator
}) => {
const mxClient = (0, _react.useContext)(_MatrixClientContext.default);
const roomContext = (0, _react.useContext)(_RoomContext.default);
const timelinePanel = (0, _react.useRef)(null);
const card = (0, _react.useRef)(null);
const closeButonRef = (0, _react.useRef)(null);
const [filterOption, setFilterOption] = (0, _react.useState)(ThreadFilterType.All);
const [room, setRoom] = (0, _react.useState)(null);
const [narrow, setNarrow] = (0, _react.useState)(false);
const timelineSet = filterOption === ThreadFilterType.My ? room?.threadsTimelineSets[1] : room?.threadsTimelineSets[0];
const hasThreads = Boolean(room?.threadsTimelineSets?.[0]?.getLiveTimeline()?.getEvents()?.length);
(0, _react.useEffect)(() => {
const room = mxClient.getRoom(roomId);
room?.createThreadsTimelineSets().then(() => room.fetchRoomThreads()).then(() => {
setFilterOption(ThreadFilterType.All);
setRoom(room);
});
}, [mxClient, roomId]);
(0, _react.useEffect)(() => {
if (timelineSet && !_matrix.Thread.hasServerSideSupport) {
timelinePanel.current?.refreshTimeline();
}
}, [timelineSet, timelinePanel]);
return /*#__PURE__*/_react.default.createElement(_RoomContext.default.Provider, {
value: _objectSpread(_objectSpread({}, roomContext), {}, {
timelineRenderingType: _RoomContext.TimelineRenderingType.ThreadsList,
showHiddenEvents: true,
narrow
})
}, /*#__PURE__*/_react.default.createElement(_BaseCard.default, {
header: hasThreads && /*#__PURE__*/_react.default.createElement(ThreadPanelHeader, {
filterOption: filterOption,
setFilterOption: setFilterOption
}),
id: "thread-panel",
className: "mx_ThreadPanel",
ariaLabelledBy: "thread-panel-tab",
role: "tabpanel",
onClose: onClose,
withoutScrollContainer: true,
ref: card,
closeButtonRef: closeButonRef
}, card.current && /*#__PURE__*/_react.default.createElement(_Measured.default, {
sensor: card.current,
onMeasurement: setNarrow
}), timelineSet ? /*#__PURE__*/_react.default.createElement(_TimelinePanel.default, {
key: filterOption + ":" + (timelineSet.getFilter()?.filterId ?? roomId),
ref: timelinePanel,
showReadReceipts: false // No RR support in thread's list
,
manageReadReceipts: false // No RR support in thread's list
,
manageReadMarkers: false // No RM support in thread's list
,
sendReadReceiptOnLoad: false // No RR support in thread's list
,
timelineSet: timelineSet,
showUrlPreview: false // No URL previews at the threads list level
,
empty: /*#__PURE__*/_react.default.createElement(_EmptyState.default, {
Icon: _threads.default,
title: (0, _languageHandler._t)("threads|empty_title"),
description: (0, _languageHandler._t)("threads|empty_description", {
replyInThread: (0, _languageHandler._t)("action|reply_in_thread")
})
}),
alwaysShowTimestamps: true,
layout: _Layout.Layout.Group,
hideThreadedMessages: false,
hidden: false,
showReactions: false,
className: "mx_RoomView_messagePanel",
membersLoaded: true,
permalinkCreator: permalinkCreator,
disableGrouping: true
}) : /*#__PURE__*/_react.default.createElement("div", {
className: "mx_AutoHideScrollbar"
}, /*#__PURE__*/_react.default.createElement(_Spinner.default, null))));
};
var _default = exports.default = ThreadPanel;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_matrix","_compoundWeb","_logger","_threads","_interopRequireDefault","_checkAll","_BaseCard","_MatrixClientContext","_languageHandler","_ContextMenuButton","_ContextMenu","_RoomContext","_TimelinePanel","_Layout","_Measured","_PosthogTrackers","_Spinner","_notifications","_EmptyState","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","ownKeys","keys","getOwnPropertySymbols","o","filter","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","getOwnPropertyDescriptors","defineProperties","ThreadFilterType","exports","ThreadPanelHeaderFilterOptionItem","label","description","onClick","isSelected","createElement","MenuItemRadio","active","className","ThreadPanelHeader","filterOption","setFilterOption","mxClient","useMatrixClientContext","roomContext","useRoomContext","menuDisplayed","button","openMenu","closeMenu","useContextMenu","options","_t","key","All","My","value","find","option","contextMenuOptions","map","opt","contextMenu","top","right","onFinished","chevronFace","ChevronFace","Top","wrapperClassName","onMarkAllThreadsReadClick","React","useCallback","PosthogTrackers","trackInteraction","room","logger","error","clearRoomNotification","catch","Tooltip","IconButton","size","Icon","ContextMenuButton","ref","isExpanded","ev","ThreadPanel","roomId","onClose","permalinkCreator","useContext","MatrixClientContext","RoomContext","timelinePanel","useRef","card","closeButonRef","useState","setRoom","narrow","setNarrow","timelineSet","threadsTimelineSets","hasThreads","Boolean","getLiveTimeline","getEvents","useEffect","getRoom","createThreadsTimelineSets","then","fetchRoomThreads","Thread","hasServerSideSupport","current","refreshTimeline","Provider","timelineRenderingType","TimelineRenderingType","ThreadsList","showHiddenEvents","header","id","ariaLabelledBy","role","withoutScrollContainer","closeButtonRef","sensor","onMeasurement","getFilter","filterId","showReadReceipts","manageReadReceipts","manageReadMarkers","sendReadReceiptOnLoad","showUrlPreview","empty","ThreadsIcon","title","replyInThread","alwaysShowTimestamps","layout","Layout","Group","hideThreadedMessages","hidden","showReactions","membersLoaded","disableGrouping","_default"],"sources":["../../../src/components/structures/ThreadPanel.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2021-2023 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport { Optional } from \"matrix-events-sdk\";\nimport React, { useContext, useEffect, useRef, useState } from \"react\";\nimport { EventTimelineSet, Room, Thread } from \"matrix-js-sdk/src/matrix\";\nimport { IconButton, Tooltip } from \"@vector-im/compound-web\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\nimport ThreadsIcon from \"@vector-im/compound-design-tokens/assets/web/icons/threads\";\n\nimport { Icon as MarkAllThreadsReadIcon } from \"../../../res/img/element-icons/check-all.svg\";\nimport BaseCard from \"../views/right_panel/BaseCard\";\nimport ResizeNotifier from \"../../utils/ResizeNotifier\";\nimport MatrixClientContext, { useMatrixClientContext } from \"../../contexts/MatrixClientContext\";\nimport { _t } from \"../../languageHandler\";\nimport { ContextMenuButton } from \"../../accessibility/context_menu/ContextMenuButton\";\nimport ContextMenu, { ChevronFace, MenuItemRadio, useContextMenu } from \"./ContextMenu\";\nimport RoomContext, { TimelineRenderingType, useRoomContext } from \"../../contexts/RoomContext\";\nimport TimelinePanel from \"./TimelinePanel\";\nimport { Layout } from \"../../settings/enums/Layout\";\nimport { RoomPermalinkCreator } from \"../../utils/permalinks/Permalinks\";\nimport Measured from \"../views/elements/Measured\";\nimport PosthogTrackers from \"../../PosthogTrackers\";\nimport { ButtonEvent } from \"../views/elements/AccessibleButton\";\nimport Spinner from \"../views/elements/Spinner\";\nimport { clearRoomNotification } from \"../../utils/notifications\";\nimport EmptyState from \"../views/right_panel/EmptyState\";\n\ninterface IProps {\n    roomId: string;\n    onClose: () => void;\n    resizeNotifier: ResizeNotifier;\n    permalinkCreator: RoomPermalinkCreator;\n}\n\nexport enum ThreadFilterType {\n    \"My\",\n    \"All\",\n}\n\ntype ThreadPanelHeaderOption = {\n    label: string;\n    description: string;\n    key: ThreadFilterType;\n};\n\nexport const ThreadPanelHeaderFilterOptionItem: React.FC<\n    ThreadPanelHeaderOption & {\n        onClick: () => void;\n        isSelected: boolean;\n    }\n> = ({ label, description, onClick, isSelected }) => {\n    return (\n        <MenuItemRadio active={isSelected} className=\"mx_ThreadPanel_Header_FilterOptionItem\" onClick={onClick}>\n            <span>{label}</span>\n            <span>{description}</span>\n        </MenuItemRadio>\n    );\n};\n\nexport const ThreadPanelHeader: React.FC<{\n    filterOption: ThreadFilterType;\n    setFilterOption: (filterOption: ThreadFilterType) => void;\n}> = ({ filterOption, setFilterOption }) => {\n    const mxClient = useMatrixClientContext();\n    const roomContext = useRoomContext();\n    const [menuDisplayed, button, openMenu, closeMenu] = useContextMenu<HTMLElement>();\n    const options: readonly ThreadPanelHeaderOption[] = [\n        {\n            label: _t(\"threads|all_threads\"),\n            description: _t(\"threads|all_threads_description\"),\n            key: ThreadFilterType.All,\n        },\n        {\n            label: _t(\"threads|my_threads\"),\n            description: _t(\"threads|my_threads_description\"),\n            key: ThreadFilterType.My,\n        },\n    ];\n\n    const value = options.find((option) => option.key === filterOption);\n    const contextMenuOptions = options.map((opt) => (\n        <ThreadPanelHeaderFilterOptionItem\n            key={opt.key}\n            label={opt.label}\n            description={opt.description}\n            onClick={() => {\n                setFilterOption(opt.key);\n                closeMenu();\n            }}\n            isSelected={opt === value}\n        />\n    ));\n    const contextMenu = menuDisplayed ? (\n        <ContextMenu\n            top={108}\n            right={33}\n            onFinished={closeMenu}\n            chevronFace={ChevronFace.Top}\n            wrapperClassName=\"mx_BaseCard_header_title\"\n        >\n            {contextMenuOptions}\n        </ContextMenu>\n    ) : null;\n\n    const onMarkAllThreadsReadClick = React.useCallback(\n        (e: React.MouseEvent) => {\n            PosthogTrackers.trackInteraction(\"WebThreadsMarkAllReadButton\", e);\n            if (!roomContext.room) {\n                logger.error(\"No room in context to mark all threads read\");\n                return;\n            }\n            // This actually clears all room notifications by sending an unthreaded read receipt.\n            // We'd have to loop over all unread threads (pagninating back to find any we don't\n            // know about yet) and send threaded receipts for all of them... or implement a\n            // specific API for it. In practice, the user will have to be viewing the room to\n            // see this button, so will have marked the room itself read anyway.\n            clearRoomNotification(roomContext.room, mxClient).catch((e) => {\n                logger.error(\"Failed to mark all threads read\", e);\n            });\n        },\n        [roomContext.room, mxClient],\n    );\n\n    return (\n        <div className=\"mx_BaseCard_header_title\">\n            <Tooltip label={_t(\"threads|mark_all_read\")}>\n                <IconButton onClick={onMarkAllThreadsReadClick} size=\"24px\">\n                    <MarkAllThreadsReadIcon />\n                </IconButton>\n            </Tooltip>\n            <div className=\"mx_ThreadPanel_vertical_separator\" />\n            <ContextMenuButton\n                className=\"mx_ThreadPanel_dropdown\"\n                ref={button}\n                isExpanded={menuDisplayed}\n                onClick={(ev: ButtonEvent) => {\n                    openMenu();\n                    PosthogTrackers.trackInteraction(\"WebRightPanelThreadPanelFilterDropdown\", ev);\n                }}\n            >\n                {`${_t(\"threads|show_thread_filter\")} ${value?.label}`}\n            </ContextMenuButton>\n            {contextMenu}\n        </div>\n    );\n};\n\nconst ThreadPanel: React.FC<IProps> = ({ roomId, onClose, permalinkCreator }) => {\n    const mxClient = useContext(MatrixClientContext);\n    const roomContext = useContext(RoomContext);\n    const timelinePanel = useRef<TimelinePanel | null>(null);\n    const card = useRef<HTMLDivElement | null>(null);\n    const closeButonRef = useRef<HTMLButtonElement | null>(null);\n\n    const [filterOption, setFilterOption] = useState<ThreadFilterType>(ThreadFilterType.All);\n    const [room, setRoom] = useState<Room | null>(null);\n    const [narrow, setNarrow] = useState<boolean>(false);\n\n    const timelineSet: Optional<EventTimelineSet> =\n        filterOption === ThreadFilterType.My ? room?.threadsTimelineSets[1] : room?.threadsTimelineSets[0];\n    const hasThreads = Boolean(room?.threadsTimelineSets?.[0]?.getLiveTimeline()?.getEvents()?.length);\n\n    useEffect(() => {\n        const room = mxClient.getRoom(roomId);\n        room\n            ?.createThreadsTimelineSets()\n            .then(() => room.fetchRoomThreads())\n            .then(() => {\n                setFilterOption(ThreadFilterType.All);\n                setRoom(room);\n            });\n    }, [mxClient, roomId]);\n\n    useEffect(() => {\n        if (timelineSet && !Thread.hasServerSideSupport) {\n            timelinePanel.current?.refreshTimeline();\n        }\n    }, [timelineSet, timelinePanel]);\n\n    return (\n        <RoomContext.Provider\n            value={{\n                ...roomContext,\n                timelineRenderingType: TimelineRenderingType.ThreadsList,\n                showHiddenEvents: true,\n                narrow,\n            }}\n        >\n            <BaseCard\n                header={\n                    hasThreads && <ThreadPanelHeader filterOption={filterOption} setFilterOption={setFilterOption} />\n                }\n                id=\"thread-panel\"\n                className=\"mx_ThreadPanel\"\n                ariaLabelledBy=\"thread-panel-tab\"\n                role=\"tabpanel\"\n                onClose={onClose}\n                withoutScrollContainer={true}\n                ref={card}\n                closeButtonRef={closeButonRef}\n            >\n                {card.current && <Measured sensor={card.current} onMeasurement={setNarrow} />}\n                {timelineSet ? (\n                    <TimelinePanel\n                        key={filterOption + \":\" + (timelineSet.getFilter()?.filterId ?? roomId)}\n                        ref={timelinePanel}\n                        showReadReceipts={false} // No RR support in thread's list\n                        manageReadReceipts={false} // No RR support in thread's list\n                        manageReadMarkers={false} // No RM support in thread's list\n                        sendReadReceiptOnLoad={false} // No RR support in thread's list\n                        timelineSet={timelineSet}\n                        showUrlPreview={false} // No URL previews at the threads list level\n                        empty={\n                            <EmptyState\n                                Icon={ThreadsIcon}\n                                title={_t(\"threads|empty_title\")}\n                                description={_t(\"threads|empty_description\", {\n                                    replyInThread: _t(\"action|reply_in_thread\"),\n                                })}\n                            />\n                        }\n                        alwaysShowTimestamps={true}\n                        layout={Layout.Group}\n                        hideThreadedMessages={false}\n                        hidden={false}\n                        showReactions={false}\n                        className=\"mx_RoomView_messagePanel\"\n                        membersLoaded={true}\n                        permalinkCreator={permalinkCreator}\n                        disableGrouping={true}\n                    />\n                ) : (\n                    <div className=\"mx_AutoHideScrollbar\">\n                        <Spinner />\n                    </div>\n                )}\n            </BaseCard>\n        </RoomContext.Provider>\n    );\n};\nexport default ThreadPanel;\n"],"mappings":"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","ignoreList":[]}