mirror of
https://github.com/element-hq/element-web.git
synced 2024-12-16 21:21:46 +03:00
621 lines
25 KiB
TypeScript
621 lines
25 KiB
TypeScript
/*
|
|
Copyright 2015-2021 The Matrix.org Foundation C.I.C.
|
|
|
|
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 {createClient} from 'matrix-js-sdk/src/matrix';
|
|
import React, {ReactNode} from 'react';
|
|
import {MatrixClient} from "matrix-js-sdk/src/client";
|
|
|
|
import * as sdk from '../../../index';
|
|
import { _t, _td } from '../../../languageHandler';
|
|
import { messageForResourceLimitError } from '../../../utils/ErrorUtils';
|
|
import AutoDiscoveryUtils, {ValidatedServerConfig} from "../../../utils/AutoDiscoveryUtils";
|
|
import classNames from "classnames";
|
|
import * as Lifecycle from '../../../Lifecycle';
|
|
import {MatrixClientPeg} from "../../../MatrixClientPeg";
|
|
import AuthPage from "../../views/auth/AuthPage";
|
|
import Login, {ISSOFlow} from "../../../Login";
|
|
import dis from "../../../dispatcher/dispatcher";
|
|
import SSOButtons from "../../views/elements/SSOButtons";
|
|
import ServerPicker from '../../views/elements/ServerPicker';
|
|
import {replaceableComponent} from "../../../utils/replaceableComponent";
|
|
|
|
interface IProps {
|
|
serverConfig: ValidatedServerConfig;
|
|
defaultDeviceDisplayName: string;
|
|
email?: string;
|
|
brand?: string;
|
|
clientSecret?: string;
|
|
sessionId?: string;
|
|
idSid?: string;
|
|
fragmentAfterLogin?: string;
|
|
|
|
// Called when the user has logged in. Params:
|
|
// - object with userId, deviceId, homeserverUrl, identityServerUrl, accessToken
|
|
// - The user's password, if available and applicable (may be cached in memory
|
|
// for a short time so the user is not required to re-enter their password
|
|
// for operations like uploading cross-signing keys).
|
|
onLoggedIn(params: {
|
|
userId: string;
|
|
deviceId: string
|
|
homeserverUrl: string;
|
|
identityServerUrl?: string;
|
|
accessToken: string;
|
|
}, password: string): void;
|
|
makeRegistrationUrl(params: {
|
|
/* eslint-disable camelcase */
|
|
client_secret: string;
|
|
hs_url: string;
|
|
is_url?: string;
|
|
session_id: string;
|
|
/* eslint-enable camelcase */
|
|
}): string;
|
|
// registration shouldn't know or care how login is done.
|
|
onLoginClick(): void;
|
|
onServerConfigChange(config: ValidatedServerConfig): void;
|
|
}
|
|
|
|
interface IState {
|
|
busy: boolean;
|
|
errorText?: ReactNode;
|
|
// true if we're waiting for the user to complete
|
|
// We remember the values entered by the user because
|
|
// the registration form will be unmounted during the
|
|
// course of registration, but if there's an error we
|
|
// want to bring back the registration form with the
|
|
// values the user entered still in it. We can keep
|
|
// them in this component's state since this component
|
|
// persist for the duration of the registration process.
|
|
formVals: Record<string, string>;
|
|
// user-interactive auth
|
|
// If we've been given a session ID, we're resuming
|
|
// straight back into UI auth
|
|
doingUIAuth: boolean;
|
|
// If set, we've registered but are not going to log
|
|
// the user in to their new account automatically.
|
|
completedNoSignin: boolean;
|
|
flows: {
|
|
stages: string[];
|
|
}[];
|
|
// We perform liveliness checks later, but for now suppress the errors.
|
|
// We also track the server dead errors independently of the regular errors so
|
|
// that we can render it differently, and override any other error the user may
|
|
// be seeing.
|
|
serverIsAlive: boolean;
|
|
serverErrorIsFatal: boolean;
|
|
serverDeadError?: ReactNode;
|
|
|
|
// Our matrix client - part of state because we can't render the UI auth
|
|
// component without it.
|
|
matrixClient?: MatrixClient;
|
|
// The user ID we've just registered
|
|
registeredUsername?: string;
|
|
// if a different user ID to the one we just registered is logged in,
|
|
// this is the user ID that's logged in.
|
|
differentLoggedInUserId?: string;
|
|
// the SSO flow definition, this is fetched from /login as that's the only
|
|
// place it is exposed.
|
|
ssoFlow?: ISSOFlow;
|
|
}
|
|
|
|
@replaceableComponent("structures.auth.Registration")
|
|
export default class Registration extends React.Component<IProps, IState> {
|
|
loginLogic: Login;
|
|
|
|
constructor(props) {
|
|
super(props);
|
|
|
|
this.state = {
|
|
busy: false,
|
|
errorText: null,
|
|
formVals: {
|
|
email: this.props.email,
|
|
},
|
|
doingUIAuth: Boolean(this.props.sessionId),
|
|
flows: null,
|
|
completedNoSignin: false,
|
|
serverIsAlive: true,
|
|
serverErrorIsFatal: false,
|
|
serverDeadError: "",
|
|
};
|
|
|
|
const {hsUrl, isUrl} = this.props.serverConfig;
|
|
this.loginLogic = new Login(hsUrl, isUrl, null, {
|
|
defaultDeviceDisplayName: "Element login check", // We shouldn't ever be used
|
|
});
|
|
}
|
|
|
|
componentDidMount() {
|
|
this.replaceClient(this.props.serverConfig);
|
|
}
|
|
|
|
// TODO: [REACT-WARNING] Replace with appropriate lifecycle event
|
|
// eslint-disable-next-line camelcase
|
|
UNSAFE_componentWillReceiveProps(newProps) {
|
|
if (newProps.serverConfig.hsUrl === this.props.serverConfig.hsUrl &&
|
|
newProps.serverConfig.isUrl === this.props.serverConfig.isUrl) return;
|
|
|
|
this.replaceClient(newProps.serverConfig);
|
|
}
|
|
|
|
private async replaceClient(serverConfig: ValidatedServerConfig) {
|
|
this.setState({
|
|
errorText: null,
|
|
serverDeadError: null,
|
|
serverErrorIsFatal: false,
|
|
// busy while we do liveness check (we need to avoid trying to render
|
|
// the UI auth component while we don't have a matrix client)
|
|
busy: true,
|
|
});
|
|
|
|
// Do a liveliness check on the URLs
|
|
try {
|
|
await AutoDiscoveryUtils.validateServerConfigWithStaticUrls(
|
|
serverConfig.hsUrl,
|
|
serverConfig.isUrl,
|
|
);
|
|
this.setState({
|
|
serverIsAlive: true,
|
|
serverErrorIsFatal: false,
|
|
});
|
|
} catch (e) {
|
|
this.setState({
|
|
busy: false,
|
|
...AutoDiscoveryUtils.authComponentStateForError(e, "register"),
|
|
});
|
|
if (this.state.serverErrorIsFatal) {
|
|
return; // Server is dead - do not continue.
|
|
}
|
|
}
|
|
|
|
const {hsUrl, isUrl} = serverConfig;
|
|
const cli = createClient({
|
|
baseUrl: hsUrl,
|
|
idBaseUrl: isUrl,
|
|
});
|
|
|
|
this.loginLogic.setHomeserverUrl(hsUrl);
|
|
this.loginLogic.setIdentityServerUrl(isUrl);
|
|
|
|
let ssoFlow: ISSOFlow;
|
|
try {
|
|
const loginFlows = await this.loginLogic.getFlows();
|
|
ssoFlow = loginFlows.find(f => f.type === "m.login.sso" || f.type === "m.login.cas") as ISSOFlow;
|
|
} catch (e) {
|
|
console.error("Failed to get login flows to check for SSO support", e);
|
|
}
|
|
|
|
this.setState({
|
|
matrixClient: cli,
|
|
ssoFlow,
|
|
busy: false,
|
|
});
|
|
const showGenericError = (e) => {
|
|
this.setState({
|
|
errorText: _t("Unable to query for supported registration methods."),
|
|
// add empty flows array to get rid of spinner
|
|
flows: [],
|
|
});
|
|
};
|
|
try {
|
|
// We do the first registration request ourselves to discover whether we need to
|
|
// do SSO instead. If we've already started the UI Auth process though, we don't
|
|
// need to.
|
|
if (!this.state.doingUIAuth) {
|
|
await this.makeRegisterRequest(null);
|
|
// This should never succeed since we specified no auth object.
|
|
console.log("Expecting 401 from register request but got success!");
|
|
}
|
|
} catch (e) {
|
|
if (e.httpStatus === 401) {
|
|
this.setState({
|
|
flows: e.data.flows,
|
|
});
|
|
} else if (e.httpStatus === 403 || e.errcode === "M_FORBIDDEN") {
|
|
// Check for 403 or M_FORBIDDEN, Synapse used to send 403 M_UNKNOWN but now sends 403 M_FORBIDDEN.
|
|
// At this point registration is pretty much disabled, but before we do that let's
|
|
// quickly check to see if the server supports SSO instead. If it does, we'll send
|
|
// the user off to the login page to figure their account out.
|
|
if (ssoFlow) {
|
|
// Redirect to login page - server probably expects SSO only
|
|
dis.dispatch({action: 'start_login'});
|
|
} else {
|
|
this.setState({
|
|
serverErrorIsFatal: true, // fatal because user cannot continue on this server
|
|
errorText: _t("Registration has been disabled on this homeserver."),
|
|
// add empty flows array to get rid of spinner
|
|
flows: [],
|
|
});
|
|
}
|
|
} else {
|
|
console.log("Unable to query for supported registration methods.", e);
|
|
showGenericError(e);
|
|
}
|
|
}
|
|
}
|
|
|
|
private onFormSubmit = formVals => {
|
|
this.setState({
|
|
errorText: "",
|
|
busy: true,
|
|
formVals: formVals,
|
|
doingUIAuth: true,
|
|
});
|
|
};
|
|
|
|
private requestEmailToken = (emailAddress, clientSecret, sendAttempt, sessionId) => {
|
|
return this.state.matrixClient.requestRegisterEmailToken(
|
|
emailAddress,
|
|
clientSecret,
|
|
sendAttempt,
|
|
this.props.makeRegistrationUrl({
|
|
client_secret: clientSecret,
|
|
hs_url: this.state.matrixClient.getHomeserverUrl(),
|
|
is_url: this.state.matrixClient.getIdentityServerUrl(),
|
|
session_id: sessionId,
|
|
}),
|
|
);
|
|
}
|
|
|
|
private onUIAuthFinished = async (success, response, extra) => {
|
|
if (!success) {
|
|
let msg = response.message || response.toString();
|
|
// can we give a better error message?
|
|
if (response.errcode === 'M_RESOURCE_LIMIT_EXCEEDED') {
|
|
const errorTop = messageForResourceLimitError(
|
|
response.data.limit_type,
|
|
response.data.admin_contact,
|
|
{
|
|
'monthly_active_user': _td("This homeserver has hit its Monthly Active User limit."),
|
|
'hs_blocked': _td("This homeserver has been blocked by it's administrator."),
|
|
'': _td("This homeserver has exceeded one of its resource limits."),
|
|
},
|
|
);
|
|
const errorDetail = messageForResourceLimitError(
|
|
response.data.limit_type,
|
|
response.data.admin_contact,
|
|
{
|
|
'': _td("Please <a>contact your service administrator</a> to continue using this service."),
|
|
},
|
|
);
|
|
msg = <div>
|
|
<p>{errorTop}</p>
|
|
<p>{errorDetail}</p>
|
|
</div>;
|
|
} else if (response.required_stages && response.required_stages.indexOf('m.login.msisdn') > -1) {
|
|
let msisdnAvailable = false;
|
|
for (const flow of response.available_flows) {
|
|
msisdnAvailable = msisdnAvailable || flow.stages.includes('m.login.msisdn');
|
|
}
|
|
if (!msisdnAvailable) {
|
|
msg = _t('This server does not support authentication with a phone number.');
|
|
}
|
|
} else if (response.errcode === "M_USER_IN_USE") {
|
|
msg = _t("That username already exists, please try another.");
|
|
}
|
|
this.setState({
|
|
busy: false,
|
|
doingUIAuth: false,
|
|
errorText: msg,
|
|
});
|
|
return;
|
|
}
|
|
|
|
MatrixClientPeg.setJustRegisteredUserId(response.user_id);
|
|
|
|
const newState = {
|
|
doingUIAuth: false,
|
|
registeredUsername: response.user_id,
|
|
differentLoggedInUserId: null,
|
|
completedNoSignin: false,
|
|
// we're still busy until we get unmounted: don't show the registration form again
|
|
busy: true,
|
|
};
|
|
|
|
// The user came in through an email validation link. To avoid overwriting
|
|
// their session, check to make sure the session isn't someone else, and
|
|
// isn't a guest user since we'll usually have set a guest user session before
|
|
// starting the registration process. This isn't perfect since it's possible
|
|
// the user had a separate guest session they didn't actually mean to replace.
|
|
const [sessionOwner, sessionIsGuest] = await Lifecycle.getStoredSessionOwner();
|
|
if (sessionOwner && !sessionIsGuest && sessionOwner !== response.userId) {
|
|
console.log(
|
|
`Found a session for ${sessionOwner} but ${response.userId} has just registered.`,
|
|
);
|
|
newState.differentLoggedInUserId = sessionOwner;
|
|
}
|
|
|
|
if (response.access_token) {
|
|
await this.props.onLoggedIn({
|
|
userId: response.user_id,
|
|
deviceId: response.device_id,
|
|
homeserverUrl: this.state.matrixClient.getHomeserverUrl(),
|
|
identityServerUrl: this.state.matrixClient.getIdentityServerUrl(),
|
|
accessToken: response.access_token,
|
|
}, this.state.formVals.password);
|
|
|
|
this.setupPushers();
|
|
} else {
|
|
newState.busy = false;
|
|
newState.completedNoSignin = true;
|
|
}
|
|
|
|
this.setState(newState);
|
|
};
|
|
|
|
private setupPushers() {
|
|
if (!this.props.brand) {
|
|
return Promise.resolve();
|
|
}
|
|
const matrixClient = MatrixClientPeg.get();
|
|
return matrixClient.getPushers().then((resp)=>{
|
|
const pushers = resp.pushers;
|
|
for (let i = 0; i < pushers.length; ++i) {
|
|
if (pushers[i].kind === 'email') {
|
|
const emailPusher = pushers[i];
|
|
emailPusher.data = { brand: this.props.brand };
|
|
matrixClient.setPusher(emailPusher).then(() => {
|
|
console.log("Set email branding to " + this.props.brand);
|
|
}, (error) => {
|
|
console.error("Couldn't set email branding: " + error);
|
|
});
|
|
}
|
|
}
|
|
}, (error) => {
|
|
console.error("Couldn't get pushers: " + error);
|
|
});
|
|
}
|
|
|
|
private onLoginClick = ev => {
|
|
ev.preventDefault();
|
|
ev.stopPropagation();
|
|
this.props.onLoginClick();
|
|
};
|
|
|
|
private onGoToFormClicked = ev => {
|
|
ev.preventDefault();
|
|
ev.stopPropagation();
|
|
this.replaceClient(this.props.serverConfig);
|
|
this.setState({
|
|
busy: false,
|
|
doingUIAuth: false,
|
|
});
|
|
};
|
|
|
|
private makeRegisterRequest = auth => {
|
|
// We inhibit login if we're trying to register with an email address: this
|
|
// avoids a lot of complex race conditions that can occur if we try to log
|
|
// the user in one one or both of the tabs they might end up with after
|
|
// clicking the email link.
|
|
let inhibitLogin = Boolean(this.state.formVals.email);
|
|
|
|
// Only send inhibitLogin if we're sending username / pw params
|
|
// (Since we need to send no params at all to use the ones saved in the
|
|
// session).
|
|
if (!this.state.formVals.password) inhibitLogin = null;
|
|
|
|
const registerParams = {
|
|
username: this.state.formVals.username,
|
|
password: this.state.formVals.password,
|
|
initial_device_display_name: this.props.defaultDeviceDisplayName,
|
|
auth: undefined,
|
|
inhibit_login: undefined,
|
|
};
|
|
if (auth) registerParams.auth = auth;
|
|
if (inhibitLogin !== undefined && inhibitLogin !== null) registerParams.inhibit_login = inhibitLogin;
|
|
return this.state.matrixClient.registerRequest(registerParams);
|
|
};
|
|
|
|
private getUIAuthInputs() {
|
|
return {
|
|
emailAddress: this.state.formVals.email,
|
|
phoneCountry: this.state.formVals.phoneCountry,
|
|
phoneNumber: this.state.formVals.phoneNumber,
|
|
};
|
|
}
|
|
|
|
// Links to the login page shown after registration is completed are routed through this
|
|
// which checks the user hasn't already logged in somewhere else (perhaps we should do
|
|
// this more generally?)
|
|
private onLoginClickWithCheck = async ev => {
|
|
ev.preventDefault();
|
|
|
|
const sessionLoaded = await Lifecycle.loadSession({ignoreGuest: true});
|
|
if (!sessionLoaded) {
|
|
// ok fine, there's still no session: really go to the login page
|
|
this.props.onLoginClick();
|
|
}
|
|
|
|
return sessionLoaded;
|
|
};
|
|
|
|
private renderRegisterComponent() {
|
|
const InteractiveAuth = sdk.getComponent('structures.InteractiveAuth');
|
|
const Spinner = sdk.getComponent('elements.Spinner');
|
|
const RegistrationForm = sdk.getComponent('auth.RegistrationForm');
|
|
|
|
if (this.state.matrixClient && this.state.doingUIAuth) {
|
|
return <InteractiveAuth
|
|
matrixClient={this.state.matrixClient}
|
|
makeRequest={this.makeRegisterRequest}
|
|
onAuthFinished={this.onUIAuthFinished}
|
|
inputs={this.getUIAuthInputs()}
|
|
requestEmailToken={this.requestEmailToken}
|
|
sessionId={this.props.sessionId}
|
|
clientSecret={this.props.clientSecret}
|
|
emailSid={this.props.idSid}
|
|
poll={true}
|
|
/>;
|
|
} else if (!this.state.matrixClient && !this.state.busy) {
|
|
return null;
|
|
} else if (this.state.busy || !this.state.flows) {
|
|
return <div className="mx_AuthBody_spinner">
|
|
<Spinner />
|
|
</div>;
|
|
} else if (this.state.flows.length) {
|
|
let ssoSection;
|
|
if (this.state.ssoFlow) {
|
|
let continueWithSection;
|
|
const providers = this.state.ssoFlow.identity_providers || [];
|
|
// when there is only a single (or 0) providers we show a wide button with `Continue with X` text
|
|
if (providers.length > 1) {
|
|
// i18n: ssoButtons is a placeholder to help translators understand context
|
|
continueWithSection = <h3 className="mx_AuthBody_centered">
|
|
{ _t("Continue with %(ssoButtons)s", { ssoButtons: "" }).trim() }
|
|
</h3>;
|
|
}
|
|
|
|
// i18n: ssoButtons & usernamePassword are placeholders to help translators understand context
|
|
ssoSection = <React.Fragment>
|
|
{ continueWithSection }
|
|
<SSOButtons
|
|
matrixClient={this.loginLogic.createTemporaryClient()}
|
|
flow={this.state.ssoFlow}
|
|
loginType={this.state.ssoFlow.type === "m.login.sso" ? "sso" : "cas"}
|
|
fragmentAfterLogin={this.props.fragmentAfterLogin}
|
|
/>
|
|
<h3 className="mx_AuthBody_centered">
|
|
{ _t("%(ssoButtons)s Or %(usernamePassword)s", { ssoButtons: "", usernamePassword: ""}).trim() }
|
|
</h3>
|
|
</React.Fragment>;
|
|
}
|
|
|
|
return <React.Fragment>
|
|
{ ssoSection }
|
|
<RegistrationForm
|
|
defaultUsername={this.state.formVals.username}
|
|
defaultEmail={this.state.formVals.email}
|
|
defaultPhoneCountry={this.state.formVals.phoneCountry}
|
|
defaultPhoneNumber={this.state.formVals.phoneNumber}
|
|
defaultPassword={this.state.formVals.password}
|
|
onRegisterClick={this.onFormSubmit}
|
|
flows={this.state.flows}
|
|
serverConfig={this.props.serverConfig}
|
|
canSubmit={!this.state.serverErrorIsFatal}
|
|
/>
|
|
</React.Fragment>;
|
|
}
|
|
}
|
|
|
|
render() {
|
|
const AuthHeader = sdk.getComponent('auth.AuthHeader');
|
|
const AuthBody = sdk.getComponent("auth.AuthBody");
|
|
const AccessibleButton = sdk.getComponent('elements.AccessibleButton');
|
|
|
|
let errorText;
|
|
const err = this.state.errorText;
|
|
if (err) {
|
|
errorText = <div className="mx_Login_error">{ err }</div>;
|
|
}
|
|
|
|
let serverDeadSection;
|
|
if (!this.state.serverIsAlive) {
|
|
const classes = classNames({
|
|
"mx_Login_error": true,
|
|
"mx_Login_serverError": true,
|
|
"mx_Login_serverErrorNonFatal": !this.state.serverErrorIsFatal,
|
|
});
|
|
serverDeadSection = (
|
|
<div className={classes}>
|
|
{this.state.serverDeadError}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const signIn = <span className="mx_AuthBody_changeFlow">
|
|
{_t("Already have an account? <a>Sign in here</a>", {}, {
|
|
a: sub => <a onClick={this.onLoginClick} href="#">{ sub }</a>,
|
|
})}
|
|
</span>;
|
|
|
|
// Only show the 'go back' button if you're not looking at the form
|
|
let goBack;
|
|
if (this.state.doingUIAuth) {
|
|
goBack = <a className="mx_AuthBody_changeFlow" onClick={this.onGoToFormClicked} href="#">
|
|
{ _t('Go back') }
|
|
</a>;
|
|
}
|
|
|
|
let body;
|
|
if (this.state.completedNoSignin) {
|
|
let regDoneText;
|
|
if (this.state.differentLoggedInUserId) {
|
|
regDoneText = <div>
|
|
<p>{_t(
|
|
"Your new account (%(newAccountId)s) is registered, but you're already " +
|
|
"logged into a different account (%(loggedInUserId)s).", {
|
|
newAccountId: this.state.registeredUsername,
|
|
loggedInUserId: this.state.differentLoggedInUserId,
|
|
},
|
|
)}</p>
|
|
<p><AccessibleButton element="span" className="mx_linkButton" onClick={async event => {
|
|
const sessionLoaded = await this.onLoginClickWithCheck(event);
|
|
if (sessionLoaded) {
|
|
dis.dispatch({action: "view_welcome_page"});
|
|
}
|
|
}}>
|
|
{_t("Continue with previous account")}
|
|
</AccessibleButton></p>
|
|
</div>;
|
|
} else if (this.state.formVals.password) {
|
|
// We're the client that started the registration
|
|
regDoneText = <h3>{_t(
|
|
"<a>Log in</a> to your new account.", {},
|
|
{
|
|
a: (sub) => <a href="#/login" onClick={this.onLoginClickWithCheck}>{sub}</a>,
|
|
},
|
|
)}</h3>;
|
|
} else {
|
|
// We're not the original client: the user probably got to us by clicking the
|
|
// email validation link. We can't offer a 'go straight to your account' link
|
|
// as we don't have the original creds.
|
|
regDoneText = <h3>{_t(
|
|
"You can now close this window or <a>log in</a> to your new account.", {},
|
|
{
|
|
a: (sub) => <a href="#/login" onClick={this.onLoginClickWithCheck}>{sub}</a>,
|
|
},
|
|
)}</h3>;
|
|
}
|
|
body = <div>
|
|
<h2>{_t("Registration Successful")}</h2>
|
|
{ regDoneText }
|
|
</div>;
|
|
} else {
|
|
body = <div>
|
|
<h2>{ _t('Create account') }</h2>
|
|
{ errorText }
|
|
{ serverDeadSection }
|
|
<ServerPicker
|
|
title={_t("Host account on")}
|
|
dialogTitle={_t("Decide where your account is hosted")}
|
|
serverConfig={this.props.serverConfig}
|
|
onServerConfigChange={this.state.doingUIAuth ? undefined : this.props.onServerConfigChange}
|
|
/>
|
|
{ this.renderRegisterComponent() }
|
|
{ goBack }
|
|
{ signIn }
|
|
</div>;
|
|
}
|
|
|
|
return (
|
|
<AuthPage>
|
|
<AuthHeader />
|
|
<AuthBody>
|
|
{ body }
|
|
</AuthBody>
|
|
</AuthPage>
|
|
);
|
|
}
|
|
}
|