matrix-react-sdk
Version:
SDK for matrix.org using React
171 lines (166 loc) • 25.6 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _react = _interopRequireWildcard(require("react"));
var _edit = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/edit"));
var _share = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/share"));
var _delete = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/delete"));
var _compoundWeb = require("@vector-im/compound-web");
var _classnames = _interopRequireDefault(require("classnames"));
var _languageHandler = require("../../../languageHandler");
var _Media = require("../../../customisations/Media");
var _BrowserWorkarounds = require("../../../utils/BrowserWorkarounds");
var _useId = require("../../../utils/useId");
var _AccessibleButton = _interopRequireDefault(require("../elements/AccessibleButton"));
var _BaseAvatar = _interopRequireDefault(require("../avatars/BaseAvatar"));
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-2024 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.
*/
const AvatarSettingContextMenu = ({
trigger,
onUploadSelect,
onRemoveSelect,
menuOpen,
onOpenChange
}) => {
return /*#__PURE__*/_react.default.createElement(_compoundWeb.Menu, {
trigger: trigger,
title: (0, _languageHandler._t)("action|set_avatar"),
showTitle: false,
open: menuOpen,
onOpenChange: onOpenChange
}, /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, {
as: "div",
Icon: /*#__PURE__*/_react.default.createElement(_share.default, {
width: "24px",
height: "24px"
}),
label: (0, _languageHandler._t)("action|upload_file"),
onSelect: onUploadSelect
}), onRemoveSelect && /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, {
as: "div",
Icon: /*#__PURE__*/_react.default.createElement(_delete.default, {
width: "24px",
height: "24px"
}),
className: "mx_AvatarSetting_removeMenuItem",
label: (0, _languageHandler._t)("action|remove"),
onSelect: onRemoveSelect
}));
};
/**
* Component for setting or removing an avatar on something (eg. a user or a room)
*/
const AvatarSetting = ({
avatar,
avatarAltText,
onChange,
removeAvatar,
disabled,
placeholderId,
placeholderName
}) => {
const fileInputRef = /*#__PURE__*/(0, _react.createRef)();
// Real URL that we can supply to the img element, either a data URL or whatever mediaFromMxc gives
// This represents whatever avatar the user has chosen at the time
const [avatarURL, setAvatarURL] = (0, _react.useState)(undefined);
(0, _react.useEffect)(() => {
if (avatar instanceof File) {
const reader = new FileReader();
reader.onload = () => {
setAvatarURL(reader.result);
};
reader.readAsDataURL(avatar);
} else if (avatar) {
setAvatarURL((0, _Media.mediaFromMxc)(avatar).getSquareThumbnailHttp(96) ?? undefined);
} else {
setAvatarURL(undefined);
}
}, [avatar]);
// Prevents ID collisions when this component is used more than once on the same page.
const a11yId = (0, _useId.useId)();
const onFileChanged = (0, _react.useCallback)(e => {
if (e.target.files) onChange?.(e.target.files[0]);
}, [onChange]);
const uploadAvatar = (0, _react.useCallback)(() => {
fileInputRef.current?.click();
}, [fileInputRef]);
const [menuOpen, setMenuOpen] = (0, _react.useState)(false);
const onOpenChange = (0, _react.useCallback)(newOpen => {
setMenuOpen(newOpen);
}, []);
let avatarElement = /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
element: "div",
onClick: uploadAvatar,
className: "mx_AvatarSetting_avatarPlaceholder mx_AvatarSetting_avatarDisplay",
"aria-labelledby": disabled ? undefined : a11yId
// Inhibit tab stop as we have explicit upload/remove buttons
,
tabIndex: -1,
disabled: disabled
}, /*#__PURE__*/_react.default.createElement(_BaseAvatar.default, {
idName: placeholderId,
name: placeholderName,
size: "90px"
}));
if (avatarURL) {
avatarElement = /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
element: "img",
className: "mx_AvatarSetting_avatarDisplay",
src: avatarURL,
alt: avatarAltText,
onClick: uploadAvatar
// Inhibit tab stop as we have explicit upload/remove buttons
,
tabIndex: -1,
disabled: disabled
});
}
let uploadAvatarBtn;
if (!disabled) {
const uploadButtonClasses = (0, _classnames.default)("mx_AvatarSetting_uploadButton", {
mx_AvatarSetting_uploadButton_active: menuOpen
});
uploadAvatarBtn = /*#__PURE__*/_react.default.createElement("div", {
className: uploadButtonClasses
}, /*#__PURE__*/_react.default.createElement(_edit.default, {
width: "20px",
height: "20px"
}));
}
const content = /*#__PURE__*/_react.default.createElement("div", {
className: "mx_AvatarSetting_avatar",
role: "group",
"aria-label": avatarAltText
}, avatarElement, uploadAvatarBtn);
if (disabled) {
return content;
}
return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(AvatarSettingContextMenu, {
trigger: content,
onUploadSelect: uploadAvatar,
onRemoveSelect: removeAvatar,
menuOpen: menuOpen,
onOpenChange: onOpenChange
}), /*#__PURE__*/_react.default.createElement("input", {
type: "file",
style: {
display: "none"
},
ref: fileInputRef,
onClick: _BrowserWorkarounds.chromeFileInputFix,
onChange: onFileChanged,
accept: "image/*",
alt: (0, _languageHandler._t)("action|upload")
}));
};
var _default = exports.default = AvatarSetting;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_edit","_interopRequireDefault","_share","_delete","_compoundWeb","_classnames","_languageHandler","_Media","_BrowserWorkarounds","_useId","_AccessibleButton","_BaseAvatar","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","AvatarSettingContextMenu","trigger","onUploadSelect","onRemoveSelect","menuOpen","onOpenChange","createElement","Menu","title","_t","showTitle","open","MenuItem","as","Icon","width","height","label","onSelect","className","AvatarSetting","avatar","avatarAltText","onChange","removeAvatar","disabled","placeholderId","placeholderName","fileInputRef","createRef","avatarURL","setAvatarURL","useState","undefined","useEffect","File","reader","FileReader","onload","result","readAsDataURL","mediaFromMxc","getSquareThumbnailHttp","a11yId","useId","onFileChanged","useCallback","target","files","uploadAvatar","current","click","setMenuOpen","newOpen","avatarElement","element","onClick","tabIndex","idName","name","size","src","alt","uploadAvatarBtn","uploadButtonClasses","classNames","mx_AvatarSetting_uploadButton_active","content","role","Fragment","type","style","display","ref","chromeFileInputFix","accept","_default","exports"],"sources":["../../../../src/components/views/settings/AvatarSetting.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2019-2024 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, { ReactNode, createRef, useCallback, useEffect, useState } from \"react\";\nimport EditIcon from \"@vector-im/compound-design-tokens/assets/web/icons/edit\";\nimport UploadIcon from \"@vector-im/compound-design-tokens/assets/web/icons/share\";\nimport DeleteIcon from \"@vector-im/compound-design-tokens/assets/web/icons/delete\";\nimport { Menu, MenuItem } from \"@vector-im/compound-web\";\nimport classNames from \"classnames\";\n\nimport { _t } from \"../../../languageHandler\";\nimport { mediaFromMxc } from \"../../../customisations/Media\";\nimport { chromeFileInputFix } from \"../../../utils/BrowserWorkarounds\";\nimport { useId } from \"../../../utils/useId\";\nimport AccessibleButton from \"../elements/AccessibleButton\";\nimport BaseAvatar from \"../avatars/BaseAvatar\";\n\ninterface MenuProps {\n    trigger: ReactNode;\n    onUploadSelect: () => void;\n    onRemoveSelect?: () => void;\n    menuOpen: boolean;\n    onOpenChange: (newOpen: boolean) => void;\n}\n\nconst AvatarSettingContextMenu: React.FC<MenuProps> = ({\n    trigger,\n    onUploadSelect,\n    onRemoveSelect,\n    menuOpen,\n    onOpenChange,\n}) => {\n    return (\n        <Menu\n            trigger={trigger}\n            title={_t(\"action|set_avatar\")}\n            showTitle={false}\n            open={menuOpen}\n            onOpenChange={onOpenChange}\n        >\n            <MenuItem\n                as=\"div\"\n                Icon={<UploadIcon width=\"24px\" height=\"24px\" />}\n                label={_t(\"action|upload_file\")}\n                onSelect={onUploadSelect}\n            />\n            {onRemoveSelect && (\n                <MenuItem\n                    as=\"div\"\n                    Icon={<DeleteIcon width=\"24px\" height=\"24px\" />}\n                    className=\"mx_AvatarSetting_removeMenuItem\"\n                    label={_t(\"action|remove\")}\n                    onSelect={onRemoveSelect}\n                />\n            )}\n        </Menu>\n    );\n};\n\ninterface IProps {\n    /**\n     * The current value of the avatar URL, as an mxc URL or a File.\n     * Generally, an mxc URL would be specified until the user selects a file, then\n     * the file supplied by the onChange callback would be supplied here until it's\n     * saved.\n     */\n    avatar?: string | File;\n\n    /**\n     * If true, the user cannot change the avatar\n     */\n    disabled?: boolean;\n\n    /**\n     * Called when the user has selected a new avatar\n     * The callback is passed a File object for the new avatar data\n     */\n    onChange?: (f: File) => void;\n\n    /**\n     * Called when the user wishes to remove the avatar\n     */\n    removeAvatar?: () => void;\n\n    /**\n     * The alt text for the avatar\n     */\n    avatarAltText: string;\n\n    /**\n     * String to use for computing the colour of the placeholder avatar if no avatar is set\n     */\n    placeholderId: string;\n\n    /**\n     * String to use for the placeholder display if no avatar is set\n     */\n    placeholderName: string;\n}\n\n/**\n * Component for setting or removing an avatar on something (eg. a user or a room)\n */\nconst AvatarSetting: React.FC<IProps> = ({\n    avatar,\n    avatarAltText,\n    onChange,\n    removeAvatar,\n    disabled,\n    placeholderId,\n    placeholderName,\n}) => {\n    const fileInputRef = createRef<HTMLInputElement>();\n\n    // Real URL that we can supply to the img element, either a data URL or whatever mediaFromMxc gives\n    // This represents whatever avatar the user has chosen at the time\n    const [avatarURL, setAvatarURL] = useState<string | undefined>(undefined);\n    useEffect(() => {\n        if (avatar instanceof File) {\n            const reader = new FileReader();\n            reader.onload = () => {\n                setAvatarURL(reader.result as string);\n            };\n            reader.readAsDataURL(avatar);\n        } else if (avatar) {\n            setAvatarURL(mediaFromMxc(avatar).getSquareThumbnailHttp(96) ?? undefined);\n        } else {\n            setAvatarURL(undefined);\n        }\n    }, [avatar]);\n\n    // Prevents ID collisions when this component is used more than once on the same page.\n    const a11yId = useId();\n\n    const onFileChanged = useCallback(\n        (e: React.ChangeEvent<HTMLInputElement>) => {\n            if (e.target.files) onChange?.(e.target.files[0]);\n        },\n        [onChange],\n    );\n\n    const uploadAvatar = useCallback((): void => {\n        fileInputRef.current?.click();\n    }, [fileInputRef]);\n\n    const [menuOpen, setMenuOpen] = useState(false);\n\n    const onOpenChange = useCallback((newOpen: boolean) => {\n        setMenuOpen(newOpen);\n    }, []);\n\n    let avatarElement = (\n        <AccessibleButton\n            element=\"div\"\n            onClick={uploadAvatar}\n            className=\"mx_AvatarSetting_avatarPlaceholder mx_AvatarSetting_avatarDisplay\"\n            aria-labelledby={disabled ? undefined : a11yId}\n            // Inhibit tab stop as we have explicit upload/remove buttons\n            tabIndex={-1}\n            disabled={disabled}\n        >\n            <BaseAvatar idName={placeholderId} name={placeholderName} size=\"90px\" />\n        </AccessibleButton>\n    );\n    if (avatarURL) {\n        avatarElement = (\n            <AccessibleButton\n                element=\"img\"\n                className=\"mx_AvatarSetting_avatarDisplay\"\n                src={avatarURL}\n                alt={avatarAltText}\n                onClick={uploadAvatar}\n                // Inhibit tab stop as we have explicit upload/remove buttons\n                tabIndex={-1}\n                disabled={disabled}\n            />\n        );\n    }\n\n    let uploadAvatarBtn: JSX.Element | undefined;\n    if (!disabled) {\n        const uploadButtonClasses = classNames(\"mx_AvatarSetting_uploadButton\", {\n            mx_AvatarSetting_uploadButton_active: menuOpen,\n        });\n        uploadAvatarBtn = (\n            <div className={uploadButtonClasses}>\n                <EditIcon width=\"20px\" height=\"20px\" />\n            </div>\n        );\n    }\n\n    const content = (\n        <div className=\"mx_AvatarSetting_avatar\" role=\"group\" aria-label={avatarAltText}>\n            {avatarElement}\n            {uploadAvatarBtn}\n        </div>\n    );\n\n    if (disabled) {\n        return content;\n    }\n\n    return (\n        <>\n            <AvatarSettingContextMenu\n                trigger={content}\n                onUploadSelect={uploadAvatar}\n                onRemoveSelect={removeAvatar}\n                menuOpen={menuOpen}\n                onOpenChange={onOpenChange}\n            />\n            <input\n                type=\"file\"\n                style={{ display: \"none\" }}\n                ref={fileInputRef}\n                onClick={chromeFileInputFix}\n                onChange={onFileChanged}\n                accept=\"image/*\"\n                alt={_t(\"action|upload\")}\n            />\n        </>\n    );\n};\n\nexport default AvatarSetting;\n"],"mappings":"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","ignoreList":[]}