matrix-react-sdk
Version:
SDK for matrix.org using React
124 lines (119 loc) • 18.9 kB
JavaScript
;
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":";;;;;;;;;AAWA,IAAAA,MAAA,GAAAC,uBAAA,CAAAC,OAAA;AACA,IAAAC,WAAA,GAAAC,sBAAA,CAAAF,OAAA;AACA,IAAAG,OAAA,GAAAH,OAAA;AACA,IAAAI,YAAA,GAAAJ,OAAA;AAEA,IAAAK,cAAA,GAAAH,sBAAA,CAAAF,OAAA;AAEA,IAAAM,YAAA,GAAAJ,sBAAA,CAAAF,OAAA;AACA,IAAAO,oBAAA,GAAAL,sBAAA,CAAAF,OAAA;AACA,IAAAQ,gBAAA,GAAAR,OAAA;AACA,IAAAS,gBAAA,GAAAT,OAAA;AAA8C,MAAAU,SAAA;AArB9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA,SAAAC,yBAAAC,CAAA,6BAAAC,OAAA,mBAAAC,CAAA,OAAAD,OAAA,IAAAE,CAAA,OAAAF,OAAA,YAAAF,wBAAA,YAAAA,CAAAC,CAAA,WAAAA,CAAA,GAAAG,CAAA,GAAAD,CAAA,KAAAF,CAAA;AAAA,SAAAb,wBAAAa,CAAA,EAAAE,CAAA,SAAAA,CAAA,IAAAF,CAAA,IAAAA,CAAA,CAAAI,UAAA,SAAAJ,CAAA,eAAAA,CAAA,uBAAAA,CAAA,yBAAAA,CAAA,WAAAK,OAAA,EAAAL,CAAA,QAAAG,CAAA,GAAAJ,wBAAA,CAAAG,CAAA,OAAAC,CAAA,IAAAA,CAAA,CAAAG,GAAA,CAAAN,CAAA,UAAAG,CAAA,CAAAI,GAAA,CAAAP,CAAA,OAAAQ,CAAA,KAAAC,SAAA,UAAAC,CAAA,GAAAC,MAAA,CAAAC,cAAA,IAAAD,MAAA,CAAAE,wBAAA,WAAAC,CAAA,IAAAd,CAAA,oBAAAc,CAAA,OAAAC,cAAA,CAAAC,IAAA,CAAAhB,CAAA,EAAAc,CAAA,SAAAG,CAAA,GAAAP,CAAA,GAAAC,MAAA,CAAAE,wBAAA,CAAAb,CAAA,EAAAc,CAAA,UAAAG,CAAA,KAAAA,CAAA,CAAAV,GAAA,IAAAU,CAAA,CAAAC,GAAA,IAAAP,MAAA,CAAAC,cAAA,CAAAJ,CAAA,EAAAM,CAAA,EAAAG,CAAA,IAAAT,CAAA,CAAAM,CAAA,IAAAd,CAAA,CAAAc,CAAA,YAAAN,CAAA,CAAAH,OAAA,GAAAL,CAAA,EAAAG,CAAA,IAAAA,CAAA,CAAAe,GAAA,CAAAlB,CAAA,EAAAQ,CAAA,GAAAA,CAAA;AAsCA,MAAMW,aAAa,GAAGA,CAACC,GAAmB,EAAEC,IAAe,EAAEC,YAAY,GAAG,KAAK,KAAe;EAC5F;EACA;;EAEA,IAAIC,KAAe,GAAG,EAAE;EACxB,IAAI,CAACD,YAAY,EAAE;IACfC,KAAK,GAAGF,IAAI,IAAI,EAAE;IAElB,IAAID,GAAG,EAAE;MACL;MACAG,KAAK,GAAG,CAACH,GAAG,EAAE,GAAGG,KAAK,CAAC;IAC3B;EACJ;;EAEA;EACA,OAAOC,KAAK,CAACC,IAAI,CAAC,IAAIC,GAAG,CAACH,KAAK,CAAC,CAAC;AACrC,CAAC;AAED,MAAMI,WAAW,GAAGA,CAAC;EAAEP,GAAG;EAAEC;AAA+C,CAAC,KAA2B;EACnG;EACA;EACA,MAAMO,WAAW,GAAG,IAAAC,iBAAU,EAACC,oBAAW,CAAC;EAC3C,MAAMR,YAAY,GAAGM,WAAW,GAAGA,WAAW,CAACN,YAAY,GAAGS,sBAAa,CAACC,QAAQ,CAAC,cAAc,CAAC;EAEpG,MAAM,CAACC,SAAS,EAAEC,OAAO,CAAC,GAAG,IAAAC,eAAQ,EAAWhB,aAAa,CAACC,GAAG,EAAEC,IAAI,EAAEC,YAAY,CAAC,CAAC;EACvF,MAAM,CAACc,SAAS,EAAEC,QAAQ,CAAC,GAAG,IAAAF,eAAQ,EAAS,CAAC,CAAC;EAEjD,MAAMG,OAAO,GAAG,IAAAC,kBAAW,EAAC,MAAM;IAC9BF,QAAQ,CAAEpB,CAAC,IAAKA,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;EAC5B,CAAC,EAAE,EAAE,CAAC;EAEN,IAAAuB,gBAAS,EAAC,MAAM;IACZN,OAAO,CAACf,aAAa,CAACC,GAAG,EAAEC,IAAI,EAAEC,YAAY,CAAC,CAAC;IAC/Ce,QAAQ,CAAC,CAAC,CAAC;EACf,CAAC,EAAE,CAACjB,GAAG,EAAEqB,IAAI,CAACC,SAAS,CAACrB,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;;EAEjC,MAAMsB,GAAG,GAAG,IAAAd,iBAAU,EAACe,4BAAmB,CAAC;EAC3C,MAAMC,YAAY,GAAG,IAAAN,kBAAW,EAAC,CAACO,SAAoB,EAAEC,SAA2B,KAAK;IACpF;IACA;IACA,MAAMC,WAAW,GAAGF,SAAS,KAAK,OAAO,IAAIC,SAAS,KAAKD,SAAS;IACpE,IAAIE,WAAW,EAAE;MACbX,QAAQ,CAAC,CAAC,CAAC;IACf;EACJ,CAAC,EAAE,EAAE,CAAC;EACN,IAAAY,qCAAoB,EAACN,GAAG,EAAEO,mBAAW,CAACC,IAAI,EAAEN,YAAY,CAAC;EAEzD,MAAMO,QAAQ,GAAGnB,SAAS,CAACG,SAAS,CAAC;EACrC,OAAO,CAACgB,QAAQ,EAAEd,OAAO,CAAC;AAC9B,CAAC;AAED,MAAMe,UAAU,gBAAG,IAAAC,iBAAU,EAAsB,CAACC,KAAK,EAAEC,GAAG,KAAK;EAC/D,MAAM;MACFC,IAAI;MACJC,MAAM;MACNC,KAAK;MACLvC,GAAG;MACHC,IAAI;MACJuC,IAAI,GAAG,MAAM;MACbC,OAAO;MACPC,SAAS;MACTC,IAAI,GAAG,OAAO;MACdC,OAAO,GAAG,IAAAC,mBAAE,EAAC,eAAe;IAEhC,CAAC,GAAGV,KAAK;IADFW,UAAU,OAAAC,yBAAA,CAAA9D,OAAA,EACbkD,KAAK,EAAAzD,SAAA;EAET,MAAM,CAACsD,QAAQ,EAAEd,OAAO,CAAC,GAAGX,WAAW,CAAC;IAAEP,GAAG;IAAEC;EAAK,CAAC,CAAC;EAEtD,MAAM+C,UAAwD,GAAG,CAAC,CAAC;EAEnE,IAAIP,OAAO,EAAE;IACTO,UAAU,CAAC,WAAW,CAAC,GAAG,KAAK;IAC/BA,UAAU,CAAC,MAAM,CAAC,GAAG,QAAQ;EACjC,CAAC,MAAM,IAAI,CAAChB,QAAQ,EAAE;IAClBgB,UAAU,CAAC,MAAM,CAAC,GAAG,cAAc;IACnCA,UAAU,CAAC,YAAY,CAAC,GAAGC,SAAS;EACxC,CAAC,MAAM;IACHD,UAAU,CAAC,MAAM,CAAC,GAAGC,SAAS;EAClC;EAEA,oBACInF,MAAA,CAAAmB,OAAA,CAAAiE,aAAA,CAAC9E,YAAA,CAAA+E,MAAM,MAAAC,SAAA,CAAAnE,OAAA;IACHmD,GAAG,EAAEA,GAAI;IACTiB,GAAG,EAAErB,QAAS;IACdsB,EAAE,EAAEhB,MAAM,IAAI,EAAG;IACjBD,IAAI,EAAEA,IAAI,IAAI,EAAG;IACjBM,IAAI,EAAEA,IAAK;IACXH,IAAI,EAAEA,IAAK;IACXE,SAAS,EAAE,IAAAa,mBAAU,EAAC,eAAe,EAAEb,SAAS,CAAE;IAClD,cAAYE,OAAQ;IACpB1B,OAAO,EAAEA,OAAQ;IACjBqB,KAAK,EAAEA,KAAM;IACbE,OAAO,EAAEA;EAAQ,GACbO,UAAU,EACVF,UAAU;IACd,eAAY;EAAY,EAC3B,CAAC;AAEV,CAAC,CAAC;AAAC,IAAAU,QAAA,GAAAC,OAAA,CAAAxE,OAAA,GAEYgD,UAAU","ignoreList":[]}