UNPKG

matrix-react-sdk

Version:
124 lines (119 loc) 18.9 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends")); var _objectWithoutProperties2 = _interopRequireDefault(require("@babel/runtime/helpers/objectWithoutProperties")); var _react = _interopRequireWildcard(require("react")); var _classnames = _interopRequireDefault(require("classnames")); var _matrix = require("matrix-js-sdk/src/matrix"); var _compoundWeb = require("@vector-im/compound-web"); var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore")); var _RoomContext = _interopRequireDefault(require("../../../contexts/RoomContext")); var _MatrixClientContext = _interopRequireDefault(require("../../../contexts/MatrixClientContext")); var _useEventEmitter = require("../../../hooks/useEventEmitter"); var _languageHandler = require("../../../languageHandler"); const _excluded = ["name", "idName", "title", "url", "urls", "size", "onClick", "className", "type", "altText"]; /* Copyright 2024 New Vector Ltd. Copyright 2019, 2020 The Matrix.org Foundation C.I.C. Copyright 2019 Michael Telatynski <7t3chguy@gmail.com> Copyright 2018 New Vector Ltd Copyright 2015, 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. */ 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; } const calculateUrls = (url, urls, lowBandwidth = false) => { // work out the full set of urls to try to load. This is formed like so: // imageUrls: [ props.url, ...props.urls ] let _urls = []; if (!lowBandwidth) { _urls = urls || []; if (url) { // copy urls and put url first _urls = [url, ..._urls]; } } // deduplicate URLs return Array.from(new Set(_urls)); }; const useImageUrl = ({ url, urls }) => { // Since this is a hot code path and the settings store can be slow, we // use the cached lowBandwidth value from the room context if it exists const roomContext = (0, _react.useContext)(_RoomContext.default); const lowBandwidth = roomContext ? roomContext.lowBandwidth : _SettingsStore.default.getValue("lowBandwidth"); const [imageUrls, setUrls] = (0, _react.useState)(calculateUrls(url, urls, lowBandwidth)); const [urlsIndex, setIndex] = (0, _react.useState)(0); const onError = (0, _react.useCallback)(() => { setIndex(i => i + 1); // try the next one }, []); (0, _react.useEffect)(() => { setUrls(calculateUrls(url, urls, lowBandwidth)); setIndex(0); }, [url, JSON.stringify(urls)]); // eslint-disable-line react-hooks/exhaustive-deps const cli = (0, _react.useContext)(_MatrixClientContext.default); const onClientSync = (0, _react.useCallback)((syncState, prevState) => { // Consider the client reconnected if there is no error with syncing. // This means the state could be RECONNECTING, SYNCING, PREPARED or CATCHUP. const reconnected = syncState !== "ERROR" && prevState !== syncState; if (reconnected) { setIndex(0); } }, []); (0, _useEventEmitter.useTypedEventEmitter)(cli, _matrix.ClientEvent.Sync, onClientSync); const imageUrl = imageUrls[urlsIndex]; return [imageUrl, onError]; }; const BaseAvatar = /*#__PURE__*/(0, _react.forwardRef)((props, ref) => { const { name, idName, title, url, urls, size = "40px", onClick, className, type = "round", altText = (0, _languageHandler._t)("common|avatar") } = props, otherProps = (0, _objectWithoutProperties2.default)(props, _excluded); const [imageUrl, onError] = useImageUrl({ url, urls }); const extraProps = {}; if (onClick) { extraProps["aria-live"] = "off"; extraProps["role"] = "button"; } else if (!imageUrl) { extraProps["role"] = "presentation"; extraProps["aria-label"] = undefined; } else { extraProps["role"] = undefined; } return /*#__PURE__*/_react.default.createElement(_compoundWeb.Avatar, (0, _extends2.default)({ ref: ref, src: imageUrl, id: idName ?? "", name: name ?? "", type: type, size: size, className: (0, _classnames.default)("mx_BaseAvatar", className), "aria-label": altText, onError: onError, title: title, onClick: onClick }, extraProps, otherProps, { "data-testid": "avatar-img" })); }); var _default = exports.default = BaseAvatar; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_classnames","_interopRequireDefault","_matrix","_compoundWeb","_SettingsStore","_RoomContext","_MatrixClientContext","_useEventEmitter","_languageHandler","_excluded","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","calculateUrls","url","urls","lowBandwidth","_urls","Array","from","Set","useImageUrl","roomContext","useContext","RoomContext","SettingsStore","getValue","imageUrls","setUrls","useState","urlsIndex","setIndex","onError","useCallback","useEffect","JSON","stringify","cli","MatrixClientContext","onClientSync","syncState","prevState","reconnected","useTypedEventEmitter","ClientEvent","Sync","imageUrl","BaseAvatar","forwardRef","props","ref","name","idName","title","size","onClick","className","type","altText","_t","otherProps","_objectWithoutProperties2","extraProps","undefined","createElement","Avatar","_extends2","src","id","classNames","_default","exports"],"sources":["../../../../src/components/views/avatars/BaseAvatar.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2019, 2020 The Matrix.org Foundation C.I.C.\nCopyright 2019 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2018 New Vector Ltd\nCopyright 2015, 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, { AriaRole, forwardRef, useCallback, useContext, useEffect, useState } from \"react\";\nimport classNames from \"classnames\";\nimport { ClientEvent, SyncState } from \"matrix-js-sdk/src/matrix\";\nimport { Avatar } from \"@vector-im/compound-web\";\n\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport { ButtonEvent } from \"../elements/AccessibleButton\";\nimport RoomContext from \"../../../contexts/RoomContext\";\nimport MatrixClientContext from \"../../../contexts/MatrixClientContext\";\nimport { useTypedEventEmitter } from \"../../../hooks/useEventEmitter\";\nimport { _t } from \"../../../languageHandler\";\n\ninterface IProps {\n    name?: React.ComponentProps<typeof Avatar>[\"name\"]; // The name (first initial used as default)\n    idName?: React.ComponentProps<typeof Avatar>[\"id\"]; // ID for generating hash colours\n    title?: string; // onHover title text\n    url?: string | null; // highest priority of them all, shortcut to set in urls[0]\n    urls?: string[]; // [highest_priority, ... , lowest_priority]\n    type?: React.ComponentProps<typeof Avatar>[\"type\"];\n    size: string;\n    onClick?: (ev: ButtonEvent) => void;\n    className?: string;\n    tabIndex?: number;\n    altText?: string;\n    role?: AriaRole;\n}\n\nconst calculateUrls = (url?: string | null, urls?: string[], lowBandwidth = false): string[] => {\n    // work out the full set of urls to try to load. This is formed like so:\n    // imageUrls: [ props.url, ...props.urls ]\n\n    let _urls: string[] = [];\n    if (!lowBandwidth) {\n        _urls = urls || [];\n\n        if (url) {\n            // copy urls and put url first\n            _urls = [url, ..._urls];\n        }\n    }\n\n    // deduplicate URLs\n    return Array.from(new Set(_urls));\n};\n\nconst useImageUrl = ({ url, urls }: { url?: string | null; urls?: string[] }): [string, () => void] => {\n    // Since this is a hot code path and the settings store can be slow, we\n    // use the cached lowBandwidth value from the room context if it exists\n    const roomContext = useContext(RoomContext);\n    const lowBandwidth = roomContext ? roomContext.lowBandwidth : SettingsStore.getValue(\"lowBandwidth\");\n\n    const [imageUrls, setUrls] = useState<string[]>(calculateUrls(url, urls, lowBandwidth));\n    const [urlsIndex, setIndex] = useState<number>(0);\n\n    const onError = useCallback(() => {\n        setIndex((i) => i + 1); // try the next one\n    }, []);\n\n    useEffect(() => {\n        setUrls(calculateUrls(url, urls, lowBandwidth));\n        setIndex(0);\n    }, [url, JSON.stringify(urls)]); // eslint-disable-line react-hooks/exhaustive-deps\n\n    const cli = useContext(MatrixClientContext);\n    const onClientSync = useCallback((syncState: SyncState, prevState: SyncState | null) => {\n        // Consider the client reconnected if there is no error with syncing.\n        // This means the state could be RECONNECTING, SYNCING, PREPARED or CATCHUP.\n        const reconnected = syncState !== \"ERROR\" && prevState !== syncState;\n        if (reconnected) {\n            setIndex(0);\n        }\n    }, []);\n    useTypedEventEmitter(cli, ClientEvent.Sync, onClientSync);\n\n    const imageUrl = imageUrls[urlsIndex];\n    return [imageUrl, onError];\n};\n\nconst BaseAvatar = forwardRef<HTMLElement, IProps>((props, ref) => {\n    const {\n        name,\n        idName,\n        title,\n        url,\n        urls,\n        size = \"40px\",\n        onClick,\n        className,\n        type = \"round\",\n        altText = _t(\"common|avatar\"),\n        ...otherProps\n    } = props;\n\n    const [imageUrl, onError] = useImageUrl({ url, urls });\n\n    const extraProps: Partial<React.ComponentProps<typeof Avatar>> = {};\n\n    if (onClick) {\n        extraProps[\"aria-live\"] = \"off\";\n        extraProps[\"role\"] = \"button\";\n    } else if (!imageUrl) {\n        extraProps[\"role\"] = \"presentation\";\n        extraProps[\"aria-label\"] = undefined;\n    } else {\n        extraProps[\"role\"] = undefined;\n    }\n\n    return (\n        <Avatar\n            ref={ref}\n            src={imageUrl}\n            id={idName ?? \"\"}\n            name={name ?? \"\"}\n            type={type}\n            size={size}\n            className={classNames(\"mx_BaseAvatar\", className)}\n            aria-label={altText}\n            onError={onError}\n            title={title}\n            onClick={onClick}\n            {...extraProps}\n            {...otherProps}\n            data-testid=\"avatar-img\"\n        />\n    );\n});\n\nexport default BaseAvatar;\nexport type BaseAvatarType = React.FC<IProps>;\n"],"mappings":"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","ignoreList":[]}