matrix-react-sdk
Version:
SDK for matrix.org using React
211 lines (207 loc) • 34 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = exports.REACTION_SHORTCODE_KEY = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends"));
var _react = _interopRequireDefault(require("react"));
var _classnames = _interopRequireDefault(require("classnames"));
var _matrix = require("matrix-js-sdk/src/matrix");
var _lodash = require("lodash");
var _NamespacedValue = require("matrix-js-sdk/src/NamespacedValue");
var _languageHandler = require("../../../languageHandler");
var _EventUtils = require("../../../utils/EventUtils");
var _ContextMenuTooltipButton = require("../../../accessibility/context_menu/ContextMenuTooltipButton");
var _ContextMenu = _interopRequireWildcard(require("../../structures/ContextMenu"));
var _ReactionPicker = _interopRequireDefault(require("../emojipicker/ReactionPicker"));
var _ReactionsRowButton = _interopRequireDefault(require("./ReactionsRowButton"));
var _RoomContext = _interopRequireDefault(require("../../../contexts/RoomContext"));
var _AccessibleButton = _interopRequireDefault(require("../elements/AccessibleButton"));
var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore"));
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 2019-2021 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 maximum number of reactions to initially show on a message.
const MAX_ITEMS_WHEN_LIMITED = 8;
const REACTION_SHORTCODE_KEY = exports.REACTION_SHORTCODE_KEY = new _NamespacedValue.UnstableValue("shortcode", "com.beeper.reaction.shortcode");
const ReactButton = ({
mxEvent,
reactions
}) => {
const [menuDisplayed, button, openMenu, closeMenu] = (0, _ContextMenu.useContextMenu)();
let contextMenu;
if (menuDisplayed && button.current) {
const buttonRect = button.current.getBoundingClientRect();
contextMenu = /*#__PURE__*/_react.default.createElement(_ContextMenu.default, (0, _extends2.default)({}, (0, _ContextMenu.aboveLeftOf)(buttonRect), {
onFinished: closeMenu,
managed: false
}), /*#__PURE__*/_react.default.createElement(_ReactionPicker.default, {
mxEvent: mxEvent,
reactions: reactions,
onFinished: closeMenu
}));
}
return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_ContextMenuTooltipButton.ContextMenuTooltipButton, {
className: (0, _classnames.default)("mx_ReactionsRow_addReactionButton", {
mx_ReactionsRow_addReactionButton_active: menuDisplayed
}),
title: (0, _languageHandler._t)("timeline|reactions|add_reaction_prompt"),
onClick: openMenu,
onContextMenu: e => {
e.preventDefault();
openMenu();
},
isExpanded: menuDisplayed,
ref: button
}), contextMenu);
};
class ReactionsRow extends _react.default.PureComponent {
constructor(props, context) {
super(props, context);
(0, _defineProperty2.default)(this, "onDecrypted", () => {
// Decryption changes whether the event is actionable
this.forceUpdate();
});
(0, _defineProperty2.default)(this, "onReactionsChange", () => {
// TODO: Call `onHeightChanged` as needed
this.setState({
myReactions: this.getMyReactions()
});
// Using `forceUpdate` for the moment, since we know the overall set of reactions
// has changed (this is triggered by events for that purpose only) and
// `PureComponent`s shallow state / props compare would otherwise filter this out.
this.forceUpdate();
});
(0, _defineProperty2.default)(this, "onShowAllClick", () => {
this.setState({
showAll: true
});
});
this.state = {
myReactions: this.getMyReactions(),
showAll: false
};
}
componentDidMount() {
const {
mxEvent,
reactions
} = this.props;
if (mxEvent.isBeingDecrypted() || mxEvent.shouldAttemptDecryption()) {
mxEvent.once(_matrix.MatrixEventEvent.Decrypted, this.onDecrypted);
}
if (reactions) {
reactions.on(_matrix.RelationsEvent.Add, this.onReactionsChange);
reactions.on(_matrix.RelationsEvent.Remove, this.onReactionsChange);
reactions.on(_matrix.RelationsEvent.Redaction, this.onReactionsChange);
}
}
componentWillUnmount() {
const {
mxEvent,
reactions
} = this.props;
mxEvent.off(_matrix.MatrixEventEvent.Decrypted, this.onDecrypted);
if (reactions) {
reactions.off(_matrix.RelationsEvent.Add, this.onReactionsChange);
reactions.off(_matrix.RelationsEvent.Remove, this.onReactionsChange);
reactions.off(_matrix.RelationsEvent.Redaction, this.onReactionsChange);
}
}
componentDidUpdate(prevProps) {
if (this.props.reactions && prevProps.reactions !== this.props.reactions) {
this.props.reactions.on(_matrix.RelationsEvent.Add, this.onReactionsChange);
this.props.reactions.on(_matrix.RelationsEvent.Remove, this.onReactionsChange);
this.props.reactions.on(_matrix.RelationsEvent.Redaction, this.onReactionsChange);
this.onReactionsChange();
}
}
getMyReactions() {
const reactions = this.props.reactions;
if (!reactions) {
return null;
}
const userId = this.context.room?.client.getUserId();
if (!userId) return null;
const myReactions = reactions.getAnnotationsBySender()?.[userId];
if (!myReactions) {
return null;
}
return [...myReactions.values()];
}
render() {
const {
mxEvent,
reactions
} = this.props;
const {
myReactions,
showAll
} = this.state;
if (!reactions || !(0, _EventUtils.isContentActionable)(mxEvent)) {
return null;
}
const customReactionImagesEnabled = _SettingsStore.default.getValue("feature_render_reaction_images");
let items = reactions.getSortedAnnotationsByKey()?.map(([content, events]) => {
const count = events.size;
if (!count) {
return null;
}
// Deduplicate the events as per the spec https://spec.matrix.org/v1.7/client-server-api/#annotations-client-behaviour
// This isn't done by the underlying data model as applications may still need access to the whole list of events
// for moderation purposes.
const deduplicatedEvents = (0, _lodash.uniqBy)([...events], e => e.getSender());
const myReactionEvent = myReactions?.find(mxEvent => {
if (mxEvent.isRedacted()) {
return false;
}
return mxEvent.getRelation()?.key === content;
});
return /*#__PURE__*/_react.default.createElement(_ReactionsRowButton.default, {
key: content,
content: content,
count: deduplicatedEvents.length,
mxEvent: mxEvent,
reactionEvents: deduplicatedEvents,
myReactionEvent: myReactionEvent,
customReactionImagesEnabled: customReactionImagesEnabled,
disabled: !this.context.canReact || myReactionEvent && !myReactionEvent.isRedacted() && !this.context.canSelfRedact
});
}).filter(item => !!item);
if (!items?.length) return null;
// Show the first MAX_ITEMS if there are MAX_ITEMS + 1 or more items.
// The "+ 1" ensure that the "show all" reveals something that takes up
// more space than the button itself.
let showAllButton;
if (items.length > MAX_ITEMS_WHEN_LIMITED + 1 && !showAll) {
items = items.slice(0, MAX_ITEMS_WHEN_LIMITED);
showAllButton = /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
kind: "link_inline",
className: "mx_ReactionsRow_showAll",
onClick: this.onShowAllClick
}, (0, _languageHandler._t)("action|show_all"));
}
let addReactionButton;
if (this.context.canReact) {
addReactionButton = /*#__PURE__*/_react.default.createElement(ReactButton, {
mxEvent: mxEvent,
reactions: reactions
});
}
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ReactionsRow",
role: "toolbar",
"aria-label": (0, _languageHandler._t)("common|reactions")
}, items, showAllButton, addReactionButton);
}
}
exports.default = ReactionsRow;
(0, _defineProperty2.default)(ReactionsRow, "contextType", _RoomContext.default);
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_classnames","_matrix","_lodash","_NamespacedValue","_languageHandler","_EventUtils","_ContextMenuTooltipButton","_ContextMenu","_interopRequireWildcard","_ReactionPicker","_ReactionsRowButton","_RoomContext","_AccessibleButton","_SettingsStore","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","MAX_ITEMS_WHEN_LIMITED","REACTION_SHORTCODE_KEY","exports","UnstableValue","ReactButton","mxEvent","reactions","menuDisplayed","button","openMenu","closeMenu","useContextMenu","contextMenu","current","buttonRect","getBoundingClientRect","createElement","_extends2","aboveLeftOf","onFinished","managed","Fragment","ContextMenuTooltipButton","className","classNames","mx_ReactionsRow_addReactionButton_active","title","_t","onClick","onContextMenu","preventDefault","isExpanded","ref","ReactionsRow","React","PureComponent","constructor","props","context","_defineProperty2","forceUpdate","setState","myReactions","getMyReactions","showAll","state","componentDidMount","isBeingDecrypted","shouldAttemptDecryption","once","MatrixEventEvent","Decrypted","onDecrypted","on","RelationsEvent","Add","onReactionsChange","Remove","Redaction","componentWillUnmount","off","componentDidUpdate","prevProps","userId","room","client","getUserId","getAnnotationsBySender","values","render","isContentActionable","customReactionImagesEnabled","SettingsStore","getValue","items","getSortedAnnotationsByKey","map","content","events","count","size","deduplicatedEvents","uniqBy","getSender","myReactionEvent","find","isRedacted","getRelation","key","length","reactionEvents","disabled","canReact","canSelfRedact","filter","item","showAllButton","slice","kind","onShowAllClick","addReactionButton","role","RoomContext"],"sources":["../../../../src/components/views/messages/ReactionsRow.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2019-2021 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 React, { SyntheticEvent } from \"react\";\nimport classNames from \"classnames\";\nimport { MatrixEvent, MatrixEventEvent, Relations, RelationsEvent } from \"matrix-js-sdk/src/matrix\";\nimport { uniqBy } from \"lodash\";\nimport { UnstableValue } from \"matrix-js-sdk/src/NamespacedValue\";\n\nimport { _t } from \"../../../languageHandler\";\nimport { isContentActionable } from \"../../../utils/EventUtils\";\nimport { ContextMenuTooltipButton } from \"../../../accessibility/context_menu/ContextMenuTooltipButton\";\nimport ContextMenu, { aboveLeftOf, useContextMenu } from \"../../structures/ContextMenu\";\nimport ReactionPicker from \"../emojipicker/ReactionPicker\";\nimport ReactionsRowButton from \"./ReactionsRowButton\";\nimport RoomContext from \"../../../contexts/RoomContext\";\nimport AccessibleButton from \"../elements/AccessibleButton\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\n\n// The maximum number of reactions to initially show on a message.\nconst MAX_ITEMS_WHEN_LIMITED = 8;\n\nexport const REACTION_SHORTCODE_KEY = new UnstableValue(\"shortcode\", \"com.beeper.reaction.shortcode\");\n\nconst ReactButton: React.FC<IProps> = ({ mxEvent, reactions }) => {\n    const [menuDisplayed, button, openMenu, closeMenu] = useContextMenu();\n\n    let contextMenu: JSX.Element | undefined;\n    if (menuDisplayed && button.current) {\n        const buttonRect = button.current.getBoundingClientRect();\n        contextMenu = (\n            <ContextMenu {...aboveLeftOf(buttonRect)} onFinished={closeMenu} managed={false}>\n                <ReactionPicker mxEvent={mxEvent} reactions={reactions} onFinished={closeMenu} />\n            </ContextMenu>\n        );\n    }\n\n    return (\n        <React.Fragment>\n            <ContextMenuTooltipButton\n                className={classNames(\"mx_ReactionsRow_addReactionButton\", {\n                    mx_ReactionsRow_addReactionButton_active: menuDisplayed,\n                })}\n                title={_t(\"timeline|reactions|add_reaction_prompt\")}\n                onClick={openMenu}\n                onContextMenu={(e: SyntheticEvent): void => {\n                    e.preventDefault();\n                    openMenu();\n                }}\n                isExpanded={menuDisplayed}\n                ref={button}\n            />\n\n            {contextMenu}\n        </React.Fragment>\n    );\n};\n\ninterface IProps {\n    // The event we're displaying reactions for\n    mxEvent: MatrixEvent;\n    // The Relations model from the JS SDK for reactions to `mxEvent`\n    reactions?: Relations | null | undefined;\n}\n\ninterface IState {\n    myReactions: MatrixEvent[] | null;\n    showAll: boolean;\n}\n\nexport default class ReactionsRow extends React.PureComponent<IProps, IState> {\n    public static contextType = RoomContext;\n    public declare context: React.ContextType<typeof RoomContext>;\n\n    public constructor(props: IProps, context: React.ContextType<typeof RoomContext>) {\n        super(props, context);\n\n        this.state = {\n            myReactions: this.getMyReactions(),\n            showAll: false,\n        };\n    }\n\n    public componentDidMount(): void {\n        const { mxEvent, reactions } = this.props;\n\n        if (mxEvent.isBeingDecrypted() || mxEvent.shouldAttemptDecryption()) {\n            mxEvent.once(MatrixEventEvent.Decrypted, this.onDecrypted);\n        }\n\n        if (reactions) {\n            reactions.on(RelationsEvent.Add, this.onReactionsChange);\n            reactions.on(RelationsEvent.Remove, this.onReactionsChange);\n            reactions.on(RelationsEvent.Redaction, this.onReactionsChange);\n        }\n    }\n\n    public componentWillUnmount(): void {\n        const { mxEvent, reactions } = this.props;\n\n        mxEvent.off(MatrixEventEvent.Decrypted, this.onDecrypted);\n\n        if (reactions) {\n            reactions.off(RelationsEvent.Add, this.onReactionsChange);\n            reactions.off(RelationsEvent.Remove, this.onReactionsChange);\n            reactions.off(RelationsEvent.Redaction, this.onReactionsChange);\n        }\n    }\n\n    public componentDidUpdate(prevProps: IProps): void {\n        if (this.props.reactions && prevProps.reactions !== this.props.reactions) {\n            this.props.reactions.on(RelationsEvent.Add, this.onReactionsChange);\n            this.props.reactions.on(RelationsEvent.Remove, this.onReactionsChange);\n            this.props.reactions.on(RelationsEvent.Redaction, this.onReactionsChange);\n            this.onReactionsChange();\n        }\n    }\n\n    private onDecrypted = (): void => {\n        // Decryption changes whether the event is actionable\n        this.forceUpdate();\n    };\n\n    private onReactionsChange = (): void => {\n        // TODO: Call `onHeightChanged` as needed\n        this.setState({\n            myReactions: this.getMyReactions(),\n        });\n        // Using `forceUpdate` for the moment, since we know the overall set of reactions\n        // has changed (this is triggered by events for that purpose only) and\n        // `PureComponent`s shallow state / props compare would otherwise filter this out.\n        this.forceUpdate();\n    };\n\n    private getMyReactions(): MatrixEvent[] | null {\n        const reactions = this.props.reactions;\n        if (!reactions) {\n            return null;\n        }\n        const userId = this.context.room?.client.getUserId();\n        if (!userId) return null;\n        const myReactions = reactions.getAnnotationsBySender()?.[userId];\n        if (!myReactions) {\n            return null;\n        }\n        return [...myReactions.values()];\n    }\n\n    private onShowAllClick = (): void => {\n        this.setState({\n            showAll: true,\n        });\n    };\n\n    public render(): React.ReactNode {\n        const { mxEvent, reactions } = this.props;\n        const { myReactions, showAll } = this.state;\n\n        if (!reactions || !isContentActionable(mxEvent)) {\n            return null;\n        }\n        const customReactionImagesEnabled = SettingsStore.getValue(\"feature_render_reaction_images\");\n\n        let items = reactions\n            .getSortedAnnotationsByKey()\n            ?.map(([content, events]) => {\n                const count = events.size;\n                if (!count) {\n                    return null;\n                }\n                // Deduplicate the events as per the spec https://spec.matrix.org/v1.7/client-server-api/#annotations-client-behaviour\n                // This isn't done by the underlying data model as applications may still need access to the whole list of events\n                // for moderation purposes.\n                const deduplicatedEvents = uniqBy([...events], (e) => e.getSender());\n                const myReactionEvent = myReactions?.find((mxEvent) => {\n                    if (mxEvent.isRedacted()) {\n                        return false;\n                    }\n                    return mxEvent.getRelation()?.key === content;\n                });\n                return (\n                    <ReactionsRowButton\n                        key={content}\n                        content={content}\n                        count={deduplicatedEvents.length}\n                        mxEvent={mxEvent}\n                        reactionEvents={deduplicatedEvents}\n                        myReactionEvent={myReactionEvent}\n                        customReactionImagesEnabled={customReactionImagesEnabled}\n                        disabled={\n                            !this.context.canReact ||\n                            (myReactionEvent && !myReactionEvent.isRedacted() && !this.context.canSelfRedact)\n                        }\n                    />\n                );\n            })\n            .filter((item) => !!item);\n\n        if (!items?.length) return null;\n\n        // Show the first MAX_ITEMS if there are MAX_ITEMS + 1 or more items.\n        // The \"+ 1\" ensure that the \"show all\" reveals something that takes up\n        // more space than the button itself.\n        let showAllButton: JSX.Element | undefined;\n        if (items.length > MAX_ITEMS_WHEN_LIMITED + 1 && !showAll) {\n            items = items.slice(0, MAX_ITEMS_WHEN_LIMITED);\n            showAllButton = (\n                <AccessibleButton kind=\"link_inline\" className=\"mx_ReactionsRow_showAll\" onClick={this.onShowAllClick}>\n                    {_t(\"action|show_all\")}\n                </AccessibleButton>\n            );\n        }\n\n        let addReactionButton: JSX.Element | undefined;\n        if (this.context.canReact) {\n            addReactionButton = <ReactButton mxEvent={mxEvent} reactions={reactions} />;\n        }\n\n        return (\n            <div className=\"mx_ReactionsRow\" role=\"toolbar\" aria-label={_t(\"common|reactions\")}>\n                {items}\n                {showAllButton}\n                {addReactionButton}\n            </div>\n        );\n    }\n}\n"],"mappings":"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","ignoreList":[]}