wallet-core/src/pages/tree.tsx

426 lines
11 KiB
TypeScript
Raw Normal View History

2016-10-12 02:55:53 +02:00
/*
This file is part of TALER
(C) 2016 Inria
TALER is free software; you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation; either version 3, or (at your option) any later version.
TALER is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
*/
/**
* Show contents of the wallet as a tree.
*
* @author Florian Dold
*/
import {ExchangeRecord, DenominationRecord, CoinStatus} from "src/types";
2016-11-15 15:07:17 +01:00
import { ReserveRecord, CoinRecord, PreCoinRecord, Denomination } from "src/types";
2016-11-13 23:30:18 +01:00
import { ImplicitStateComponent, StateHolder } from "src/components";
2016-10-13 02:36:33 +02:00
import {
getReserves, getExchanges, getCoins, getPreCoins,
refresh, getDenoms
2016-11-13 23:30:18 +01:00
} from "src/wxApi";
2016-11-19 15:59:17 +01:00
import { prettyAmount } from "src/renderHtml";
2016-11-13 23:30:18 +01:00
import { getTalerStampDate } from "src/helpers";
2016-10-12 02:55:53 +02:00
interface ReserveViewProps {
2016-10-13 02:23:24 +02:00
reserve: ReserveRecord;
2016-10-12 02:55:53 +02:00
}
class ReserveView extends React.Component<ReserveViewProps, void> {
2016-10-12 02:55:53 +02:00
render(): JSX.Element {
2016-10-13 02:23:24 +02:00
let r: ReserveRecord = this.props.reserve;
2016-10-12 02:55:53 +02:00
return (
<div className="tree-item">
<ul>
<li>Key: {r.reserve_pub}</li>
<li>Created: {(new Date(r.created * 1000).toString())}</li>
<li>Current: {r.current_amount ? prettyAmount(r.current_amount!) : "null"}</li>
<li>Requested: {prettyAmount(r.requested_amount)}</li>
<li>Confirmed: {r.confirmed}</li>
2016-10-12 02:55:53 +02:00
</ul>
</div>
);
}
}
interface ReserveListProps {
exchangeBaseUrl: string;
}
interface ToggleProps {
expanded: StateHolder<boolean>;
}
class Toggle extends ImplicitStateComponent<ToggleProps> {
renderButton() {
let show = () => {
this.props.expanded(true);
this.setState({});
};
let hide = () => {
this.props.expanded(false);
this.setState({});
};
if (this.props.expanded()) {
return <button onClick={hide}>hide</button>;
}
return <button onClick={show}>show</button>;
}
render() {
return (
<div style={{display: "inline"}}>
2016-10-12 02:55:53 +02:00
{this.renderButton()}
{this.props.expanded() ? this.props.children : []}
</div>);
}
}
interface CoinViewProps {
2016-11-15 15:07:17 +01:00
coin: CoinRecord;
2016-10-12 02:55:53 +02:00
}
2016-10-12 23:30:10 +02:00
interface RefreshDialogProps {
2016-11-15 15:07:17 +01:00
coin: CoinRecord;
2016-10-12 23:30:10 +02:00
}
class RefreshDialog extends ImplicitStateComponent<RefreshDialogProps> {
refreshRequested = this.makeState<boolean>(false);
render(): JSX.Element {
if (!this.refreshRequested()) {
return (
<div style={{display: "inline"}}>
2016-10-12 23:30:10 +02:00
<button onClick={() => this.refreshRequested(true)}>refresh</button>
</div>
);
}
return (
<div>
Refresh amount: <input type="text" size={10} />
2016-10-13 02:36:33 +02:00
<button onClick={() => refresh(this.props.coin.coinPub)}>ok</button>
2016-10-12 23:30:10 +02:00
<button onClick={() => this.refreshRequested(false)}>cancel</button>
</div>
);
}
}
class CoinView extends React.Component<CoinViewProps, void> {
2016-10-12 02:55:53 +02:00
render() {
let c = this.props.coin;
return (
<div className="tree-item">
<ul>
<li>Key: {c.coinPub}</li>
<li>Current amount: {prettyAmount(c.currentAmount)}</li>
2016-11-19 15:59:17 +01:00
<li>Denomination: <ExpanderText text={c.denomPub} /></li>
2016-10-12 02:55:53 +02:00
<li>Suspended: {(c.suspended || false).toString()}</li>
<li>Status: {CoinStatus[c.status]}</li>
2016-10-12 23:30:10 +02:00
<li><RefreshDialog coin={c} /></li>
2016-10-12 02:55:53 +02:00
</ul>
</div>
);
}
}
interface PreCoinViewProps {
2016-11-15 15:07:17 +01:00
precoin: PreCoinRecord;
2016-10-12 02:55:53 +02:00
}
class PreCoinView extends React.Component<PreCoinViewProps, void> {
2016-10-12 02:55:53 +02:00
render() {
let c = this.props.precoin;
return (
<div className="tree-item">
<ul>
<li>Key: {c.coinPub}</li>
</ul>
</div>
);
}
}
interface CoinListProps {
exchangeBaseUrl: string;
}
class CoinList extends ImplicitStateComponent<CoinListProps> {
2016-11-15 15:07:17 +01:00
coins = this.makeState<CoinRecord[] | null>(null);
2016-10-12 02:55:53 +02:00
expanded = this.makeState<boolean>(false);
constructor(props: CoinListProps) {
super(props);
this.update(props);
2016-10-12 02:55:53 +02:00
}
async update(props: CoinListProps) {
let coins = await getCoins(props.exchangeBaseUrl);
2016-10-12 02:55:53 +02:00
this.coins(coins);
}
componentWillReceiveProps(newProps: CoinListProps) {
this.update(newProps);
}
2016-10-12 02:55:53 +02:00
render(): JSX.Element {
if (!this.coins()) {
return <div>...</div>;
}
return (
<div className="tree-item">
Coins ({this.coins() !.length.toString()})
{" "}
<Toggle expanded={this.expanded}>
{this.coins() !.map((c) => <CoinView coin={c} />)}
</Toggle>
</div>
);
}
}
interface PreCoinListProps {
exchangeBaseUrl: string;
}
class PreCoinList extends ImplicitStateComponent<PreCoinListProps> {
2016-11-15 15:07:17 +01:00
precoins = this.makeState<PreCoinRecord[] | null>(null);
2016-10-12 02:55:53 +02:00
expanded = this.makeState<boolean>(false);
constructor(props: PreCoinListProps) {
super(props);
this.update();
}
async update() {
let precoins = await getPreCoins(this.props.exchangeBaseUrl);
this.precoins(precoins);
}
render(): JSX.Element {
if (!this.precoins()) {
return <div>...</div>;
}
return (
<div className="tree-item">
2016-11-19 15:59:17 +01:00
Planchets ({this.precoins() !.length.toString()})
2016-10-12 02:55:53 +02:00
{" "}
<Toggle expanded={this.expanded}>
{this.precoins() !.map((c) => <PreCoinView precoin={c} />)}
</Toggle>
</div>
);
}
}
interface DenominationListProps {
2016-11-15 15:07:17 +01:00
exchange: ExchangeRecord;
2016-10-12 02:55:53 +02:00
}
interface ExpanderTextProps {
text: string;
}
class ExpanderText extends ImplicitStateComponent<ExpanderTextProps> {
expanded = this.makeState<boolean>(false);
textArea: any = undefined;
componentDidUpdate() {
if (this.expanded() && this.textArea) {
this.textArea.focus();
this.textArea.scrollTop = 0;
}
}
render(): JSX.Element {
if (!this.expanded()) {
return (
<span onClick={() => { this.expanded(true); }}>
{(this.props.text.length <= 10)
? this.props.text
: (
<span>
{this.props.text.substring(0,10)}
<span style={{textDecoration: "underline"}}>...</span>
</span>
)
}
</span>
);
}
return (
<textarea
readOnly
style={{display: "block"}}
onBlur={() => this.expanded(false)}
ref={(e) => this.textArea = e}>
{this.props.text}
</textarea>
);
}
}
2016-10-12 02:55:53 +02:00
class DenominationList extends ImplicitStateComponent<DenominationListProps> {
expanded = this.makeState<boolean>(false);
denoms = this.makeState<undefined|DenominationRecord[]>(undefined);
2016-10-12 02:55:53 +02:00
constructor(props: DenominationListProps) {
super(props);
this.update();
}
async update() {
let d = await getDenoms(this.props.exchange.baseUrl);
this.denoms(d);
}
renderDenom(d: DenominationRecord) {
2016-10-12 02:55:53 +02:00
return (
<div className="tree-item">
<ul>
<li>Offered: {d.isOffered ? "yes" : "no"}</li>
2016-10-12 02:55:53 +02:00
<li>Value: {prettyAmount(d.value)}</li>
<li>Withdraw fee: {prettyAmount(d.feeWithdraw)}</li>
<li>Refresh fee: {prettyAmount(d.feeRefresh)}</li>
<li>Deposit fee: {prettyAmount(d.feeDeposit)}</li>
<li>Refund fee: {prettyAmount(d.feeRefund)}</li>
<li>Start: {getTalerStampDate(d.stampStart)!.toString()}</li>
<li>Withdraw expiration: {getTalerStampDate(d.stampExpireWithdraw)!.toString()}</li>
<li>Legal expiration: {getTalerStampDate(d.stampExpireLegal)!.toString()}</li>
<li>Deposit expiration: {getTalerStampDate(d.stampExpireDeposit)!.toString()}</li>
<li>Denom pub: <ExpanderText text={d.denomPub} /></li>
2016-10-12 02:55:53 +02:00
</ul>
</div>
);
}
render(): JSX.Element {
let denoms = this.denoms()
if (!denoms) {
return (
<div className="tree-item">
Denominations (...)
{" "}
<Toggle expanded={this.expanded}>
...
</Toggle>
</div>
);
}
2016-10-12 02:55:53 +02:00
return (
<div className="tree-item">
Denominations ({denoms.length.toString()})
2016-10-12 02:55:53 +02:00
{" "}
<Toggle expanded={this.expanded}>
{denoms.map((d) => this.renderDenom(d))}
2016-10-12 02:55:53 +02:00
</Toggle>
</div>
);
}
}
class ReserveList extends ImplicitStateComponent<ReserveListProps> {
2016-10-13 02:23:24 +02:00
reserves = this.makeState<ReserveRecord[] | null>(null);
2016-10-12 02:55:53 +02:00
expanded = this.makeState<boolean>(false);
constructor(props: ReserveListProps) {
super(props);
this.update();
}
async update() {
let reserves = await getReserves(this.props.exchangeBaseUrl);
this.reserves(reserves);
}
render(): JSX.Element {
if (!this.reserves()) {
return <div>...</div>;
}
return (
<div className="tree-item">
Reserves ({this.reserves() !.length.toString()})
{" "}
<Toggle expanded={this.expanded}>
{this.reserves() !.map((r) => <ReserveView reserve={r} />)}
</Toggle>
</div>
);
}
}
interface ExchangeProps {
2016-11-15 15:07:17 +01:00
exchange: ExchangeRecord;
2016-10-12 02:55:53 +02:00
}
class ExchangeView extends React.Component<ExchangeProps, void> {
2016-10-12 02:55:53 +02:00
render(): JSX.Element {
let e = this.props.exchange;
return (
<div className="tree-item">
<ul>
<li>Exchange Base Url: {this.props.exchange.baseUrl}</li>
<li>Master public key: <ExpanderText text={this.props.exchange.masterPublicKey} /></li>
</ul>
2016-10-12 02:55:53 +02:00
<DenominationList exchange={e} />
<ReserveList exchangeBaseUrl={this.props.exchange.baseUrl} />
<CoinList exchangeBaseUrl={this.props.exchange.baseUrl} />
<PreCoinList exchangeBaseUrl={this.props.exchange.baseUrl} />
</div>
);
}
}
interface ExchangesListState {
2016-11-15 15:07:17 +01:00
exchanges?: ExchangeRecord[];
2016-10-12 02:55:53 +02:00
}
class ExchangesList extends React.Component<any, ExchangesListState> {
2016-10-12 02:55:53 +02:00
constructor() {
super();
let port = chrome.runtime.connect();
port.onMessage.addListener((msg: any) => {
if (msg.notify) {
console.log("got notified");
this.update();
}
});
2016-10-12 02:55:53 +02:00
this.update();
2016-11-13 10:17:39 +01:00
this.state = {} as any;
2016-10-12 02:55:53 +02:00
}
async update() {
let exchanges = await getExchanges();
console.log("exchanges: ", exchanges);
this.setState({ exchanges });
}
render(): JSX.Element {
2016-11-13 10:17:39 +01:00
let exchanges = this.state.exchanges;
if (!exchanges) {
2016-10-12 02:55:53 +02:00
return <span>...</span>;
}
return (
<div className="tree-item">
2016-11-13 10:17:39 +01:00
Exchanges ({exchanges.length.toString()}):
{exchanges.map(e => <ExchangeView exchange={e} />)}
2016-10-12 02:55:53 +02:00
</div>
);
}
}
export function main() {
2016-11-13 10:17:39 +01:00
ReactDOM.render(<ExchangesList />, document.getElementById("container")!);
2016-10-12 02:55:53 +02:00
}