matrix-react-sdk
Version:
SDK for matrix.org using React
250 lines (243 loc) • 45.2 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireWildcard(require("react"));
var _matrix = require("matrix-js-sdk/src/matrix");
var _logger = require("matrix-js-sdk/src/logger");
var _files = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/files"));
var _MatrixClientPeg = require("../../MatrixClientPeg");
var _EventIndexPeg = _interopRequireDefault(require("../../indexing/EventIndexPeg"));
var _languageHandler = require("../../languageHandler");
var _SearchWarning = _interopRequireWildcard(require("../views/elements/SearchWarning"));
var _BaseCard = _interopRequireDefault(require("../views/right_panel/BaseCard"));
var _TimelinePanel = _interopRequireDefault(require("./TimelinePanel"));
var _Spinner = _interopRequireDefault(require("../views/elements/Spinner"));
var _Layout = require("../../settings/enums/Layout");
var _RoomContext = _interopRequireWildcard(require("../../contexts/RoomContext"));
var _Measured = _interopRequireDefault(require("../views/elements/Measured"));
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 2019-2022 The Matrix.org Foundation C.I.C.
Copyright 2016 OpenMarket Ltd
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
/*
* Component which shows the filtered file using a TimelinePanel
*/
class FilePanel extends _react.default.Component {
constructor(...args) {
super(...args);
// This is used to track if a decrypted event was a live event and should be
// added to the timeline.
(0, _defineProperty2.default)(this, "decryptingEvents", new Set());
(0, _defineProperty2.default)(this, "noRoom", false);
(0, _defineProperty2.default)(this, "card", /*#__PURE__*/(0, _react.createRef)());
(0, _defineProperty2.default)(this, "state", {
timelineSet: null,
narrow: false
});
(0, _defineProperty2.default)(this, "onRoomTimeline", (ev, room, toStartOfTimeline, removed, data) => {
if (room?.roomId !== this.props.roomId) return;
if (toStartOfTimeline || !data || !data.liveEvent || ev.isRedacted()) return;
const client = _MatrixClientPeg.MatrixClientPeg.safeGet();
client.decryptEventIfNeeded(ev);
if (ev.isBeingDecrypted()) {
this.decryptingEvents.add(ev.getId());
} else {
this.addEncryptedLiveEvent(ev);
}
});
(0, _defineProperty2.default)(this, "onEventDecrypted", (ev, err) => {
if (ev.getRoomId() !== this.props.roomId) return;
const eventId = ev.getId();
if (!this.decryptingEvents.delete(eventId)) return;
if (err) return;
this.addEncryptedLiveEvent(ev);
});
(0, _defineProperty2.default)(this, "onPaginationRequest", (timelineWindow, direction, limit) => {
const client = _MatrixClientPeg.MatrixClientPeg.safeGet();
const eventIndex = _EventIndexPeg.default.get();
const roomId = this.props.roomId;
const room = client.getRoom(roomId);
// We override the pagination request for encrypted rooms so that we ask
// the event index to fulfill the pagination request. Asking the server
// to paginate won't ever work since the server can't correctly filter
// out events containing URLs
if (room && client.isRoomEncrypted(roomId) && eventIndex !== null) {
return eventIndex.paginateTimelineWindow(room, timelineWindow, direction, limit);
} else {
return timelineWindow.paginate(direction, limit);
}
});
(0, _defineProperty2.default)(this, "onMeasurement", narrow => {
this.setState({
narrow
});
});
}
addEncryptedLiveEvent(ev) {
if (!this.state.timelineSet) return;
const timeline = this.state.timelineSet.getLiveTimeline();
if (ev.getType() !== "m.room.message") return;
if (!["m.file", "m.image", "m.video", "m.audio"].includes(ev.getContent().msgtype)) {
return;
}
if (!this.state.timelineSet.eventIdToTimeline(ev.getId())) {
this.state.timelineSet.addEventToTimeline(ev, timeline, false);
}
}
async componentDidMount() {
const client = _MatrixClientPeg.MatrixClientPeg.safeGet();
await this.updateTimelineSet(this.props.roomId);
if (!client.isRoomEncrypted(this.props.roomId)) return;
// The timelineSets filter makes sure that encrypted events that contain
// URLs never get added to the timeline, even if they are live events.
// These methods are here to manually listen for such events and add
// them despite the filter's best efforts.
//
// We do this only for encrypted rooms and if an event index exists,
// this could be made more general in the future or the filter logic
// could be fixed.
if (_EventIndexPeg.default.get() !== null) {
client.on(_matrix.RoomEvent.Timeline, this.onRoomTimeline);
client.on(_matrix.MatrixEventEvent.Decrypted, this.onEventDecrypted);
}
}
componentWillUnmount() {
const client = _MatrixClientPeg.MatrixClientPeg.get();
if (client === null) return;
if (!client.isRoomEncrypted(this.props.roomId)) return;
if (_EventIndexPeg.default.get() !== null) {
client.removeListener(_matrix.RoomEvent.Timeline, this.onRoomTimeline);
client.removeListener(_matrix.MatrixEventEvent.Decrypted, this.onEventDecrypted);
}
}
async fetchFileEventsServer(room) {
const client = _MatrixClientPeg.MatrixClientPeg.safeGet();
const filter = new _matrix.Filter(client.getSafeUserId());
filter.setDefinition({
room: {
timeline: {
contains_url: true,
types: ["m.room.message"]
}
}
});
filter.filterId = await client.getOrCreateFilter("FILTER_FILES_" + client.credentials.userId, filter);
return room.getOrCreateFilteredTimelineSet(filter);
}
async updateTimelineSet(roomId) {
const client = _MatrixClientPeg.MatrixClientPeg.safeGet();
const room = client.getRoom(roomId);
const eventIndex = _EventIndexPeg.default.get();
this.noRoom = !room;
if (room) {
let timelineSet;
try {
timelineSet = await this.fetchFileEventsServer(room);
// If this room is encrypted the file panel won't be populated
// correctly since the defined filter doesn't support encrypted
// events and the server can't check if encrypted events contain
// URLs.
//
// This is where our event index comes into place, we ask the
// event index to populate the timelineSet for us. This call
// will add 10 events to the live timeline of the set. More can
// be requested using pagination.
if (client.isRoomEncrypted(roomId) && eventIndex !== null) {
const timeline = timelineSet.getLiveTimeline();
await eventIndex.populateFileTimeline(timelineSet, timeline, room, 10);
}
this.setState({
timelineSet: timelineSet
});
} catch (error) {
_logger.logger.error("Failed to get or create file panel filter", error);
}
} else {
_logger.logger.error("Failed to add filtered timelineSet for FilePanel as no room!");
}
}
render() {
if (_MatrixClientPeg.MatrixClientPeg.safeGet().isGuest()) {
return /*#__PURE__*/_react.default.createElement(_BaseCard.default, {
className: "mx_FilePanel mx_RoomView_messageListWrapper",
onClose: this.props.onClose,
header: (0, _languageHandler._t)("right_panel|files_button")
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_RoomView_empty"
}, (0, _languageHandler._t)("file_panel|guest_note", {}, {
a: sub => /*#__PURE__*/_react.default.createElement("a", {
href: "#/register",
key: "sub"
}, sub)
})));
} else if (this.noRoom) {
return /*#__PURE__*/_react.default.createElement(_BaseCard.default, {
className: "mx_FilePanel mx_RoomView_messageListWrapper",
onClose: this.props.onClose,
header: (0, _languageHandler._t)("right_panel|files_button")
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_RoomView_empty"
}, (0, _languageHandler._t)("file_panel|peek_note")));
}
// wrap a TimelinePanel with the jump-to-event bits turned off.
const emptyState = /*#__PURE__*/_react.default.createElement(_EmptyState.default, {
Icon: _files.default,
title: (0, _languageHandler._t)("file_panel|empty_heading"),
description: (0, _languageHandler._t)("file_panel|empty_description")
});
const isRoomEncrypted = this.noRoom ? false : _MatrixClientPeg.MatrixClientPeg.safeGet().isRoomEncrypted(this.props.roomId);
if (this.state.timelineSet) {
return /*#__PURE__*/_react.default.createElement(_RoomContext.default.Provider, {
value: _objectSpread(_objectSpread({}, this.context), {}, {
timelineRenderingType: _RoomContext.TimelineRenderingType.File,
narrow: this.state.narrow
})
}, /*#__PURE__*/_react.default.createElement(_BaseCard.default, {
className: "mx_FilePanel",
onClose: this.props.onClose,
withoutScrollContainer: true,
ref: this.card,
header: (0, _languageHandler._t)("right_panel|files_button")
}, this.card.current && /*#__PURE__*/_react.default.createElement(_Measured.default, {
sensor: this.card.current,
onMeasurement: this.onMeasurement
}), /*#__PURE__*/_react.default.createElement(_SearchWarning.default, {
isRoomEncrypted: isRoomEncrypted,
kind: _SearchWarning.WarningKind.Files
}), /*#__PURE__*/_react.default.createElement(_TimelinePanel.default, {
manageReadReceipts: false,
manageReadMarkers: false,
timelineSet: this.state.timelineSet,
showUrlPreview: false,
onPaginationRequest: this.onPaginationRequest,
resizeNotifier: this.props.resizeNotifier,
empty: emptyState,
layout: _Layout.Layout.Group
})));
} else {
return /*#__PURE__*/_react.default.createElement(_RoomContext.default.Provider, {
value: _objectSpread(_objectSpread({}, this.context), {}, {
timelineRenderingType: _RoomContext.TimelineRenderingType.File
})
}, /*#__PURE__*/_react.default.createElement(_BaseCard.default, {
className: "mx_FilePanel",
onClose: this.props.onClose,
header: (0, _languageHandler._t)("right_panel|files_button")
}, /*#__PURE__*/_react.default.createElement(_Spinner.default, null)));
}
}
}
(0, _defineProperty2.default)(FilePanel, "contextType", _RoomContext.default);
var _default = exports.default = FilePanel;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_matrix","_logger","_files","_interopRequireDefault","_MatrixClientPeg","_EventIndexPeg","_languageHandler","_SearchWarning","_BaseCard","_TimelinePanel","_Spinner","_Layout","_RoomContext","_Measured","_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","FilePanel","React","Component","constructor","args","Set","createRef","timelineSet","narrow","ev","room","toStartOfTimeline","removed","data","roomId","props","liveEvent","isRedacted","client","MatrixClientPeg","safeGet","decryptEventIfNeeded","isBeingDecrypted","decryptingEvents","add","getId","addEncryptedLiveEvent","err","getRoomId","eventId","delete","timelineWindow","direction","limit","eventIndex","EventIndexPeg","getRoom","isRoomEncrypted","paginateTimelineWindow","paginate","setState","state","timeline","getLiveTimeline","getType","includes","getContent","msgtype","eventIdToTimeline","addEventToTimeline","componentDidMount","updateTimelineSet","on","RoomEvent","Timeline","onRoomTimeline","MatrixEventEvent","Decrypted","onEventDecrypted","componentWillUnmount","removeListener","fetchFileEventsServer","Filter","getSafeUserId","setDefinition","contains_url","types","filterId","getOrCreateFilter","credentials","userId","getOrCreateFilteredTimelineSet","noRoom","populateFileTimeline","error","logger","render","isGuest","createElement","className","onClose","header","_t","sub","href","key","emptyState","Icon","FilesIcon","title","description","Provider","value","context","timelineRenderingType","TimelineRenderingType","File","withoutScrollContainer","ref","card","current","sensor","onMeasurement","kind","WarningKind","Files","manageReadReceipts","manageReadMarkers","showUrlPreview","onPaginationRequest","resizeNotifier","empty","layout","Layout","Group","RoomContext","_default","exports"],"sources":["../../../src/components/structures/FilePanel.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2019-2022 The Matrix.org Foundation C.I.C.\nCopyright 2016 OpenMarket Ltd\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 React, { createRef } from \"react\";\nimport {\n    Filter,\n    EventTimelineSet,\n    IRoomTimelineData,\n    Direction,\n    MatrixEvent,\n    MatrixEventEvent,\n    Room,\n    RoomEvent,\n    TimelineWindow,\n} from \"matrix-js-sdk/src/matrix\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\nimport FilesIcon from \"@vector-im/compound-design-tokens/assets/web/icons/files\";\n\nimport { MatrixClientPeg } from \"../../MatrixClientPeg\";\nimport EventIndexPeg from \"../../indexing/EventIndexPeg\";\nimport { _t } from \"../../languageHandler\";\nimport SearchWarning, { WarningKind } from \"../views/elements/SearchWarning\";\nimport BaseCard from \"../views/right_panel/BaseCard\";\nimport ResizeNotifier from \"../../utils/ResizeNotifier\";\nimport TimelinePanel from \"./TimelinePanel\";\nimport Spinner from \"../views/elements/Spinner\";\nimport { Layout } from \"../../settings/enums/Layout\";\nimport RoomContext, { TimelineRenderingType } from \"../../contexts/RoomContext\";\nimport Measured from \"../views/elements/Measured\";\nimport EmptyState from \"../views/right_panel/EmptyState\";\n\ninterface IProps {\n    roomId: string;\n    onClose: () => void;\n    resizeNotifier: ResizeNotifier;\n}\n\ninterface IState {\n    timelineSet: EventTimelineSet | null;\n    narrow: boolean;\n}\n\n/*\n * Component which shows the filtered file using a TimelinePanel\n */\nclass FilePanel extends React.Component<IProps, IState> {\n    public static contextType = RoomContext;\n    public declare context: React.ContextType<typeof RoomContext>;\n\n    // This is used to track if a decrypted event was a live event and should be\n    // added to the timeline.\n    private decryptingEvents = new Set<string>();\n    public noRoom = false;\n    private card = createRef<HTMLDivElement>();\n\n    public state: IState = {\n        timelineSet: null,\n        narrow: false,\n    };\n\n    private onRoomTimeline = (\n        ev: MatrixEvent,\n        room: Room | undefined,\n        toStartOfTimeline: boolean | undefined,\n        removed: boolean,\n        data: IRoomTimelineData,\n    ): void => {\n        if (room?.roomId !== this.props.roomId) return;\n        if (toStartOfTimeline || !data || !data.liveEvent || ev.isRedacted()) return;\n\n        const client = MatrixClientPeg.safeGet();\n        client.decryptEventIfNeeded(ev);\n\n        if (ev.isBeingDecrypted()) {\n            this.decryptingEvents.add(ev.getId()!);\n        } else {\n            this.addEncryptedLiveEvent(ev);\n        }\n    };\n\n    private onEventDecrypted = (ev: MatrixEvent, err?: any): void => {\n        if (ev.getRoomId() !== this.props.roomId) return;\n        const eventId = ev.getId()!;\n\n        if (!this.decryptingEvents.delete(eventId)) return;\n        if (err) return;\n\n        this.addEncryptedLiveEvent(ev);\n    };\n\n    public addEncryptedLiveEvent(ev: MatrixEvent): void {\n        if (!this.state.timelineSet) return;\n\n        const timeline = this.state.timelineSet.getLiveTimeline();\n        if (ev.getType() !== \"m.room.message\") return;\n        if (![\"m.file\", \"m.image\", \"m.video\", \"m.audio\"].includes(ev.getContent().msgtype!)) {\n            return;\n        }\n\n        if (!this.state.timelineSet.eventIdToTimeline(ev.getId()!)) {\n            this.state.timelineSet.addEventToTimeline(ev, timeline, false);\n        }\n    }\n\n    public async componentDidMount(): Promise<void> {\n        const client = MatrixClientPeg.safeGet();\n\n        await this.updateTimelineSet(this.props.roomId);\n\n        if (!client.isRoomEncrypted(this.props.roomId)) return;\n\n        // The timelineSets filter makes sure that encrypted events that contain\n        // URLs never get added to the timeline, even if they are live events.\n        // These methods are here to manually listen for such events and add\n        // them despite the filter's best efforts.\n        //\n        // We do this only for encrypted rooms and if an event index exists,\n        // this could be made more general in the future or the filter logic\n        // could be fixed.\n        if (EventIndexPeg.get() !== null) {\n            client.on(RoomEvent.Timeline, this.onRoomTimeline);\n            client.on(MatrixEventEvent.Decrypted, this.onEventDecrypted);\n        }\n    }\n\n    public componentWillUnmount(): void {\n        const client = MatrixClientPeg.get();\n        if (client === null) return;\n\n        if (!client.isRoomEncrypted(this.props.roomId)) return;\n\n        if (EventIndexPeg.get() !== null) {\n            client.removeListener(RoomEvent.Timeline, this.onRoomTimeline);\n            client.removeListener(MatrixEventEvent.Decrypted, this.onEventDecrypted);\n        }\n    }\n\n    public async fetchFileEventsServer(room: Room): Promise<EventTimelineSet> {\n        const client = MatrixClientPeg.safeGet();\n\n        const filter = new Filter(client.getSafeUserId());\n        filter.setDefinition({\n            room: {\n                timeline: {\n                    contains_url: true,\n                    types: [\"m.room.message\"],\n                },\n            },\n        });\n\n        filter.filterId = await client.getOrCreateFilter(\"FILTER_FILES_\" + client.credentials.userId, filter);\n        return room.getOrCreateFilteredTimelineSet(filter);\n    }\n\n    private onPaginationRequest = (\n        timelineWindow: TimelineWindow,\n        direction: Direction,\n        limit: number,\n    ): Promise<boolean> => {\n        const client = MatrixClientPeg.safeGet();\n        const eventIndex = EventIndexPeg.get();\n        const roomId = this.props.roomId;\n\n        const room = client.getRoom(roomId);\n\n        // We override the pagination request for encrypted rooms so that we ask\n        // the event index to fulfill the pagination request. Asking the server\n        // to paginate won't ever work since the server can't correctly filter\n        // out events containing URLs\n        if (room && client.isRoomEncrypted(roomId) && eventIndex !== null) {\n            return eventIndex.paginateTimelineWindow(room, timelineWindow, direction, limit);\n        } else {\n            return timelineWindow.paginate(direction, limit);\n        }\n    };\n\n    private onMeasurement = (narrow: boolean): void => {\n        this.setState({ narrow });\n    };\n\n    public async updateTimelineSet(roomId: string): Promise<void> {\n        const client = MatrixClientPeg.safeGet();\n        const room = client.getRoom(roomId);\n        const eventIndex = EventIndexPeg.get();\n\n        this.noRoom = !room;\n\n        if (room) {\n            let timelineSet;\n\n            try {\n                timelineSet = await this.fetchFileEventsServer(room);\n\n                // If this room is encrypted the file panel won't be populated\n                // correctly since the defined filter doesn't support encrypted\n                // events and the server can't check if encrypted events contain\n                // URLs.\n                //\n                // This is where our event index comes into place, we ask the\n                // event index to populate the timelineSet for us. This call\n                // will add 10 events to the live timeline of the set. More can\n                // be requested using pagination.\n                if (client.isRoomEncrypted(roomId) && eventIndex !== null) {\n                    const timeline = timelineSet.getLiveTimeline();\n                    await eventIndex.populateFileTimeline(timelineSet, timeline, room, 10);\n                }\n\n                this.setState({ timelineSet: timelineSet });\n            } catch (error) {\n                logger.error(\"Failed to get or create file panel filter\", error);\n            }\n        } else {\n            logger.error(\"Failed to add filtered timelineSet for FilePanel as no room!\");\n        }\n    }\n\n    public render(): React.ReactNode {\n        if (MatrixClientPeg.safeGet().isGuest()) {\n            return (\n                <BaseCard\n                    className=\"mx_FilePanel mx_RoomView_messageListWrapper\"\n                    onClose={this.props.onClose}\n                    header={_t(\"right_panel|files_button\")}\n                >\n                    <div className=\"mx_RoomView_empty\">\n                        {_t(\n                            \"file_panel|guest_note\",\n                            {},\n                            {\n                                a: (sub) => (\n                                    <a href=\"#/register\" key=\"sub\">\n                                        {sub}\n                                    </a>\n                                ),\n                            },\n                        )}\n                    </div>\n                </BaseCard>\n            );\n        } else if (this.noRoom) {\n            return (\n                <BaseCard\n                    className=\"mx_FilePanel mx_RoomView_messageListWrapper\"\n                    onClose={this.props.onClose}\n                    header={_t(\"right_panel|files_button\")}\n                >\n                    <div className=\"mx_RoomView_empty\">{_t(\"file_panel|peek_note\")}</div>\n                </BaseCard>\n            );\n        }\n\n        // wrap a TimelinePanel with the jump-to-event bits turned off.\n\n        const emptyState = (\n            <EmptyState\n                Icon={FilesIcon}\n                title={_t(\"file_panel|empty_heading\")}\n                description={_t(\"file_panel|empty_description\")}\n            />\n        );\n\n        const isRoomEncrypted = this.noRoom ? false : MatrixClientPeg.safeGet().isRoomEncrypted(this.props.roomId);\n\n        if (this.state.timelineSet) {\n            return (\n                <RoomContext.Provider\n                    value={{\n                        ...this.context,\n                        timelineRenderingType: TimelineRenderingType.File,\n                        narrow: this.state.narrow,\n                    }}\n                >\n                    <BaseCard\n                        className=\"mx_FilePanel\"\n                        onClose={this.props.onClose}\n                        withoutScrollContainer\n                        ref={this.card}\n                        header={_t(\"right_panel|files_button\")}\n                    >\n                        {this.card.current && (\n                            <Measured sensor={this.card.current} onMeasurement={this.onMeasurement} />\n                        )}\n                        <SearchWarning isRoomEncrypted={isRoomEncrypted} kind={WarningKind.Files} />\n                        <TimelinePanel\n                            manageReadReceipts={false}\n                            manageReadMarkers={false}\n                            timelineSet={this.state.timelineSet}\n                            showUrlPreview={false}\n                            onPaginationRequest={this.onPaginationRequest}\n                            resizeNotifier={this.props.resizeNotifier}\n                            empty={emptyState}\n                            layout={Layout.Group}\n                        />\n                    </BaseCard>\n                </RoomContext.Provider>\n            );\n        } else {\n            return (\n                <RoomContext.Provider\n                    value={{\n                        ...this.context,\n                        timelineRenderingType: TimelineRenderingType.File,\n                    }}\n                >\n                    <BaseCard\n                        className=\"mx_FilePanel\"\n                        onClose={this.props.onClose}\n                        header={_t(\"right_panel|files_button\")}\n                    >\n                        <Spinner />\n                    </BaseCard>\n                </RoomContext.Provider>\n            );\n        }\n    }\n}\n\nexport default FilePanel;\n"],"mappings":";;;;;;;;AASA,IAAAA,MAAA,GAAAC,uBAAA,CAAAC,OAAA;AACA,IAAAC,OAAA,GAAAD,OAAA;AAWA,IAAAE,OAAA,GAAAF,OAAA;AACA,IAAAG,MAAA,GAAAC,sBAAA,CAAAJ,OAAA;AAEA,IAAAK,gBAAA,GAAAL,OAAA;AACA,IAAAM,cAAA,GAAAF,sBAAA,CAAAJ,OAAA;AACA,IAAAO,gBAAA,GAAAP,OAAA;AACA,IAAAQ,cAAA,GAAAT,uBAAA,CAAAC,OAAA;AACA,IAAAS,SAAA,GAAAL,sBAAA,CAAAJ,OAAA;AAEA,IAAAU,cAAA,GAAAN,sBAAA,CAAAJ,OAAA;AACA,IAAAW,QAAA,GAAAP,sBAAA,CAAAJ,OAAA;AACA,IAAAY,OAAA,GAAAZ,OAAA;AACA,IAAAa,YAAA,GAAAd,uBAAA,CAAAC,OAAA;AACA,IAAAc,SAAA,GAAAV,sBAAA,CAAAJ,OAAA;AACA,IAAAe,WAAA,GAAAX,sBAAA,CAAAJ,OAAA;AAAyD,SAAAgB,yBAAAC,CAAA,6BAAAC,OAAA,mBAAAC,CAAA,OAAAD,OAAA,IAAAE,CAAA,OAAAF,OAAA,YAAAF,wBAAA,YAAAA,CAAAC,CAAA,WAAAA,CAAA,GAAAG,CAAA,GAAAD,CAAA,KAAAF,CAAA;AAAA,SAAAlB,wBAAAkB,CAAA,EAAAE,CAAA,SAAAA,CAAA,IAAAF,CAAA,IAAAA,CAAA,CAAAI,UAAA,SAAAJ,CAAA,eAAAA,CAAA,uBAAAA,CAAA,yBAAAA,CAAA,WAAAK,OAAA,EAAAL,CAAA,QAAAG,CAAA,GAAAJ,wBAAA,CAAAG,CAAA,OAAAC,CAAA,IAAAA,CAAA,CAAAG,GAAA,CAAAN,CAAA,UAAAG,CAAA,CAAAI,GAAA,CAAAP,CAAA,OAAAQ,CAAA,KAAAC,SAAA,UAAAC,CAAA,GAAAC,MAAA,CAAAC,cAAA,IAAAD,MAAA,CAAAE,wBAAA,WAAAC,CAAA,IAAAd,CAAA,oBAAAc,CAAA,OAAAC,cAAA,CAAAC,IAAA,CAAAhB,CAAA,EAAAc,CAAA,SAAAG,CAAA,GAAAP,CAAA,GAAAC,MAAA,CAAAE,wBAAA,CAAAb,CAAA,EAAAc,CAAA,UAAAG,CAAA,KAAAA,CAAA,CAAAV,GAAA,IAAAU,CAAA,CAAAC,GAAA,IAAAP,MAAA,CAAAC,cAAA,CAAAJ,CAAA,EAAAM,CAAA,EAAAG,CAAA,IAAAT,CAAA,CAAAM,CAAA,IAAAd,CAAA,CAAAc,CAAA,YAAAN,CAAA,CAAAH,OAAA,GAAAL,CAAA,EAAAG,CAAA,IAAAA,CAAA,CAAAe,GAAA,CAAAlB,CAAA,EAAAQ,CAAA,GAAAA,CAAA;AAAA,SAAAW,QAAAnB,CAAA,EAAAE,CAAA,QAAAC,CAAA,GAAAQ,MAAA,CAAAS,IAAA,CAAApB,CAAA,OAAAW,MAAA,CAAAU,qBAAA,QAAAC,CAAA,GAAAX,MAAA,CAAAU,qBAAA,CAAArB,CAAA,GAAAE,CAAA,KAAAoB,CAAA,GAAAA,CAAA,CAAAC,MAAA,WAAArB,CAAA,WAAAS,MAAA,CAAAE,wBAAA,CAAAb,CAAA,EAAAE,CAAA,EAAAsB,UAAA,OAAArB,CAAA,CAAAsB,IAAA,CAAAC,KAAA,CAAAvB,CAAA,EAAAmB,CAAA,YAAAnB,CAAA;AAAA,SAAAwB,cAAA3B,CAAA,aAAAE,CAAA,MAAAA,CAAA,GAAA0B,SAAA,CAAAC,MAAA,EAAA3B,CAAA,UAAAC,CAAA,WAAAyB,SAAA,CAAA1B,CAAA,IAAA0B,SAAA,CAAA1B,CAAA,QAAAA,CAAA,OAAAiB,OAAA,CAAAR,MAAA,CAAAR,CAAA,OAAA2B,OAAA,WAAA5B,CAAA,QAAA6B,gBAAA,CAAA1B,OAAA,EAAAL,CAAA,EAAAE,CAAA,EAAAC,CAAA,CAAAD,CAAA,SAAAS,MAAA,CAAAqB,yBAAA,GAAArB,MAAA,CAAAsB,gBAAA,CAAAjC,CAAA,EAAAW,MAAA,CAAAqB,yBAAA,CAAA7B,CAAA,KAAAgB,OAAA,CAAAR,MAAA,CAAAR,CAAA,GAAA2B,OAAA,WAAA5B,CAAA,IAAAS,MAAA,CAAAC,cAAA,CAAAZ,CAAA,EAAAE,CAAA,EAAAS,MAAA,CAAAE,wBAAA,CAAAV,CAAA,EAAAD,CAAA,iBAAAF,CAAA,IAnCzD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAyCA;AACA;AACA;AACA,MAAMkC,SAAS,SAASC,cAAK,CAACC,SAAS,CAAiB;EAAAC,YAAA,GAAAC,IAAA;IAAA,SAAAA,IAAA;IAIpD;IACA;IAAA,IAAAP,gBAAA,CAAA1B,OAAA,4BAC2B,IAAIkC,GAAG,CAAS,CAAC;IAAA,IAAAR,gBAAA,CAAA1B,OAAA,kBAC5B,KAAK;IAAA,IAAA0B,gBAAA,CAAA1B,OAAA,6BACN,IAAAmC,gBAAS,EAAiB,CAAC;IAAA,IAAAT,gBAAA,CAAA1B,OAAA,iBAEnB;MACnBoC,WAAW,EAAE,IAAI;MACjBC,MAAM,EAAE;IACZ,CAAC;IAAA,IAAAX,gBAAA,CAAA1B,OAAA,0BAEwB,CACrBsC,EAAe,EACfC,IAAsB,EACtBC,iBAAsC,EACtCC,OAAgB,EAChBC,IAAuB,KAChB;MACP,IAAIH,IAAI,EAAEI,MAAM,KAAK,IAAI,CAACC,KAAK,CAACD,MAAM,EAAE;MACxC,IAAIH,iBAAiB,IAAI,CAACE,IAAI,IAAI,CAACA,IAAI,CAACG,SAAS,IAAIP,EAAE,CAACQ,UAAU,CAAC,CAAC,EAAE;MAEtE,MAAMC,MAAM,GAAGC,gCAAe,CAACC,OAAO,CAAC,CAAC;MACxCF,MAAM,CAACG,oBAAoB,CAACZ,EAAE,CAAC;MAE/B,IAAIA,EAAE,CAACa,gBAAgB,CAAC,CAAC,EAAE;QACvB,IAAI,CAACC,gBAAgB,CAACC,GAAG,CAACf,EAAE,CAACgB,KAAK,CAAC,CAAE,CAAC;MAC1C,CAAC,MAAM;QACH,IAAI,CAACC,qBAAqB,CAACjB,EAAE,CAAC;MAClC;IACJ,CAAC;IAAA,IAAAZ,gBAAA,CAAA1B,OAAA,4BAE0B,CAACsC,EAAe,EAAEkB,GAAS,KAAW;MAC7D,IAAIlB,EAAE,CAACmB,SAAS,CAAC,CAAC,KAAK,IAAI,CAACb,KAAK,CAACD,MAAM,EAAE;MAC1C,MAAMe,OAAO,GAAGpB,EAAE,CAACgB,KAAK,CAAC,CAAE;MAE3B,IAAI,CAAC,IAAI,CAACF,gBAAgB,CAACO,MAAM,CAACD,OAAO,CAAC,EAAE;MAC5C,IAAIF,GAAG,EAAE;MAET,IAAI,CAACD,qBAAqB,CAACjB,EAAE,CAAC;IAClC,CAAC;IAAA,IAAAZ,gBAAA,CAAA1B,OAAA,+BAkE6B,CAC1B4D,cAA8B,EAC9BC,SAAoB,EACpBC,KAAa,KACM;MACnB,MAAMf,MAAM,GAAGC,gCAAe,CAACC,OAAO,CAAC,CAAC;MACxC,MAAMc,UAAU,GAAGC,sBAAa,CAAC9D,GAAG,CAAC,CAAC;MACtC,MAAMyC,MAAM,GAAG,IAAI,CAACC,KAAK,CAACD,MAAM;MAEhC,MAAMJ,IAAI,GAAGQ,MAAM,CAACkB,OAAO,CAACtB,MAAM,CAAC;;MAEnC;MACA;MACA;MACA;MACA,IAAIJ,IAAI,IAAIQ,MAAM,CAACmB,eAAe,CAACvB,MAAM,CAAC,IAAIoB,UAAU,KAAK,IAAI,EAAE;QAC/D,OAAOA,UAAU,CAACI,sBAAsB,CAAC5B,IAAI,EAAEqB,cAAc,EAAEC,SAAS,EAAEC,KAAK,CAAC;MACpF,CAAC,MAAM;QACH,OAAOF,cAAc,CAACQ,QAAQ,CAACP,SAAS,EAAEC,KAAK,CAAC;MACpD;IACJ,CAAC;IAAA,IAAApC,gBAAA,CAAA1B,OAAA,yBAEwBqC,MAAe,IAAW;MAC/C,IAAI,CAACgC,QAAQ,CAAC;QAAEhC;MAAO,CAAC,CAAC;IAC7B,CAAC;EAAA;EAxFMkB,qBAAqBA,CAACjB,EAAe,EAAQ;IAChD,IAAI,CAAC,IAAI,CAACgC,KAAK,CAAClC,WAAW,EAAE;IAE7B,MAAMmC,QAAQ,GAAG,IAAI,CAACD,KAAK,CAAClC,WAAW,CAACoC,eAAe,CAAC,CAAC;IACzD,IAAIlC,EAAE,CAACmC,OAAO,CAAC,CAAC,KAAK,gBAAgB,EAAE;IACvC,IAAI,CAAC,CAAC,QAAQ,EAAE,SAAS,EAAE,SAAS,EAAE,SAAS,CAAC,CAACC,QAAQ,CAACpC,EAAE,CAACqC,UAAU,CAAC,CAAC,CAACC,OAAQ,CAAC,EAAE;MACjF;IACJ;IAEA,IAAI,CAAC,IAAI,CAACN,KAAK,CAAClC,WAAW,CAACyC,iBAAiB,CAACvC,EAAE,CAACgB,KAAK,CAAC,CAAE,CAAC,EAAE;MACxD,IAAI,CAACgB,KAAK,CAAClC,WAAW,CAAC0C,kBAAkB,CAACxC,EAAE,EAAEiC,QAAQ,EAAE,KAAK,CAAC;IAClE;EACJ;EAEA,MAAaQ,iBAAiBA,CAAA,EAAkB;IAC5C,MAAMhC,MAAM,GAAGC,gCAAe,CAACC,OAAO,CAAC,CAAC;IAExC,MAAM,IAAI,CAAC+B,iBAAiB,CAAC,IAAI,CAACpC,KAAK,CAACD,MAAM,CAAC;IAE/C,IAAI,CAACI,MAAM,CAACmB,eAAe,CAAC,IAAI,CAACtB,KAAK,CAACD,MAAM,CAAC,EAAE;;IAEhD;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA,IAAIqB,sBAAa,CAAC9D,GAAG,CAAC,CAAC,KAAK,IAAI,EAAE;MAC9B6C,MAAM,CAACkC,EAAE,CAACC,iBAAS,CAACC,QAAQ,EAAE,IAAI,CAACC,cAAc,CAAC;MAClDrC,MAAM,CAACkC,EAAE,CAACI,wBAAgB,CAACC,SAAS,EAAE,IAAI,CAACC,gBAAgB,CAAC;IAChE;EACJ;EAEOC,oBAAoBA,CAAA,EAAS;IAChC,MAAMzC,MAAM,GAAGC,gCAAe,CAAC9C,GAAG,CAAC,CAAC;IACpC,IAAI6C,MAAM,KAAK,IAAI,EAAE;IAErB,IAAI,CAACA,MAAM,CAACmB,eAAe,CAAC,IAAI,CAACtB,KAAK,CAACD,MAAM,CAAC,EAAE;IAEhD,IAAIqB,sBAAa,CAAC9D,GAAG,CAAC,CAAC,KAAK,IAAI,EAAE;MAC9B6C,MAAM,CAAC0C,cAAc,CAACP,iBAAS,CAACC,QAAQ,EAAE,IAAI,CAACC,cAAc,CAAC;MAC9DrC,MAAM,CAAC0C,cAAc,CAACJ,wBAAgB,CAACC,SAAS,EAAE,IAAI,CAACC,gBAAgB,CAAC;IAC5E;EACJ;EAEA,MAAaG,qBAAqBA,CAACnD,IAAU,EAA6B;IACtE,MAAMQ,MAAM,GAAGC,gCAAe,CAACC,OAAO,CAAC,CAAC;IAExC,MAAM/B,MAAM,GAAG,IAAIyE,cAAM,CAAC5C,MAAM,CAAC6C,aAAa,CAAC,CAAC,CAAC;IACjD1E,MAAM,CAAC2E,aAAa,CAAC;MACjBtD,IAAI,EAAE;QACFgC,QAAQ,EAAE;UACNuB,YAAY,EAAE,IAAI;UAClBC,KAAK,EAAE,CAAC,gBAAgB;QAC5B;MACJ;IACJ,CAAC,CAAC;IAEF7E,MAAM,CAAC8E,QAAQ,GAAG,MAAMjD,MAAM,CAACkD,iBAAiB,CAAC,eAAe,GAAGlD,MAAM,CAACmD,WAAW,CAACC,MAAM,EAAEjF,MAAM,CAAC;IACrG,OAAOqB,IAAI,CAAC6D,8BAA8B,CAAClF,MAAM,CAAC;EACtD;EA4BA,MAAa8D,iBAAiBA,CAACrC,MAAc,EAAiB;IAC1D,MAAMI,MAAM,GAAGC,gCAAe,CAACC,OAAO,CAAC,CAAC;IACxC,MAAMV,IAAI,GAAGQ,MAAM,CAACkB,OAAO,CAACtB,MAAM,CAAC;IACnC,MAAMoB,UAAU,GAAGC,sBAAa,CAAC9D,GAAG,CAAC,CAAC;IAEtC,IAAI,CAACmG,MAAM,GAAG,CAAC9D,IAAI;IAEnB,IAAIA,IAAI,EAAE;MACN,IAAIH,WAAW;MAEf,IAAI;QACAA,WAAW,GAAG,MAAM,IAAI,CAACsD,qBAAqB,CAACnD,IAAI,CAAC;;QAEpD;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA,IAAIQ,MAAM,CAACmB,eAAe,CAACvB,MAAM,CAAC,IAAIoB,UAAU,KAAK,IAAI,EAAE;UACvD,MAAMQ,QAAQ,GAAGnC,WAAW,CAACoC,eAAe,CAAC,CAAC;UAC9C,MAAMT,UAAU,CAACuC,oBAAoB,CAAClE,WAAW,EAAEmC,QAAQ,EAAEhC,IAAI,EAAE,EAAE,CAAC;QAC1E;QAEA,IAAI,CAAC8B,QAAQ,CAAC;UAAEjC,WAAW,EAAEA;QAAY,CAAC,CAAC;MAC/C,CAAC,CAAC,OAAOmE,KAAK,EAAE;QACZC,cAAM,CAACD,KAAK,CAAC,2CAA2C,EAAEA,KAAK,CAAC;MACpE;IACJ,CAAC,MAAM;MACHC,cAAM,CAACD,KAAK,CAAC,8DAA8D,CAAC;IAChF;EACJ;EAEOE,MAAMA,CAAA,EAAoB;IAC7B,IAAIzD,gCAAe,CAACC,OAAO,CAAC,CAAC,CAACyD,OAAO,CAAC,CAAC,EAAE;MACrC,oBACIlI,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAACxH,SAAA,CAAAa,OAAQ;QACL4G,SAAS,EAAC,6CAA6C;QACvDC,OAAO,EAAE,IAAI,CAACjE,KAAK,CAACiE,OAAQ;QAC5BC,MAAM,EAAE,IAAAC,mBAAE,EAAC,0BAA0B;MAAE,gBAEvCvI,MAAA,CAAAwB,OAAA,CAAA2G,aAAA;QAAKC,SAAS,EAAC;MAAmB,GAC7B,IAAAG,mBAAE,EACC,uBAAuB,EACvB,CAAC,CAAC,EACF;QACI1G,CAAC,EAAG2G,GAAG,iBACHxI,MAAA,CAAAwB,OAAA,CAAA2G,aAAA;UAAGM,IAAI,EAAC,YAAY;UAACC,GAAG,EAAC;QAAK,GACzBF,GACF;MAEX,CACJ,CACC,CACC,CAAC;IAEnB,CAAC,MAAM,IAAI,IAAI,CAACX,MAAM,EAAE;MACpB,oBACI7H,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAACxH,SAAA,CAAAa,OAAQ;QACL4G,SAAS,EAAC,6CAA6C;QACvDC,OAAO,EAAE,IAAI,CAACjE,KAAK,CAACiE,OAAQ;QAC5BC,MAAM,EAAE,IAAAC,mBAAE,EAAC,0BAA0B;MAAE,gBAEvCvI,MAAA,CAAAwB,OAAA,CAAA2G,aAAA;QAAKC,SAAS,EAAC;MAAmB,GAAE,IAAAG,mBAAE,EAAC,sBAAsB,CAAO,CAC9D,CAAC;IAEnB;;IAEA;;IAEA,MAAMI,UAAU,gBACZ3I,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAAClH,WAAA,CAAAO,OAAU;MACPoH,IAAI,EAAEC,cAAU;MAChBC,KAAK,EAAE,IAAAP,mBAAE,EAAC,0BAA0B,CAAE;MACtCQ,WAAW,EAAE,IAAAR,mBAAE,EAAC,8BAA8B;IAAE,CACnD,CACJ;IAED,MAAM7C,eAAe,GAAG,IAAI,CAACmC,MAAM,GAAG,KAAK,GAAGrD,gCAAe,CAACC,OAAO,CAAC,CAAC,CAACiB,eAAe,CAAC,IAAI,CAACtB,KAAK,CAACD,MAAM,CAAC;IAE1G,IAAI,IAAI,CAAC2B,KAAK,CAAClC,WAAW,EAAE;MACxB,oBACI5D,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAACpH,YAAA,CAAAS,OAAW,CAACwH,QAAQ;QACjBC,KAAK,EAAAnG,aAAA,CAAAA,aAAA,KACE,IAAI,CAACoG,OAAO;UACfC,qBAAqB,EAAEC,kCAAqB,CAACC,IAAI;UACjDxF,MAAM,EAAE,IAAI,CAACiC,KAAK,CAACjC;QAAM;MAC3B,gBAEF7D,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAACxH,SAAA,CAAAa,OAAQ;QACL4G,SAAS,EAAC,cAAc;QACxBC,OAAO,EAAE,IAAI,CAACjE,KAAK,CAACiE,OAAQ;QAC5BiB,sBAAsB;QACtBC,GAAG,EAAE,IAAI,CAACC,IAAK;QACflB,MAAM,EAAE,IAAAC,mBAAE,EAAC,0BAA0B;MAAE,GAEtC,IAAI,CAACiB,IAAI,CAACC,OAAO,iBACdzJ,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAACnH,SAAA,CAAAQ,OAAQ;QAACkI,MAAM,EAAE,IAAI,CAACF,IAAI,CAACC,OAAQ;QAACE,aAAa,EAAE,IAAI,CAACA;MAAc,CAAE,CAC5E,eACD3J,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAACzH,cAAA,CAAAc,OAAa;QAACkE,eAAe,EAAEA,eAAgB;QAACkE,IAAI,EAAEC,0BAAW,CAACC;MAAM,CAAE,CAAC,eAC5E9J,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAACvH,cAAA,CAAAY,OAAa;QACVuI,kBAAkB,EAAE,KAAM;QAC1BC,iBAAiB,EAAE,KAAM;QACzBpG,WAAW,EAAE,IAAI,CAACkC,KAAK,CAAClC,WAAY;QACpCqG,cAAc,EAAE,KAAM;QACtBC,mBAAmB,EAAE,IAAI,CAACA,mBAAoB;QAC9CC,cAAc,EAAE,IAAI,CAAC/F,KAAK,CAAC+F,cAAe;QAC1CC,KAAK,EAAEzB,UAAW;QAClB0B,MAAM,EAAEC,cAAM,CAACC;MAAM,CACxB,CACK,CACQ,CAAC;IAE/B,CAAC,MAAM;MACH,oBACIvK,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAACpH,YAAA,CAAAS,OAAW,CAACwH,QAAQ;QACjBC,KAAK,EAAAnG,aAAA,CAAAA,aAAA,KACE,IAAI,CAACoG,OAAO;UACfC,qBAAqB,EAAEC,kCAAqB,CAACC;QAAI;MACnD,gBAEFrJ,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAACxH,SAAA,CAAAa,OAAQ;QACL4G,SAAS,EAAC,cAAc;QACxBC,OAAO,EAAE,IAAI,CAACjE,KAAK,CAACiE,OAAQ;QAC5BC,MAAM,EAAE,IAAAC,mBAAE,EAAC,0BAA0B;MAAE,gBAEvCvI,MAAA,CAAAwB,OAAA,CAAA2G,aAAA,CAACtH,QAAA,CAAAW,OAAO,MAAE,CACJ,CACQ,CAAC;IAE/B;EACJ;AACJ;AAAC,IAAA0B,gBAAA,CAAA1B,OAAA,EA9QK6B,SAAS,iBACiBmH,oBAAW;AAAA,IAAAC,QAAA,GAAAC,OAAA,CAAAlJ,OAAA,GA+Q5B6B,SAAS","ignoreList":[]}