UNPKG

matrix-react-sdk

Version:
171 lines (166 loc) 25.6 kB
"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":[]}