Edit user profile complete

pull/738/head
Vincent 5 years ago
parent a1a582d7ea
commit c369b0c553

@ -952,6 +952,9 @@
"cancel": {
"message": "Cancel"
},
"copy": {
"message": "Copy"
},
"skip": {
"message": "Skip"
},
@ -2161,6 +2164,9 @@
"description":
"Button action that the user can click to view their unique seed"
},
"yourSessionID": {
"message": "Your Session ID"
},
"setAccountPasswordTitle": {
"message": "Set Account Password",
"description": "Prompt for user to set account password in settings view"
@ -2255,6 +2261,11 @@
"description":
"A toast message telling the user that the mnemonic seed was copied"
},
"copiedSessionID": {
"message": "Copied Session ID to clipboard",
"description":
"A toast message telling the user that their Session ID was copied"
},
"passwordViewTitle": {
"message": "Type in your password",
@ -2423,7 +2434,7 @@
},
"editProfileModalTitle": {
"message": "Edit Profile",
"message": "Profile",
"description": "Title for the Edit Profile modal"
},

@ -820,8 +820,107 @@
confirmDialog.render();
};
window.showQRDialog = window.owsDesktopApp.appView.showQRDialog;
window.showSeedDialog = window.owsDesktopApp.appView.showSeedDialog;
window.showPasswordDialog = window.owsDesktopApp.appView.showPasswordDialog;
window.showEditProfileDialog = async () => {
const ourNumber = window.storage.get('primaryDevicePubKey');
const conversation = await ConversationController.getOrCreateAndWait(
ourNumber,
'private'
);
const readFile = attachment =>
new Promise((resolve, reject) => {
const fileReader = new FileReader();
fileReader.onload = e => {
const data = e.target.result;
resolve({
...attachment,
data,
size: data.byteLength,
});
};
fileReader.onerror = reject;
fileReader.onabort = reject;
fileReader.readAsArrayBuffer(attachment.file);
});
const avatarPath = conversation.getAvatarPath();
const profile = conversation.getLokiProfile();
const displayName = profile && profile.displayName;
if (appView) {
appView.showEditProfileDialog({
profileName: displayName,
pubkey: ourNumber,
avatarPath,
avatarColor: conversation.getColor(),
onOk: async (newName, avatar) => {
let newAvatarPath = '';
let url = null;
let profileKey = null;
if (avatar) {
const data = await readFile({ file: avatar });
// For simplicity we use the same attachment pointer that would send to
// others, which means we need to wait for the database response.
// To avoid the wait, we create a temporary url for the local image
// and use it until we the the response from the server
const tempUrl = window.URL.createObjectURL(avatar);
conversation.setLokiProfile({ displayName: newName });
conversation.set('avatar', tempUrl);
// Encrypt with a new key every time
profileKey = libsignal.crypto.getRandomBytes(32);
const encryptedData = await textsecure.crypto.encryptProfile(
data.data,
profileKey
);
const avatarPointer = await textsecure.messaging.uploadAvatar({
...data,
data: encryptedData,
size: encryptedData.byteLength,
});
({ url } = avatarPointer);
storage.put('profileKey', profileKey);
conversation.set('avatarPointer', url);
const upgraded = await Signal.Migrations.processNewAttachment({
isRaw: true,
data: data.data,
url,
});
newAvatarPath = upgraded.path;
}
// Replace our temporary image with the attachment pointer from the server:
conversation.set('avatar', null);
conversation.setLokiProfile({
displayName: newName,
avatar: newAvatarPath,
});
// inform all your registered public servers
// could put load on all the servers
// if they just keep changing their names without sending messages
// so we could disable this here
// or least it enable for the quickest response
window.lokiPublicChatAPI.setProfileName(newName);
window
.getConversations()
.filter(convo => convo.isPublic() && !convo.isRss())
.forEach(convo =>
convo.trigger('ourAvatarChanged', { url, profileKey })
);
},
});
}
};
window.generateID = () =>
Math.random()
@ -1031,104 +1130,6 @@
});
});
Whisper.events.on('onEditProfile', async () => {
const ourNumber = window.storage.get('primaryDevicePubKey');
const conversation = await ConversationController.getOrCreateAndWait(
ourNumber,
'private'
);
const readFile = attachment =>
new Promise((resolve, reject) => {
const fileReader = new FileReader();
fileReader.onload = e => {
const data = e.target.result;
resolve({
...attachment,
data,
size: data.byteLength,
});
};
fileReader.onerror = reject;
fileReader.onabort = reject;
fileReader.readAsArrayBuffer(attachment.file);
});
const avatarPath = conversation.getAvatarPath();
const profile = conversation.getLokiProfile();
const displayName = profile && profile.displayName;
if (appView) {
appView.showEditProfileDialog({
profileName: displayName,
pubkey: ourNumber,
avatarPath,
avatarColor: conversation.getColor(),
onOk: async (newName, avatar) => {
let newAvatarPath = '';
let url = null;
let profileKey = null;
if (avatar) {
const data = await readFile({ file: avatar });
// For simplicity we use the same attachment pointer that would send to
// others, which means we need to wait for the database response.
// To avoid the wait, we create a temporary url for the local image
// and use it until we the the response from the server
const tempUrl = window.URL.createObjectURL(avatar);
conversation.setLokiProfile({ displayName: newName });
conversation.set('avatar', tempUrl);
// Encrypt with a new key every time
profileKey = libsignal.crypto.getRandomBytes(32);
const encryptedData = await textsecure.crypto.encryptProfile(
data.data,
profileKey
);
const avatarPointer = await textsecure.messaging.uploadAvatar({
...data,
data: encryptedData,
size: encryptedData.byteLength,
});
({ url } = avatarPointer);
storage.put('profileKey', profileKey);
conversation.set('avatarPointer', url);
const upgraded = await Signal.Migrations.processNewAttachment({
isRaw: true,
data: data.data,
url,
});
newAvatarPath = upgraded.path;
}
// Replace our temporary image with the attachment pointer from the server:
conversation.set('avatar', null);
conversation.setLokiProfile({
displayName: newName,
avatar: newAvatarPath,
});
// inform all your registered public servers
// could put load on all the servers
// if they just keep changing their names without sending messages
// so we could disable this here
// or least it enable for the quickest response
window.lokiPublicChatAPI.setProfileName(newName);
window
.getConversations()
.filter(convo => convo.isPublic() && !convo.isRss())
.forEach(convo =>
convo.trigger('ourAvatarChanged', { url, profileKey })
);
},
});
}
});
Whisper.events.on('onShowUserDetails', async ({ userPubKey }) => {
const conversation = await ConversationController.getOrCreateAndWait(
userPubKey,

@ -27,7 +27,7 @@
font-weight: bold;
}
@font-face {
font-family: 'SFPro';
font-family: 'SF Pro Text';
src: url('../fonts/SFProText-Regular.ttf') format('truetype');
}
@ -87,6 +87,12 @@ $session-margin-sm: 10px;
$session-margin-md: 15px;
$session-margin-lg: 20px;
$session-font-xs: 11px;
$session-font-sm: 13px;
$session-font-md: 15px;
$session-font-lg: 18px;
$session-font-xl: 24px;
$session-search-input-height: 34px;
$main-view-header-height: 85px;
@ -322,7 +328,7 @@ $session_message-container-border-radius: 5px;
height: 45px;
line-height: 40px;
padding: 0;
font-size: 15px;
font-size: $session-font-md;
font-family: $session-font-family;
border-radius: 500px;
@ -340,7 +346,7 @@ $session_message-container-border-radius: 5px;
height: 33px;
padding: 0px 18px;
line-height: 33px;
font-size: 13px;
font-size: $session-font-sm;
}
&.square,
@ -415,7 +421,7 @@ $session_message-container-border-radius: 5px;
.notification-count {
position: absolute;
font-size: 12px;
font-size: $session-font-xs;
font-family: $session-font-family;
top: 20px;
right: 20px;
@ -581,8 +587,8 @@ label {
}
.title {
font-size: 15px;
line-height: 13px;
font-size: $session-font-md;
line-height: $session-font-sm;
margin-bottom: $session-margin-sm;
padding-top: 0px;
color: $session-color-white;
@ -590,7 +596,7 @@ label {
}
.description {
font-size: 12px;
font-size: $session-font-xs;
@include session-color-subtle($session-color-white);
}
@ -640,15 +646,30 @@ label {
display: flex;
flex-direction: row;
justify-content: space-between;
align-items: center;
padding: $session-margin-lg;
font-family: 'Wasa';
text-align: center;
line-height: 18px;
font-size: 15px;
font-size: $session-font-md;
font-weight: 700;
&.reverse {
flex-direction: row-reverse;
.session-modal__header__close > div {
float: right;
}
.session-modal__header__icons > div {
float: left;
padding-left: 0px;
padding-right: 10px;
}
}
&__icons,
&__close {
width: 60px;
@ -668,8 +689,8 @@ label {
&__body {
padding: 0px $session-margin-lg $session-margin-lg $session-margin-lg;
font-family: 'Wasa';
line-height: 16px;
font-size: 13px;
line-height: $session-font-md;
font-size: $session-font-sm;
.message {
text-align: center;
@ -703,7 +724,7 @@ label {
font-family: monospace;
font-style: normal;
font-size: 11px;
font-size: $session-font-xs;
}
}
@ -715,7 +736,7 @@ label {
}
&-main-message {
font-size: 15px;
font-size: $session-font-md;
}
&-sub-message {
margin-top: 5px;
@ -775,7 +796,7 @@ label {
color: $session-color-white;
font-family: 'Wasa';
font-size: 12px;
font-size: $session-font-sm;
line-height: $session-icon-size-sm;
font-weight: 700;
@ -811,7 +832,7 @@ label {
color: $session-color-white;
font-family: 'Wasa';
font-size: 10px;
font-size: $session-font-xs;
line-height: $session-icon-size-sm;
font-weight: 700;
@ -837,10 +858,132 @@ label {
}
}
.edit-profile-dialog .image-upload-section {
position: absolute;
margin-top: 50px;
margin-left: 75px;
.edit-profile-dialog {
.session-modal__header__title {
font-size: $session-font-lg;
}
.session-modal {
width: $session-modal-size-md;
&__header {
height: 68.45px;
}
}
.avatar-center-inner {
position: relative;
.module-avatar {
box-shadow: 0 0 23px 0 rgba($session-color-black, 0.78);
}
.qr-view-button {
cursor: pointer;
display: flex;
align-items: center;
justify-content: center;
position: absolute;
right: -3px;
height: 26px;
width: 26px;
border-radius: 50%;
padding-top: 3px;
background-color: $session-color-white;
transition: $session-transition-duration;
&:hover{
filter: brightness(90%);
}
.session-icon-button {
opacity: 1;
}
}
}
.image-upload-section {
display: flex;
align-items: center;
justify-content: center;
position: absolute;
cursor: pointer;
width: 80px;
height: 80px;
border-radius: 100%;
background-color: rgba($session-color-black, 0.72);
box-shadow: 0px 0px 3px 0.5px rgba(0,0,0,0.75);
opacity: 0;
transition: $session-transition-duration;
&:after{
content: "Edit";
}
&:hover{
opacity: 1;
}
}
.qr-image {
display: flex;
justify-content: center;
svg {
width: 135px;
height: 135px;
border-radius: 3px;
padding: $session-margin-xs;
background-color: $session-color-white;
}
}
.session-id-section {
.panel-text-divider {
margin-top: 35px;
margin-bottom: 35px;
}
&-display{
text-align: center;
word-break: break-all;
font-size: $session-font-md;
padding: 0px $session-margin-lg;
font-family: "SF Pro Text";
font-weight: 100;
color: rgba($session-color-white, 0.80);
font-size: $session-font-md;
padding: 0px $session-margin-sm;
}
}
.profile-name {
display: flex;
justify-content: center;
margin-top: $session-margin-lg;
&-input {
height: 38px;
width: 142px;
border-radius: 5px;
text-align: center;
font-size: $session-font-md;
background-color: $session-shade-5 !important;
}
&-uneditable {
display: flex;
align-items: center;
justify-content: center;
margin-left: $session-margin-lg;
p {
font-size: $session-font-md;
padding: 0px $session-margin-sm;
}
}
}
}
.conversation-loader {
@ -932,7 +1075,7 @@ label {
&-title {
line-height: $main-view-header-height;
font-weight: bold;
font-size: 18px;
font-size: $session-font-lg;
text-align: center;
flex-grow: 1;
}
@ -944,7 +1087,7 @@ label {
}
&-item {
font-size: 15px;
font-size: $session-font-md;
color: $session-color-white;
background-color: $session-shade-1;
@ -963,12 +1106,12 @@ label {
&__title {
line-height: 1.7;
font-size: 16px;
font-size: $session-font-lg;
font-weight: bold;
}
&__description {
font-size: 13px;
font-size: $session-font-sm;
font-weight: 100;
@include session-color-subtle($session-color-white);
}
@ -1014,7 +1157,7 @@ label {
border: none;
border-radius: 2px;
text-align: center;
font-size: 25px;
font-size: $session-font-xl;
letter-spacing: 5px;
font-family: 'SF Pro Text';
}
@ -1025,6 +1168,8 @@ label {
#qr svg {
width: $session-modal-size-sm;
height: $session-modal-size-sm;
padding: $session-margin-xs;
background-color: $session-color-white;
border-radius: 3px;
}
@ -1059,7 +1204,7 @@ label {
border: none;
margin: 0px;
padding: 0px $session-margin-lg;
font-size: 15px;
font-size: $session-font-md;
line-height: 60px;
@include session-color-subtle($session-color-white);
@ -1173,7 +1318,7 @@ input {
width: 600px;
&__header {
font-size: 17px;
font-size: $session-font-lg;
}
&__body > div:first-child {

@ -1,8 +1,10 @@
import React from 'react';
import classNames from 'classnames';
import { QRCode } from 'react-qr-svg';
import { Avatar } from './Avatar';
import { SessionButton, SessionButtonColor } from './session/SessionButton';
import { SessionButton, SessionButtonColor, SessionButtonType } from './session/SessionButton';
import {
SessionIconButton,
SessionIconSize,
@ -28,9 +30,9 @@ interface Props {
interface State {
profileName: string;
errorDisplayed: boolean;
errorMessage: string;
setProfileName: string;
avatar: string;
mode: 'default' | 'edit' | 'qr';
}
export class EditProfileDialog extends React.Component<Props, State> {
@ -42,15 +44,14 @@ export class EditProfileDialog extends React.Component<Props, State> {
this.onNameEdited = this.onNameEdited.bind(this);
this.closeDialog = this.closeDialog.bind(this);
this.onClickOK = this.onClickOK.bind(this);
this.showError = this.showError.bind(this);
this.onKeyUp = this.onKeyUp.bind(this);
this.onFileSelected = this.onFileSelected.bind(this);
this.state = {
profileName: this.props.profileName,
errorDisplayed: false,
errorMessage: 'placeholder',
setProfileName: this.props.profileName,
avatar: this.props.avatarPath,
mode: 'default',
};
this.inputEl = React.createRef();
@ -61,25 +62,81 @@ export class EditProfileDialog extends React.Component<Props, State> {
public render() {
const i18n = this.props.i18n;
const cancelText = i18n('cancel');
const okText = i18n('ok');
const placeholderText = i18n('profileName');
const viewDefault = this.state.mode === 'default';
const viewEdit = this.state.mode === 'edit';
const viewQR = this.state.mode === 'qr';
const sessionID = window.textsecure.storage.user.getNumber();
const errorMessageClasses = classNames(
'error-message',
this.state.errorDisplayed ? 'error-shown' : 'error-faded'
);
const backButton = (viewEdit || viewQR)
? [{
iconType: SessionIconType.Chevron,
iconRotation: 90,
onClick: () => this.setState({mode: 'default'}),
}]
: undefined;
return (
<SessionModal
title={i18n('editProfileModalTitle')}
onOk={this.onClickOK}
onClose={this.closeDialog}
headerReverse={viewEdit || viewQR}
headerIconButtons={backButton}
>
<div className="spacer-md"></div>
{viewQR && this.renderQRView(sessionID)}
{viewDefault && this.renderDefaultView()}
{viewEdit && this.renderEditView()}
<div className="session-id-section">
<div className="panel-text-divider">
<span>{window.i18n('yourSessionID')}</span>
</div>
<p className="session-id-section-display">
{ sessionID }
</p>
<div className="spacer-lg"></div>
{ (viewDefault || viewQR) ? (
<SessionButton
text={window.i18n('copy')}
buttonType={SessionButtonType.BrandOutline}
buttonColor={SessionButtonColor.Green}
onClick={() => this.copySessionID(sessionID)}
/>
) : (
<SessionButton
text={window.i18n('save')}
buttonType={SessionButtonType.BrandOutline}
buttonColor={SessionButtonColor.White}
onClick={this.onClickOK}
/>
)}
<div className="spacer-lg"></div>
</div>
</SessionModal>
);
}
private renderProfileHeader() {
return (
<>
<div className="avatar-center">
<div className="avatar-center-inner">
{this.renderAvatar()}
<div className="image-upload-section">
<div
className="image-upload-section"
onClick={() => {
const el = this.inputEl.current;
if (el) {
el.click();
}
}}
>
<input
type="file"
ref={this.inputEl}
@ -88,51 +145,72 @@ export class EditProfileDialog extends React.Component<Props, State> {
name="name"
onChange={this.onFileSelected}
/>
</div>
<div
className="qr-view-button"
onClick={() => this.setState({mode: 'qr'})}
>
<SessionIconButton
iconType={SessionIconType.Upload}
iconSize={SessionIconSize.Huge}
onClick={() => {
const el = this.inputEl.current;
if (el) {
el.click();
}
}}
iconType={SessionIconType.QR}
iconSize={SessionIconSize.Small}
iconColor={"#000000"}
/>
</div>
</div>
</div>
</>
);
}
<div className="spacer md" />
<input
type="text"
className="profile-name"
value={this.state.profileName}
placeholder={placeholderText}
onChange={this.onNameEdited}
tabIndex={0}
required={true}
aria-required={true}
/>
<div className="message">{i18n('editProfileDisplayNameWarning')}</div>
<span className={errorMessageClasses}>{this.state.errorMessage}</span>
<div className="spacer-lg" />
<div className="session-modal__button-group">
<SessionButton
text={okText}
buttonColor={SessionButtonColor.Secondary}
onClick={this.onClickOK}
private renderDefaultView(){
return (
<>
{ this.renderProfileHeader() }
<div className="profile-name-uneditable">
<p>{ this.state.setProfileName }</p>
<SessionIconButton
iconType={SessionIconType.Pencil}
iconSize={SessionIconSize.Medium}
onClick={() => this.setState({mode: 'edit'})}
/>
<SessionButton
text={cancelText}
buttonColor={SessionButtonColor.Primary}
onClick={this.closeDialog}
</div>
</>
);
}
private renderEditView() {
const placeholderText = window.i18n('displayName');
return (
<>
{ this.renderProfileHeader() }
<div className="profile-name">
<input
type="text"
className="profile-name-input"
value={this.state.profileName}
placeholder={placeholderText}
onChange={this.onNameEdited}
tabIndex={0}
required={true}
aria-required={true}
/>
</div>
</SessionModal>
</>
);
}
private renderQRView(sessionID: string){
const bgColor = '#FFFFFF';
const fgColor = '#1B1B1B';
return (
<div className="qr-image">
<QRCode value={sessionID} bgColor={bgColor} fgColor={fgColor} level="L" />
</div>
);
}
@ -189,29 +267,20 @@ export class EditProfileDialog extends React.Component<Props, State> {
}
}
private showError(msg: string) {
if (this.state.errorDisplayed) {
return;
}
private copySessionID(sessionID: string) {
window.clipboard.writeText(sessionID);
this.setState({
errorDisplayed: true,
errorMessage: msg,
window.pushToast({
title: window.i18n('copiedSessionID'),
type: 'success',
id: 'copiedSessionID',
});
setTimeout(() => {
this.setState({
errorDisplayed: false,
});
}, 3000);
}
private onClickOK() {
const newName = this.state.profileName.trim();
if (newName === '') {
this.showError(this.props.i18n('emptyProfileNameError'));
return;
}
@ -224,7 +293,11 @@ export class EditProfileDialog extends React.Component<Props, State> {
: null;
this.props.onOk(newName, avatar);
this.closeDialog();
this.setState({
mode: 'default',
setProfileName: this.state.profileName,
});
}
private closeDialog() {

@ -40,9 +40,9 @@ const Section = ({
}) => {
const handleClick = onSelect
? () => {
if (type !== SectionType.Profile) {
onSelect(type);
}
type === SectionType.Profile
? window.showEditProfileDialog()
: onSelect(type);
}
: undefined;

@ -1,6 +1,8 @@
import React from 'react';
import classNames from 'classnames';
import { SessionIconButton, SessionIconSize, SessionIconType } from './icon/';
import { SessionButton, SessionButtonType, SessionButtonColor } from './SessionButton';
interface Props {
title: string;
@ -8,9 +10,17 @@ interface Props {
onOk: any;
showExitIcon?: boolean;
showHeader?: boolean;
headerReverse?: boolean;
//Maximum of two icons in header
headerIconButtons?: Array<{
type: SessionIconType;
iconType: SessionIconType;
iconRotation: number;
onClick?: any;
}>;
headerButtons?: Array<{
buttonType: SessionButtonType;
buttonColor: SessionButtonColor;
text: string;
onClick?: any;
}>;
}
@ -23,6 +33,7 @@ export class SessionModal extends React.PureComponent<Props, State> {
public static defaultProps = {
showExitIcon: true,
showHeader: true,
headerReverse: false,
};
constructor(props: any) {
@ -38,14 +49,14 @@ export class SessionModal extends React.PureComponent<Props, State> {
}
public render() {
const { title, headerIconButtons, showExitIcon, showHeader } = this.props;
const { title, headerIconButtons, showExitIcon, showHeader, headerReverse } = this.props;
const { isVisible } = this.state;
return isVisible ? (
<div className={'session-modal'}>
{showHeader ? (
<>
<div className="session-modal__header">
<div className={classNames("session-modal__header", headerReverse && "reverse")}>
<div className="session-modal__header__close">
{showExitIcon ? (
<SessionIconButton
@ -61,9 +72,11 @@ export class SessionModal extends React.PureComponent<Props, State> {
? headerIconButtons.map((iconItem: any) => {
return (
<SessionIconButton
key={iconItem.type}
iconType={iconItem.type}
iconSize={SessionIconSize.Medium}
key={iconItem.iconType}
iconType={iconItem.iconType}
iconSize={SessionIconSize.Large}
iconRotation={iconItem.iconRotation}
onClick={iconItem.onClick}
/>
);
})

@ -17,11 +17,9 @@ export class SessionQRModal extends React.Component<Props> {
public render() {
const { value, onClose } = this.props;
const theme = window.Events.getThemeSetting();
// Foreground equivalent to .session-modal background color
const bgColor = 'rgba(0, 0, 0, 0)';
const fgColor = theme === 'dark' ? '#FFFFFF' : '#1B1B1B';
const bgColor = '#FFFFFF';
const fgColor = '#1B1B1B';
return (
<SessionModal

@ -22,6 +22,7 @@ export enum SessionIconType {
Lock = 'lock',
Microphone = 'microphone',
Moon = 'moon',
Pencil = 'pencil',
Reply = 'reply',
Search = 'search',
Send = 'send',
@ -154,6 +155,11 @@ export const icons = {
'M11.1441877,12.8180303 C8.90278993,10.5766325 8.24397847,7.29260898 9.27752593,4.437982 C6.09633644,5.5873034 3.89540402,8.67837285 4.00385273,12.2078365 C4.13368986,16.4333868 7.52883112,19.8285281 11.7543814,19.9583652 C15.2838451,20.0668139 18.3749145,17.8658815 19.5242359,14.684692 C16.669609,15.7182395 13.3855854,15.059428 11.1441877,12.8180303 Z M21.9576498,12.8823459 C21.4713729,18.1443552 16.9748949,22.1197182 11.692957,21.9574217 C6.41101918,21.7951253 2.16709261,17.5511988 2.00479619,12.2692609 C1.84249977,6.98732307 5.81786273,2.49084501 11.0798721,2.00456809 C11.9400195,1.92507947 12.4895134,2.90008536 11.9760569,3.59473245 C10.2106529,5.98311963 10.4582768,9.30369233 12.5584012,11.4038167 C14.6585256,13.5039411 17.9790983,13.7515651 20.3674855,11.986161 C21.0621326,11.4727046 22.0371385,12.0221984 21.9576498,12.8823459 lZ',
viewBox: '0.5 0.5 22 22',
},
[SessionIconType.Pencil]: {
path:
'M4,16.4142136 L4,20 L7.58578644,20 L19.5857864,8 L16,4.41421356 L4,16.4142136 Z M16.7071068,2.29289322 L21.7071068,7.29289322 C22.0976311,7.68341751 22.0976311,8.31658249 21.7071068,8.70710678 L8.70710678,21.7071068 C8.5195704,21.8946432 8.26521649,22 8,22 L3,22 C2.44771525,22 2,21.5522847 2,21 L2,16 C2,15.7347835 2.10535684,15.4804296 2.29289322,15.2928932 L15.2928932,2.29289322 C15.6834175,1.90236893 16.3165825,1.90236893 16.7071068,2.29289322 Z',
viewBox: '1 1 21 21',
},
[SessionIconType.Reply]: {
path:
'M4,3 C4.55228475,3 5,3.44771525 5,4 L5,4 L5,11 C5,12.6568542 6.34314575,14 8,14 L8,14 L17.585,14 L14.2928932,10.7071068 C13.9324093,10.3466228 13.9046797,9.77939176 14.2097046,9.38710056 L14.2928932,9.29289322 C14.6834175,8.90236893 15.3165825,8.90236893 15.7071068,9.29289322 L15.7071068,9.29289322 L20.7071068,14.2928932 C20.7355731,14.3213595 20.7623312,14.3515341 20.787214,14.3832499 C20.788658,14.3849951 20.7902348,14.3870172 20.7918027,14.389044 C20.8140715,14.4179625 20.8348358,14.4480862 20.8539326,14.4793398 C20.8613931,14.4913869 20.8685012,14.5036056 20.8753288,14.5159379 C20.8862061,14.5357061 20.8966234,14.5561086 20.9063462,14.5769009 C20.914321,14.5939015 20.9218036,14.6112044 20.9287745,14.628664 C20.9366843,14.6484208 20.9438775,14.6682023 20.9504533,14.6882636 C20.9552713,14.7031487 20.9599023,14.7185367 20.9641549,14.734007 C20.9701664,14.7555635 20.9753602,14.7772539 20.9798348,14.7992059 C20.9832978,14.8166247 20.9863719,14.834051 20.9889822,14.8515331 C20.9962388,14.8996379 21,14.9493797 21,15 L20.9962979,14.9137692 C20.9978436,14.9317345 20.9989053,14.9497336 20.9994829,14.9677454 L21,15 C21,15.0112225 20.9998151,15.0224019 20.9994483,15.0335352 C20.9988772,15.050591 20.997855,15.0679231 20.996384,15.0852242 C20.994564,15.1070574 20.9920941,15.1281144 20.9889807,15.1489612 C20.9863719,15.165949 20.9832978,15.1833753 20.9797599,15.2007258 C20.9753602,15.2227461 20.9701664,15.2444365 20.964279,15.2658396 C20.9599023,15.2814633 20.9552713,15.2968513 20.9502619,15.3121425 C20.9438775,15.3317977 20.9366843,15.3515792 20.928896,15.3710585 C20.9218036,15.3887956 20.914321,15.4060985 20.9063266,15.4232215 C20.8974314,15.4421635 20.8879327,15.4609002 20.8778732,15.4792864 C20.8703855,15.4931447 20.862375,15.5070057 20.8540045,15.5207088 C20.8382813,15.546275 20.8215099,15.5711307 20.8036865,15.5951593 C20.774687,15.6343256 20.7425008,15.6717127 20.7071068,15.7071068 L20.787214,15.6167501 C20.7849289,15.6196628 20.7826279,15.6225624 20.7803112,15.625449 L20.7071068,15.7071068 L15.7071068,20.7071068 C15.3165825,21.0976311 14.6834175,21.0976311 14.2928932,20.7071068 C13.9023689,20.3165825 13.9023689,19.6834175 14.2928932,19.2928932 L14.2928932,19.2928932 L17.585,16 L8,16 C5.3112453,16 3.11818189,13.8776933 3.00461951,11.2168896 L3,11 L3,4 C3,3.44771525 3.44771525,3 4,3 lZ',

@ -216,7 +216,7 @@ export class SettingsView extends React.Component<SettingsViewProps, State> {
return (
<div className="session-settings">
<SettingsHeader category={category} />
<SettingsHeader disableLinkDeviceButton={shouldRenderPasswordLock} category={category} />
{shouldRenderPasswordLock ? (
this.renderPasswordLock()
) : (
@ -294,7 +294,7 @@ export class SettingsView extends React.Component<SettingsViewProps, State> {
{
id: 'theme-setting',
title: window.i18n('themeToggleTitle'),
description: 'Choose the theme best suited to you',
description: window.i18n('themeToggleDescription'),
hidden: true,
comparisonValue: 'light',
type: SessionSettingType.Toggle,

@ -4,11 +4,19 @@ import { SessionIconButton, SessionIconSize, SessionIconType } from '../icon';
import { SessionSettingCategory, SettingsViewProps } from './SessionSettings';
import { SessionButton } from '../SessionButton';
export class SettingsHeader extends React.Component<SettingsViewProps, any> {
interface Props extends SettingsViewProps{
disableLinkDeviceButton: boolean | null;
}
export class SettingsHeader extends React.Component<Props, any> {
public static defaultProps = {
disableLinkDeviceButton: true,
}
public constructor(props: any) {
super(props);
this.state = {
disableLinkDeviceButton: true,
disableLinkDeviceButton: this.props.disableLinkDeviceButton,
};
this.showAddLinkedDeviceModal = this.showAddLinkedDeviceModal.bind(this);
}

2
ts/global.d.ts vendored

@ -28,8 +28,10 @@ interface Window {
pushToast: any;
confirmationDialog: any;
showQRDialog: any;
showSeedDialog: any;
showPasswordDialog: any;
showEditProfileDialog: any;
deleteAccount: any;

Loading…
Cancel
Save