matrix-react-sdk
Version:
SDK for matrix.org using React
222 lines (190 loc) • 28.6 kB
JavaScript
"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 _languageHandler = require("../../../languageHandler");
var _MatrixClientPeg = require("../../../MatrixClientPeg");
var _Field = _interopRequireDefault(require("../elements/Field"));
var _HostingLink = require("../../../utils/HostingLink");
var sdk = _interopRequireWildcard(require("../../../index"));
var _OwnProfileStore = require("../../../stores/OwnProfileStore");
var _Modal = _interopRequireDefault(require("../../../Modal"));
var _ErrorDialog = _interopRequireDefault(require("../dialogs/ErrorDialog"));
var _replaceableComponent = require("../../../utils/replaceableComponent");
var _Media = require("../../../customisations/Media");
var _dec, _class, _temp;
let ProfileSettings = (_dec = (0, _replaceableComponent.replaceableComponent)("views.settings.ProfileSettings"), _dec(_class = (_temp = class ProfileSettings extends _react.default.Component {
constructor() {
super();
(0, _defineProperty2.default)(this, "_uploadAvatar", () => {
this._avatarUpload.current.click();
});
(0, _defineProperty2.default)(this, "_removeAvatar", () => {
// clear file upload field so same file can be selected
this._avatarUpload.current.value = "";
this.setState({
avatarUrl: null,
avatarFile: null,
enableProfileSave: true
});
});
(0, _defineProperty2.default)(this, "_cancelProfileChanges", async e => {
e.stopPropagation();
e.preventDefault();
if (!this.state.enableProfileSave) return;
this.setState({
enableProfileSave: false,
displayName: this.state.originalDisplayName,
avatarUrl: this.state.originalAvatarUrl,
avatarFile: null
});
});
(0, _defineProperty2.default)(this, "_saveProfile", async e => {
e.stopPropagation();
e.preventDefault();
if (!this.state.enableProfileSave) return;
this.setState({
enableProfileSave: false
});
const client = _MatrixClientPeg.MatrixClientPeg.get();
const newState = {};
const displayName = this.state.displayName.trim();
try {
if (this.state.originalDisplayName !== this.state.displayName) {
await client.setDisplayName(displayName);
newState.originalDisplayName = displayName;
newState.displayName = displayName;
}
if (this.state.avatarFile) {
console.log(`Uploading new avatar, ${this.state.avatarFile.name} of type ${this.state.avatarFile.type},` + ` (${this.state.avatarFile.size}) bytes`);
const uri = await client.uploadContent(this.state.avatarFile);
await client.setAvatarUrl(uri);
newState.avatarUrl = (0, _Media.mediaFromMxc)(uri).getSquareThumbnailHttp(96);
newState.originalAvatarUrl = newState.avatarUrl;
newState.avatarFile = null;
} else if (this.state.originalAvatarUrl !== this.state.avatarUrl) {
await client.setAvatarUrl(""); // use empty string as Synapse 500s on undefined
}
} catch (err) {
console.log("Failed to save profile", err);
_Modal.default.createTrackedDialog('Failed to save profile', '', _ErrorDialog.default, {
title: (0, _languageHandler._t)("Failed to save your profile"),
description: err && err.message ? err.message : (0, _languageHandler._t)("The operation could not be completed")
});
}
this.setState(newState);
});
(0, _defineProperty2.default)(this, "_onDisplayNameChanged", e => {
this.setState({
displayName: e.target.value,
enableProfileSave: true
});
});
(0, _defineProperty2.default)(this, "_onAvatarChanged", e => {
if (!e.target.files || !e.target.files.length) {
this.setState({
avatarUrl: this.state.originalAvatarUrl,
avatarFile: null,
enableProfileSave: false
});
return;
}
const file = e.target.files[0];
const reader = new FileReader();
reader.onload = ev => {
this.setState({
avatarUrl: ev.target.result,
avatarFile: file,
enableProfileSave: true
});
};
reader.readAsDataURL(file);
});
const _client = _MatrixClientPeg.MatrixClientPeg.get();
let avatarUrl = _OwnProfileStore.OwnProfileStore.instance.avatarMxc;
if (avatarUrl) avatarUrl = (0, _Media.mediaFromMxc)(avatarUrl).getSquareThumbnailHttp(96);
this.state = {
userId: _client.getUserId(),
originalDisplayName: _OwnProfileStore.OwnProfileStore.instance.displayName,
displayName: _OwnProfileStore.OwnProfileStore.instance.displayName,
originalAvatarUrl: avatarUrl,
avatarUrl: avatarUrl,
avatarFile: null,
enableProfileSave: false
};
this._avatarUpload = /*#__PURE__*/(0, _react.createRef)();
}
render() {
const hostingSignupLink = (0, _HostingLink.getHostingLink)('user-settings');
let hostingSignup = null;
if (hostingSignupLink) {
hostingSignup = /*#__PURE__*/_react.default.createElement("span", {
className: "mx_ProfileSettings_hostingSignup"
}, (0, _languageHandler._t)("<a>Upgrade</a> to your own domain", {}, {
a: sub => /*#__PURE__*/_react.default.createElement("a", {
href: hostingSignupLink,
target: "_blank",
rel: "noreferrer noopener"
}, sub)
}), /*#__PURE__*/_react.default.createElement("a", {
href: hostingSignupLink,
target: "_blank",
rel: "noreferrer noopener"
}, /*#__PURE__*/_react.default.createElement("img", {
src: require("../../../../res/img/external-link.svg"),
width: "11",
height: "10",
alt: ""
})));
}
const AccessibleButton = sdk.getComponent('elements.AccessibleButton');
const AvatarSetting = sdk.getComponent('settings.AvatarSetting');
return /*#__PURE__*/_react.default.createElement("form", {
onSubmit: this._saveProfile,
autoComplete: "off",
noValidate: true,
className: "mx_ProfileSettings_profileForm"
}, /*#__PURE__*/_react.default.createElement("input", {
type: "file",
ref: this._avatarUpload,
className: "mx_ProfileSettings_avatarUpload",
onChange: this._onAvatarChanged,
accept: "image/*"
}), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ProfileSettings_profile"
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ProfileSettings_controls"
}, /*#__PURE__*/_react.default.createElement("span", {
className: "mx_SettingsTab_subheading"
}, (0, _languageHandler._t)("Profile")), /*#__PURE__*/_react.default.createElement(_Field.default, {
label: (0, _languageHandler._t)("Display Name"),
type: "text",
value: this.state.displayName,
autoComplete: "off",
onChange: this._onDisplayNameChanged
}), /*#__PURE__*/_react.default.createElement("p", null, this.state.userId, hostingSignup)), /*#__PURE__*/_react.default.createElement(AvatarSetting, {
avatarUrl: this.state.avatarUrl,
avatarName: this.state.displayName || this.state.userId,
avatarAltText: (0, _languageHandler._t)("Profile picture"),
uploadAvatar: this._uploadAvatar,
removeAvatar: this._removeAvatar
})), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ProfileSettings_buttons"
}, /*#__PURE__*/_react.default.createElement(AccessibleButton, {
onClick: this._cancelProfileChanges,
kind: "link",
disabled: !this.state.enableProfileSave
}, (0, _languageHandler._t)("Cancel")), /*#__PURE__*/_react.default.createElement(AccessibleButton, {
onClick: this._saveProfile,
kind: "primary",
disabled: !this.state.enableProfileSave
}, (0, _languageHandler._t)("Save"))));
}
}, _temp)) || _class);
exports.default = ProfileSettings;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/settings/ProfileSettings.js"],"names":["ProfileSettings","React","Component","constructor","_avatarUpload","current","click","value","setState","avatarUrl","avatarFile","enableProfileSave","e","stopPropagation","preventDefault","state","displayName","originalDisplayName","originalAvatarUrl","client","MatrixClientPeg","get","newState","trim","setDisplayName","console","log","name","type","size","uri","uploadContent","setAvatarUrl","getSquareThumbnailHttp","err","Modal","createTrackedDialog","ErrorDialog","title","description","message","target","files","length","file","reader","FileReader","onload","ev","result","readAsDataURL","OwnProfileStore","instance","avatarMxc","userId","getUserId","render","hostingSignupLink","hostingSignup","a","sub","require","AccessibleButton","sdk","getComponent","AvatarSetting","_saveProfile","_onAvatarChanged","_onDisplayNameChanged","_uploadAvatar","_removeAvatar","_cancelProfileChanges"],"mappings":"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","sourcesContent":["/*\nCopyright 2019 New Vector Ltd\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 {_t} from \"../../../languageHandler\";\nimport {MatrixClientPeg} from \"../../../MatrixClientPeg\";\nimport Field from \"../elements/Field\";\nimport { getHostingLink } from '../../../utils/HostingLink';\nimport * as sdk from \"../../../index\";\nimport {OwnProfileStore} from \"../../../stores/OwnProfileStore\";\nimport Modal from \"../../../Modal\";\nimport ErrorDialog from \"../dialogs/ErrorDialog\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\nimport {mediaFromMxc} from \"../../../customisations/Media\";\n\n@replaceableComponent(\"views.settings.ProfileSettings\")\nexport default class ProfileSettings extends React.Component {\n    constructor() {\n        super();\n\n        const client = MatrixClientPeg.get();\n        let avatarUrl = OwnProfileStore.instance.avatarMxc;\n        if (avatarUrl) avatarUrl = mediaFromMxc(avatarUrl).getSquareThumbnailHttp(96);\n        this.state = {\n            userId: client.getUserId(),\n            originalDisplayName: OwnProfileStore.instance.displayName,\n            displayName: OwnProfileStore.instance.displayName,\n            originalAvatarUrl: avatarUrl,\n            avatarUrl: avatarUrl,\n            avatarFile: null,\n            enableProfileSave: false,\n        };\n\n        this._avatarUpload = createRef();\n    }\n\n    _uploadAvatar = () => {\n        this._avatarUpload.current.click();\n    };\n\n    _removeAvatar = () => {\n        // clear file upload field so same file can be selected\n        this._avatarUpload.current.value = \"\";\n        this.setState({\n            avatarUrl: null,\n            avatarFile: null,\n            enableProfileSave: true,\n        });\n    };\n\n    _cancelProfileChanges = async (e) => {\n        e.stopPropagation();\n        e.preventDefault();\n\n        if (!this.state.enableProfileSave) return;\n        this.setState({\n            enableProfileSave: false,\n            displayName: this.state.originalDisplayName,\n            avatarUrl: this.state.originalAvatarUrl,\n            avatarFile: null,\n        });\n    };\n\n    _saveProfile = async (e) => {\n        e.stopPropagation();\n        e.preventDefault();\n\n        if (!this.state.enableProfileSave) return;\n        this.setState({enableProfileSave: false});\n\n        const client = MatrixClientPeg.get();\n        const newState = {};\n\n        const displayName = this.state.displayName.trim();\n        try {\n            if (this.state.originalDisplayName !== this.state.displayName) {\n                await client.setDisplayName(displayName);\n                newState.originalDisplayName = displayName;\n                newState.displayName = displayName;\n            }\n\n            if (this.state.avatarFile) {\n                console.log(\n                    `Uploading new avatar, ${this.state.avatarFile.name} of type ${this.state.avatarFile.type},` +\n                    ` (${this.state.avatarFile.size}) bytes`);\n                const uri = await client.uploadContent(this.state.avatarFile);\n                await client.setAvatarUrl(uri);\n                newState.avatarUrl = mediaFromMxc(uri).getSquareThumbnailHttp(96);\n                newState.originalAvatarUrl = newState.avatarUrl;\n                newState.avatarFile = null;\n            } else if (this.state.originalAvatarUrl !== this.state.avatarUrl) {\n                await client.setAvatarUrl(\"\"); // use empty string as Synapse 500s on undefined\n            }\n        } catch (err) {\n            console.log(\"Failed to save profile\", err);\n            Modal.createTrackedDialog('Failed to save profile', '', ErrorDialog, {\n                title: _t(\"Failed to save your profile\"),\n                description: ((err && err.message) ? err.message : _t(\"The operation could not be completed\")),\n            });\n        }\n\n        this.setState(newState);\n    };\n\n    _onDisplayNameChanged = (e) => {\n        this.setState({\n            displayName: e.target.value,\n            enableProfileSave: true,\n        });\n    };\n\n    _onAvatarChanged = (e) => {\n        if (!e.target.files || !e.target.files.length) {\n            this.setState({\n                avatarUrl: this.state.originalAvatarUrl,\n                avatarFile: null,\n                enableProfileSave: false,\n            });\n            return;\n        }\n\n        const file = e.target.files[0];\n        const reader = new FileReader();\n        reader.onload = (ev) => {\n            this.setState({\n                avatarUrl: ev.target.result,\n                avatarFile: file,\n                enableProfileSave: true,\n            });\n        };\n        reader.readAsDataURL(file);\n    };\n\n    render() {\n        const hostingSignupLink = getHostingLink('user-settings');\n        let hostingSignup = null;\n        if (hostingSignupLink) {\n            hostingSignup = <span className=\"mx_ProfileSettings_hostingSignup\">\n                {_t(\n                    \"<a>Upgrade</a> to your own domain\", {},\n                    {\n                        a: sub => <a href={hostingSignupLink} target=\"_blank\" rel=\"noreferrer noopener\">{sub}</a>,\n                    },\n                )}\n                <a href={hostingSignupLink} target=\"_blank\" rel=\"noreferrer noopener\">\n                    <img src={require(\"../../../../res/img/external-link.svg\")} width=\"11\" height=\"10\" alt='' />\n                </a>\n            </span>;\n        }\n\n        const AccessibleButton = sdk.getComponent('elements.AccessibleButton');\n        const AvatarSetting = sdk.getComponent('settings.AvatarSetting');\n        return (\n            <form\n                onSubmit={this._saveProfile}\n                autoComplete=\"off\"\n                noValidate={true}\n                className=\"mx_ProfileSettings_profileForm\"\n            >\n                <input\n                    type=\"file\"\n                    ref={this._avatarUpload} className=\"mx_ProfileSettings_avatarUpload\"\n                    onChange={this._onAvatarChanged}\n                    accept=\"image/*\"\n                />\n                <div className=\"mx_ProfileSettings_profile\">\n                    <div className=\"mx_ProfileSettings_controls\">\n                        <span className=\"mx_SettingsTab_subheading\">{_t(\"Profile\")}</span>\n                        <Field\n                            label={_t(\"Display Name\")}\n                            type=\"text\" value={this.state.displayName}\n                            autoComplete=\"off\"\n                            onChange={this._onDisplayNameChanged}\n                        />\n                        <p>\n                            {this.state.userId}\n                            {hostingSignup}\n                        </p>\n                    </div>\n                    <AvatarSetting\n                        avatarUrl={this.state.avatarUrl}\n                        avatarName={this.state.displayName || this.state.userId}\n                        avatarAltText={_t(\"Profile picture\")}\n                        uploadAvatar={this._uploadAvatar}\n                        removeAvatar={this._removeAvatar} />\n                </div>\n                <div className=\"mx_ProfileSettings_buttons\">\n                    <AccessibleButton\n                        onClick={this._cancelProfileChanges}\n                        kind=\"link\"\n                        disabled={!this.state.enableProfileSave}\n                    >\n                        {_t(\"Cancel\")}\n                    </AccessibleButton>\n                    <AccessibleButton\n                        onClick={this._saveProfile}\n                        kind=\"primary\"\n                        disabled={!this.state.enableProfileSave}\n                    >\n                        {_t(\"Save\")}\n                    </AccessibleButton>\n                </div>\n            </form>\n        );\n    }\n}\n"]}