UNPKG

matrix-react-sdk

Version:
205 lines (168 loc) 23.1 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireWildcard(require("react")); var _propTypes = _interopRequireDefault(require("prop-types")); var _languageHandler = require("../../../languageHandler"); var _DateUtils = require("../../../DateUtils"); var _NodeAnimator = _interopRequireDefault(require("../../../NodeAnimator")); var sdk = _interopRequireWildcard(require("../../../index")); var _units = require("../../../utils/units"); var _replaceableComponent = require("../../../utils/replaceableComponent"); var _dec, _class, _class2, _temp; let ReadReceiptMarker = (_dec = (0, _replaceableComponent.replaceableComponent)("views.rooms.ReadReceiptMarker"), _dec(_class = (_temp = _class2 = class ReadReceiptMarker extends _react.default.PureComponent { constructor(props) { super(props); 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.readReceiptInfo; 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; } const avatarNode = this._avatar.current; rrInfo.top = avatarNode.offsetTop; rrInfo.left = avatarNode.offsetLeft; rrInfo.parent = avatarNode.offsetParent; } componentDidMount() { if (!this.state.suppressDisplay) { // we've already done our display - nothing more to do. return; } this._animateMarker(); } componentDidUpdate(prevProps) { const differentLeftOffset = prevProps.leftOffset !== this.props.leftOffset; const visibilityChanged = prevProps.hidden !== this.props.hidden; if (differentLeftOffset || visibilityChanged) { this._animateMarker(); } } _animateMarker() { // treat new RRs as though they were off the top of the screen let oldTop = -15; const oldInfo = this.props.readReceiptInfo; if (oldInfo && oldInfo.parent) { oldTop = oldInfo.top + oldInfo.parent.getBoundingClientRect().top; } const newElement = this._avatar.current; let startTopOffset; if (!newElement.offsetParent) { // 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. console.warn(`ReadReceiptMarker for ${this.props.fallbackUserId} in has no offsetParent`); startTopOffset = 0; } else { startTopOffset = oldTop - newElement.offsetParent.getBoundingClientRect().top; } const startStyles = []; if (oldInfo && oldInfo.left) { // start at the old height and in the old h pos startStyles.push({ top: startTopOffset + "px", left: (0, _units.toPx)(oldInfo.left) }); } startStyles.push({ top: startTopOffset + 'px', left: '0' }); this.setState({ suppressDisplay: false, startStyles: startStyles }); } render() { const MemberAvatar = sdk.getComponent('avatars.MemberAvatar'); if (this.state.suppressDisplay) { return /*#__PURE__*/_react.default.createElement("div", { ref: this._avatar }); } const style = { left: (0, _units.toPx)(this.props.leftOffset), top: '0px' }; let title; if (this.props.timestamp) { const dateString = (0, _DateUtils.formatDate)(new Date(this.props.timestamp), this.props.showTwelveHour); if (!this.props.member || this.props.fallbackUserId === this.props.member.rawDisplayName) { title = (0, _languageHandler._t)("Seen by %(userName)s at %(dateTime)s", { userName: this.props.fallbackUserId, dateTime: dateString }); } else { title = (0, _languageHandler._t)("Seen by %(displayName)s (%(userName)s) at %(dateTime)s", { displayName: this.props.member.rawDisplayName, userName: this.props.fallbackUserId, dateTime: dateString }); } } return /*#__PURE__*/_react.default.createElement(_NodeAnimator.default, { startStyles: this.state.startStyles }, /*#__PURE__*/_react.default.createElement(MemberAvatar, { member: this.props.member, fallbackUserId: this.props.fallbackUserId, "aria-hidden": "true", width: 14, height: 14, resizeMethod: "crop", style: style, title: title, onClick: this.props.onClick, inputRef: this._avatar })); } }, (0, _defineProperty2.default)(_class2, "propTypes", { // the RoomMember to show the RR for member: _propTypes.default.object, // userId to fallback the avatar to // if the member hasn't been loaded yet fallbackUserId: _propTypes.default.string.isRequired, // number of pixels to offset the avatar from the right of its parent; // typically a negative value. leftOffset: _propTypes.default.number, // true to hide the avatar (it will still be animated) hidden: _propTypes.default.bool, // don't animate this RR into position suppressAnimation: _propTypes.default.bool, // an opaque object for storing information about this user's RR in // this room readReceiptInfo: _propTypes.default.object, // A function which is used to check if the parent panel is being // unmounted, to avoid unnecessary work. Should return true if we // are being unmounted. checkUnmounting: _propTypes.default.func, // callback for clicks on this RR onClick: _propTypes.default.func, // Timestamp when the receipt was read timestamp: _propTypes.default.number, // True to show twelve hour format, false otherwise showTwelveHour: _propTypes.default.bool }), (0, _defineProperty2.default)(_class2, "defaultProps", { leftOffset: 0 }), _temp)) || _class); exports.default = ReadReceiptMarker; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/rooms/ReadReceiptMarker.js"],"names":["ReadReceiptMarker","React","PureComponent","constructor","props","_avatar","state","suppressDisplay","suppressAnimation","componentWillUnmount","rrInfo","readReceiptInfo","checkUnmounting","avatarNode","current","top","offsetTop","left","offsetLeft","parent","offsetParent","componentDidMount","_animateMarker","componentDidUpdate","prevProps","differentLeftOffset","leftOffset","visibilityChanged","hidden","oldTop","oldInfo","getBoundingClientRect","newElement","startTopOffset","console","warn","fallbackUserId","startStyles","push","setState","render","MemberAvatar","sdk","getComponent","style","title","timestamp","dateString","Date","showTwelveHour","member","rawDisplayName","userName","dateTime","displayName","onClick","PropTypes","object","string","isRequired","number","bool","func"],"mappings":"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","sourcesContent":["/*\nCopyright 2016 OpenMarket Ltd\nCopyright 2019 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport React, {createRef} from 'react';\nimport PropTypes from 'prop-types';\nimport { _t } from '../../../languageHandler';\nimport {formatDate} from '../../../DateUtils';\nimport NodeAnimator from \"../../../NodeAnimator\";\nimport * as sdk from \"../../../index\";\nimport {toPx} from \"../../../utils/units\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\n\n@replaceableComponent(\"views.rooms.ReadReceiptMarker\")\nexport default class ReadReceiptMarker extends React.PureComponent {\n    static propTypes = {\n        // the RoomMember to show the RR for\n        member: PropTypes.object,\n        // userId to fallback the avatar to\n        // if the member hasn't been loaded yet\n        fallbackUserId: PropTypes.string.isRequired,\n\n        // number of pixels to offset the avatar from the right of its parent;\n        // typically a negative value.\n        leftOffset: PropTypes.number,\n\n        // true to hide the avatar (it will still be animated)\n        hidden: PropTypes.bool,\n\n        // don't animate this RR into position\n        suppressAnimation: PropTypes.bool,\n\n        // an opaque object for storing information about this user's RR in\n        // this room\n        readReceiptInfo: PropTypes.object,\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: PropTypes.func,\n\n        // callback for clicks on this RR\n        onClick: PropTypes.func,\n\n        // Timestamp when the receipt was read\n        timestamp: PropTypes.number,\n\n        // True to show twelve hour format, false otherwise\n        showTwelveHour: PropTypes.bool,\n    };\n\n    static defaultProps = {\n        leftOffset: 0,\n    };\n\n    constructor(props) {\n        super(props);\n\n        this._avatar = createRef();\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    componentWillUnmount() {\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.readReceiptInfo;\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        const avatarNode = this._avatar.current;\n        rrInfo.top = avatarNode.offsetTop;\n        rrInfo.left = avatarNode.offsetLeft;\n        rrInfo.parent = avatarNode.offsetParent;\n    }\n\n    componentDidMount() {\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    componentDidUpdate(prevProps) {\n        const differentLeftOffset = prevProps.leftOffset !== this.props.leftOffset;\n        const visibilityChanged = prevProps.hidden !== this.props.hidden;\n        if (differentLeftOffset || visibilityChanged) {\n            this._animateMarker();\n        }\n    }\n\n    _animateMarker() {\n        // treat new RRs as though they were off the top of the screen\n        let oldTop = -15;\n\n        const oldInfo = this.props.readReceiptInfo;\n        if (oldInfo && oldInfo.parent) {\n            oldTop = oldInfo.top + oldInfo.parent.getBoundingClientRect().top;\n        }\n\n        const newElement = this._avatar.current;\n        let startTopOffset;\n        if (!newElement.offsetParent) {\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            console.warn(\n                `ReadReceiptMarker for ${this.props.fallbackUserId} in has no offsetParent`,\n            );\n            startTopOffset = 0;\n        } else {\n            startTopOffset = oldTop - newElement.offsetParent.getBoundingClientRect().top;\n        }\n\n        const startStyles = [];\n\n        if (oldInfo && oldInfo.left) {\n            // start at the old height and in the old h pos\n            startStyles.push({ top: startTopOffset+\"px\",\n                               left: toPx(oldInfo.left) });\n        }\n\n        startStyles.push({ top: startTopOffset+'px', left: '0' });\n\n        this.setState({\n            suppressDisplay: false,\n            startStyles: startStyles,\n        });\n    }\n\n    render() {\n        const MemberAvatar = sdk.getComponent('avatars.MemberAvatar');\n        if (this.state.suppressDisplay) {\n            return <div ref={this._avatar} />;\n        }\n\n        const style = {\n            left: toPx(this.props.leftOffset),\n            top: '0px',\n        };\n\n        let title;\n        if (this.props.timestamp) {\n            const dateString = formatDate(new Date(this.props.timestamp), this.props.showTwelveHour);\n            if (!this.props.member || this.props.fallbackUserId === this.props.member.rawDisplayName) {\n                title = _t(\n                    \"Seen by %(userName)s at %(dateTime)s\",\n                    {userName: this.props.fallbackUserId,\n                    dateTime: dateString},\n                );\n            } else {\n                title = _t(\n                    \"Seen by %(displayName)s (%(userName)s) at %(dateTime)s\",\n                    {displayName: this.props.member.rawDisplayName,\n                    userName: this.props.fallbackUserId,\n                    dateTime: dateString},\n                );\n            }\n        }\n\n        return (\n            <NodeAnimator startStyles={this.state.startStyles}>\n                <MemberAvatar\n                    member={this.props.member}\n                    fallbackUserId={this.props.fallbackUserId}\n                    aria-hidden=\"true\"\n                    width={14} height={14} resizeMethod=\"crop\"\n                    style={style}\n                    title={title}\n                    onClick={this.props.onClick}\n                    inputRef={this._avatar}\n                />\n            </NodeAnimator>\n        );\n    }\n}\n"]}