|
|
@ -2,7 +2,7 @@ import React from 'react';
|
|
|
|
|
|
|
|
|
|
|
|
import { SessionModal } from './SessionModal';
|
|
|
|
import { SessionModal } from './SessionModal';
|
|
|
|
import { SessionButton, SessionButtonColor } from './SessionButton';
|
|
|
|
import { SessionButton, SessionButtonColor } from './SessionButton';
|
|
|
|
import { PasswordUtil } from '../../util/';
|
|
|
|
import { missingCaseError, PasswordUtil } from '../../util/';
|
|
|
|
import { ToastUtils } from '../../session/utils';
|
|
|
|
import { ToastUtils } from '../../session/utils';
|
|
|
|
import { toast } from 'react-toastify';
|
|
|
|
import { toast } from 'react-toastify';
|
|
|
|
import { SessionToast, SessionToastType } from './SessionToast';
|
|
|
|
import { SessionToast, SessionToastType } from './SessionToast';
|
|
|
@ -46,8 +46,6 @@ class SessionPasswordModalInner extends React.Component<Props, State> {
|
|
|
|
|
|
|
|
|
|
|
|
this.onPasswordInput = this.onPasswordInput.bind(this);
|
|
|
|
this.onPasswordInput = this.onPasswordInput.bind(this);
|
|
|
|
this.onPasswordConfirmInput = this.onPasswordConfirmInput.bind(this);
|
|
|
|
this.onPasswordConfirmInput = this.onPasswordConfirmInput.bind(this);
|
|
|
|
|
|
|
|
|
|
|
|
this.onPaste = this.onPaste.bind(this);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public componentDidMount() {
|
|
|
|
public componentDidMount() {
|
|
|
@ -86,8 +84,6 @@ class SessionPasswordModalInner extends React.Component<Props, State> {
|
|
|
|
}}
|
|
|
|
}}
|
|
|
|
placeholder={placeholders[0]}
|
|
|
|
placeholder={placeholders[0]}
|
|
|
|
onKeyUp={this.onPasswordInput}
|
|
|
|
onKeyUp={this.onPasswordInput}
|
|
|
|
maxLength={window.CONSTANTS.MAX_PASSWORD_LENGTH}
|
|
|
|
|
|
|
|
onPaste={this.onPaste}
|
|
|
|
|
|
|
|
/>
|
|
|
|
/>
|
|
|
|
{action !== PasswordAction.Remove && (
|
|
|
|
{action !== PasswordAction.Remove && (
|
|
|
|
<input
|
|
|
|
<input
|
|
|
@ -95,8 +91,6 @@ class SessionPasswordModalInner extends React.Component<Props, State> {
|
|
|
|
id="password-modal-input-confirm"
|
|
|
|
id="password-modal-input-confirm"
|
|
|
|
placeholder={placeholders[1]}
|
|
|
|
placeholder={placeholders[1]}
|
|
|
|
onKeyUp={this.onPasswordConfirmInput}
|
|
|
|
onKeyUp={this.onPasswordConfirmInput}
|
|
|
|
maxLength={window.CONSTANTS.MAX_PASSWORD_LENGTH}
|
|
|
|
|
|
|
|
onPaste={this.onPaste}
|
|
|
|
|
|
|
|
/>
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
@ -108,7 +102,7 @@ class SessionPasswordModalInner extends React.Component<Props, State> {
|
|
|
|
<SessionButton
|
|
|
|
<SessionButton
|
|
|
|
text={window.i18n('ok')}
|
|
|
|
text={window.i18n('ok')}
|
|
|
|
buttonColor={confirmButtonColor}
|
|
|
|
buttonColor={confirmButtonColor}
|
|
|
|
onClick={async () => this.setPassword(onOk)}
|
|
|
|
onClick={this.setPassword}
|
|
|
|
/>
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
|
|
<SessionButton
|
|
|
|
<SessionButton
|
|
|
@ -145,144 +139,147 @@ class SessionPasswordModalInner extends React.Component<Props, State> {
|
|
|
|
);
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// tslint:disable-next-line: cyclomatic-complexity
|
|
|
|
/**
|
|
|
|
private async setPassword(onSuccess?: any) {
|
|
|
|
* Returns false and set the state error field in the input is not a valid password
|
|
|
|
const { action } = this.props;
|
|
|
|
* or returns true
|
|
|
|
const {
|
|
|
|
*/
|
|
|
|
currentPasswordEntered,
|
|
|
|
private validatePassword(firstPassword: string) {
|
|
|
|
currentPasswordConfirmEntered,
|
|
|
|
|
|
|
|
} = this.state;
|
|
|
|
|
|
|
|
const { Set, Remove, Change } = PasswordAction;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Trim leading / trailing whitespace for UX
|
|
|
|
|
|
|
|
const enteredPassword = (currentPasswordEntered || '').trim();
|
|
|
|
|
|
|
|
const enteredPasswordConfirm = (currentPasswordConfirmEntered || '').trim();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// if user did not fill the first password field, we can't do anything
|
|
|
|
// if user did not fill the first password field, we can't do anything
|
|
|
|
const errorFirstInput = PasswordUtil.validatePassword(
|
|
|
|
const errorFirstInput = PasswordUtil.validatePassword(
|
|
|
|
enteredPassword,
|
|
|
|
firstPassword,
|
|
|
|
window.i18n
|
|
|
|
window.i18n
|
|
|
|
);
|
|
|
|
);
|
|
|
|
if (errorFirstInput !== null) {
|
|
|
|
if (errorFirstInput !== null) {
|
|
|
|
this.setState({
|
|
|
|
this.setState({
|
|
|
|
error: errorFirstInput,
|
|
|
|
error: errorFirstInput,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private async handleActionSet(
|
|
|
|
|
|
|
|
enteredPassword: string,
|
|
|
|
|
|
|
|
enteredPasswordConfirm: string
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
// be sure both password are valid
|
|
|
|
|
|
|
|
if (!this.validatePassword(enteredPassword)) {
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// no need to validate second password. we just need to check that enteredPassword is valid, and that both password matches
|
|
|
|
|
|
|
|
|
|
|
|
// if action is Set or Change, we need a valid ConfirmPassword
|
|
|
|
if (enteredPassword !== enteredPasswordConfirm) {
|
|
|
|
if (action === Set || action === Change) {
|
|
|
|
|
|
|
|
const errorSecondInput = PasswordUtil.validatePassword(
|
|
|
|
|
|
|
|
enteredPasswordConfirm,
|
|
|
|
|
|
|
|
window.i18n
|
|
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
if (errorSecondInput !== null) {
|
|
|
|
|
|
|
|
this.setState({
|
|
|
|
this.setState({
|
|
|
|
error: errorSecondInput,
|
|
|
|
error: window.i18n('setPasswordInvalid'),
|
|
|
|
});
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
await window.setPassword(enteredPassword, null);
|
|
|
|
|
|
|
|
ToastUtils.pushToastSuccess(
|
|
|
|
// Passwords match or remove password successful
|
|
|
|
'setPasswordSuccessToast',
|
|
|
|
const newPassword = action === Remove ? null : enteredPasswordConfirm;
|
|
|
|
window.i18n('setPasswordTitle'),
|
|
|
|
const oldPassword = action === Set ? null : enteredPassword;
|
|
|
|
window.i18n('setPasswordToastDescription'),
|
|
|
|
|
|
|
|
SessionIconType.Lock
|
|
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
// Check if password match, when setting, changing or removing
|
|
|
|
this.props.onOk(this.props.action);
|
|
|
|
let valid;
|
|
|
|
this.closeDialog();
|
|
|
|
if (action === Set) {
|
|
|
|
|
|
|
|
valid = enteredPassword === enteredPasswordConfirm;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
valid = Boolean(await this.validatePasswordHash(oldPassword));
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!valid) {
|
|
|
|
private async handleActionChange(oldPassword: string, newPassword: string) {
|
|
|
|
let str;
|
|
|
|
// We don't validate oldPassword on change: this is validate on the validatePasswordHash below
|
|
|
|
switch (action) {
|
|
|
|
// we only validate the newPassword here
|
|
|
|
case Set:
|
|
|
|
if (!this.validatePassword(newPassword)) {
|
|
|
|
str = window.i18n('setPasswordInvalid');
|
|
|
|
return;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Change:
|
|
|
|
|
|
|
|
str = window.i18n('changePasswordInvalid');
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Remove:
|
|
|
|
|
|
|
|
str = window.i18n('removePasswordInvalid');
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
|
|
|
throw new Error(`Invalid action ${action}`);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const isValidWithStoredInDB = Boolean(
|
|
|
|
|
|
|
|
await this.validatePasswordHash(oldPassword)
|
|
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!isValidWithStoredInDB) {
|
|
|
|
this.setState({
|
|
|
|
this.setState({
|
|
|
|
error: str,
|
|
|
|
error: window.i18n('changePasswordInvalid'),
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
await window.setPassword(newPassword, oldPassword);
|
|
|
|
await window.setPassword(newPassword, oldPassword);
|
|
|
|
let title;
|
|
|
|
|
|
|
|
let description;
|
|
|
|
|
|
|
|
switch (action) {
|
|
|
|
|
|
|
|
case Set:
|
|
|
|
|
|
|
|
title = window.i18n('setPasswordTitle');
|
|
|
|
|
|
|
|
description = window.i18n('setPasswordToastDescription');
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Change:
|
|
|
|
|
|
|
|
title = window.i18n('changePasswordTitle');
|
|
|
|
|
|
|
|
description = window.i18n('changePasswordToastDescription');
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Remove:
|
|
|
|
|
|
|
|
title = window.i18n('removePasswordTitle');
|
|
|
|
|
|
|
|
description = window.i18n('removePasswordToastDescription');
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
|
|
|
throw new Error(`Invalid action ${action}`);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (action !== Remove) {
|
|
|
|
|
|
|
|
ToastUtils.pushToastSuccess(
|
|
|
|
ToastUtils.pushToastSuccess(
|
|
|
|
'setPasswordSuccessToast',
|
|
|
|
'setPasswordSuccessToast',
|
|
|
|
title,
|
|
|
|
window.i18n('changePasswordTitle'),
|
|
|
|
description,
|
|
|
|
window.i18n('changePasswordToastDescription'),
|
|
|
|
SessionIconType.Lock
|
|
|
|
SessionIconType.Lock
|
|
|
|
);
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
|
|
|
|
|
|
|
|
this.props.onOk(this.props.action);
|
|
|
|
|
|
|
|
this.closeDialog();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private async handleActionRemove(oldPassword: string) {
|
|
|
|
|
|
|
|
// We don't validate oldPassword on change: this is validate on the validatePasswordHash below
|
|
|
|
|
|
|
|
const isValidWithStoredInDB = Boolean(
|
|
|
|
|
|
|
|
await this.validatePasswordHash(oldPassword)
|
|
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!isValidWithStoredInDB) {
|
|
|
|
|
|
|
|
this.setState({
|
|
|
|
|
|
|
|
error: window.i18n('removePasswordInvalid'),
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
await window.setPassword(null, oldPassword);
|
|
|
|
|
|
|
|
|
|
|
|
ToastUtils.pushToastWarning(
|
|
|
|
ToastUtils.pushToastWarning(
|
|
|
|
'setPasswordSuccessToast',
|
|
|
|
'setPasswordSuccessToast',
|
|
|
|
title,
|
|
|
|
window.i18n('removePasswordTitle'),
|
|
|
|
description
|
|
|
|
window.i18n('removePasswordToastDescription')
|
|
|
|
);
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
onSuccess(this.props.action);
|
|
|
|
this.props.onOk(this.props.action);
|
|
|
|
this.closeDialog();
|
|
|
|
this.closeDialog();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private closeDialog() {
|
|
|
|
// tslint:disable-next-line: cyclomatic-complexity
|
|
|
|
if (this.props.onClose) {
|
|
|
|
private async setPassword() {
|
|
|
|
this.props.onClose();
|
|
|
|
const { action } = this.props;
|
|
|
|
}
|
|
|
|
const {
|
|
|
|
}
|
|
|
|
currentPasswordEntered,
|
|
|
|
|
|
|
|
currentPasswordConfirmEntered,
|
|
|
|
|
|
|
|
} = this.state;
|
|
|
|
|
|
|
|
const { Set, Remove, Change } = PasswordAction;
|
|
|
|
|
|
|
|
|
|
|
|
private onPaste(event: any) {
|
|
|
|
// Trim leading / trailing whitespace for UX
|
|
|
|
const clipboard = event.clipboardData.getData('text');
|
|
|
|
const firstPasswordEntered = (currentPasswordEntered || '').trim();
|
|
|
|
|
|
|
|
const secondPasswordEntered = (currentPasswordConfirmEntered || '').trim();
|
|
|
|
|
|
|
|
|
|
|
|
if (clipboard.length > window.CONSTANTS.MAX_PASSWORD_LENGTH) {
|
|
|
|
switch (action) {
|
|
|
|
const title = String(
|
|
|
|
case Set: {
|
|
|
|
window.i18n(
|
|
|
|
await this.handleActionSet(firstPasswordEntered, secondPasswordEntered);
|
|
|
|
'pasteLongPasswordToastTitle',
|
|
|
|
return;
|
|
|
|
window.CONSTANTS.MAX_PASSWORD_LENGTH
|
|
|
|
}
|
|
|
|
)
|
|
|
|
case Change: {
|
|
|
|
|
|
|
|
await this.handleActionChange(
|
|
|
|
|
|
|
|
firstPasswordEntered,
|
|
|
|
|
|
|
|
secondPasswordEntered
|
|
|
|
);
|
|
|
|
);
|
|
|
|
ToastUtils.pushToastWarning('passwordModal', title);
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
case Remove: {
|
|
|
|
|
|
|
|
await this.handleActionRemove(firstPasswordEntered);
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
|
|
|
throw missingCaseError(action);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Prevent pating into input
|
|
|
|
private closeDialog() {
|
|
|
|
return false;
|
|
|
|
if (this.props.onClose) {
|
|
|
|
|
|
|
|
this.props.onClose();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private async onPasswordInput(event: any) {
|
|
|
|
private async onPasswordInput(event: any) {
|
|
|
|
if (event.key === 'Enter') {
|
|
|
|
if (event.key === 'Enter') {
|
|
|
|
return this.setPassword(this.props.onOk);
|
|
|
|
return this.setPassword();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const currentPasswordEntered = event.target.value;
|
|
|
|
const currentPasswordEntered = event.target.value;
|
|
|
|
|
|
|
|
|
|
|
@ -291,7 +288,7 @@ class SessionPasswordModalInner extends React.Component<Props, State> {
|
|
|
|
|
|
|
|
|
|
|
|
private async onPasswordConfirmInput(event: any) {
|
|
|
|
private async onPasswordConfirmInput(event: any) {
|
|
|
|
if (event.key === 'Enter') {
|
|
|
|
if (event.key === 'Enter') {
|
|
|
|
return this.setPassword(this.props.onOk);
|
|
|
|
return this.setPassword();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const currentPasswordConfirmEntered = event.target.value;
|
|
|
|
const currentPasswordConfirmEntered = event.target.value;
|
|
|
|
|
|
|
|
|
|
|
|