matrix-react-sdk
Version:
SDK for matrix.org using React
224 lines (219 loc) • 34.5 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.READ_AVATAR_SIZE = void 0;
exports.ReadReceiptGroup = ReadReceiptGroup;
exports.ReadReceiptPerson = ReadReceiptPerson;
exports.determineAvatarPosition = determineAvatarPosition;
exports.readReceiptTooltip = readReceiptTooltip;
var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends"));
var _react = _interopRequireDefault(require("react"));
var _compoundWeb = require("@vector-im/compound-web");
var _ReadReceiptMarker = _interopRequireDefault(require("./ReadReceiptMarker"));
var _AccessibleButton = _interopRequireDefault(require("../elements/AccessibleButton"));
var _MemberAvatar = _interopRequireDefault(require("../avatars/MemberAvatar"));
var _AutoHideScrollbar = _interopRequireDefault(require("../../structures/AutoHideScrollbar"));
var _DateUtils = require("../../../DateUtils");
var _actions = require("../../../dispatcher/actions");
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _ContextMenu = _interopRequireWildcard(require("../../structures/ContextMenu"));
var _languageHandler = require("../../../languageHandler");
var _RovingTabIndex = require("../../../accessibility/RovingTabIndex");
var _FormattingUtils = require("../../../utils/FormattingUtils");
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 2022 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.
*/
// #20547 Design specified that we should show the three latest read receipts
const MAX_READ_AVATARS_PLUS_N = 3;
// #21935 If we’ve got just 4, don’t show +1, just show all of them
const MAX_READ_AVATARS = MAX_READ_AVATARS_PLUS_N + 1;
const READ_AVATAR_OFFSET = 10;
const READ_AVATAR_SIZE = exports.READ_AVATAR_SIZE = 16;
function determineAvatarPosition(index, max) {
if (index < max) {
return {
hidden: false,
position: index
};
} else {
return {
hidden: true,
position: 0
};
}
}
function readReceiptTooltip(members, maxAvatars) {
return (0, _FormattingUtils.formatList)(members, maxAvatars);
}
function ReadReceiptGroup({
readReceipts,
readReceiptMap,
checkUnmounting,
suppressAnimation,
isTwelveHour
}) {
const [menuDisplayed, button, openMenu, closeMenu] = (0, _ContextMenu.useContextMenu)();
// If we are above MAX_READ_AVATARS, we’ll have to remove a few to have space for the +n count.
const hasMore = readReceipts.length > MAX_READ_AVATARS;
const maxAvatars = hasMore ? MAX_READ_AVATARS_PLUS_N : MAX_READ_AVATARS;
const tooltipMembers = readReceipts.map(it => it.roomMember?.name ?? it.userId);
const tooltipText = readReceiptTooltip(tooltipMembers, maxAvatars);
// return early if there are no read receipts
if (readReceipts.length === 0) {
// We currently must include `mx_ReadReceiptGroup_container` in
// the DOM of all events, as it is the positioned parent of the
// animated read receipts. We can't let it unmount when a receipt
// moves events, so for now we mount it for all events. Without
// it, the animation will start from the top of the timeline
// (because it lost its container).
// See also https://github.com/vector-im/element-web/issues/17561
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_EventTile_msgOption"
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ReadReceiptGroup"
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ReadReceiptGroup_button"
}, /*#__PURE__*/_react.default.createElement("span", {
className: "mx_ReadReceiptGroup_container"
}))));
}
const avatars = readReceipts.map((receipt, index) => {
const {
hidden,
position
} = determineAvatarPosition(index, maxAvatars);
const userId = receipt.userId;
let readReceiptPosition;
if (readReceiptMap) {
readReceiptPosition = readReceiptMap[userId];
if (!readReceiptPosition) {
readReceiptPosition = {};
readReceiptMap[userId] = readReceiptPosition;
}
}
return /*#__PURE__*/_react.default.createElement(_ReadReceiptMarker.default, {
key: userId,
member: receipt.roomMember,
fallbackUserId: userId,
offset: position * READ_AVATAR_OFFSET,
hidden: hidden,
readReceiptPosition: readReceiptPosition,
checkUnmounting: checkUnmounting,
suppressAnimation: suppressAnimation,
timestamp: receipt.ts,
showTwelveHour: isTwelveHour
});
}).reverse();
let remText;
const remainder = readReceipts.length - maxAvatars;
if (remainder > 0) {
remText = /*#__PURE__*/_react.default.createElement("span", {
className: "mx_ReadReceiptGroup_remainder",
"aria-live": "off"
}, "+", remainder);
}
let contextMenu;
if (menuDisplayed && button.current) {
const buttonRect = button.current.getBoundingClientRect();
contextMenu = /*#__PURE__*/_react.default.createElement(_ContextMenu.default, (0, _extends2.default)({
menuClassName: "mx_ReadReceiptGroup_popup",
onFinished: closeMenu
}, (0, _ContextMenu.aboveLeftOf)(buttonRect)), /*#__PURE__*/_react.default.createElement(_AutoHideScrollbar.default, null, /*#__PURE__*/_react.default.createElement(SectionHeader, {
className: "mx_ReadReceiptGroup_title"
}, (0, _languageHandler._t)("timeline|read_receipt_title", {
count: readReceipts.length
})), readReceipts.map(receipt => /*#__PURE__*/_react.default.createElement(ReadReceiptPerson, (0, _extends2.default)({
key: receipt.userId
}, receipt, {
isTwelveHour: isTwelveHour,
onAfterClick: closeMenu
})))));
}
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_EventTile_msgOption"
}, /*#__PURE__*/_react.default.createElement(_compoundWeb.Tooltip, {
label: (0, _languageHandler._t)("timeline|read_receipt_title", {
count: readReceipts.length
}),
caption: tooltipText,
placement: "top-end"
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ReadReceiptGroup",
role: "group",
"aria-label": (0, _languageHandler._t)("timeline|read_receipts_label")
}, /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
className: "mx_ReadReceiptGroup_button",
ref: button,
"aria-label": tooltipText,
"aria-haspopup": "true",
onClick: openMenu
}, remText, /*#__PURE__*/_react.default.createElement("span", {
className: "mx_ReadReceiptGroup_container",
style: {
width: Math.min(maxAvatars, readReceipts.length) * READ_AVATAR_OFFSET + READ_AVATAR_SIZE - READ_AVATAR_OFFSET
}
}, avatars)), contextMenu)));
}
// Export for testing
function ReadReceiptPerson({
userId,
roomMember,
ts,
isTwelveHour,
onAfterClick
}) {
return /*#__PURE__*/_react.default.createElement(_compoundWeb.Tooltip, {
description: roomMember?.rawDisplayName ?? userId,
caption: userId,
placement: "top"
}, /*#__PURE__*/_react.default.createElement("div", null, /*#__PURE__*/_react.default.createElement(_ContextMenu.MenuItem, {
className: "mx_ReadReceiptGroup_person",
onClick: () => {
_dispatcher.default.dispatch({
action: _actions.Action.ViewUser,
// XXX: We should be using a real member object and not assuming what the receiver wants.
// The ViewUser action leads to the RightPanelStore, and RightPanelStoreIPanelState defines the
// member property of IRightPanelCardState as `RoomMember | User`, so we’re fine for now, but we
// should definitely clean this up later
member: roomMember ?? {
userId
},
push: false
});
onAfterClick?.();
}
}, /*#__PURE__*/_react.default.createElement(_MemberAvatar.default, {
member: roomMember,
fallbackUserId: userId,
size: "24px",
"aria-hidden": "true",
"aria-live": "off",
resizeMethod: "crop",
hideTitle: true
}), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ReadReceiptGroup_name"
}, /*#__PURE__*/_react.default.createElement("p", null, roomMember?.name ?? userId), /*#__PURE__*/_react.default.createElement("p", {
className: "mx_ReadReceiptGroup_secondary"
}, (0, _DateUtils.formatDate)(new Date(ts), isTwelveHour))))));
}
function SectionHeader({
className,
children
}) {
const [onFocus,, ref] = (0, _RovingTabIndex.useRovingTabIndex)();
return /*#__PURE__*/_react.default.createElement("h3", {
className: className,
role: "menuitem",
onFocus: onFocus,
tabIndex: -1,
ref: ref
}, children);
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_compoundWeb","_ReadReceiptMarker","_AccessibleButton","_MemberAvatar","_AutoHideScrollbar","_DateUtils","_actions","_dispatcher","_ContextMenu","_interopRequireWildcard","_languageHandler","_RovingTabIndex","_FormattingUtils","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","MAX_READ_AVATARS_PLUS_N","MAX_READ_AVATARS","READ_AVATAR_OFFSET","READ_AVATAR_SIZE","exports","determineAvatarPosition","index","max","hidden","position","readReceiptTooltip","members","maxAvatars","formatList","ReadReceiptGroup","readReceipts","readReceiptMap","checkUnmounting","suppressAnimation","isTwelveHour","menuDisplayed","button","openMenu","closeMenu","useContextMenu","hasMore","length","tooltipMembers","map","it","roomMember","name","userId","tooltipText","createElement","className","avatars","receipt","readReceiptPosition","key","member","fallbackUserId","offset","timestamp","ts","showTwelveHour","reverse","remText","remainder","contextMenu","current","buttonRect","getBoundingClientRect","_extends2","menuClassName","onFinished","aboveLeftOf","SectionHeader","_t","count","ReadReceiptPerson","onAfterClick","Tooltip","label","caption","placement","role","ref","onClick","style","width","Math","min","description","rawDisplayName","MenuItem","dis","dispatch","action","Action","ViewUser","push","size","resizeMethod","hideTitle","formatDate","Date","children","onFocus","useRovingTabIndex","tabIndex"],"sources":["../../../../src/components/views/rooms/ReadReceiptGroup.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2022 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, { PropsWithChildren } from \"react\";\nimport { User } from \"matrix-js-sdk/src/matrix\";\nimport { Tooltip } from \"@vector-im/compound-web\";\n\nimport ReadReceiptMarker, { IReadReceiptPosition } from \"./ReadReceiptMarker\";\nimport { IReadReceiptProps } from \"./EventTile\";\nimport AccessibleButton from \"../elements/AccessibleButton\";\nimport MemberAvatar from \"../avatars/MemberAvatar\";\nimport AutoHideScrollbar from \"../../structures/AutoHideScrollbar\";\nimport { formatDate } from \"../../../DateUtils\";\nimport { Action } from \"../../../dispatcher/actions\";\nimport dis from \"../../../dispatcher/dispatcher\";\nimport ContextMenu, { aboveLeftOf, MenuItem, useContextMenu } from \"../../structures/ContextMenu\";\nimport { _t } from \"../../../languageHandler\";\nimport { useRovingTabIndex } from \"../../../accessibility/RovingTabIndex\";\nimport { formatList } from \"../../../utils/FormattingUtils\";\n\n// #20547 Design specified that we should show the three latest read receipts\nconst MAX_READ_AVATARS_PLUS_N = 3;\n// #21935 If we’ve got just 4, don’t show +1, just show all of them\nconst MAX_READ_AVATARS = MAX_READ_AVATARS_PLUS_N + 1;\n\nconst READ_AVATAR_OFFSET = 10;\nexport const READ_AVATAR_SIZE = 16;\n\ninterface Props {\n    readReceipts: IReadReceiptProps[];\n    readReceiptMap: { [userId: string]: IReadReceiptPosition };\n    checkUnmounting?: () => boolean;\n    suppressAnimation: boolean;\n    isTwelveHour?: boolean;\n}\n\ninterface IAvatarPosition {\n    hidden: boolean;\n    position: number;\n}\n\nexport function determineAvatarPosition(index: number, max: number): IAvatarPosition {\n    if (index < max) {\n        return {\n            hidden: false,\n            position: index,\n        };\n    } else {\n        return {\n            hidden: true,\n            position: 0,\n        };\n    }\n}\n\nexport function readReceiptTooltip(members: string[], maxAvatars: number): string | undefined {\n    return formatList(members, maxAvatars);\n}\n\nexport function ReadReceiptGroup({\n    readReceipts,\n    readReceiptMap,\n    checkUnmounting,\n    suppressAnimation,\n    isTwelveHour,\n}: Props): JSX.Element {\n    const [menuDisplayed, button, openMenu, closeMenu] = useContextMenu();\n\n    // If we are above MAX_READ_AVATARS, we’ll have to remove a few to have space for the +n count.\n    const hasMore = readReceipts.length > MAX_READ_AVATARS;\n    const maxAvatars = hasMore ? MAX_READ_AVATARS_PLUS_N : MAX_READ_AVATARS;\n\n    const tooltipMembers: string[] = readReceipts.map((it) => it.roomMember?.name ?? it.userId);\n    const tooltipText = readReceiptTooltip(tooltipMembers, maxAvatars);\n\n    // return early if there are no read receipts\n    if (readReceipts.length === 0) {\n        // We currently must include `mx_ReadReceiptGroup_container` in\n        // the DOM of all events, as it is the positioned parent of the\n        // animated read receipts. We can't let it unmount when a receipt\n        // moves events, so for now we mount it for all events. Without\n        // it, the animation will start from the top of the timeline\n        // (because it lost its container).\n        // See also https://github.com/vector-im/element-web/issues/17561\n        return (\n            <div className=\"mx_EventTile_msgOption\">\n                <div className=\"mx_ReadReceiptGroup\">\n                    <div className=\"mx_ReadReceiptGroup_button\">\n                        <span className=\"mx_ReadReceiptGroup_container\" />\n                    </div>\n                </div>\n            </div>\n        );\n    }\n\n    const avatars = readReceipts\n        .map((receipt, index) => {\n            const { hidden, position } = determineAvatarPosition(index, maxAvatars);\n\n            const userId = receipt.userId;\n            let readReceiptPosition: IReadReceiptPosition | undefined;\n\n            if (readReceiptMap) {\n                readReceiptPosition = readReceiptMap[userId];\n                if (!readReceiptPosition) {\n                    readReceiptPosition = {};\n                    readReceiptMap[userId] = readReceiptPosition;\n                }\n            }\n\n            return (\n                <ReadReceiptMarker\n                    key={userId}\n                    member={receipt.roomMember}\n                    fallbackUserId={userId}\n                    offset={position * READ_AVATAR_OFFSET}\n                    hidden={hidden}\n                    readReceiptPosition={readReceiptPosition}\n                    checkUnmounting={checkUnmounting}\n                    suppressAnimation={suppressAnimation}\n                    timestamp={receipt.ts}\n                    showTwelveHour={isTwelveHour}\n                />\n            );\n        })\n        .reverse();\n\n    let remText: JSX.Element | undefined;\n    const remainder = readReceipts.length - maxAvatars;\n    if (remainder > 0) {\n        remText = (\n            <span className=\"mx_ReadReceiptGroup_remainder\" aria-live=\"off\">\n                +{remainder}\n            </span>\n        );\n    }\n\n    let contextMenu: JSX.Element | undefined;\n    if (menuDisplayed && button.current) {\n        const buttonRect = button.current.getBoundingClientRect();\n        contextMenu = (\n            <ContextMenu menuClassName=\"mx_ReadReceiptGroup_popup\" onFinished={closeMenu} {...aboveLeftOf(buttonRect)}>\n                <AutoHideScrollbar>\n                    <SectionHeader className=\"mx_ReadReceiptGroup_title\">\n                        {_t(\"timeline|read_receipt_title\", { count: readReceipts.length })}\n                    </SectionHeader>\n                    {readReceipts.map((receipt) => (\n                        <ReadReceiptPerson\n                            key={receipt.userId}\n                            {...receipt}\n                            isTwelveHour={isTwelveHour}\n                            onAfterClick={closeMenu}\n                        />\n                    ))}\n                </AutoHideScrollbar>\n            </ContextMenu>\n        );\n    }\n\n    return (\n        <div className=\"mx_EventTile_msgOption\">\n            <Tooltip\n                label={_t(\"timeline|read_receipt_title\", { count: readReceipts.length })}\n                caption={tooltipText}\n                placement=\"top-end\"\n            >\n                <div className=\"mx_ReadReceiptGroup\" role=\"group\" aria-label={_t(\"timeline|read_receipts_label\")}>\n                    <AccessibleButton\n                        className=\"mx_ReadReceiptGroup_button\"\n                        ref={button}\n                        aria-label={tooltipText}\n                        aria-haspopup=\"true\"\n                        onClick={openMenu}\n                    >\n                        {remText}\n                        <span\n                            className=\"mx_ReadReceiptGroup_container\"\n                            style={{\n                                width:\n                                    Math.min(maxAvatars, readReceipts.length) * READ_AVATAR_OFFSET +\n                                    READ_AVATAR_SIZE -\n                                    READ_AVATAR_OFFSET,\n                            }}\n                        >\n                            {avatars}\n                        </span>\n                    </AccessibleButton>\n                    {contextMenu}\n                </div>\n            </Tooltip>\n        </div>\n    );\n}\n\ninterface ReadReceiptPersonProps extends IReadReceiptProps {\n    isTwelveHour?: boolean;\n    onAfterClick?: () => void;\n}\n\n// Export for testing\nexport function ReadReceiptPerson({\n    userId,\n    roomMember,\n    ts,\n    isTwelveHour,\n    onAfterClick,\n}: ReadReceiptPersonProps): JSX.Element {\n    return (\n        <Tooltip description={roomMember?.rawDisplayName ?? userId} caption={userId} placement=\"top\">\n            <div>\n                <MenuItem\n                    className=\"mx_ReadReceiptGroup_person\"\n                    onClick={() => {\n                        dis.dispatch({\n                            action: Action.ViewUser,\n                            // XXX: We should be using a real member object and not assuming what the receiver wants.\n                            // The ViewUser action leads to the RightPanelStore, and RightPanelStoreIPanelState defines the\n                            // member property of IRightPanelCardState as `RoomMember | User`, so we’re fine for now, but we\n                            // should definitely clean this up later\n                            member: roomMember ?? ({ userId } as User),\n                            push: false,\n                        });\n                        onAfterClick?.();\n                    }}\n                >\n                    <MemberAvatar\n                        member={roomMember}\n                        fallbackUserId={userId}\n                        size=\"24px\"\n                        aria-hidden=\"true\"\n                        aria-live=\"off\"\n                        resizeMethod=\"crop\"\n                        hideTitle\n                    />\n                    <div className=\"mx_ReadReceiptGroup_name\">\n                        <p>{roomMember?.name ?? userId}</p>\n                        <p className=\"mx_ReadReceiptGroup_secondary\">{formatDate(new Date(ts), isTwelveHour)}</p>\n                    </div>\n                </MenuItem>\n            </div>\n        </Tooltip>\n    );\n}\n\ninterface ISectionHeaderProps {\n    className?: string;\n}\n\nfunction SectionHeader({ className, children }: PropsWithChildren<ISectionHeaderProps>): JSX.Element {\n    const [onFocus, , ref] = useRovingTabIndex<HTMLHeadingElement>();\n\n    return (\n        <h3 className={className} role=\"menuitem\" onFocus={onFocus} tabIndex={-1} ref={ref}>\n            {children}\n        </h3>\n    );\n}\n"],"mappings":"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","ignoreList":[]}