matrix-react-sdk
Version:
SDK for matrix.org using React
266 lines (249 loc) • 40.2 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.PinnedMessageBanner = PinnedMessageBanner;
var _react = _interopRequireWildcard(require("react"));
var _pinSolid = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/pin-solid"));
var _compoundWeb = require("@vector-im/compound-web");
var _matrix = require("matrix-js-sdk/src/matrix");
var _classnames = _interopRequireDefault(require("classnames"));
var _usePinnedEvents = require("../../../hooks/usePinnedEvents");
var _languageHandler = require("../../../languageHandler");
var _RightPanelStore = _interopRequireDefault(require("../../../stores/right-panel/RightPanelStore"));
var _RightPanelStorePhases = require("../../../stores/right-panel/RightPanelStorePhases");
var _useEventEmitter = require("../../../hooks/useEventEmitter");
var _AsyncStore = require("../../../stores/AsyncStore");
var _MessagePreviewStore = require("../../../stores/room-list/MessagePreviewStore");
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _actions = require("../../../dispatcher/actions");
var _MessageEvent = _interopRequireDefault(require("../messages/MessageEvent"));
var _PosthogTrackers = _interopRequireDefault(require("../../../PosthogTrackers.ts"));
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; }
/*
* Copyright 2024 New Vector Ltd.
* Copyright 2024 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.
*/
/**
* The props for the {@link PinnedMessageBanner} component.
*/
/**
* A banner that displays the pinned messages in a room.
*/
function PinnedMessageBanner({
room,
permalinkCreator
}) {
const pinnedEventIds = (0, _usePinnedEvents.usePinnedEvents)(room);
const pinnedEvents = (0, _usePinnedEvents.useSortedFetchedPinnedEvents)(room, pinnedEventIds);
const eventCount = pinnedEvents.length;
const isSinglePinnedEvent = eventCount === 1;
const [currentEventIndex, setCurrentEventIndex] = (0, _react.useState)(eventCount - 1);
// When the number of pinned messages changes, we want to display the last message
(0, _react.useEffect)(() => {
setCurrentEventIndex(() => eventCount - 1);
}, [eventCount]);
const pinnedEvent = pinnedEvents[currentEventIndex];
if (!pinnedEvent) return null;
const shouldUseMessageEvent = pinnedEvent.isRedacted() || pinnedEvent.isDecryptionFailure();
const onBannerClick = () => {
_PosthogTrackers.default.trackInteraction("PinnedMessageBannerClick");
// Scroll to the pinned message
_dispatcher.default.dispatch({
action: _actions.Action.ViewRoom,
event_id: pinnedEvent.getId(),
highlighted: true,
room_id: room.roomId,
metricsTrigger: undefined // room doesn't change
});
// Cycle through the pinned messages
// When we reach the first message, we go back to the last message
setCurrentEventIndex(currentEventIndex => --currentEventIndex === -1 ? eventCount - 1 : currentEventIndex);
};
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_PinnedMessageBanner",
"data-single-message": isSinglePinnedEvent,
"aria-label": (0, _languageHandler._t)("room|pinned_message_banner|description"),
"data-testid": "pinned-message-banner"
}, /*#__PURE__*/_react.default.createElement("button", {
"aria-label": (0, _languageHandler._t)("room|pinned_message_banner|go_to_message"),
type: "button",
className: "mx_PinnedMessageBanner_main",
onClick: onBannerClick
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_PinnedMessageBanner_content"
}, /*#__PURE__*/_react.default.createElement(Indicators, {
count: eventCount,
currentIndex: currentEventIndex
}), /*#__PURE__*/_react.default.createElement(_pinSolid.default, {
width: "20px",
height: "20px",
className: "mx_PinnedMessageBanner_PinIcon"
}), !isSinglePinnedEvent && /*#__PURE__*/_react.default.createElement("div", {
className: "mx_PinnedMessageBanner_title",
"data-testid": "banner-counter"
}, (0, _languageHandler._t)("room|pinned_message_banner|title", {
index: currentEventIndex + 1,
length: eventCount
}, {
bold: sub => /*#__PURE__*/_react.default.createElement("span", {
className: "mx_PinnedMessageBanner_title_counter"
}, sub)
})), /*#__PURE__*/_react.default.createElement(EventPreview, {
pinnedEvent: pinnedEvent
}), shouldUseMessageEvent && /*#__PURE__*/_react.default.createElement("div", {
className: "mx_PinnedMessageBanner_redactedMessage"
}, /*#__PURE__*/_react.default.createElement(_MessageEvent.default, {
mxEvent: pinnedEvent,
maxImageHeight: 20,
permalinkCreator: permalinkCreator,
replacingEventId: pinnedEvent.replacingEventId()
})))), !isSinglePinnedEvent && /*#__PURE__*/_react.default.createElement(BannerButton, {
room: room
}));
}
/**
* The props for the {@link EventPreview} component.
*/
/**
* A component that displays a preview for the pinned event.
*/
function EventPreview({
pinnedEvent
}) {
const preview = useEventPreview(pinnedEvent);
if (!preview) return null;
const prefix = getPreviewPrefix(pinnedEvent.getType(), pinnedEvent.getContent().msgtype);
if (!prefix) return /*#__PURE__*/_react.default.createElement("span", {
className: "mx_PinnedMessageBanner_message",
"data-testid": "banner-message"
}, preview);
return /*#__PURE__*/_react.default.createElement("span", {
className: "mx_PinnedMessageBanner_message",
"data-testid": "banner-message"
}, (0, _languageHandler._t)("room|pinned_message_banner|preview", {
prefix,
preview
}, {
bold: sub => /*#__PURE__*/_react.default.createElement("span", {
className: "mx_PinnedMessageBanner_prefix"
}, sub)
}));
}
/**
* Hooks to generate a preview for the pinned event.
* @param pinnedEvent
*/
function useEventPreview(pinnedEvent) {
return (0, _react.useMemo)(() => {
if (!pinnedEvent || pinnedEvent.isRedacted() || pinnedEvent.isDecryptionFailure()) return null;
return _MessagePreviewStore.MessagePreviewStore.instance.generatePreviewForEvent(pinnedEvent);
}, [pinnedEvent]);
}
/**
* Get the prefix for the preview based on the type and the message type.
* @param type
* @param msgType
*/
function getPreviewPrefix(type, msgType) {
switch (type) {
case _matrix.M_POLL_START.name:
return (0, _languageHandler._t)("room|pinned_message_banner|prefix|poll");
default:
}
switch (msgType) {
case _matrix.MsgType.Audio:
return (0, _languageHandler._t)("room|pinned_message_banner|prefix|audio");
case _matrix.MsgType.Image:
return (0, _languageHandler._t)("room|pinned_message_banner|prefix|image");
case _matrix.MsgType.Video:
return (0, _languageHandler._t)("room|pinned_message_banner|prefix|video");
case _matrix.MsgType.File:
return (0, _languageHandler._t)("room|pinned_message_banner|prefix|file");
default:
return null;
}
}
const MAX_INDICATORS = 3;
/**
* The props for the {@link IndicatorsProps} component.
*/
/**
* A component that displays vertical indicators for the pinned messages.
*/
function Indicators({
count,
currentIndex
}) {
// We only display a maximum of 3 indicators at one time.
// When there is more than 3 messages pinned, we will cycle through the indicators
// If there is only 2 messages pinned, we will display 2 indicators
// In case of 1 message pinned, the indicators are not displayed, see {@link PinnedMessageBanner} logic.
const numberOfIndicators = Math.min(count, MAX_INDICATORS);
// The index of the active indicator
const index = currentIndex % numberOfIndicators;
// We hide the indicators when we are on the last cycle and there are less than 3 remaining messages pinned
const numberOfCycles = Math.ceil(count / numberOfIndicators);
// If the current index is greater than the last cycle index, we are on the last cycle
const isLastCycle = currentIndex >= (numberOfCycles - 1) * MAX_INDICATORS;
// The index of the last message in the last cycle
const lastCycleIndex = numberOfIndicators - (numberOfCycles * numberOfIndicators - count);
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_PinnedMessageBanner_Indicators"
}, Array.from({
length: numberOfIndicators
}).map((_, i) => /*#__PURE__*/_react.default.createElement(Indicator, {
key: i,
active: i === index,
hidden: isLastCycle && lastCycleIndex <= i
})));
}
/**
* The props for the {@link Indicator} component.
*/
/**
* A component that displays a vertical indicator for a pinned message.
*/
function Indicator({
active,
hidden
}) {
return /*#__PURE__*/_react.default.createElement("div", {
"data-testid": "banner-indicator",
className: (0, _classnames.default)("mx_PinnedMessageBanner_Indicator", {
"mx_PinnedMessageBanner_Indicator--active": active,
"mx_PinnedMessageBanner_Indicator--hidden": hidden
})
});
}
function getRightPanelPhase(roomId) {
if (!_RightPanelStore.default.instance.isOpenForRoom(roomId)) return null;
return _RightPanelStore.default.instance.currentCard.phase;
}
/**
* The props for the {@link BannerButton} component.
*/
/**
* A button that allows the user to view or close the list of pinned messages.
*/
function BannerButton({
room
}) {
const [currentPhase, setCurrentPhase] = (0, _react.useState)(getRightPanelPhase(room.roomId));
(0, _useEventEmitter.useEventEmitter)(_RightPanelStore.default.instance, _AsyncStore.UPDATE_EVENT, () => setCurrentPhase(getRightPanelPhase(room.roomId)));
const isPinnedMessagesPhase = currentPhase === _RightPanelStorePhases.RightPanelPhases.PinnedMessages;
return /*#__PURE__*/_react.default.createElement(_compoundWeb.Button, {
className: "mx_PinnedMessageBanner_actions",
kind: "tertiary",
onClick: () => {
if (isPinnedMessagesPhase) _PosthogTrackers.default.trackInteraction("PinnedMessageBannerCloseListButton");else _PosthogTrackers.default.trackInteraction("PinnedMessageBannerViewAllButton");
_RightPanelStore.default.instance.showOrHidePhase(_RightPanelStorePhases.RightPanelPhases.PinnedMessages);
}
}, isPinnedMessagesPhase ? (0, _languageHandler._t)("room|pinned_message_banner|button_close_list") : (0, _languageHandler._t)("room|pinned_message_banner|button_view_all"));
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_pinSolid","_interopRequireDefault","_compoundWeb","_matrix","_classnames","_usePinnedEvents","_languageHandler","_RightPanelStore","_RightPanelStorePhases","_useEventEmitter","_AsyncStore","_MessagePreviewStore","_dispatcher","_actions","_MessageEvent","_PosthogTrackers","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","PinnedMessageBanner","room","permalinkCreator","pinnedEventIds","usePinnedEvents","pinnedEvents","useSortedFetchedPinnedEvents","eventCount","length","isSinglePinnedEvent","currentEventIndex","setCurrentEventIndex","useState","useEffect","pinnedEvent","shouldUseMessageEvent","isRedacted","isDecryptionFailure","onBannerClick","PosthogTrackers","trackInteraction","dis","dispatch","action","Action","ViewRoom","event_id","getId","highlighted","room_id","roomId","metricsTrigger","undefined","createElement","className","_t","type","onClick","Indicators","count","currentIndex","width","height","index","bold","sub","EventPreview","mxEvent","maxImageHeight","replacingEventId","BannerButton","preview","useEventPreview","prefix","getPreviewPrefix","getType","getContent","msgtype","useMemo","MessagePreviewStore","instance","generatePreviewForEvent","msgType","M_POLL_START","name","MsgType","Audio","Image","Video","File","MAX_INDICATORS","numberOfIndicators","Math","min","numberOfCycles","ceil","isLastCycle","lastCycleIndex","Array","from","map","_","Indicator","key","active","hidden","classNames","getRightPanelPhase","RightPanelStore","isOpenForRoom","currentCard","phase","currentPhase","setCurrentPhase","useEventEmitter","UPDATE_EVENT","isPinnedMessagesPhase","RightPanelPhases","PinnedMessages","Button","kind","showOrHidePhase"],"sources":["../../../../src/components/views/rooms/PinnedMessageBanner.tsx"],"sourcesContent":["/*\n * Copyright 2024 New Vector Ltd.\n * Copyright 2024 The Matrix.org Foundation C.I.C.\n *\n * SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\n * Please see LICENSE files in the repository root for full details.\n */\n\nimport React, { JSX, useEffect, useMemo, useState } from \"react\";\nimport PinIcon from \"@vector-im/compound-design-tokens/assets/web/icons/pin-solid\";\nimport { Button } from \"@vector-im/compound-web\";\nimport { M_POLL_START, MatrixEvent, MsgType, Room } from \"matrix-js-sdk/src/matrix\";\nimport classNames from \"classnames\";\n\nimport { usePinnedEvents, useSortedFetchedPinnedEvents } from \"../../../hooks/usePinnedEvents\";\nimport { _t } from \"../../../languageHandler\";\nimport RightPanelStore from \"../../../stores/right-panel/RightPanelStore\";\nimport { RightPanelPhases } from \"../../../stores/right-panel/RightPanelStorePhases\";\nimport { useEventEmitter } from \"../../../hooks/useEventEmitter\";\nimport { UPDATE_EVENT } from \"../../../stores/AsyncStore\";\nimport { RoomPermalinkCreator } from \"../../../utils/permalinks/Permalinks\";\nimport { MessagePreviewStore } from \"../../../stores/room-list/MessagePreviewStore\";\nimport dis from \"../../../dispatcher/dispatcher\";\nimport { ViewRoomPayload } from \"../../../dispatcher/payloads/ViewRoomPayload\";\nimport { Action } from \"../../../dispatcher/actions\";\nimport MessageEvent from \"../messages/MessageEvent\";\nimport PosthogTrackers from \"../../../PosthogTrackers.ts\";\n\n/**\n * The props for the {@link PinnedMessageBanner} component.\n */\ninterface PinnedMessageBannerProps {\n    /**\n     * The permalink creator to use.\n     */\n    permalinkCreator: RoomPermalinkCreator;\n    /**\n     * The room where the banner is displayed\n     */\n    room: Room;\n}\n\n/**\n * A banner that displays the pinned messages in a room.\n */\nexport function PinnedMessageBanner({ room, permalinkCreator }: PinnedMessageBannerProps): JSX.Element | null {\n    const pinnedEventIds = usePinnedEvents(room);\n    const pinnedEvents = useSortedFetchedPinnedEvents(room, pinnedEventIds);\n    const eventCount = pinnedEvents.length;\n    const isSinglePinnedEvent = eventCount === 1;\n\n    const [currentEventIndex, setCurrentEventIndex] = useState(eventCount - 1);\n    // When the number of pinned messages changes, we want to display the last message\n    useEffect(() => {\n        setCurrentEventIndex(() => eventCount - 1);\n    }, [eventCount]);\n\n    const pinnedEvent = pinnedEvents[currentEventIndex];\n    if (!pinnedEvent) return null;\n\n    const shouldUseMessageEvent = pinnedEvent.isRedacted() || pinnedEvent.isDecryptionFailure();\n\n    const onBannerClick = (): void => {\n        PosthogTrackers.trackInteraction(\"PinnedMessageBannerClick\");\n\n        // Scroll to the pinned message\n        dis.dispatch<ViewRoomPayload>({\n            action: Action.ViewRoom,\n            event_id: pinnedEvent.getId(),\n            highlighted: true,\n            room_id: room.roomId,\n            metricsTrigger: undefined, // room doesn't change\n        });\n\n        // Cycle through the pinned messages\n        // When we reach the first message, we go back to the last message\n        setCurrentEventIndex((currentEventIndex) => (--currentEventIndex === -1 ? eventCount - 1 : currentEventIndex));\n    };\n\n    return (\n        <div\n            className=\"mx_PinnedMessageBanner\"\n            data-single-message={isSinglePinnedEvent}\n            aria-label={_t(\"room|pinned_message_banner|description\")}\n            data-testid=\"pinned-message-banner\"\n        >\n            <button\n                aria-label={_t(\"room|pinned_message_banner|go_to_message\")}\n                type=\"button\"\n                className=\"mx_PinnedMessageBanner_main\"\n                onClick={onBannerClick}\n            >\n                <div className=\"mx_PinnedMessageBanner_content\">\n                    <Indicators count={eventCount} currentIndex={currentEventIndex} />\n                    <PinIcon width=\"20px\" height=\"20px\" className=\"mx_PinnedMessageBanner_PinIcon\" />\n                    {!isSinglePinnedEvent && (\n                        <div className=\"mx_PinnedMessageBanner_title\" data-testid=\"banner-counter\">\n                            {_t(\n                                \"room|pinned_message_banner|title\",\n                                {\n                                    index: currentEventIndex + 1,\n                                    length: eventCount,\n                                },\n                                { bold: (sub) => <span className=\"mx_PinnedMessageBanner_title_counter\">{sub}</span> },\n                            )}\n                        </div>\n                    )}\n                    <EventPreview pinnedEvent={pinnedEvent} />\n                    {/* In case of redacted event, we want to display the nice sentence of the message event like in the timeline or in the pinned message list */}\n                    {shouldUseMessageEvent && (\n                        <div className=\"mx_PinnedMessageBanner_redactedMessage\">\n                            <MessageEvent\n                                mxEvent={pinnedEvent}\n                                maxImageHeight={20}\n                                permalinkCreator={permalinkCreator}\n                                replacingEventId={pinnedEvent.replacingEventId()}\n                            />\n                        </div>\n                    )}\n                </div>\n            </button>\n            {!isSinglePinnedEvent && <BannerButton room={room} />}\n        </div>\n    );\n}\n\n/**\n * The props for the {@link EventPreview} component.\n */\ninterface EventPreviewProps {\n    /**\n     * The pinned event to display the preview for\n     */\n    pinnedEvent: MatrixEvent;\n}\n\n/**\n * A component that displays a preview for the pinned event.\n */\nfunction EventPreview({ pinnedEvent }: EventPreviewProps): JSX.Element | null {\n    const preview = useEventPreview(pinnedEvent);\n    if (!preview) return null;\n\n    const prefix = getPreviewPrefix(pinnedEvent.getType(), pinnedEvent.getContent().msgtype as MsgType);\n    if (!prefix)\n        return (\n            <span className=\"mx_PinnedMessageBanner_message\" data-testid=\"banner-message\">\n                {preview}\n            </span>\n        );\n\n    return (\n        <span className=\"mx_PinnedMessageBanner_message\" data-testid=\"banner-message\">\n            {_t(\n                \"room|pinned_message_banner|preview\",\n                {\n                    prefix,\n                    preview,\n                },\n                {\n                    bold: (sub) => <span className=\"mx_PinnedMessageBanner_prefix\">{sub}</span>,\n                },\n            )}\n        </span>\n    );\n}\n\n/**\n * Hooks to generate a preview for the pinned event.\n * @param pinnedEvent\n */\nfunction useEventPreview(pinnedEvent: MatrixEvent | null): string | null {\n    return useMemo(() => {\n        if (!pinnedEvent || pinnedEvent.isRedacted() || pinnedEvent.isDecryptionFailure()) return null;\n        return MessagePreviewStore.instance.generatePreviewForEvent(pinnedEvent);\n    }, [pinnedEvent]);\n}\n\n/**\n * Get the prefix for the preview based on the type and the message type.\n * @param type\n * @param msgType\n */\nfunction getPreviewPrefix(type: string, msgType: MsgType): string | null {\n    switch (type) {\n        case M_POLL_START.name:\n            return _t(\"room|pinned_message_banner|prefix|poll\");\n        default:\n    }\n\n    switch (msgType) {\n        case MsgType.Audio:\n            return _t(\"room|pinned_message_banner|prefix|audio\");\n        case MsgType.Image:\n            return _t(\"room|pinned_message_banner|prefix|image\");\n        case MsgType.Video:\n            return _t(\"room|pinned_message_banner|prefix|video\");\n        case MsgType.File:\n            return _t(\"room|pinned_message_banner|prefix|file\");\n        default:\n            return null;\n    }\n}\n\nconst MAX_INDICATORS = 3;\n\n/**\n * The props for the {@link IndicatorsProps} component.\n */\ninterface IndicatorsProps {\n    /**\n     * The number of messages pinned\n     */\n    count: number;\n    /**\n     * The current index of the pinned message\n     */\n    currentIndex: number;\n}\n\n/**\n * A component that displays vertical indicators for the pinned messages.\n */\nfunction Indicators({ count, currentIndex }: IndicatorsProps): JSX.Element {\n    // We only display a maximum of 3 indicators at one time.\n    // When there is more than 3 messages pinned, we will cycle through the indicators\n\n    // If there is only 2 messages pinned, we will display 2 indicators\n    // In case of 1 message pinned, the indicators are not displayed, see {@link PinnedMessageBanner} logic.\n    const numberOfIndicators = Math.min(count, MAX_INDICATORS);\n    // The index of the active indicator\n    const index = currentIndex % numberOfIndicators;\n\n    // We hide the indicators when we are on the last cycle and there are less than 3 remaining messages pinned\n    const numberOfCycles = Math.ceil(count / numberOfIndicators);\n    // If the current index is greater than the last cycle index, we are on the last cycle\n    const isLastCycle = currentIndex >= (numberOfCycles - 1) * MAX_INDICATORS;\n    // The index of the last message in the last cycle\n    const lastCycleIndex = numberOfIndicators - (numberOfCycles * numberOfIndicators - count);\n\n    return (\n        <div className=\"mx_PinnedMessageBanner_Indicators\">\n            {Array.from({ length: numberOfIndicators }).map((_, i) => (\n                <Indicator key={i} active={i === index} hidden={isLastCycle && lastCycleIndex <= i} />\n            ))}\n        </div>\n    );\n}\n\n/**\n * The props for the {@link Indicator} component.\n */\ninterface IndicatorProps {\n    /**\n     * Whether the indicator is active\n     */\n    active: boolean;\n    /**\n     * Whether the indicator is hidden\n     */\n    hidden: boolean;\n}\n\n/**\n * A component that displays a vertical indicator for a pinned message.\n */\nfunction Indicator({ active, hidden }: IndicatorProps): JSX.Element {\n    return (\n        <div\n            data-testid=\"banner-indicator\"\n            className={classNames(\"mx_PinnedMessageBanner_Indicator\", {\n                \"mx_PinnedMessageBanner_Indicator--active\": active,\n                \"mx_PinnedMessageBanner_Indicator--hidden\": hidden,\n            })}\n        />\n    );\n}\n\nfunction getRightPanelPhase(roomId: string): RightPanelPhases | null {\n    if (!RightPanelStore.instance.isOpenForRoom(roomId)) return null;\n    return RightPanelStore.instance.currentCard.phase;\n}\n\n/**\n * The props for the {@link BannerButton} component.\n */\ninterface BannerButtonProps {\n    /**\n     * The room where the banner is displayed\n     */\n    room: Room;\n}\n\n/**\n * A button that allows the user to view or close the list of pinned messages.\n */\nfunction BannerButton({ room }: BannerButtonProps): JSX.Element {\n    const [currentPhase, setCurrentPhase] = useState<RightPanelPhases | null>(getRightPanelPhase(room.roomId));\n    useEventEmitter(RightPanelStore.instance, UPDATE_EVENT, () => setCurrentPhase(getRightPanelPhase(room.roomId)));\n    const isPinnedMessagesPhase = currentPhase === RightPanelPhases.PinnedMessages;\n\n    return (\n        <Button\n            className=\"mx_PinnedMessageBanner_actions\"\n            kind=\"tertiary\"\n            onClick={() => {\n                if (isPinnedMessagesPhase) PosthogTrackers.trackInteraction(\"PinnedMessageBannerCloseListButton\");\n                else PosthogTrackers.trackInteraction(\"PinnedMessageBannerViewAllButton\");\n\n                RightPanelStore.instance.showOrHidePhase(RightPanelPhases.PinnedMessages);\n            }}\n        >\n            {isPinnedMessagesPhase\n                ? _t(\"room|pinned_message_banner|button_close_list\")\n                : _t(\"room|pinned_message_banner|button_view_all\")}\n        </Button>\n    );\n}\n"],"mappings":"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","ignoreList":[]}