UNPKG

matrix-react-sdk

Version:
139 lines (134 loc) 21.4 kB
"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 _logger = require("matrix-js-sdk/src/logger"); var _NodeAnimator = _interopRequireDefault(require("../../../NodeAnimator")); var _units = require("../../../utils/units"); var _MemberAvatar = _interopRequireDefault(require("../avatars/MemberAvatar")); var _ReadReceiptGroup = require("./ReadReceiptGroup"); 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 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. */ // The top & right from the bounding client rect of each read receipt class ReadReceiptMarker extends _react.default.PureComponent { constructor(props) { super(props); (0, _defineProperty2.default)(this, "avatar", /*#__PURE__*/(0, _react.createRef)()); this.state = { // if we are going to animate the RR, we don't show it on first render, // and instead just add a placeholder to the DOM; once we've been // mounted, we start an animation which moves the RR from its old // position. suppressDisplay: !this.props.suppressAnimation }; } componentWillUnmount() { // before we remove the rr, store its location in the map, so that if // it reappears, it can be animated from the right place. const rrInfo = this.props.readReceiptPosition; if (!rrInfo) { return; } // checking the DOM properties can force a re-layout, which can be // quite expensive; so if the parent messagepanel is being unmounted, // then don't bother with this. if (this.props.checkUnmounting && this.props.checkUnmounting()) { return; } this.buildReadReceiptInfo(rrInfo); } componentDidMount() { if (!this.state.suppressDisplay) { // we've already done our display - nothing more to do. return; } this.animateMarker(); } componentDidUpdate(prevProps) { const differentOffset = prevProps.offset !== this.props.offset; const visibilityChanged = prevProps.hidden !== this.props.hidden; if (differentOffset || visibilityChanged) { this.animateMarker(); } } buildReadReceiptInfo(target = {}) { const element = this.avatar.current; // this is the mx_ReadReceiptsGroup_container const horizontalContainer = element?.offsetParent; if (!horizontalContainer || !horizontalContainer.getBoundingClientRect) { // this seems to happen sometimes for reasons I don't understand // the docs for `offsetParent` say it may be null if `display` is // `none`, but I can't see why that would happen. _logger.logger.warn(`ReadReceiptMarker for ${this.props.fallbackUserId} has no valid horizontalContainer`); target.top = 0; target.right = 0; return target; } const elementRect = element.getBoundingClientRect(); target.top = elementRect.top; target.right = elementRect.right - horizontalContainer.getBoundingClientRect().right; return target; } animateMarker() { const oldInfo = this.props.readReceiptPosition; const newInfo = this.buildReadReceiptInfo(); const newPosition = newInfo.top ?? 0; const oldPosition = oldInfo && oldInfo.top !== undefined ? // start at the old height and in the old h pos oldInfo.top : // treat new RRs as though they were off the top of the screen -_ReadReceiptGroup.READ_AVATAR_SIZE; const startStyles = []; if (oldInfo?.right) { startStyles.push({ top: oldPosition - newPosition, right: oldInfo.right }); } startStyles.push({ top: oldPosition - newPosition, right: 0 }); this.setState({ suppressDisplay: false, startStyles }); } render() { if (this.state.suppressDisplay) { return /*#__PURE__*/_react.default.createElement("div", { ref: this.avatar }); } const style = { right: (0, _units.toPx)(this.props.offset), top: "0px" }; return /*#__PURE__*/_react.default.createElement(_NodeAnimator.default, { startStyles: this.state.startStyles, innerRef: this.avatar }, /*#__PURE__*/_react.default.createElement(_MemberAvatar.default, { member: this.props.member ?? null, fallbackUserId: this.props.fallbackUserId, "aria-hidden": "true", "aria-live": "off", size: "14px", style: style, hideTitle: true, tabIndex: -1 })); } } exports.default = ReadReceiptMarker; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_logger","_NodeAnimator","_interopRequireDefault","_units","_MemberAvatar","_ReadReceiptGroup","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","ReadReceiptMarker","React","PureComponent","constructor","props","_defineProperty2","createRef","state","suppressDisplay","suppressAnimation","componentWillUnmount","rrInfo","readReceiptPosition","checkUnmounting","buildReadReceiptInfo","componentDidMount","animateMarker","componentDidUpdate","prevProps","differentOffset","offset","visibilityChanged","hidden","target","element","avatar","current","horizontalContainer","offsetParent","getBoundingClientRect","logger","warn","fallbackUserId","top","right","elementRect","oldInfo","newInfo","newPosition","oldPosition","undefined","READ_AVATAR_SIZE","startStyles","push","setState","render","createElement","ref","style","toPx","innerRef","member","size","hideTitle","tabIndex","exports"],"sources":["../../../../src/components/views/rooms/ReadReceiptMarker.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2019 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 { RoomMember } from \"matrix-js-sdk/src/matrix\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\n\nimport NodeAnimator from \"../../../NodeAnimator\";\nimport { toPx } from \"../../../utils/units\";\nimport MemberAvatar from \"../avatars/MemberAvatar\";\nimport { READ_AVATAR_SIZE } from \"./ReadReceiptGroup\";\n\n// The top & right from the bounding client rect of each read receipt\nexport interface IReadReceiptPosition {\n    top?: number;\n    right?: number;\n}\n\ninterface IProps {\n    // the RoomMember to show the RR for\n    member?: RoomMember | null;\n    // userId to fallback the avatar to\n    // if the member hasn't been loaded yet\n    fallbackUserId: string;\n\n    // number of pixels to offset the avatar from the right of its parent;\n    // typically a negative value.\n    offset: number;\n\n    // true to hide the avatar (it will still be animated)\n    hidden?: boolean;\n\n    // don't animate this RR into position\n    suppressAnimation?: boolean;\n\n    // an opaque object for storing information about this user's RR in this room\n    readReceiptPosition?: IReadReceiptPosition;\n\n    // A function which is used to check if the parent panel is being\n    // unmounted, to avoid unnecessary work. Should return true if we\n    // are being unmounted.\n    checkUnmounting?: () => boolean;\n\n    // Timestamp when the receipt was read\n    timestamp?: number;\n\n    // True to show twelve hour format, false otherwise\n    showTwelveHour?: boolean;\n}\n\ninterface IState {\n    suppressDisplay: boolean;\n    startStyles?: IReadReceiptMarkerStyle[];\n}\n\ninterface IReadReceiptMarkerStyle {\n    top: number;\n    right: number;\n}\n\nexport default class ReadReceiptMarker extends React.PureComponent<IProps, IState> {\n    private avatar = createRef<HTMLDivElement>();\n\n    public constructor(props: IProps) {\n        super(props);\n\n        this.state = {\n            // if we are going to animate the RR, we don't show it on first render,\n            // and instead just add a placeholder to the DOM; once we've been\n            // mounted, we start an animation which moves the RR from its old\n            // position.\n            suppressDisplay: !this.props.suppressAnimation,\n        };\n    }\n\n    public componentWillUnmount(): void {\n        // before we remove the rr, store its location in the map, so that if\n        // it reappears, it can be animated from the right place.\n        const rrInfo = this.props.readReceiptPosition;\n        if (!rrInfo) {\n            return;\n        }\n\n        // checking the DOM properties can force a re-layout, which can be\n        // quite expensive; so if the parent messagepanel is being unmounted,\n        // then don't bother with this.\n        if (this.props.checkUnmounting && this.props.checkUnmounting()) {\n            return;\n        }\n\n        this.buildReadReceiptInfo(rrInfo);\n    }\n\n    public componentDidMount(): void {\n        if (!this.state.suppressDisplay) {\n            // we've already done our display - nothing more to do.\n            return;\n        }\n        this.animateMarker();\n    }\n\n    public componentDidUpdate(prevProps: IProps): void {\n        const differentOffset = prevProps.offset !== this.props.offset;\n        const visibilityChanged = prevProps.hidden !== this.props.hidden;\n        if (differentOffset || visibilityChanged) {\n            this.animateMarker();\n        }\n    }\n\n    private buildReadReceiptInfo(target: IReadReceiptPosition = {}): IReadReceiptPosition {\n        const element = this.avatar.current;\n        // this is the mx_ReadReceiptsGroup_container\n        const horizontalContainer = element?.offsetParent;\n        if (!horizontalContainer || !horizontalContainer.getBoundingClientRect) {\n            // this seems to happen sometimes for reasons I don't understand\n            // the docs for `offsetParent` say it may be null if `display` is\n            // `none`, but I can't see why that would happen.\n            logger.warn(`ReadReceiptMarker for ${this.props.fallbackUserId} has no valid horizontalContainer`);\n\n            target.top = 0;\n            target.right = 0;\n            return target;\n        }\n\n        const elementRect = element.getBoundingClientRect();\n\n        target.top = elementRect.top;\n        target.right = elementRect.right - horizontalContainer.getBoundingClientRect().right;\n        return target;\n    }\n\n    private animateMarker(): void {\n        const oldInfo = this.props.readReceiptPosition;\n        const newInfo = this.buildReadReceiptInfo();\n\n        const newPosition = newInfo.top ?? 0;\n        const oldPosition =\n            oldInfo && oldInfo.top !== undefined\n                ? // start at the old height and in the old h pos\n                  oldInfo.top\n                : // treat new RRs as though they were off the top of the screen\n                  -READ_AVATAR_SIZE;\n\n        const startStyles: IReadReceiptMarkerStyle[] = [];\n        if (oldInfo?.right) {\n            startStyles.push({\n                top: oldPosition - newPosition,\n                right: oldInfo.right,\n            });\n        }\n        startStyles.push({\n            top: oldPosition - newPosition,\n            right: 0,\n        });\n\n        this.setState({\n            suppressDisplay: false,\n            startStyles,\n        });\n    }\n\n    public render(): React.ReactNode {\n        if (this.state.suppressDisplay) {\n            return <div ref={this.avatar} />;\n        }\n\n        const style = {\n            right: toPx(this.props.offset),\n            top: \"0px\",\n        };\n\n        return (\n            <NodeAnimator startStyles={this.state.startStyles} innerRef={this.avatar}>\n                <MemberAvatar\n                    member={this.props.member ?? null}\n                    fallbackUserId={this.props.fallbackUserId}\n                    aria-hidden=\"true\"\n                    aria-live=\"off\"\n                    size=\"14px\"\n                    style={style}\n                    hideTitle\n                    tabIndex={-1}\n                />\n            </NodeAnimator>\n        );\n    }\n}\n"],"mappings":"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","ignoreList":[]}