matrix-react-sdk
Version:
SDK for matrix.org using React
220 lines (217 loc) • 35.5 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 = _interopRequireDefault(require("react"));
var _classnames = _interopRequireDefault(require("classnames"));
var _languageHandler = require("../../../languageHandler");
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _Permalinks = require("../../../utils/permalinks/Permalinks");
var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore"));
var _FormattingUtils = require("../../../utils/FormattingUtils");
var _actions = require("../../../dispatcher/actions");
var _Spinner = _interopRequireDefault(require("./Spinner"));
var _ReplyTile = _interopRequireDefault(require("../rooms/ReplyTile"));
var _Pill = require("./Pill");
var _AccessibleButton = _interopRequireDefault(require("./AccessibleButton"));
var _Reply = require("../../../utils/Reply");
var _RoomContext = _interopRequireDefault(require("../../../contexts/RoomContext"));
var _MatrixClientPeg = require("../../../MatrixClientPeg");
/*
Copyright 2024 New Vector Ltd.
Copyright 2017-2023 The Matrix.org Foundation C.I.C.
Copyright 2019 Michael Telatynski <7t3chguy@gmail.com>
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
/**
* This number is based on the previous behavior - if we have message of height
* over 60px then we want to show button that will allow to expand it.
*/
const SHOW_EXPAND_QUOTE_PIXELS = 60;
// This component does no cycle detection, simply because the only way to make such a cycle would be to
// craft event_id's, using a homeserver that generates predictable event IDs; even then the impact would
// be low as each event being loaded (after the first) is triggered by an explicit user action.
class ReplyChain extends _react.default.Component {
constructor(props, context) {
super(props, context);
(0, _defineProperty2.default)(this, "unmounted", false);
(0, _defineProperty2.default)(this, "room", void 0);
(0, _defineProperty2.default)(this, "blockquoteRef", /*#__PURE__*/_react.default.createRef());
(0, _defineProperty2.default)(this, "canCollapse", () => {
return this.state.events.length > 1;
});
(0, _defineProperty2.default)(this, "collapse", () => {
this.initialize();
});
(0, _defineProperty2.default)(this, "onQuoteClick", async () => {
if (!this.state.loadedEv) return;
const events = [this.state.loadedEv, ...this.state.events];
let loadedEv = null;
if (events.length > 0) {
loadedEv = await this.getNextEvent(events[0]);
}
this.setState({
loadedEv,
events
});
_dispatcher.default.fire(_actions.Action.FocusSendMessageComposer);
});
this.state = {
events: [],
loadedEv: null,
loading: true,
err: false
};
this.room = this.matrixClient.getRoom(this.props.parentEv.getRoomId());
}
get matrixClient() {
return _MatrixClientPeg.MatrixClientPeg.safeGet();
}
componentDidMount() {
this.initialize();
this.trySetExpandableQuotes();
}
componentDidUpdate() {
this.props.onHeightChanged?.();
this.trySetExpandableQuotes();
}
componentWillUnmount() {
this.unmounted = true;
}
trySetExpandableQuotes() {
if (this.props.isQuoteExpanded === undefined && this.blockquoteRef.current) {
const el = this.blockquoteRef.current.querySelector(".mx_EventTile_body");
if (el) {
const code = el.querySelector("code");
const isCodeEllipsisShown = code ? code.offsetHeight >= SHOW_EXPAND_QUOTE_PIXELS : false;
const isElipsisShown = el.offsetHeight >= SHOW_EXPAND_QUOTE_PIXELS || isCodeEllipsisShown;
if (isElipsisShown) {
this.props.setQuoteExpanded(false);
}
}
}
}
async initialize() {
const {
parentEv
} = this.props;
// at time of making this component we checked that props.parentEv has a parentEventId
const ev = await this.getEvent((0, _Reply.getParentEventId)(parentEv));
if (this.unmounted) return;
if (ev) {
const loadedEv = await this.getNextEvent(ev);
this.setState({
events: [ev],
loadedEv,
loading: false
});
} else {
this.setState({
err: true
});
}
}
async getNextEvent(ev) {
try {
const inReplyToEventId = (0, _Reply.getParentEventId)(ev);
if (!inReplyToEventId) return null;
return await this.getEvent(inReplyToEventId);
} catch (e) {
return null;
}
}
async getEvent(eventId) {
if (!eventId) return null;
const event = this.room.findEventById(eventId);
if (event) return event;
try {
// ask the client to fetch the event we want using the context API, only interface to do so is to ask
// for a timeline with that event, but once it is loaded we can use findEventById to look up the ev map
await this.matrixClient.getEventTimeline(this.room.getUnfilteredTimelineSet(), eventId);
} catch (e) {
// if it fails catch the error and return early, there's no point trying to find the event in this case.
// Return null as it is falsy and thus should be treated as an error (as the event cannot be resolved).
return null;
}
return this.room.findEventById(eventId) ?? null;
}
getReplyChainColorClass(ev) {
return (0, _FormattingUtils.getUserNameColorClass)(ev.getSender()).replace("Username", "ReplyChain");
}
render() {
let header;
if (this.state.err) {
header = /*#__PURE__*/_react.default.createElement("blockquote", {
className: "mx_ReplyChain mx_ReplyChain_error"
}, (0, _languageHandler._t)("timeline|reply|error_loading"));
} else if (this.state.loadedEv && (0, _Reply.shouldDisplayReply)(this.state.events[0])) {
const ev = this.state.loadedEv;
const room = this.matrixClient.getRoom(ev.getRoomId());
header = /*#__PURE__*/_react.default.createElement("blockquote", {
className: `mx_ReplyChain ${this.getReplyChainColorClass(ev)}`
}, (0, _languageHandler._t)("timeline|reply|in_reply_to", {}, {
a: sub => /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
kind: "link_inline",
className: "mx_ReplyChain_show",
onClick: this.onQuoteClick
}, sub),
pill: /*#__PURE__*/_react.default.createElement(_Pill.Pill, {
type: _Pill.PillType.UserMention,
room: room ?? undefined,
url: (0, _Permalinks.makeUserPermalink)(ev.getSender()),
shouldShowPillAvatar: _SettingsStore.default.getValue("Pill.shouldShowPillAvatar")
})
}));
} else if (this.props.forExport) {
const eventId = (0, _Reply.getParentEventId)(this.props.parentEv);
header = /*#__PURE__*/_react.default.createElement("p", {
className: "mx_ReplyChain_Export"
}, (0, _languageHandler._t)("timeline|reply|in_reply_to_for_export", {}, {
a: sub => /*#__PURE__*/_react.default.createElement("a", {
className: "mx_reply_anchor",
href: `#${eventId}`,
"data-scroll-to": eventId
}, " ", sub, " ")
}));
} else if (this.state.loading) {
header = /*#__PURE__*/_react.default.createElement(_Spinner.default, {
w: 16,
h: 16
});
}
const {
isQuoteExpanded
} = this.props;
const evTiles = this.state.events.map(ev => {
const classname = (0, _classnames.default)({
"mx_ReplyChain": true,
[this.getReplyChainColorClass(ev)]: true,
// We don't want to add the class if it's undefined, it should only be expanded/collapsed when it's true/false
"mx_ReplyChain--expanded": isQuoteExpanded === true,
// We don't want to add the class if it's undefined, it should only be expanded/collapsed when it's true/false
"mx_ReplyChain--collapsed": isQuoteExpanded === false
});
return /*#__PURE__*/_react.default.createElement("blockquote", {
ref: this.blockquoteRef,
className: classname,
key: ev.getId()
}, /*#__PURE__*/_react.default.createElement(_ReplyTile.default, {
mxEvent: ev,
onHeightChanged: this.props.onHeightChanged,
permalinkCreator: this.props.permalinkCreator,
toggleExpandedQuote: () => this.props.setQuoteExpanded(!this.props.isQuoteExpanded),
getRelationsForEvent: this.props.getRelationsForEvent
}));
});
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ReplyChain_wrapper"
}, /*#__PURE__*/_react.default.createElement("div", null, header), /*#__PURE__*/_react.default.createElement("div", null, evTiles));
}
}
exports.default = ReplyChain;
(0, _defineProperty2.default)(ReplyChain, "contextType", _RoomContext.default);
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_classnames","_languageHandler","_dispatcher","_Permalinks","_SettingsStore","_FormattingUtils","_actions","_Spinner","_ReplyTile","_Pill","_AccessibleButton","_Reply","_RoomContext","_MatrixClientPeg","SHOW_EXPAND_QUOTE_PIXELS","ReplyChain","React","Component","constructor","props","context","_defineProperty2","default","createRef","state","events","length","initialize","loadedEv","getNextEvent","setState","dis","fire","Action","FocusSendMessageComposer","loading","err","room","matrixClient","getRoom","parentEv","getRoomId","MatrixClientPeg","safeGet","componentDidMount","trySetExpandableQuotes","componentDidUpdate","onHeightChanged","componentWillUnmount","unmounted","isQuoteExpanded","undefined","blockquoteRef","current","el","querySelector","code","isCodeEllipsisShown","offsetHeight","isElipsisShown","setQuoteExpanded","ev","getEvent","getParentEventId","inReplyToEventId","e","eventId","event","findEventById","getEventTimeline","getUnfilteredTimelineSet","getReplyChainColorClass","getUserNameColorClass","getSender","replace","render","header","createElement","className","_t","shouldDisplayReply","a","sub","kind","onClick","onQuoteClick","pill","Pill","type","PillType","UserMention","url","makeUserPermalink","shouldShowPillAvatar","SettingsStore","getValue","forExport","href","w","h","evTiles","map","classname","classNames","ref","key","getId","mxEvent","permalinkCreator","toggleExpandedQuote","getRelationsForEvent","exports","RoomContext"],"sources":["../../../../src/components/views/elements/ReplyChain.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2017-2023 The Matrix.org Foundation C.I.C.\nCopyright 2019 Michael Telatynski <7t3chguy@gmail.com>\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 from \"react\";\nimport classNames from \"classnames\";\nimport { MatrixEvent, Room, MatrixClient } from \"matrix-js-sdk/src/matrix\";\n\nimport { _t } from \"../../../languageHandler\";\nimport dis from \"../../../dispatcher/dispatcher\";\nimport { makeUserPermalink, RoomPermalinkCreator } from \"../../../utils/permalinks/Permalinks\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport { Layout } from \"../../../settings/enums/Layout\";\nimport { getUserNameColorClass } from \"../../../utils/FormattingUtils\";\nimport { Action } from \"../../../dispatcher/actions\";\nimport Spinner from \"./Spinner\";\nimport ReplyTile from \"../rooms/ReplyTile\";\nimport { Pill, PillType } from \"./Pill\";\nimport AccessibleButton from \"./AccessibleButton\";\nimport { getParentEventId, shouldDisplayReply } from \"../../../utils/Reply\";\nimport RoomContext from \"../../../contexts/RoomContext\";\nimport { MatrixClientPeg } from \"../../../MatrixClientPeg\";\nimport { GetRelationsForEvent } from \"../rooms/EventTile\";\n\n/**\n * This number is based on the previous behavior - if we have message of height\n * over 60px then we want to show button that will allow to expand it.\n */\nconst SHOW_EXPAND_QUOTE_PIXELS = 60;\n\ninterface IProps {\n    // the latest event in this chain of replies\n    parentEv: MatrixEvent;\n    // called when the ReplyChain contents has changed, including EventTiles thereof\n    onHeightChanged?: () => void;\n    permalinkCreator?: RoomPermalinkCreator;\n    // Specifies which layout to use.\n    layout?: Layout;\n    // Whether to always show a timestamp\n    alwaysShowTimestamps?: boolean;\n    forExport?: boolean;\n    isQuoteExpanded?: boolean;\n    setQuoteExpanded: (isExpanded: boolean) => void;\n    getRelationsForEvent?: GetRelationsForEvent;\n}\n\ninterface IState {\n    // The loaded events to be rendered as linear-replies\n    events: MatrixEvent[];\n    // The latest loaded event which has not yet been shown\n    loadedEv: MatrixEvent | null;\n    // Whether the component is still loading more events\n    loading: boolean;\n    // Whether as error was encountered fetching a replied to event.\n    err: boolean;\n}\n\n// This component does no cycle detection, simply because the only way to make such a cycle would be to\n// craft event_id's, using a homeserver that generates predictable event IDs; even then the impact would\n// be low as each event being loaded (after the first) is triggered by an explicit user action.\nexport default class ReplyChain extends React.Component<IProps, IState> {\n    public static contextType = RoomContext;\n    public declare context: React.ContextType<typeof RoomContext>;\n\n    private unmounted = false;\n    private room: Room;\n    private blockquoteRef = React.createRef<HTMLQuoteElement>();\n\n    public constructor(props: IProps, context: React.ContextType<typeof RoomContext>) {\n        super(props, context);\n\n        this.state = {\n            events: [],\n            loadedEv: null,\n            loading: true,\n            err: false,\n        };\n\n        this.room = this.matrixClient.getRoom(this.props.parentEv.getRoomId())!;\n    }\n\n    private get matrixClient(): MatrixClient {\n        return MatrixClientPeg.safeGet();\n    }\n\n    public componentDidMount(): void {\n        this.initialize();\n        this.trySetExpandableQuotes();\n    }\n\n    public componentDidUpdate(): void {\n        this.props.onHeightChanged?.();\n        this.trySetExpandableQuotes();\n    }\n\n    public componentWillUnmount(): void {\n        this.unmounted = true;\n    }\n\n    private trySetExpandableQuotes(): void {\n        if (this.props.isQuoteExpanded === undefined && this.blockquoteRef.current) {\n            const el: HTMLElement | null = this.blockquoteRef.current.querySelector(\".mx_EventTile_body\");\n            if (el) {\n                const code: HTMLElement | null = el.querySelector(\"code\");\n                const isCodeEllipsisShown = code ? code.offsetHeight >= SHOW_EXPAND_QUOTE_PIXELS : false;\n                const isElipsisShown = el.offsetHeight >= SHOW_EXPAND_QUOTE_PIXELS || isCodeEllipsisShown;\n                if (isElipsisShown) {\n                    this.props.setQuoteExpanded(false);\n                }\n            }\n        }\n    }\n\n    private async initialize(): Promise<void> {\n        const { parentEv } = this.props;\n        // at time of making this component we checked that props.parentEv has a parentEventId\n        const ev = await this.getEvent(getParentEventId(parentEv));\n\n        if (this.unmounted) return;\n\n        if (ev) {\n            const loadedEv = await this.getNextEvent(ev);\n            this.setState({\n                events: [ev],\n                loadedEv,\n                loading: false,\n            });\n        } else {\n            this.setState({ err: true });\n        }\n    }\n\n    private async getNextEvent(ev: MatrixEvent): Promise<MatrixEvent | null> {\n        try {\n            const inReplyToEventId = getParentEventId(ev);\n            if (!inReplyToEventId) return null;\n            return await this.getEvent(inReplyToEventId);\n        } catch (e) {\n            return null;\n        }\n    }\n\n    private async getEvent(eventId?: string): Promise<MatrixEvent | null> {\n        if (!eventId) return null;\n        const event = this.room.findEventById(eventId);\n        if (event) return event;\n\n        try {\n            // ask the client to fetch the event we want using the context API, only interface to do so is to ask\n            // for a timeline with that event, but once it is loaded we can use findEventById to look up the ev map\n            await this.matrixClient.getEventTimeline(this.room.getUnfilteredTimelineSet(), eventId);\n        } catch (e) {\n            // if it fails catch the error and return early, there's no point trying to find the event in this case.\n            // Return null as it is falsy and thus should be treated as an error (as the event cannot be resolved).\n            return null;\n        }\n        return this.room.findEventById(eventId) ?? null;\n    }\n\n    public canCollapse = (): boolean => {\n        return this.state.events.length > 1;\n    };\n\n    public collapse = (): void => {\n        this.initialize();\n    };\n\n    private onQuoteClick = async (): Promise<void> => {\n        if (!this.state.loadedEv) return;\n        const events = [this.state.loadedEv, ...this.state.events];\n\n        let loadedEv: MatrixEvent | null = null;\n        if (events.length > 0) {\n            loadedEv = await this.getNextEvent(events[0]);\n        }\n\n        this.setState({\n            loadedEv,\n            events,\n        });\n\n        dis.fire(Action.FocusSendMessageComposer);\n    };\n\n    private getReplyChainColorClass(ev: MatrixEvent): string {\n        return getUserNameColorClass(ev.getSender()!).replace(\"Username\", \"ReplyChain\");\n    }\n\n    public render(): React.ReactNode {\n        let header: JSX.Element | undefined;\n        if (this.state.err) {\n            header = (\n                <blockquote className=\"mx_ReplyChain mx_ReplyChain_error\">\n                    {_t(\"timeline|reply|error_loading\")}\n                </blockquote>\n            );\n        } else if (this.state.loadedEv && shouldDisplayReply(this.state.events[0])) {\n            const ev = this.state.loadedEv;\n            const room = this.matrixClient.getRoom(ev.getRoomId());\n            header = (\n                <blockquote className={`mx_ReplyChain ${this.getReplyChainColorClass(ev)}`}>\n                    {_t(\n                        \"timeline|reply|in_reply_to\",\n                        {},\n                        {\n                            a: (sub) => (\n                                <AccessibleButton\n                                    kind=\"link_inline\"\n                                    className=\"mx_ReplyChain_show\"\n                                    onClick={this.onQuoteClick}\n                                >\n                                    {sub}\n                                </AccessibleButton>\n                            ),\n                            pill: (\n                                <Pill\n                                    type={PillType.UserMention}\n                                    room={room ?? undefined}\n                                    url={makeUserPermalink(ev.getSender()!)}\n                                    shouldShowPillAvatar={SettingsStore.getValue(\"Pill.shouldShowPillAvatar\")}\n                                />\n                            ),\n                        },\n                    )}\n                </blockquote>\n            );\n        } else if (this.props.forExport) {\n            const eventId = getParentEventId(this.props.parentEv);\n            header = (\n                <p className=\"mx_ReplyChain_Export\">\n                    {_t(\n                        \"timeline|reply|in_reply_to_for_export\",\n                        {},\n                        {\n                            a: (sub) => (\n                                <a className=\"mx_reply_anchor\" href={`#${eventId}`} data-scroll-to={eventId}>\n                                    {\" \"}\n                                    {sub}{\" \"}\n                                </a>\n                            ),\n                        },\n                    )}\n                </p>\n            );\n        } else if (this.state.loading) {\n            header = <Spinner w={16} h={16} />;\n        }\n\n        const { isQuoteExpanded } = this.props;\n        const evTiles = this.state.events.map((ev) => {\n            const classname = classNames({\n                \"mx_ReplyChain\": true,\n                [this.getReplyChainColorClass(ev)]: true,\n                // We don't want to add the class if it's undefined, it should only be expanded/collapsed when it's true/false\n                \"mx_ReplyChain--expanded\": isQuoteExpanded === true,\n                // We don't want to add the class if it's undefined, it should only be expanded/collapsed when it's true/false\n                \"mx_ReplyChain--collapsed\": isQuoteExpanded === false,\n            });\n            return (\n                <blockquote ref={this.blockquoteRef} className={classname} key={ev.getId()}>\n                    <ReplyTile\n                        mxEvent={ev}\n                        onHeightChanged={this.props.onHeightChanged}\n                        permalinkCreator={this.props.permalinkCreator}\n                        toggleExpandedQuote={() => this.props.setQuoteExpanded(!this.props.isQuoteExpanded)}\n                        getRelationsForEvent={this.props.getRelationsForEvent}\n                    />\n                </blockquote>\n            );\n        });\n\n        return (\n            <div className=\"mx_ReplyChain_wrapper\">\n                <div>{header}</div>\n                <div>{evTiles}</div>\n            </div>\n        );\n    }\n}\n"],"mappings":"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","ignoreList":[]}