/* Copyright 2015, 2016 OpenMarket Ltd Copyright 2017 Vector Creations Ltd Copyright 2019 New Vector Ltd. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ import React from 'react'; import PropTypes from 'prop-types'; import classNames from 'classnames'; import * as sdk from '../../../index'; import { _t } from '../../../languageHandler'; import SdkConfig from '../../../SdkConfig'; import {ValidatedServerConfig} from "../../../utils/AutoDiscoveryUtils"; import AccessibleButton from "../elements/AccessibleButton"; /** * A pure UI component which displays a username/password form. */ export default class PasswordLogin extends React.Component { static propTypes = { onSubmit: PropTypes.func.isRequired, // fn(username, password) onError: PropTypes.func, onEditServerDetailsClick: PropTypes.func, onForgotPasswordClick: PropTypes.func, // fn() initialUsername: PropTypes.string, initialPhoneCountry: PropTypes.string, initialPhoneNumber: PropTypes.string, initialPassword: PropTypes.string, onUsernameChanged: PropTypes.func, onPhoneCountryChanged: PropTypes.func, onPhoneNumberChanged: PropTypes.func, onPasswordChanged: PropTypes.func, loginIncorrect: PropTypes.bool, disableSubmit: PropTypes.bool, serverConfig: PropTypes.instanceOf(ValidatedServerConfig).isRequired, busy: PropTypes.bool, }; static defaultProps = { onError: function() {}, onEditServerDetailsClick: null, onUsernameChanged: function() {}, onUsernameBlur: function() {}, onPasswordChanged: function() {}, onPhoneCountryChanged: function() {}, onPhoneNumberChanged: function() {}, onPhoneNumberBlur: function() {}, initialUsername: "", initialPhoneCountry: "", initialPhoneNumber: "", initialPassword: "", loginIncorrect: false, disableSubmit: false, }; static LOGIN_FIELD_EMAIL = "login_field_email"; static LOGIN_FIELD_MXID = "login_field_mxid"; static LOGIN_FIELD_PHONE = "login_field_phone"; constructor(props) { super(props); this.state = { username: this.props.initialUsername, password: this.props.initialPassword, phoneCountry: this.props.initialPhoneCountry, phoneNumber: this.props.initialPhoneNumber, loginType: PasswordLogin.LOGIN_FIELD_MXID, }; this.onForgotPasswordClick = this.onForgotPasswordClick.bind(this); this.onSubmitForm = this.onSubmitForm.bind(this); this.onUsernameChanged = this.onUsernameChanged.bind(this); this.onUsernameBlur = this.onUsernameBlur.bind(this); this.onLoginTypeChange = this.onLoginTypeChange.bind(this); this.onPhoneCountryChanged = this.onPhoneCountryChanged.bind(this); this.onPhoneNumberChanged = this.onPhoneNumberChanged.bind(this); this.onPhoneNumberBlur = this.onPhoneNumberBlur.bind(this); this.onPasswordChanged = this.onPasswordChanged.bind(this); this.isLoginEmpty = this.isLoginEmpty.bind(this); } onForgotPasswordClick(ev) { ev.preventDefault(); ev.stopPropagation(); this.props.onForgotPasswordClick(); } onSubmitForm(ev) { ev.preventDefault(); let username = ''; // XXX: Synapse breaks if you send null here: let phoneCountry = null; let phoneNumber = null; let error; switch (this.state.loginType) { case PasswordLogin.LOGIN_FIELD_EMAIL: username = this.state.username; if (!username) { error = _t('The email field must not be blank.'); } break; case PasswordLogin.LOGIN_FIELD_MXID: username = this.state.username; if (!username) { error = _t('The username field must not be blank.'); } break; case PasswordLogin.LOGIN_FIELD_PHONE: phoneCountry = this.state.phoneCountry; phoneNumber = this.state.phoneNumber; if (!phoneNumber) { error = _t('The phone number field must not be blank.'); } break; } if (error) { this.props.onError(error); return; } if (!this.state.password) { this.props.onError(_t('The password field must not be blank.')); return; } this.props.onSubmit( username, phoneCountry, phoneNumber, this.state.password, ); } onUsernameChanged(ev) { this.setState({username: ev.target.value}); this.props.onUsernameChanged(ev.target.value); } onUsernameBlur(ev) { this.props.onUsernameBlur(ev.target.value); } onLoginTypeChange(ev) { const loginType = ev.target.value; this.props.onError(null); // send a null error to clear any error messages this.setState({ loginType: loginType, username: "", // Reset because email and username use the same state }); } onPhoneCountryChanged(country) { this.setState({ phoneCountry: country.iso2, phonePrefix: country.prefix, }); this.props.onPhoneCountryChanged(country.iso2); } onPhoneNumberChanged(ev) { this.setState({phoneNumber: ev.target.value}); this.props.onPhoneNumberChanged(ev.target.value); } onPhoneNumberBlur(ev) { this.props.onPhoneNumberBlur(ev.target.value); } onPasswordChanged(ev) { this.setState({password: ev.target.value}); this.props.onPasswordChanged(ev.target.value); } renderLoginField(loginType, autoFocus) { const Field = sdk.getComponent('elements.Field'); const classes = {}; switch (loginType) { case PasswordLogin.LOGIN_FIELD_EMAIL: classes.error = this.props.loginIncorrect && !this.state.username; return ; case PasswordLogin.LOGIN_FIELD_MXID: classes.error = this.props.loginIncorrect && !this.state.username; return ; case PasswordLogin.LOGIN_FIELD_PHONE: { const CountryDropdown = sdk.getComponent('views.auth.CountryDropdown'); classes.error = this.props.loginIncorrect && !this.state.phoneNumber; const phoneCountry = ; return ; } } } isLoginEmpty() { switch (this.state.loginType) { case PasswordLogin.LOGIN_FIELD_EMAIL: case PasswordLogin.LOGIN_FIELD_MXID: return !this.state.username; case PasswordLogin.LOGIN_FIELD_PHONE: return !this.state.phoneCountry || !this.state.phoneNumber; } } render() { const Field = sdk.getComponent('elements.Field'); const SignInToText = sdk.getComponent('views.auth.SignInToText'); let forgotPasswordJsx; if (this.props.onForgotPasswordClick) { forgotPasswordJsx = {_t('Not sure of your password? Set a new one', {}, { a: sub => ( {sub} ), })} ; } const pwFieldClass = classNames({ error: this.props.loginIncorrect && !this.isLoginEmpty(), // only error password if error isn't top field }); // If login is empty, autoFocus login, otherwise autoFocus password. // this is for when auto server discovery remounts us when the user tries to tab from username to password const autoFocusPassword = !this.isLoginEmpty(); const loginField = this.renderLoginField(this.state.loginType, !autoFocusPassword); let loginType; if (!SdkConfig.get().disable_3pid_login) { loginType = (
); } return (
{loginType} {loginField} {forgotPasswordJsx} { !this.props.busy && }
); } }