2016-11-22 14:19:08 -06:00
|
|
|
import React from 'react';
|
|
|
|
import lbry from '../lbry.js';
|
|
|
|
import {Link} from '../component/link.js';
|
|
|
|
import Modal from '../component/modal.js';
|
|
|
|
import {Address, BusyMessage, CreditAmount} from '../component/common.js';
|
|
|
|
|
|
|
|
|
2016-09-01 02:40:23 -04:00
|
|
|
var addressRefreshButtonStyle = {
|
|
|
|
fontSize: '11pt',
|
|
|
|
};
|
|
|
|
var AddressSection = React.createClass({
|
2016-10-19 03:13:39 -04:00
|
|
|
_refreshAddress: function(event) {
|
|
|
|
if (typeof event !== 'undefined') {
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
|
2016-09-01 02:40:23 -04:00
|
|
|
lbry.getNewAddress((address) => {
|
2016-11-22 14:28:16 -06:00
|
|
|
window.localStorage.setItem('wallet_address', address);
|
2016-07-18 17:40:15 -05:00
|
|
|
this.setState({
|
2016-09-01 02:40:23 -04:00
|
|
|
address: address,
|
|
|
|
});
|
2016-07-18 17:40:15 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
getInitialState: function() {
|
|
|
|
return {
|
2016-09-01 02:40:23 -04:00
|
|
|
address: null,
|
2016-10-21 03:56:37 -04:00
|
|
|
modal: null,
|
2016-09-01 02:40:23 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
componentWillMount: function() {
|
2016-11-22 14:28:16 -06:00
|
|
|
var address = window.localStorage.getItem('wallet_address');
|
2016-09-01 02:40:23 -04:00
|
|
|
if (address === null) {
|
2016-09-02 01:46:27 -04:00
|
|
|
this._refreshAddress();
|
2016-09-01 02:40:23 -04:00
|
|
|
} else {
|
2016-09-02 01:46:27 -04:00
|
|
|
lbry.checkAddressIsMine(address, (isMine) => {
|
|
|
|
if (isMine) {
|
|
|
|
this.setState({
|
|
|
|
address: address,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
this._refreshAddress();
|
|
|
|
}
|
2016-09-01 02:40:23 -04:00
|
|
|
});
|
2016-07-18 17:40:15 -05:00
|
|
|
}
|
|
|
|
},
|
|
|
|
render: function() {
|
|
|
|
return (
|
2016-08-07 20:20:14 -04:00
|
|
|
<section className="card">
|
2016-09-01 02:40:23 -04:00
|
|
|
<h3>Wallet Address</h3>
|
2016-10-19 03:13:39 -04:00
|
|
|
<Address address={this.state.address} /> <Link text="Get new address" icon='icon-refresh' onClick={this._refreshAddress} style={addressRefreshButtonStyle} />
|
|
|
|
<input type='submit' className='hidden' />
|
2016-09-01 02:40:23 -04:00
|
|
|
<div className="help">
|
|
|
|
<p>Other LBRY users may send credits to you by entering this address on the "Send" page.</p>
|
|
|
|
You can generate a new address at any time, and any previous addresses will continue to work. Using multiple addresses can be helpful for keeping track of incoming payments from multiple sources.
|
|
|
|
</div>
|
2016-08-03 04:07:36 -04:00
|
|
|
</section>
|
2016-07-18 17:40:15 -05:00
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
var SendToAddressSection = React.createClass({
|
2016-10-19 03:13:39 -04:00
|
|
|
handleSubmit: function(event) {
|
|
|
|
if (typeof event !== 'undefined') {
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
|
2016-08-02 08:35:18 -05:00
|
|
|
if ((this.state.balance - this.state.amount) < 1)
|
2016-08-02 05:34:30 -04:00
|
|
|
{
|
2016-10-21 03:56:37 -04:00
|
|
|
this.setState({
|
|
|
|
modal: 'insufficientBalance',
|
|
|
|
});
|
2016-08-02 05:34:30 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-07-18 17:40:15 -05:00
|
|
|
this.setState({
|
|
|
|
results: "",
|
|
|
|
});
|
|
|
|
|
2016-07-18 18:18:16 -05:00
|
|
|
lbry.sendToAddress(this.state.amount, this.state.address, (results) => {
|
2016-07-18 17:40:15 -05:00
|
|
|
if(results === true)
|
|
|
|
{
|
|
|
|
this.setState({
|
2016-07-22 10:59:20 -05:00
|
|
|
results: "Your transaction was successfully placed in the queue.",
|
2016-07-18 17:40:15 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.setState({
|
|
|
|
results: "Something went wrong: " + results
|
|
|
|
});
|
|
|
|
}
|
2016-07-22 10:33:05 -05:00
|
|
|
}, (error) => {
|
|
|
|
this.setState({
|
|
|
|
results: "Something went wrong: " + error.faultString + " " + error.faultCode
|
|
|
|
})
|
2016-07-18 17:40:15 -05:00
|
|
|
});
|
|
|
|
},
|
2016-10-21 03:56:37 -04:00
|
|
|
closeModal: function() {
|
|
|
|
this.setState({
|
|
|
|
modal: null,
|
|
|
|
});
|
|
|
|
},
|
2016-07-18 17:40:15 -05:00
|
|
|
getInitialState: function() {
|
|
|
|
return {
|
|
|
|
address: "",
|
|
|
|
amount: 0.0,
|
2016-08-22 11:16:43 -05:00
|
|
|
balance: <BusyMessage message="Checking balance" />,
|
2016-07-18 17:40:15 -05:00
|
|
|
results: "",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
componentWillMount: function() {
|
|
|
|
lbry.getBalance((results) => {
|
|
|
|
this.setState({
|
|
|
|
balance: results,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
setAmount: function(event) {
|
|
|
|
this.setState({
|
2016-07-22 10:33:05 -05:00
|
|
|
amount: parseFloat(event.target.value),
|
2016-07-18 17:40:15 -05:00
|
|
|
})
|
|
|
|
},
|
|
|
|
setAddress: function(event) {
|
|
|
|
this.setState({
|
|
|
|
address: event.target.value,
|
|
|
|
})
|
|
|
|
},
|
|
|
|
render: function() {
|
|
|
|
return (
|
2016-08-07 20:20:14 -04:00
|
|
|
<section className="card">
|
2016-10-19 03:13:39 -04:00
|
|
|
<form onSubmit={this.handleSubmit}>
|
|
|
|
<h3>Send Credits</h3>
|
2016-08-07 23:31:21 -04:00
|
|
|
<div className="form-row">
|
2016-10-19 03:13:39 -04:00
|
|
|
<label htmlFor="amount">Amount</label>
|
|
|
|
<input id="amount" type="text" size="10" onChange={this.setAmount}></input>
|
2016-08-07 23:31:21 -04:00
|
|
|
</div>
|
2016-10-19 03:13:39 -04:00
|
|
|
<div className="form-row">
|
|
|
|
<label htmlFor="address">Recipient address</label>
|
|
|
|
<input id="address" type="text" size="60" onChange={this.setAddress}></input>
|
|
|
|
</div>
|
|
|
|
<div className="form-row form-row-submit">
|
|
|
|
<Link button="primary" label="Send" onClick={this.handleSubmit} disabled={!(parseFloat(this.state.amount) > 0.0) || this.state.address == ""} />
|
|
|
|
<input type='submit' className='hidden' />
|
|
|
|
</div>
|
|
|
|
{
|
|
|
|
this.state.results ?
|
|
|
|
<div className="form-row">
|
|
|
|
<h4>Results</h4>
|
|
|
|
{this.state.results}
|
|
|
|
</div>
|
|
|
|
: ''
|
|
|
|
}
|
|
|
|
</form>
|
2017-01-13 17:05:09 -05:00
|
|
|
<Modal isOpen={this.state.modal === 'insufficientBalance'} contentLabel="Insufficient balance"
|
|
|
|
onConfirmed={this.closeModal}>
|
2016-10-21 03:56:37 -04:00
|
|
|
Insufficient balance: after this transaction you would have less than 1 LBC in your wallet.
|
|
|
|
</Modal>
|
2016-08-07 23:31:21 -04:00
|
|
|
</section>
|
2016-07-18 17:40:15 -05:00
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2016-08-27 10:12:56 -04:00
|
|
|
|
|
|
|
var TransactionList = React.createClass({
|
2016-08-07 20:20:14 -04:00
|
|
|
getInitialState: function() {
|
|
|
|
return {
|
2016-08-27 10:12:56 -04:00
|
|
|
transactionItems: null,
|
2016-08-07 20:20:14 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
componentWillMount: function() {
|
2016-08-20 14:52:03 -05:00
|
|
|
lbry.call('get_transaction_history', {}, (results) => {
|
|
|
|
if (results.length == 0) {
|
2016-08-27 10:12:56 -04:00
|
|
|
this.setState({ transactionItems: [] })
|
2016-08-20 14:52:03 -05:00
|
|
|
} else {
|
2016-08-27 10:12:56 -04:00
|
|
|
var transactionItems = [],
|
|
|
|
condensedTransactions = {};
|
2016-08-20 14:52:03 -05:00
|
|
|
results.forEach(function(tx) {
|
2016-08-20 16:52:31 -05:00
|
|
|
var txid = tx["txid"];
|
|
|
|
if (!(txid in condensedTransactions)) {
|
|
|
|
condensedTransactions[txid] = 0;
|
|
|
|
}
|
2016-09-05 13:20:34 -04:00
|
|
|
condensedTransactions[txid] += parseFloat(tx["value"]);
|
2016-08-20 14:52:03 -05:00
|
|
|
});
|
2016-08-24 00:08:58 -05:00
|
|
|
results.reverse().forEach(function(tx) {
|
2016-08-20 16:52:31 -05:00
|
|
|
var txid = tx["txid"];
|
2016-09-03 15:28:05 -05:00
|
|
|
if (condensedTransactions[txid] && condensedTransactions[txid] != 0)
|
2016-08-27 10:12:56 -04:00
|
|
|
{
|
|
|
|
transactionItems.push({
|
|
|
|
id: txid,
|
2016-11-11 05:09:43 -05:00
|
|
|
date: tx["timestamp"] ? (new Date(parseInt(tx["timestamp"]) * 1000)) : null,
|
2016-08-27 10:12:56 -04:00
|
|
|
amount: condensedTransactions[txid]
|
|
|
|
});
|
|
|
|
delete condensedTransactions[txid];
|
2016-08-20 16:52:31 -05:00
|
|
|
}
|
|
|
|
});
|
2016-08-27 10:12:56 -04:00
|
|
|
|
|
|
|
this.setState({ transactionItems: transactionItems });
|
2016-08-20 14:52:03 -05:00
|
|
|
}
|
2016-08-27 10:12:56 -04:00
|
|
|
});
|
|
|
|
},
|
|
|
|
render: function() {
|
|
|
|
var rows = [];
|
|
|
|
if (this.state.transactionItems && this.state.transactionItems.length > 0)
|
|
|
|
{
|
|
|
|
this.state.transactionItems.forEach(function(item) {
|
|
|
|
rows.push(
|
|
|
|
<tr key={item.id}>
|
|
|
|
<td>{ (item.amount > 0 ? '+' : '' ) + item.amount }</td>
|
2016-11-11 05:09:43 -05:00
|
|
|
<td>{ item.date ? item.date.toLocaleDateString() : <span className="empty">(Transaction pending)</span> }</td>
|
|
|
|
<td>{ item.date ? item.date.toLocaleTimeString() : <span className="empty">(Transaction pending)</span> }</td>
|
2016-08-27 10:12:56 -04:00
|
|
|
<td>
|
|
|
|
<a className="button-text" href={"https://explorer.lbry.io/tx/"+item.id} target="_blank">{item.id.substr(0, 7)}</a>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<section className="card">
|
|
|
|
<h3>Transaction History</h3>
|
|
|
|
{ this.state.transactionItems === null ? <BusyMessage message="Loading transactions" /> : '' }
|
|
|
|
{ this.state.transactionItems && rows.length === 0 ? <div className="empty">You have no transactions.</div> : '' }
|
|
|
|
{ this.state.transactionItems && rows.length > 0 ?
|
|
|
|
<table className="table-standard table-stretch">
|
|
|
|
<thead>
|
|
|
|
<tr>
|
|
|
|
<th>Amount</th>
|
|
|
|
<th>Date</th>
|
|
|
|
<th>Time</th>
|
|
|
|
<th>Transaction</th>
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
{rows}
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
: ''
|
|
|
|
}
|
|
|
|
</section>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
var WalletPage = React.createClass({
|
|
|
|
propTypes: {
|
|
|
|
viewingPage: React.PropTypes.string,
|
|
|
|
},
|
|
|
|
componentDidMount: function() {
|
|
|
|
document.title = "My Wallet";
|
|
|
|
},
|
|
|
|
/*
|
|
|
|
Below should be refactored so that balance is shared all of wallet page. Or even broader?
|
|
|
|
What is the proper React pattern for sharing a global state like balance?
|
|
|
|
*/
|
|
|
|
getInitialState: function() {
|
|
|
|
return {
|
|
|
|
balance: null,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
componentWillMount: function() {
|
|
|
|
lbry.getBalance((results) => {
|
2016-08-20 14:52:03 -05:00
|
|
|
this.setState({
|
2016-08-27 10:12:56 -04:00
|
|
|
balance: results,
|
2016-08-20 14:52:03 -05:00
|
|
|
})
|
2016-08-07 20:20:14 -04:00
|
|
|
});
|
|
|
|
},
|
2016-07-18 17:40:15 -05:00
|
|
|
render: function() {
|
|
|
|
return (
|
|
|
|
<main className="page">
|
2016-08-07 20:20:14 -04:00
|
|
|
<section className="card">
|
|
|
|
<h3>Balance</h3>
|
2016-08-27 10:12:56 -04:00
|
|
|
{ this.state.balance === null ? <BusyMessage message="Checking balance" /> : ''}
|
|
|
|
{ this.state.balance !== null ? <CreditAmount amount={this.state.balance} precision={8} /> : '' }
|
2016-08-20 14:52:03 -05:00
|
|
|
</section>
|
2016-08-27 10:12:56 -04:00
|
|
|
{ this.props.viewingPage === 'wallet' ? <TransactionList /> : '' }
|
|
|
|
{ this.props.viewingPage === 'send' ? <SendToAddressSection /> : '' }
|
2016-09-01 02:40:23 -04:00
|
|
|
{ this.props.viewingPage === 'receive' ? <AddressSection /> : '' }
|
2016-07-18 17:40:15 -05:00
|
|
|
</main>
|
|
|
|
);
|
|
|
|
}
|
2016-08-22 11:16:43 -05:00
|
|
|
});
|
2016-11-22 14:19:08 -06:00
|
|
|
|
|
|
|
export default WalletPage;
|