lbry-desktop/ui/js/component/auth.js

328 lines
10 KiB
JavaScript
Raw Normal View History

import React from "react";
2017-05-25 00:25:22 +02:00
import lbry from "../lbry.js";
import lbryio from "../lbryio.js";
import Modal from "./modal.js";
import ModalPage from "./modal-page.js";
2017-05-04 05:44:08 +02:00
import Link from "component/link"
import {RewardLink} from 'component/reward-link';
import {FormRow} from "../component/form.js";
import {CreditAmount, Address} from "../component/common.js";
2017-05-11 19:46:41 +02:00
import {getLocal, setLocal} from '../utils.js';
2017-05-25 19:38:53 +02:00
import rewards from '../rewards'
2017-04-09 17:06:23 +02:00
2017-05-17 10:10:25 +02:00
class SubmitEmailStage extends React.Component {
constructor(props) {
super(props);
this.state = {
2017-04-09 17:06:23 +02:00
rewardType: null,
email: '',
submitting: false
};
2017-05-17 10:10:25 +02:00
}
handleEmailChanged(event) {
2017-04-09 17:06:23 +02:00
this.setState({
email: event.target.value,
});
2017-05-17 10:10:25 +02:00
}
onEmailSaved(email) {
this.props.setStage("confirm", { email: email })
2017-05-17 10:10:25 +02:00
}
handleSubmit(event) {
2017-04-09 17:06:23 +02:00
event.preventDefault();
this.setState({
submitting: true,
});
lbryio.call('user_email', 'new', {email: this.state.email}, 'post').then(() => {
this.onEmailSaved(this.state.email);
2017-04-09 17:06:23 +02:00
}, (error) => {
2017-06-01 17:49:07 +02:00
if (error.xhr && (error.xhr.status == 409 || error.message == __("This email is already in use"))) {
this.onEmailSaved(this.state.email);
2017-04-18 21:45:15 +02:00
return;
} else if (this._emailRow) {
2017-04-10 14:32:40 +02:00
this._emailRow.showError(error.message)
2017-04-09 17:06:23 +02:00
}
this.setState({ submitting: false });
});
2017-05-17 10:10:25 +02:00
}
render() {
2017-04-09 17:06:23 +02:00
return (
<section>
<form onSubmit={(event) => { this.handleSubmit(event) }}>
<FormRow ref={(ref) => { this._emailRow = ref }} type="text" label={__("Email")} placeholder="scrwvwls@lbry.io"
2017-04-09 17:06:23 +02:00
name="email" value={this.state.email}
onChange={(event) => { this.handleEmailChanged(event) }} />
2017-04-10 14:32:40 +02:00
<div className="form-row-submit">
<Link button="primary" label={__("Next")} disabled={this.state.submitting} onClick={(event) => { this.handleSubmit(event) }} />
2017-04-09 17:06:23 +02:00
</div>
</form>
</section>
);
}
2017-05-17 10:10:25 +02:00
}
2017-04-09 17:06:23 +02:00
2017-05-17 10:10:25 +02:00
class ConfirmEmailStage extends React.Component {
constructor(props) {
super(props);
this.state = {
2017-04-09 17:06:23 +02:00
rewardType: null,
code: '',
submitting: false,
errorMessage: null,
};
2017-05-17 10:10:25 +02:00
}
handleCodeChanged(event) {
2017-04-09 17:06:23 +02:00
this.setState({
code: event.target.value,
});
2017-05-17 10:10:25 +02:00
}
handleSubmit(event) {
2017-04-09 17:06:23 +02:00
event.preventDefault();
this.setState({
submitting: true,
});
const onSubmitError = (error) => {
2017-04-10 14:32:40 +02:00
if (this._codeRow) {
this._codeRow.showError(error.message)
2017-04-09 17:06:23 +02:00
}
2017-04-10 14:32:40 +02:00
this.setState({ submitting: false });
};
2017-04-10 14:32:40 +02:00
2017-04-18 22:51:00 +02:00
lbryio.call('user_email', 'confirm', {verification_token: this.state.code, email: this.props.email}, 'post').then((userEmail) => {
2017-05-11 19:46:41 +02:00
if (userEmail.is_verified) {
this.props.setStage("welcome")
2017-04-10 14:32:40 +02:00
} else {
onSubmitError(new Error(__("Your email is still not verified."))) //shouldn't happen?
2017-04-10 14:32:40 +02:00
}
}, onSubmitError);
2017-05-17 10:10:25 +02:00
}
render() {
2017-04-09 17:06:23 +02:00
return (
<section>
<form onSubmit={(event) => { this.handleSubmit(event) }}>
<FormRow label={__("Verification Code")} ref={(ref) => { this._codeRow = ref }} type="text"
name="code" placeholder="a94bXXXXXXXXXXXXXX" value={this.state.code} onChange={(event) => { this.handleCodeChanged(event) }}
helper={__("A verification code is required to access this version.")}/>
<div className="form-row-submit form-row-submit--with-footer">
<Link button="primary" label={__("Verify")} disabled={this.state.submitting} onClick={(event) => { this.handleSubmit(event)}} />
2017-04-09 17:06:23 +02:00
</div>
<div className="form-field__helper">
{__("No code?")} <Link onClick={() => { this.props.setStage("nocode")}} label={__("Click here")} />.
</div>
2017-04-09 17:06:23 +02:00
</form>
</section>
);
}
2017-05-17 10:10:25 +02:00
}
class WelcomeStage extends React.Component {
2017-05-19 18:17:19 +02:00
static propTypes = {
endAuth: React.PropTypes.func,
}
2017-05-17 10:10:25 +02:00
constructor(props) {
super(props);
this.state = {
2017-04-12 22:23:20 +02:00
hasReward: false,
rewardAmount: null,
2017-05-17 10:10:25 +02:00
};
}
onRewardClaim(reward) {
2017-04-12 22:23:20 +02:00
this.setState({
hasReward: true,
rewardAmount: reward.amount
2017-04-12 22:23:20 +02:00
})
2017-05-17 10:10:25 +02:00
}
render() {
2017-04-10 14:32:40 +02:00
return (
2017-04-12 22:23:20 +02:00
!this.state.hasReward ?
<Modal type="custom" isOpen={true} contentLabel={__("Welcome to LBRY")} {...this.props}>
2017-04-12 22:23:20 +02:00
<section>
<h3 className="modal__header">{__("Welcome to LBRY.")}</h3>
<p>{__("Using LBRY is like dating a centaur. Totally normal up top, and way different underneath.")}</p>
<p>{__("Up top, LBRY is similar to popular media sites.")}</p>
<p>{__("Below, LBRY is controlled by users -- you -- via blockchain and decentralization.")}</p>
<p>{__("Thank you for making content freedom possible! Here's a nickel, kid.")}</p>
2017-04-12 22:23:20 +02:00
<div style={{textAlign: "center", marginBottom: "12px"}}>
<RewardLink type="new_user" button="primary" onRewardClaim={(event) => { this.onRewardClaim(event) }} onRewardFailure={() => this.props.setStage(null)} onConfirmed={() => { this.props.setStage(null) }} />
2017-04-12 22:23:20 +02:00
</div>
</section>
</Modal> :
<Modal type="alert" overlayClassName="modal-overlay modal-overlay--clear" isOpen={true} contentLabel={__("Welcome to LBRY")} {...this.props} onConfirmed={() => { this.props.setStage(null) }}>
2017-04-12 22:23:20 +02:00
<section>
<h3 className="modal__header">{__("About Your Reward")}</h3>
<p>{__("You earned a reward of ")} <CreditAmount amount={this.state.rewardAmount} label={false} /> {__("LBRY credits, or \"LBC\".")}</p>
<p>{__("This reward will show in your Wallet momentarily, probably while you are reading this message.")}</p>
<p>{__("LBC is used to compensate creators, to publish, and to have say in how the network works.")}</p>
<p>{__("No need to understand it all just yet! Try watching or downloading something next.")}</p>
<p>{__("Finally, know that LBRY is an early beta and that it earns the name.")}</p>
2017-04-12 22:23:20 +02:00
</section>
</Modal>
2017-04-10 14:32:40 +02:00
);
}
2017-05-17 10:10:25 +02:00
}
2017-04-10 14:32:40 +02:00
2017-05-19 18:17:19 +02:00
const ErrorStage = (props) => {
return <section>
<p>{__("An error was encountered that we cannot continue from.")}</p>
<p>{__("At least we're earning the name beta.")}</p>
{ props.errorText ? <p>{__("Message:")} {props.errorText}</p> : '' }
<Link button="alt" label={__("Try Reload")} onClick={() => { window.location.reload() } } />
2017-05-19 18:17:19 +02:00
</section>
2017-05-17 10:10:25 +02:00
}
2017-04-09 17:06:23 +02:00
2017-05-19 18:17:19 +02:00
const PendingStage = (props) => {
return <section>
<p>{__("Preparing for first access")} <span className="busy-indicator"></span></p>
2017-05-19 18:17:19 +02:00
</section>
2017-05-17 10:10:25 +02:00
}
2017-04-09 17:06:23 +02:00
2017-05-17 10:10:25 +02:00
class CodeRequiredStage extends React.Component {
constructor(props) {
super(props);
2017-05-25 00:25:22 +02:00
this._balanceSubscribeId = null
2017-05-17 10:10:25 +02:00
this.state = {
balance: 0,
address: getLocal('wallet_address')
2017-05-17 10:10:25 +02:00
};
}
2017-05-17 10:10:25 +02:00
componentWillMount() {
this._balanceSubscribeId = lbry.balanceSubscribe((balance) => {
this.setState({
balance: balance
});
})
if (!this.state.address) {
lbry.wallet_unused_address().then((address) => {
setLocal('wallet_address', address);
this.setState({ address: address });
});
}
2017-05-17 10:10:25 +02:00
}
componentWillUnmount() {
if (this._balanceSubscribeId) {
lbry.balanceUnsubscribe(this._balanceSubscribeId)
}
2017-05-17 10:10:25 +02:00
}
render() {
const disabled = this.state.balance < 1;
return (
<div>
<section className="section-spaced">
<p>{__("Access to LBRY is restricted as we build and scale the network.")}</p>
<p>{__("There are two ways in:")}</p>
<h3>{__("Own LBRY Credits")}</h3>
<p>{__("If you own at least 1 LBC, you can get in right now.")}</p>
<p style={{ textAlign: "center"}}><Link onClick={() => { setLocal('auth_bypassed', true); this.props.setStage(null); }}
disabled={disabled} label={__("Let Me In")} button={ disabled ? "alt" : "primary" } /></p>
<p>{__("Your balance is ")}<CreditAmount amount={this.state.balance} />. {__("To increase your balance, send credits to this address:")}</p>
<p><Address address={ this.state.address ? this.state.address : __("Generating Address...") } /></p>
<p>{__("If you don't understand how to send credits, then...")}</p>
</section>
<section>
<h3>{__("Wait For A Code")}</h3>
<p>{__("If you provide your email, you'll automatically receive a notification when the system is open.")}</p>
<p><Link onClick={() => { this.props.setStage("email"); }} label={__("Return")} /></p>
</section>
</div>
);
}
2017-05-17 10:10:25 +02:00
}
export class AuthOverlay extends React.Component {
constructor(props) {
super(props);
this._stages = {
pending: PendingStage,
error: ErrorStage,
nocode: CodeRequiredStage,
email: SubmitEmailStage,
confirm: ConfirmEmailStage,
welcome: WelcomeStage
}
this.state = {
2017-04-18 21:45:15 +02:00
stage: "pending",
2017-04-09 17:06:23 +02:00
stageProps: {}
};
2017-05-17 10:10:25 +02:00
}
setStage(stage, stageProps = {}) {
2017-04-10 14:32:40 +02:00
this.setState({
stage: stage,
stageProps: stageProps
})
2017-05-17 10:10:25 +02:00
}
componentWillMount() {
lbryio.authenticate().then((user) => {
2017-05-11 19:46:41 +02:00
if (!user.has_verified_email) {
if (getLocal('auth_bypassed')) {
this.setStage(null)
} else {
this.setStage("email", {})
}
2017-04-10 20:12:07 +02:00
} else {
lbryio.call('reward', 'list', {}).then((userRewards) => {
2017-04-12 22:23:20 +02:00
userRewards.filter(function(reward) {
2017-05-25 19:38:53 +02:00
return reward.reward_type == rewards.TYPE_NEW_USER && reward.transaction_id;
2017-04-12 22:23:20 +02:00
}).length ?
this.setStage(null) :
this.setStage("welcome")
});
2017-04-10 20:12:07 +02:00
}
}).catch((err) => {
this.setStage("error", { errorText: err.message })
2017-04-10 20:12:07 +02:00
document.dispatchEvent(new CustomEvent('unhandledError', {
detail: {
message: err.message,
data: err.stack
}
}));
})
2017-05-17 10:10:25 +02:00
}
render() {
2017-04-12 22:23:20 +02:00
if (!this.state.stage) {
2017-04-12 18:59:43 +02:00
return null;
2017-04-09 17:06:23 +02:00
}
const StageContent = this._stages[this.state.stage];
2017-05-19 18:17:19 +02:00
if (!StageContent) {
return <span className="empty">{__("Unknown authentication step.")}</span>
2017-05-19 18:17:19 +02:00
}
2017-04-09 17:06:23 +02:00
return (
2017-04-10 14:32:40 +02:00
this.state.stage != "welcome" ?
<ModalPage className="modal-page--full" isOpen={true} contentLabel={__("Authentication")}>
<h1>{__("LBRY Early Access")}</h1>
<StageContent {...this.state.stageProps} setStage={(stage, stageProps) => { this.setStage(stage, stageProps) }} />
2017-04-10 14:32:40 +02:00
</ModalPage> :
<StageContent setStage={(stage, stageProps) => { this.setStage(stage, stageProps) }} {...this.state.stageProps} />
2017-04-09 17:06:23 +02:00
);
}
}