deposit test case
This commit is contained in:
parent
8e468ae092
commit
c5f484d18a
67
packages/taler-wallet-webextension/dev.mjs
Executable file
67
packages/taler-wallet-webextension/dev.mjs
Executable file
@ -0,0 +1,67 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
/* eslint-disable no-undef */
|
||||||
|
|
||||||
|
import linaria from '@linaria/esbuild'
|
||||||
|
import esbuild from 'esbuild'
|
||||||
|
import { buildConfig } from "./build-fast-with-linaria.mjs"
|
||||||
|
import fs from 'fs';
|
||||||
|
import WebSocket from "ws";
|
||||||
|
import chokidar from "chokidar";
|
||||||
|
import path from "path"
|
||||||
|
|
||||||
|
const devServerBroadcastDelay = 500
|
||||||
|
const devServerPort = 8002
|
||||||
|
const wss = new WebSocket.Server({ port: devServerPort });
|
||||||
|
const toWatch = ["./src"]
|
||||||
|
|
||||||
|
function broadcast(file, event) {
|
||||||
|
setTimeout(() => {
|
||||||
|
wss.clients.forEach((client) => {
|
||||||
|
if (client.readyState === WebSocket.OPEN) {
|
||||||
|
console.log(new Date(), file)
|
||||||
|
client.send(JSON.stringify(event));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, devServerBroadcastDelay);
|
||||||
|
}
|
||||||
|
wss.addListener("connection", () => {
|
||||||
|
console.log("new client")
|
||||||
|
})
|
||||||
|
|
||||||
|
const watcher = chokidar
|
||||||
|
.watch(toWatch, {
|
||||||
|
persistent: true,
|
||||||
|
ignoreInitial: true,
|
||||||
|
awaitWriteFinish: {
|
||||||
|
stabilityThreshold: 100,
|
||||||
|
pollInterval: 100,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.on("error", (error) => console.error(error))
|
||||||
|
.on("change", async (file) => {
|
||||||
|
broadcast(file, { type: "RELOAD" });
|
||||||
|
})
|
||||||
|
.on("add", async (file) => {
|
||||||
|
broadcast(file, { type: "RELOAD" });
|
||||||
|
})
|
||||||
|
.on("unlink", async (file) => {
|
||||||
|
broadcast(file, { type: "RELOAD" });
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
fs.writeFileSync("dev-html/manifest.json", fs.readFileSync("manifest-v2.json"))
|
||||||
|
fs.writeFileSync("dev-html/mocha.css", fs.readFileSync("node_modules/mocha/mocha.css"))
|
||||||
|
fs.writeFileSync("dev-html/mocha.js", fs.readFileSync("node_modules/mocha/mocha.js"))
|
||||||
|
fs.writeFileSync("dev-html/mocha.js.map", fs.readFileSync("node_modules/mocha/mocha.js.map"))
|
||||||
|
|
||||||
|
const server = await esbuild
|
||||||
|
.serve({ servedir: 'dev-html' }, {
|
||||||
|
...buildConfig, outdir: 'dev-html/dist'
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
console.log(e)
|
||||||
|
process.exit(1)
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log("ready!", server.port);
|
||||||
|
|
@ -29,7 +29,8 @@
|
|||||||
"preact": "^10.6.5",
|
"preact": "^10.6.5",
|
||||||
"preact-router": "3.2.1",
|
"preact-router": "3.2.1",
|
||||||
"qrcode-generator": "^1.4.4",
|
"qrcode-generator": "^1.4.4",
|
||||||
"tslib": "^2.3.1"
|
"tslib": "^2.3.1",
|
||||||
|
"ws": "7.4.5"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "7.13.16",
|
"@babel/core": "7.13.16",
|
||||||
@ -59,6 +60,7 @@
|
|||||||
"babel-loader": "^8.2.3",
|
"babel-loader": "^8.2.3",
|
||||||
"babel-plugin-transform-react-jsx": "^6.24.1",
|
"babel-plugin-transform-react-jsx": "^6.24.1",
|
||||||
"chai": "^4.3.6",
|
"chai": "^4.3.6",
|
||||||
|
"chokidar": "^3.5.3",
|
||||||
"mocha": "^9.2.0",
|
"mocha": "^9.2.0",
|
||||||
"nyc": "^15.1.0",
|
"nyc": "^15.1.0",
|
||||||
"polished": "^4.1.4",
|
"polished": "^4.1.4",
|
||||||
|
@ -1,24 +0,0 @@
|
|||||||
#!/usr/bin/env node
|
|
||||||
/* eslint-disable no-undef */
|
|
||||||
|
|
||||||
import linaria from '@linaria/esbuild'
|
|
||||||
import esbuild from 'esbuild'
|
|
||||||
import { buildConfig } from "./build-fast-with-linaria.mjs"
|
|
||||||
import fs from 'fs';
|
|
||||||
|
|
||||||
fs.writeFileSync("dev-html/manifest.json", fs.readFileSync("manifest-v2.json"))
|
|
||||||
fs.writeFileSync("dev-html/mocha.css", fs.readFileSync("node_modules/mocha/mocha.css"))
|
|
||||||
fs.writeFileSync("dev-html/mocha.js", fs.readFileSync("node_modules/mocha/mocha.js"))
|
|
||||||
fs.writeFileSync("dev-html/mocha.js.map", fs.readFileSync("node_modules/mocha/mocha.js.map"))
|
|
||||||
|
|
||||||
const server = await esbuild
|
|
||||||
.serve({
|
|
||||||
servedir: 'dev-html',
|
|
||||||
}, { ...buildConfig, outdir: 'dev-html/dist' })
|
|
||||||
.catch((e) => {
|
|
||||||
console.log(e)
|
|
||||||
process.exit(1)
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log("ready!", server.port);
|
|
||||||
|
|
@ -21,7 +21,7 @@
|
|||||||
|
|
||||||
import { ContractTerms, PreparePayResultType } from "@gnu-taler/taler-util";
|
import { ContractTerms, PreparePayResultType } from "@gnu-taler/taler-util";
|
||||||
import { createExample } from "../test-utils.js";
|
import { createExample } from "../test-utils.js";
|
||||||
import { PaymentRequestView as TestedComponent } from "./Deposit.js";
|
import { View as TestedComponent } from "./Deposit.js";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
title: "cta/deposit",
|
title: "cta/deposit",
|
||||||
@ -29,140 +29,6 @@ export default {
|
|||||||
argTypes: {},
|
argTypes: {},
|
||||||
};
|
};
|
||||||
|
|
||||||
export const NoBalance = createExample(TestedComponent, {
|
export const Simple = createExample(TestedComponent, {
|
||||||
payStatus: {
|
state: { status: "ready" },
|
||||||
status: PreparePayResultType.InsufficientBalance,
|
|
||||||
noncePriv: "",
|
|
||||||
proposalId: "proposal1234",
|
|
||||||
contractTerms: {
|
|
||||||
merchant: {
|
|
||||||
name: "someone",
|
|
||||||
},
|
|
||||||
summary: "some beers",
|
|
||||||
amount: "USD:10",
|
|
||||||
} as Partial<ContractTerms> as any,
|
|
||||||
amountRaw: "USD:10",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export const NoEnoughBalance = createExample(TestedComponent, {
|
|
||||||
payStatus: {
|
|
||||||
status: PreparePayResultType.InsufficientBalance,
|
|
||||||
noncePriv: "",
|
|
||||||
proposalId: "proposal1234",
|
|
||||||
contractTerms: {
|
|
||||||
merchant: {
|
|
||||||
name: "someone",
|
|
||||||
},
|
|
||||||
summary: "some beers",
|
|
||||||
amount: "USD:10",
|
|
||||||
} as Partial<ContractTerms> as any,
|
|
||||||
amountRaw: "USD:10",
|
|
||||||
},
|
|
||||||
balance: {
|
|
||||||
currency: "USD",
|
|
||||||
fraction: 40000000,
|
|
||||||
value: 9,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export const PaymentPossible = createExample(TestedComponent, {
|
|
||||||
uri: "taler://pay/merchant-backend.taler/2021.242-01G2X4275RBWG/?c=66BE594PDZR24744J6EQK52XM0",
|
|
||||||
payStatus: {
|
|
||||||
status: PreparePayResultType.PaymentPossible,
|
|
||||||
amountEffective: "USD:10",
|
|
||||||
amountRaw: "USD:10",
|
|
||||||
noncePriv: "",
|
|
||||||
contractTerms: {
|
|
||||||
nonce: "123213123",
|
|
||||||
merchant: {
|
|
||||||
name: "someone",
|
|
||||||
},
|
|
||||||
amount: "USD:10",
|
|
||||||
summary: "some beers",
|
|
||||||
} as Partial<ContractTerms> as any,
|
|
||||||
contractTermsHash: "123456",
|
|
||||||
proposalId: "proposal1234",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export const PaymentPossibleWithFee = createExample(TestedComponent, {
|
|
||||||
uri: "taler://pay/merchant-backend.taler/2021.242-01G2X4275RBWG/?c=66BE594PDZR24744J6EQK52XM0",
|
|
||||||
payStatus: {
|
|
||||||
status: PreparePayResultType.PaymentPossible,
|
|
||||||
amountEffective: "USD:10.20",
|
|
||||||
amountRaw: "USD:10",
|
|
||||||
noncePriv: "",
|
|
||||||
contractTerms: {
|
|
||||||
nonce: "123213123",
|
|
||||||
merchant: {
|
|
||||||
name: "someone",
|
|
||||||
},
|
|
||||||
amount: "USD:10",
|
|
||||||
summary: "some beers",
|
|
||||||
} as Partial<ContractTerms> as any,
|
|
||||||
contractTermsHash: "123456",
|
|
||||||
proposalId: "proposal1234",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export const AlreadyConfirmedWithFullfilment = createExample(TestedComponent, {
|
|
||||||
payStatus: {
|
|
||||||
status: PreparePayResultType.AlreadyConfirmed,
|
|
||||||
amountEffective: "USD:10",
|
|
||||||
amountRaw: "USD:10",
|
|
||||||
contractTerms: {
|
|
||||||
merchant: {
|
|
||||||
name: "someone",
|
|
||||||
},
|
|
||||||
fulfillment_message:
|
|
||||||
"congratulations! you are looking at the fulfillment message! ",
|
|
||||||
summary: "some beers",
|
|
||||||
amount: "USD:10",
|
|
||||||
} as Partial<ContractTerms> as any,
|
|
||||||
contractTermsHash: "123456",
|
|
||||||
proposalId: "proposal1234",
|
|
||||||
paid: false,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export const AlreadyConfirmedWithoutFullfilment = createExample(
|
|
||||||
TestedComponent,
|
|
||||||
{
|
|
||||||
payStatus: {
|
|
||||||
status: PreparePayResultType.AlreadyConfirmed,
|
|
||||||
amountEffective: "USD:10",
|
|
||||||
amountRaw: "USD:10",
|
|
||||||
contractTerms: {
|
|
||||||
merchant: {
|
|
||||||
name: "someone",
|
|
||||||
},
|
|
||||||
summary: "some beers",
|
|
||||||
amount: "USD:10",
|
|
||||||
} as Partial<ContractTerms> as any,
|
|
||||||
contractTermsHash: "123456",
|
|
||||||
proposalId: "proposal1234",
|
|
||||||
paid: false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
export const AlreadyPaid = createExample(TestedComponent, {
|
|
||||||
payStatus: {
|
|
||||||
status: PreparePayResultType.AlreadyConfirmed,
|
|
||||||
amountEffective: "USD:10",
|
|
||||||
amountRaw: "USD:10",
|
|
||||||
contractTerms: {
|
|
||||||
merchant: {
|
|
||||||
name: "someone",
|
|
||||||
},
|
|
||||||
fulfillment_message:
|
|
||||||
"congratulations! you are looking at the fulfillment message! ",
|
|
||||||
summary: "some beers",
|
|
||||||
amount: "USD:10",
|
|
||||||
} as Partial<ContractTerms> as any,
|
|
||||||
contractTermsHash: "123456",
|
|
||||||
proposalId: "proposal1234",
|
|
||||||
paid: true,
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
@ -39,6 +39,8 @@ import { TalerError } from "@gnu-taler/taler-wallet-core";
|
|||||||
import { Fragment, h, VNode } from "preact";
|
import { Fragment, h, VNode } from "preact";
|
||||||
import { useEffect, useState } from "preact/hooks";
|
import { useEffect, useState } from "preact/hooks";
|
||||||
import { ErrorTalerOperation } from "../components/ErrorTalerOperation.js";
|
import { ErrorTalerOperation } from "../components/ErrorTalerOperation.js";
|
||||||
|
import { Loading } from "../components/Loading.js";
|
||||||
|
import { LoadingError } from "../components/LoadingError.js";
|
||||||
import { LogoHeader } from "../components/LogoHeader.js";
|
import { LogoHeader } from "../components/LogoHeader.js";
|
||||||
import { Part } from "../components/Part.js";
|
import { Part } from "../components/Part.js";
|
||||||
import {
|
import {
|
||||||
@ -49,157 +51,50 @@ import {
|
|||||||
WarningBox,
|
WarningBox,
|
||||||
} from "../components/styled/index.js";
|
} from "../components/styled/index.js";
|
||||||
import { useTranslationContext } from "../context/translation.js";
|
import { useTranslationContext } from "../context/translation.js";
|
||||||
import { useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
import { HookError, useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
||||||
import * as wxApi from "../wxApi.js";
|
import * as wxApi from "../wxApi.js";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
talerPayUri?: string;
|
talerDepositUri?: string;
|
||||||
goBack: () => void;
|
goBack: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function DepositPage({ talerPayUri, goBack }: Props): VNode {
|
type State = Loading | Ready;
|
||||||
const { i18n } = useTranslationContext();
|
interface Loading {
|
||||||
const [payStatus, setPayStatus] = useState<PreparePayResult | undefined>(
|
status: "loading";
|
||||||
undefined,
|
hook: HookError | undefined;
|
||||||
);
|
}
|
||||||
const [payResult, setPayResult] = useState<ConfirmPayResult | undefined>(
|
interface Ready {
|
||||||
undefined,
|
status: "ready";
|
||||||
);
|
}
|
||||||
const [payErrMsg, setPayErrMsg] = useState<TalerError | string | undefined>(
|
|
||||||
undefined,
|
|
||||||
);
|
|
||||||
|
|
||||||
const balance = useAsyncAsHook(wxApi.getBalance, [
|
function useComponentState(uri: string | undefined): State {
|
||||||
NotificationType.CoinWithdrawn,
|
return {
|
||||||
]);
|
status: "loading",
|
||||||
const balanceWithoutError = balance?.hasError
|
hook: undefined,
|
||||||
? []
|
|
||||||
: balance?.response.balances || [];
|
|
||||||
|
|
||||||
const foundBalance = balanceWithoutError.find(
|
|
||||||
(b) =>
|
|
||||||
payStatus &&
|
|
||||||
Amounts.parseOrThrow(b.available).currency ===
|
|
||||||
Amounts.parseOrThrow(payStatus?.amountRaw).currency,
|
|
||||||
);
|
|
||||||
const foundAmount = foundBalance
|
|
||||||
? Amounts.parseOrThrow(foundBalance.available)
|
|
||||||
: undefined;
|
|
||||||
// We use a string here so that dependency tracking for useEffect works properly
|
|
||||||
const foundAmountStr = foundAmount
|
|
||||||
? Amounts.stringify(foundAmount)
|
|
||||||
: undefined;
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!talerPayUri) return;
|
|
||||||
const doFetch = async (): Promise<void> => {
|
|
||||||
try {
|
|
||||||
const p = await wxApi.preparePay(talerPayUri);
|
|
||||||
setPayStatus(p);
|
|
||||||
} catch (e) {
|
|
||||||
console.log("Got error while trying to pay", e);
|
|
||||||
if (e instanceof TalerError) {
|
|
||||||
setPayErrMsg(e);
|
|
||||||
}
|
|
||||||
if (e instanceof Error) {
|
|
||||||
setPayErrMsg(e.message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
doFetch();
|
|
||||||
}, [talerPayUri, foundAmountStr]);
|
|
||||||
|
|
||||||
if (!talerPayUri) {
|
|
||||||
return (
|
|
||||||
<span>
|
|
||||||
<i18n.Translate>missing pay uri</i18n.Translate>
|
|
||||||
</span>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!payStatus) {
|
|
||||||
if (payErrMsg instanceof TalerError) {
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash payment</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<section>
|
|
||||||
<ErrorTalerOperation
|
|
||||||
title={
|
|
||||||
<i18n.Translate>
|
|
||||||
Could not get the payment information for this order
|
|
||||||
</i18n.Translate>
|
|
||||||
}
|
|
||||||
error={payErrMsg?.errorDetail}
|
|
||||||
/>
|
|
||||||
</section>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (payErrMsg) {
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash payment</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<section>
|
|
||||||
<p>
|
|
||||||
<i18n.Translate>
|
|
||||||
Could not get the payment information for this order
|
|
||||||
</i18n.Translate>
|
|
||||||
</p>
|
|
||||||
<ErrorBox>{payErrMsg}</ErrorBox>
|
|
||||||
</section>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<span>
|
|
||||||
<i18n.Translate>Loading payment information</i18n.Translate> ...
|
|
||||||
</span>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const onClick = async (): Promise<void> => {
|
|
||||||
// try {
|
|
||||||
// const res = await doPayment(payStatus);
|
|
||||||
// setPayResult(res);
|
|
||||||
// } catch (e) {
|
|
||||||
// console.error(e);
|
|
||||||
// if (e instanceof Error) {
|
|
||||||
// setPayErrMsg(e.message);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
|
||||||
<PaymentRequestView
|
|
||||||
uri={talerPayUri}
|
|
||||||
payStatus={payStatus}
|
|
||||||
payResult={payResult}
|
|
||||||
onClick={onClick}
|
|
||||||
balance={foundAmount}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface PaymentRequestViewProps {
|
export function DepositPage({ talerDepositUri, goBack }: Props): VNode {
|
||||||
payStatus: PreparePayResult;
|
const { i18n } = useTranslationContext();
|
||||||
payResult?: ConfirmPayResult;
|
|
||||||
onClick: () => void;
|
const state = useComponentState(talerDepositUri);
|
||||||
payErrMsg?: string;
|
if (state.status === "loading") {
|
||||||
uri: string;
|
if (!state.hook) return <Loading />;
|
||||||
balance: AmountJson | undefined;
|
return (
|
||||||
|
<LoadingError
|
||||||
|
title={<i18n.Translate>Could not load pay status</i18n.Translate>}
|
||||||
|
error={state.hook}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return <View state={state} />;
|
||||||
}
|
}
|
||||||
export function PaymentRequestView({
|
|
||||||
payStatus,
|
export interface ViewProps {
|
||||||
payResult,
|
state: State;
|
||||||
}: PaymentRequestViewProps): VNode {
|
}
|
||||||
const totalFees: AmountJson = Amounts.getZero(payStatus.amountRaw);
|
export function View({ state }: ViewProps): VNode {
|
||||||
const contractTerms: ContractTerms = payStatus.contractTerms;
|
|
||||||
const { i18n } = useTranslationContext();
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@ -209,78 +104,6 @@ export function PaymentRequestView({
|
|||||||
<SubTitle>
|
<SubTitle>
|
||||||
<i18n.Translate>Digital cash deposit</i18n.Translate>
|
<i18n.Translate>Digital cash deposit</i18n.Translate>
|
||||||
</SubTitle>
|
</SubTitle>
|
||||||
{payStatus.status === PreparePayResultType.AlreadyConfirmed &&
|
|
||||||
(payStatus.paid ? (
|
|
||||||
<SuccessBox>
|
|
||||||
<i18n.Translate>Already paid</i18n.Translate>
|
|
||||||
</SuccessBox>
|
|
||||||
) : (
|
|
||||||
<WarningBox>
|
|
||||||
<i18n.Translate>Already claimed</i18n.Translate>
|
|
||||||
</WarningBox>
|
|
||||||
))}
|
|
||||||
{payResult && payResult.type === ConfirmPayResultType.Done && (
|
|
||||||
<SuccessBox>
|
|
||||||
<h3>
|
|
||||||
<i18n.Translate>Payment complete</i18n.Translate>
|
|
||||||
</h3>
|
|
||||||
<p>
|
|
||||||
{!payResult.contractTerms.fulfillment_message ? (
|
|
||||||
<i18n.Translate>
|
|
||||||
You will now be sent back to the merchant you came from.
|
|
||||||
</i18n.Translate>
|
|
||||||
) : (
|
|
||||||
payResult.contractTerms.fulfillment_message
|
|
||||||
)}
|
|
||||||
</p>
|
|
||||||
</SuccessBox>
|
|
||||||
)}
|
|
||||||
<section>
|
|
||||||
{payStatus.status !== PreparePayResultType.InsufficientBalance &&
|
|
||||||
Amounts.isNonZero(totalFees) && (
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Total to pay</i18n.Translate>}
|
|
||||||
text={amountToPretty(
|
|
||||||
Amounts.parseOrThrow(payStatus.amountEffective),
|
|
||||||
)}
|
|
||||||
kind="negative"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Purchase amount</i18n.Translate>}
|
|
||||||
text={amountToPretty(Amounts.parseOrThrow(payStatus.amountRaw))}
|
|
||||||
kind="neutral"
|
|
||||||
/>
|
|
||||||
{Amounts.isNonZero(totalFees) && (
|
|
||||||
<Fragment>
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Fee</i18n.Translate>}
|
|
||||||
text={amountToPretty(totalFees)}
|
|
||||||
kind="negative"
|
|
||||||
/>
|
|
||||||
</Fragment>
|
|
||||||
)}
|
|
||||||
<Part
|
|
||||||
title={<i18n.Translate>Merchant</i18n.Translate>}
|
|
||||||
text={contractTerms.merchant.name}
|
|
||||||
kind="neutral"
|
|
||||||
/>
|
|
||||||
<Part
|
|
||||||
title={<i18n.Translate>Purchase</i18n.Translate>}
|
|
||||||
text={contractTerms.summary}
|
|
||||||
kind="neutral"
|
|
||||||
/>
|
|
||||||
{contractTerms.order_id && (
|
|
||||||
<Part
|
|
||||||
title={<i18n.Translate>Receipt</i18n.Translate>}
|
|
||||||
text={`#${contractTerms.order_id}`}
|
|
||||||
kind="neutral"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</section>
|
|
||||||
</WalletAction>
|
</WalletAction>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -65,7 +65,7 @@ import {
|
|||||||
useAsyncAsHook,
|
useAsyncAsHook,
|
||||||
useAsyncAsHook2,
|
useAsyncAsHook2,
|
||||||
} from "../hooks/useAsyncAsHook.js";
|
} from "../hooks/useAsyncAsHook.js";
|
||||||
import { ButtonHandler } from "../wallet/CreateManualWithdraw.js";
|
import { ButtonHandler } from "../mui/handlers.js";
|
||||||
import * as wxApi from "../wxApi.js";
|
import * as wxApi from "../wxApi.js";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
@ -74,32 +74,6 @@ interface Props {
|
|||||||
goBack: () => void;
|
goBack: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function doPayment(
|
|
||||||
payStatus: PreparePayResult,
|
|
||||||
api: typeof wxApi,
|
|
||||||
): Promise<ConfirmPayResultDone> {
|
|
||||||
if (payStatus.status !== "payment-possible") {
|
|
||||||
throw TalerError.fromUncheckedDetail({
|
|
||||||
code: TalerErrorCode.GENERIC_CLIENT_INTERNAL_ERROR,
|
|
||||||
hint: `payment is not possible: ${payStatus.status}`,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
const proposalId = payStatus.proposalId;
|
|
||||||
const res = await api.confirmPay(proposalId, undefined);
|
|
||||||
if (res.type !== ConfirmPayResultType.Done) {
|
|
||||||
throw TalerError.fromUncheckedDetail({
|
|
||||||
code: TalerErrorCode.GENERIC_CLIENT_INTERNAL_ERROR,
|
|
||||||
hint: `could not confirm payment`,
|
|
||||||
payResult: res,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
const fu = res.contractTerms.fulfillment_url;
|
|
||||||
if (fu) {
|
|
||||||
document.location.href = fu;
|
|
||||||
}
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
type State = Loading | Ready | Confirmed;
|
type State = Loading | Ready | Confirmed;
|
||||||
interface Loading {
|
interface Loading {
|
||||||
status: "loading";
|
status: "loading";
|
||||||
|
@ -66,7 +66,9 @@ export const TermsOfServiceNotYetLoaded = createExample(TestedComponent, {
|
|||||||
exchange: {
|
exchange: {
|
||||||
list: exchangeList,
|
list: exchangeList,
|
||||||
value: "exchange.demo.taler.net",
|
value: "exchange.demo.taler.net",
|
||||||
onChange: () => null,
|
onChange: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
showExchangeSelection: false,
|
showExchangeSelection: false,
|
||||||
mustAcceptFirst: false,
|
mustAcceptFirst: false,
|
||||||
@ -99,7 +101,9 @@ export const WithSomeFee = createExample(TestedComponent, {
|
|||||||
exchange: {
|
exchange: {
|
||||||
list: exchangeList,
|
list: exchangeList,
|
||||||
value: "exchange.demo.taler.net",
|
value: "exchange.demo.taler.net",
|
||||||
onChange: () => null,
|
onChange: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
showExchangeSelection: false,
|
showExchangeSelection: false,
|
||||||
mustAcceptFirst: false,
|
mustAcceptFirst: false,
|
||||||
@ -133,7 +137,9 @@ export const WithoutFee = createExample(TestedComponent, {
|
|||||||
exchange: {
|
exchange: {
|
||||||
list: exchangeList,
|
list: exchangeList,
|
||||||
value: "exchange.demo.taler.net",
|
value: "exchange.demo.taler.net",
|
||||||
onChange: () => null,
|
onChange: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
showExchangeSelection: false,
|
showExchangeSelection: false,
|
||||||
mustAcceptFirst: false,
|
mustAcceptFirst: false,
|
||||||
@ -167,7 +173,9 @@ export const EditExchangeUntouched = createExample(TestedComponent, {
|
|||||||
exchange: {
|
exchange: {
|
||||||
list: exchangeList,
|
list: exchangeList,
|
||||||
value: "exchange.demo.taler.net",
|
value: "exchange.demo.taler.net",
|
||||||
onChange: () => null,
|
onChange: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
showExchangeSelection: true,
|
showExchangeSelection: true,
|
||||||
mustAcceptFirst: false,
|
mustAcceptFirst: false,
|
||||||
@ -202,7 +210,9 @@ export const EditExchangeModified = createExample(TestedComponent, {
|
|||||||
list: exchangeList,
|
list: exchangeList,
|
||||||
isDirty: true,
|
isDirty: true,
|
||||||
value: "exchange.test.taler.net",
|
value: "exchange.test.taler.net",
|
||||||
onChange: () => null,
|
onChange: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
showExchangeSelection: true,
|
showExchangeSelection: true,
|
||||||
mustAcceptFirst: false,
|
mustAcceptFirst: false,
|
||||||
|
@ -42,10 +42,7 @@ import {
|
|||||||
import { useTranslationContext } from "../context/translation.js";
|
import { useTranslationContext } from "../context/translation.js";
|
||||||
import { HookError, useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
import { HookError, useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
||||||
import { buildTermsOfServiceState } from "../utils/index.js";
|
import { buildTermsOfServiceState } from "../utils/index.js";
|
||||||
import {
|
import { ButtonHandler, SelectFieldHandler } from "../mui/handlers.js";
|
||||||
ButtonHandler,
|
|
||||||
SelectFieldHandler,
|
|
||||||
} from "../wallet/CreateManualWithdraw.js";
|
|
||||||
import * as wxApi from "../wxApi.js";
|
import * as wxApi from "../wxApi.js";
|
||||||
import {
|
import {
|
||||||
Props as TermsOfServiceSectionProps,
|
Props as TermsOfServiceSectionProps,
|
||||||
@ -258,7 +255,7 @@ export function useComponentState(
|
|||||||
}
|
}
|
||||||
|
|
||||||
const exchangeHandler: SelectFieldHandler = {
|
const exchangeHandler: SelectFieldHandler = {
|
||||||
onChange: setNextExchange,
|
onChange: async (e) => setNextExchange(e),
|
||||||
value: nextExchange ?? thisExchange,
|
value: nextExchange ?? thisExchange,
|
||||||
list: exchanges,
|
list: exchanges,
|
||||||
isDirty: nextExchange !== undefined,
|
isDirty: nextExchange !== undefined,
|
||||||
|
21
packages/taler-wallet-webextension/src/mui/handlers.ts
Normal file
21
packages/taler-wallet-webextension/src/mui/handlers.ts
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import { TalerError } from "@gnu-taler/taler-wallet-core";
|
||||||
|
|
||||||
|
export interface TextFieldHandler {
|
||||||
|
onInput: (value: string) => Promise<void>;
|
||||||
|
value: string;
|
||||||
|
error?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ButtonHandler {
|
||||||
|
onClick?: () => Promise<void>;
|
||||||
|
error?: TalerError;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SelectFieldHandler {
|
||||||
|
onChange: (value: string) => Promise<void>;
|
||||||
|
error?: string;
|
||||||
|
value: string;
|
||||||
|
isDirty?: boolean;
|
||||||
|
list: Record<string, string>;
|
||||||
|
}
|
||||||
|
|
@ -69,10 +69,13 @@ const SideBar = styled.div`
|
|||||||
& > {
|
& > {
|
||||||
ol {
|
ol {
|
||||||
padding: 4px;
|
padding: 4px;
|
||||||
div {
|
div:first-child {
|
||||||
background-color: lightcoral;
|
background-color: lightcoral;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
div[data-hide="true"] {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
dd {
|
dd {
|
||||||
margin-left: 1em;
|
margin-left: 1em;
|
||||||
padding: 4px;
|
padding: 4px;
|
||||||
@ -192,12 +195,12 @@ function ExampleList({
|
|||||||
selected: ExampleItem | undefined;
|
selected: ExampleItem | undefined;
|
||||||
onSelectStory: (i: ExampleItem, id: string) => void;
|
onSelectStory: (i: ExampleItem, id: string) => void;
|
||||||
}): VNode {
|
}): VNode {
|
||||||
const [open, setOpen] = useState(true);
|
const [isOpen, setOpen] = useState(selected && selected.group === name);
|
||||||
return (
|
return (
|
||||||
<ol>
|
<ol>
|
||||||
<div onClick={() => setOpen(!open)}>{name}</div>
|
<div onClick={() => setOpen(!isOpen)}>{name}</div>
|
||||||
{open &&
|
<div data-hide={!isOpen}>
|
||||||
list.map((k) => (
|
{list.map((k) => (
|
||||||
<li key={k.name}>
|
<li key={k.name}>
|
||||||
<dl>
|
<dl>
|
||||||
<dt>{k.name}</dt>
|
<dt>{k.name}</dt>
|
||||||
@ -215,6 +218,7 @@ function ExampleList({
|
|||||||
href={`#${eId}`}
|
href={`#${eId}`}
|
||||||
onClick={(e) => {
|
onClick={(e) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
location.hash = `#${eId}`;
|
||||||
onSelectStory(r, eId);
|
onSelectStory(r, eId);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@ -226,6 +230,7 @@ function ExampleList({
|
|||||||
</dl>
|
</dl>
|
||||||
</li>
|
</li>
|
||||||
))}
|
))}
|
||||||
|
</div>
|
||||||
</ol>
|
</ol>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -335,6 +340,7 @@ function Application(): VNode {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Page>
|
<Page>
|
||||||
|
<LiveReload />
|
||||||
<SideBar>
|
<SideBar>
|
||||||
{allExamples.map((e) => (
|
{allExamples.map((e) => (
|
||||||
<ExampleList
|
<ExampleList
|
||||||
@ -382,3 +388,56 @@ function main(): void {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let liveReloadMounted = false;
|
||||||
|
function LiveReload({ port = 8002 }: { port?: number }): VNode {
|
||||||
|
const [isReloading, setIsReloading] = useState(false);
|
||||||
|
useEffect(() => {
|
||||||
|
if (!liveReloadMounted) {
|
||||||
|
setupLiveReload(port, () => {
|
||||||
|
setIsReloading(true);
|
||||||
|
window.location.reload();
|
||||||
|
});
|
||||||
|
liveReloadMounted = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (isReloading) {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
width: "100%",
|
||||||
|
height: "100%",
|
||||||
|
backgroundColor: "rgba(0,0,0,0.5)",
|
||||||
|
color: "white",
|
||||||
|
display: "flex",
|
||||||
|
justifyContent: "center",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<h1 style={{ margin: "auto" }}>reloading...</h1>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return <Fragment />;
|
||||||
|
}
|
||||||
|
|
||||||
|
function setupLiveReload(port: number, onReload: () => void): void {
|
||||||
|
const protocol = location.protocol === "https:" ? "wss:" : "ws:";
|
||||||
|
const host = location.hostname;
|
||||||
|
const socketPath = `${protocol}//${host}:${port}/socket`;
|
||||||
|
|
||||||
|
const ws = new WebSocket(socketPath);
|
||||||
|
ws.onmessage = (message) => {
|
||||||
|
const event = JSON.parse(message.data);
|
||||||
|
if (event.type === "LOG") {
|
||||||
|
console.log(event.message);
|
||||||
|
}
|
||||||
|
if (event.type === "RELOAD") {
|
||||||
|
onReload();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
ws.onerror = (error) => {
|
||||||
|
console.error(error);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
@ -21,8 +21,9 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { expect } from "chai";
|
import { expect } from "chai";
|
||||||
|
import { SelectFieldHandler, TextFieldHandler } from "../mui/handlers.js";
|
||||||
import { mountHook } from "../test-utils.js";
|
import { mountHook } from "../test-utils.js";
|
||||||
import { SelectFieldHandler, TextFieldHandler, useComponentState } from "./CreateManualWithdraw.js";
|
import { useComponentState } from "./CreateManualWithdraw.js";
|
||||||
|
|
||||||
|
|
||||||
const exchangeListWithARSandUSD = {
|
const exchangeListWithARSandUSD = {
|
||||||
|
@ -37,6 +37,7 @@ import {
|
|||||||
SubTitle,
|
SubTitle,
|
||||||
} from "../components/styled/index.js";
|
} from "../components/styled/index.js";
|
||||||
import { useTranslationContext } from "../context/translation.js";
|
import { useTranslationContext } from "../context/translation.js";
|
||||||
|
import { SelectFieldHandler, TextFieldHandler } from "../mui/handlers.js";
|
||||||
import { Pages } from "../NavigationBar.js";
|
import { Pages } from "../NavigationBar.js";
|
||||||
|
|
||||||
export interface Props {
|
export interface Props {
|
||||||
@ -55,25 +56,6 @@ export interface State {
|
|||||||
exchange: SelectFieldHandler;
|
exchange: SelectFieldHandler;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface TextFieldHandler {
|
|
||||||
onInput: (value: string) => void;
|
|
||||||
value: string;
|
|
||||||
error?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface ButtonHandler {
|
|
||||||
onClick?: () => Promise<void>;
|
|
||||||
error?: TalerError;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface SelectFieldHandler {
|
|
||||||
onChange: (value: string) => void;
|
|
||||||
error?: string;
|
|
||||||
value: string;
|
|
||||||
isDirty?: boolean;
|
|
||||||
list: Record<string, string>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useComponentState(
|
export function useComponentState(
|
||||||
exchangeUrlWithCurrency: Record<string, string>,
|
exchangeUrlWithCurrency: Record<string, string>,
|
||||||
initialAmount: string | undefined,
|
initialAmount: string | undefined,
|
||||||
@ -109,12 +91,12 @@ export function useComponentState(
|
|||||||
const [amount, setAmount] = useState(initialAmount || "");
|
const [amount, setAmount] = useState(initialAmount || "");
|
||||||
const parsedAmount = Amounts.parse(`${currency}:${amount}`);
|
const parsedAmount = Amounts.parse(`${currency}:${amount}`);
|
||||||
|
|
||||||
function changeExchange(exchange: string): void {
|
async function changeExchange(exchange: string): Promise<void> {
|
||||||
setExchange(exchange);
|
setExchange(exchange);
|
||||||
setCurrency(exchangeUrlWithCurrency[exchange]);
|
setCurrency(exchangeUrlWithCurrency[exchange]);
|
||||||
}
|
}
|
||||||
|
|
||||||
function changeCurrency(currency: string): void {
|
async function changeCurrency(currency: string): Promise<void> {
|
||||||
setCurrency(currency);
|
setCurrency(currency);
|
||||||
const found = Object.entries(exchangeUrlWithCurrency).find(
|
const found = Object.entries(exchangeUrlWithCurrency).find(
|
||||||
(e) => e[1] === currency,
|
(e) => e[1] === currency,
|
||||||
@ -140,7 +122,7 @@ export function useComponentState(
|
|||||||
},
|
},
|
||||||
amount: {
|
amount: {
|
||||||
value: amount,
|
value: amount,
|
||||||
onInput: (e: string) => setAmount(e),
|
onInput: async (e: string) => setAmount(e),
|
||||||
},
|
},
|
||||||
parsedAmount,
|
parsedAmount,
|
||||||
};
|
};
|
||||||
|
@ -20,10 +20,13 @@
|
|||||||
* @author Sebastian Javier Marchano (sebasjm)
|
* @author Sebastian Javier Marchano (sebasjm)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { Balance, parsePaytoUri } from "@gnu-taler/taler-util";
|
import { Amounts, Balance, parsePaytoUri } from "@gnu-taler/taler-util";
|
||||||
import type { DepositGroupFees } from "@gnu-taler/taler-wallet-core/src/operations/deposits.js";
|
import type { DepositGroupFees } from "@gnu-taler/taler-wallet-core/src/operations/deposits.js";
|
||||||
import { createExample } from "../test-utils.js";
|
import { createExample } from "../test-utils.js";
|
||||||
import { View as TestedComponent } from "./DepositPage.js";
|
import {
|
||||||
|
createLabelsForBankAccount,
|
||||||
|
View as TestedComponent,
|
||||||
|
} from "./DepositPage.js";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
title: "wallet/deposit",
|
title: "wallet/deposit",
|
||||||
@ -41,23 +44,44 @@ async function alwaysReturnFeeToOne(): Promise<DepositGroupFees> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const WithEmptyAccountList = createExample(TestedComponent, {
|
export const WithEmptyAccountList = createExample(TestedComponent, {
|
||||||
accounts: [],
|
state: {
|
||||||
balances: [
|
status: "no-accounts",
|
||||||
{
|
cancelHandler: {},
|
||||||
available: "USD:10",
|
},
|
||||||
} as Balance,
|
// accounts: [],
|
||||||
],
|
// balances: [
|
||||||
currency: "USD",
|
// {
|
||||||
onCalculateFee: alwaysReturnFeeToOne,
|
// available: "USD:10",
|
||||||
|
// } as Balance,
|
||||||
|
// ],
|
||||||
|
// currency: "USD",
|
||||||
|
// onCalculateFee: alwaysReturnFeeToOne,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const ac = parsePaytoUri("payto://iban/ES8877998399652238")!;
|
||||||
|
const accountMap = createLabelsForBankAccount([ac]);
|
||||||
|
|
||||||
export const WithSomeBankAccounts = createExample(TestedComponent, {
|
export const WithSomeBankAccounts = createExample(TestedComponent, {
|
||||||
accounts: [parsePaytoUri("payto://iban/ES8877998399652238")!],
|
state: {
|
||||||
balances: [
|
status: "ready",
|
||||||
{
|
account: {
|
||||||
available: "USD:10",
|
list: accountMap,
|
||||||
} as Balance,
|
value: accountMap[0],
|
||||||
],
|
onChange: async () => {
|
||||||
currency: "USD",
|
null;
|
||||||
onCalculateFee: alwaysReturnFeeToOne,
|
},
|
||||||
|
},
|
||||||
|
currency: "USD",
|
||||||
|
amount: {
|
||||||
|
onInput: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
|
value: "10:USD",
|
||||||
|
},
|
||||||
|
cancelHandler: {},
|
||||||
|
depositHandler: {},
|
||||||
|
totalFee: Amounts.getZero("USD"),
|
||||||
|
totalToDeposit: Amounts.parseOrThrow("USD:10"),
|
||||||
|
// onCalculateFee: alwaysReturnFeeToOne,
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
@ -19,46 +19,390 @@
|
|||||||
* @author Sebastian Javier Marchano (sebasjm)
|
* @author Sebastian Javier Marchano (sebasjm)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { Amounts, Balance } from "@gnu-taler/taler-util";
|
import { Amounts, Balance, BalancesResponse, parsePaytoUri } from "@gnu-taler/taler-util";
|
||||||
import { DepositGroupFees } from "@gnu-taler/taler-wallet-core/src/operations/deposits";
|
import { DepositGroupFees } from "@gnu-taler/taler-wallet-core/src/operations/deposits";
|
||||||
import { expect } from "chai";
|
import { expect } from "chai";
|
||||||
import { mountHook } from "../test-utils.js";
|
import { mountHook } from "../test-utils.js";
|
||||||
import { useComponentState } from "./DepositPage.js";
|
import { useComponentState } from "./DepositPage.js";
|
||||||
|
import * as wxApi from "../wxApi.js";
|
||||||
|
|
||||||
|
|
||||||
const currency = "EUR"
|
const currency = "EUR"
|
||||||
const feeCalculator = async (): Promise<DepositGroupFees> => ({
|
const withoutFee = async (): Promise<DepositGroupFees> => ({
|
||||||
|
coin: Amounts.parseOrThrow(`${currency}:0`),
|
||||||
|
wire: Amounts.parseOrThrow(`${currency}:0`),
|
||||||
|
refresh: Amounts.parseOrThrow(`${currency}:0`)
|
||||||
|
})
|
||||||
|
|
||||||
|
const withSomeFee = async (): Promise<DepositGroupFees> => ({
|
||||||
coin: Amounts.parseOrThrow(`${currency}:1`),
|
coin: Amounts.parseOrThrow(`${currency}:1`),
|
||||||
wire: Amounts.parseOrThrow(`${currency}:1`),
|
wire: Amounts.parseOrThrow(`${currency}:1`),
|
||||||
refresh: Amounts.parseOrThrow(`${currency}:1`)
|
refresh: Amounts.parseOrThrow(`${currency}:1`)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const freeJustForIBAN = async (account: string): Promise<DepositGroupFees> => /IBAN/i.test(account) ? withoutFee() : withSomeFee()
|
||||||
|
|
||||||
const someBalance = [{
|
const someBalance = [{
|
||||||
available: 'EUR:10'
|
available: 'EUR:10'
|
||||||
} as Balance]
|
} as Balance]
|
||||||
|
|
||||||
|
const nullFunction: any = () => null;
|
||||||
|
type VoidFunction = () => void;
|
||||||
|
|
||||||
describe("DepositPage states", () => {
|
describe("DepositPage states", () => {
|
||||||
it("should have status 'no-balance' when balance is empty", () => {
|
it("should have status 'no-balance' when balance is empty", async () => {
|
||||||
const { getLastResultOrThrow } = mountHook(() =>
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } = mountHook(() =>
|
||||||
useComponentState(currency, [], [], feeCalculator),
|
useComponentState(currency, nullFunction, nullFunction, {
|
||||||
|
getBalance: async () => ({
|
||||||
|
balances: [{ available: `${currency}:0`, }]
|
||||||
|
} as Partial<BalancesResponse>),
|
||||||
|
listKnownBankAccounts: async () => ({ accounts: [] })
|
||||||
|
} as Partial<typeof wxApi> as any)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
{
|
||||||
|
const { status } = getLastResultOrThrow()
|
||||||
|
expect(status).equal("loading")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status } = getLastResultOrThrow()
|
const { status } = getLastResultOrThrow()
|
||||||
expect(status).equal("no-balance")
|
expect(status).equal("no-balance")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await assertNoPendingUpdate()
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should have status 'no-accounts' when balance is not empty and accounts is empty", () => {
|
it("should have status 'no-accounts' when balance is not empty and accounts is empty", async () => {
|
||||||
const { getLastResultOrThrow } = mountHook(() =>
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } = mountHook(() =>
|
||||||
useComponentState(currency, [], someBalance, feeCalculator),
|
useComponentState(currency, nullFunction, nullFunction, {
|
||||||
|
getBalance: async () => ({
|
||||||
|
balances: [{ available: `${currency}:1`, }]
|
||||||
|
} as Partial<BalancesResponse>),
|
||||||
|
listKnownBankAccounts: async () => ({ accounts: [] })
|
||||||
|
} as Partial<typeof wxApi> as any)
|
||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status } = getLastResultOrThrow()
|
const { status } = getLastResultOrThrow()
|
||||||
expect(status).equal("no-accounts")
|
expect(status).equal("loading")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "no-accounts") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
await assertNoPendingUpdate()
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const ibanPayto = parsePaytoUri("payto://iban/ES8877998399652238")!;
|
||||||
|
const talerBankPayto = parsePaytoUri("payto://x-taler-bank/ES8877998399652238")!;
|
||||||
|
|
||||||
|
it("should have status 'ready' but unable to deposit ", async () => {
|
||||||
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } = mountHook(() =>
|
||||||
|
useComponentState(currency, nullFunction, nullFunction, {
|
||||||
|
getBalance: async () => ({
|
||||||
|
balances: [{ available: `${currency}:1`, }]
|
||||||
|
} as Partial<BalancesResponse>),
|
||||||
|
listKnownBankAccounts: async () => ({ accounts: [ibanPayto] })
|
||||||
|
} as Partial<typeof wxApi> as any)
|
||||||
|
);
|
||||||
|
|
||||||
|
{
|
||||||
|
const { status } = getLastResultOrThrow()
|
||||||
|
expect(status).equal("loading")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("0")
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
await assertNoPendingUpdate()
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should not be able to deposit more than the balance ", async () => {
|
||||||
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } = mountHook(() =>
|
||||||
|
useComponentState(currency, nullFunction, nullFunction, {
|
||||||
|
getBalance: async () => ({
|
||||||
|
balances: [{ available: `${currency}:1`, }]
|
||||||
|
} as Partial<BalancesResponse>),
|
||||||
|
listKnownBankAccounts: async () => ({ accounts: [ibanPayto] }),
|
||||||
|
getFeeForDeposit: withoutFee
|
||||||
|
} as Partial<typeof wxApi> as any)
|
||||||
|
);
|
||||||
|
|
||||||
|
{
|
||||||
|
const { status } = getLastResultOrThrow()
|
||||||
|
expect(status).equal("loading")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("0")
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:0`))
|
||||||
|
|
||||||
|
r.amount.onInput("10")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("10")
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:0`))
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
await assertNoPendingUpdate()
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should calculate the fee upon entering amount ", async () => {
|
||||||
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } = mountHook(() =>
|
||||||
|
useComponentState(currency, nullFunction, nullFunction, {
|
||||||
|
getBalance: async () => ({
|
||||||
|
balances: [{ available: `${currency}:1`, }]
|
||||||
|
} as Partial<BalancesResponse>),
|
||||||
|
listKnownBankAccounts: async () => ({ accounts: [ibanPayto] }),
|
||||||
|
getFeeForDeposit: withSomeFee
|
||||||
|
} as Partial<typeof wxApi> as any)
|
||||||
|
);
|
||||||
|
|
||||||
|
{
|
||||||
|
const { status } = getLastResultOrThrow()
|
||||||
|
expect(status).equal("loading")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("0")
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:0`))
|
||||||
|
|
||||||
|
r.amount.onInput("10")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("10")
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:3`))
|
||||||
|
expect(r.totalToDeposit).deep.eq(Amounts.parseOrThrow(`${currency}:7`))
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
await assertNoPendingUpdate()
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should calculate the fee upon selecting account ", async () => {
|
||||||
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } = mountHook(() =>
|
||||||
|
useComponentState(currency, nullFunction, nullFunction, {
|
||||||
|
getBalance: async () => ({
|
||||||
|
balances: [{ available: `${currency}:1`, }]
|
||||||
|
} as Partial<BalancesResponse>),
|
||||||
|
listKnownBankAccounts: async () => ({ accounts: [ibanPayto, talerBankPayto] }),
|
||||||
|
getFeeForDeposit: freeJustForIBAN
|
||||||
|
} as Partial<typeof wxApi> as any)
|
||||||
|
);
|
||||||
|
|
||||||
|
{
|
||||||
|
const { status } = getLastResultOrThrow()
|
||||||
|
expect(status).equal("loading")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("0")
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:0`))
|
||||||
|
|
||||||
|
r.account.onChange("1")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("1")
|
||||||
|
expect(r.amount.value).eq("0")
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:3`))
|
||||||
|
expect(r.totalToDeposit).deep.eq(Amounts.parseOrThrow(`${currency}:0`))
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
|
||||||
|
r.amount.onInput("10")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("1")
|
||||||
|
expect(r.amount.value).eq("10")
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:3`))
|
||||||
|
expect(r.totalToDeposit).deep.eq(Amounts.parseOrThrow(`${currency}:7`))
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
|
||||||
|
r.account.onChange("0")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("10")
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:0`))
|
||||||
|
expect(r.totalToDeposit).deep.eq(Amounts.parseOrThrow(`${currency}:10`))
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
await assertNoPendingUpdate()
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
it("should be able to deposit if has the enough balance ", async () => {
|
||||||
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } = mountHook(() =>
|
||||||
|
useComponentState(currency, nullFunction, nullFunction, {
|
||||||
|
getBalance: async () => ({
|
||||||
|
balances: [{ available: `${currency}:15`, }]
|
||||||
|
} as Partial<BalancesResponse>),
|
||||||
|
listKnownBankAccounts: async () => ({ accounts: [ibanPayto] }),
|
||||||
|
getFeeForDeposit: withSomeFee
|
||||||
|
} as Partial<typeof wxApi> as any)
|
||||||
|
);
|
||||||
|
|
||||||
|
{
|
||||||
|
const { status } = getLastResultOrThrow()
|
||||||
|
expect(status).equal("loading")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("0")
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:0`))
|
||||||
|
|
||||||
|
r.amount.onInput("10")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("10")
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:3`))
|
||||||
|
expect(r.totalToDeposit).deep.eq(Amounts.parseOrThrow(`${currency}:7`))
|
||||||
|
expect(r.depositHandler.onClick).not.undefined;
|
||||||
|
|
||||||
|
r.amount.onInput("13")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("13")
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:3`))
|
||||||
|
expect(r.totalToDeposit).deep.eq(Amounts.parseOrThrow(`${currency}:10`))
|
||||||
|
expect(r.depositHandler.onClick).not.undefined;
|
||||||
|
|
||||||
|
r.amount.onInput("15")
|
||||||
|
}
|
||||||
|
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("15")
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:3`))
|
||||||
|
expect(r.totalToDeposit).deep.eq(Amounts.parseOrThrow(`${currency}:12`))
|
||||||
|
expect(r.depositHandler.onClick).not.undefined;
|
||||||
|
r.amount.onInput("17")
|
||||||
|
}
|
||||||
|
await waitNextUpdate()
|
||||||
|
|
||||||
|
{
|
||||||
|
const r = getLastResultOrThrow()
|
||||||
|
if (r.status !== "ready") expect.fail();
|
||||||
|
expect(r.cancelHandler.onClick).not.undefined;
|
||||||
|
expect(r.currency).eq(currency);
|
||||||
|
expect(r.account.value).eq("0")
|
||||||
|
expect(r.amount.value).eq("17")
|
||||||
|
expect(r.totalFee).deep.eq(Amounts.parseOrThrow(`${currency}:3`))
|
||||||
|
expect(r.totalToDeposit).deep.eq(Amounts.parseOrThrow(`${currency}:14`))
|
||||||
|
expect(r.depositHandler.onClick).undefined;
|
||||||
|
}
|
||||||
|
await assertNoPendingUpdate()
|
||||||
|
});
|
||||||
|
|
||||||
});
|
});
|
@ -15,16 +15,10 @@
|
|||||||
TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {
|
import { AmountJson, Amounts, PaytoUri } from "@gnu-taler/taler-util";
|
||||||
AmountJson,
|
|
||||||
Amounts,
|
|
||||||
AmountString,
|
|
||||||
Balance,
|
|
||||||
PaytoUri,
|
|
||||||
} from "@gnu-taler/taler-util";
|
|
||||||
import { DepositGroupFees } from "@gnu-taler/taler-wallet-core/src/operations/deposits";
|
import { DepositGroupFees } from "@gnu-taler/taler-wallet-core/src/operations/deposits";
|
||||||
import { Fragment, h, VNode } from "preact";
|
import { Fragment, h, VNode } from "preact";
|
||||||
import { useEffect, useState } from "preact/hooks";
|
import { useState } from "preact/hooks";
|
||||||
import { Loading } from "../components/Loading.js";
|
import { Loading } from "../components/Loading.js";
|
||||||
import { LoadingError } from "../components/LoadingError.js";
|
import { LoadingError } from "../components/LoadingError.js";
|
||||||
import { SelectList } from "../components/SelectList.js";
|
import { SelectList } from "../components/SelectList.js";
|
||||||
@ -38,12 +32,13 @@ import {
|
|||||||
WarningBox,
|
WarningBox,
|
||||||
} from "../components/styled/index.js";
|
} from "../components/styled/index.js";
|
||||||
import { useTranslationContext } from "../context/translation.js";
|
import { useTranslationContext } from "../context/translation.js";
|
||||||
import { useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
import { HookError, useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
||||||
import * as wxApi from "../wxApi.js";
|
|
||||||
import {
|
import {
|
||||||
|
ButtonHandler,
|
||||||
SelectFieldHandler,
|
SelectFieldHandler,
|
||||||
TextFieldHandler,
|
TextFieldHandler,
|
||||||
} from "./CreateManualWithdraw.js";
|
} from "../mui/handlers.js";
|
||||||
|
import * as wxApi from "../wxApi.js";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
currency: string;
|
currency: string;
|
||||||
@ -51,119 +46,90 @@ interface Props {
|
|||||||
onSuccess: (currency: string) => void;
|
onSuccess: (currency: string) => void;
|
||||||
}
|
}
|
||||||
export function DepositPage({ currency, onCancel, onSuccess }: Props): VNode {
|
export function DepositPage({ currency, onCancel, onSuccess }: Props): VNode {
|
||||||
const state = useAsyncAsHook(async () => {
|
const state = useComponentState(currency, onCancel, onSuccess, wxApi);
|
||||||
const { balances } = await wxApi.getBalance();
|
|
||||||
const { accounts } = await wxApi.listKnownBankAccounts(currency);
|
|
||||||
return { accounts, balances };
|
|
||||||
});
|
|
||||||
|
|
||||||
const { i18n } = useTranslationContext();
|
return <View state={state} />;
|
||||||
|
|
||||||
async function doSend(p: PaytoUri, a: AmountJson): Promise<void> {
|
|
||||||
const account = `payto://${p.targetType}/${p.targetPath}`;
|
|
||||||
const amount = Amounts.stringify(a);
|
|
||||||
await wxApi.createDepositGroup(account, amount);
|
|
||||||
onSuccess(currency);
|
|
||||||
}
|
|
||||||
|
|
||||||
async function getFeeForAmount(
|
|
||||||
p: PaytoUri,
|
|
||||||
a: AmountJson,
|
|
||||||
): Promise<DepositGroupFees> {
|
|
||||||
const account = `payto://${p.targetType}/${p.targetPath}`;
|
|
||||||
const amount = Amounts.stringify(a);
|
|
||||||
return await wxApi.getFeeForDeposit(account, amount);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state === undefined) return <Loading />;
|
|
||||||
|
|
||||||
if (state.hasError) {
|
|
||||||
return (
|
|
||||||
<LoadingError
|
|
||||||
title={<i18n.Translate>Could not load deposit balance</i18n.Translate>}
|
|
||||||
error={state}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<View
|
|
||||||
onCancel={() => onCancel(currency)}
|
|
||||||
currency={currency}
|
|
||||||
accounts={state.response.accounts}
|
|
||||||
balances={state.response.balances}
|
|
||||||
onSend={doSend}
|
|
||||||
onCalculateFee={getFeeForAmount}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
interface ViewProps {
|
interface ViewProps {
|
||||||
accounts: Array<PaytoUri>;
|
state: State;
|
||||||
currency: string;
|
|
||||||
balances: Balance[];
|
|
||||||
onCancel: () => void;
|
|
||||||
onSend: (account: PaytoUri, amount: AmountJson) => Promise<void>;
|
|
||||||
onCalculateFee: (
|
|
||||||
account: PaytoUri,
|
|
||||||
amount: AmountJson,
|
|
||||||
) => Promise<DepositGroupFees>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type State = NoBalanceState | NoAccountsState | DepositState;
|
type State = Loading | NoBalanceState | NoAccountsState | DepositState;
|
||||||
|
|
||||||
|
interface Loading {
|
||||||
|
status: "loading";
|
||||||
|
hook: HookError | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
interface NoBalanceState {
|
interface NoBalanceState {
|
||||||
status: "no-balance";
|
status: "no-balance";
|
||||||
}
|
}
|
||||||
interface NoAccountsState {
|
interface NoAccountsState {
|
||||||
status: "no-accounts";
|
status: "no-accounts";
|
||||||
|
cancelHandler: ButtonHandler;
|
||||||
}
|
}
|
||||||
interface DepositState {
|
interface DepositState {
|
||||||
status: "deposit";
|
status: "ready";
|
||||||
|
currency: string;
|
||||||
amount: TextFieldHandler;
|
amount: TextFieldHandler;
|
||||||
account: SelectFieldHandler;
|
account: SelectFieldHandler;
|
||||||
totalFee: AmountJson;
|
totalFee: AmountJson;
|
||||||
totalToDeposit: AmountJson;
|
totalToDeposit: AmountJson;
|
||||||
unableToDeposit: boolean;
|
// currentAccount: PaytoUri;
|
||||||
selectedAccount: PaytoUri;
|
// parsedAmount: AmountJson | undefined;
|
||||||
parsedAmount: AmountJson | undefined;
|
cancelHandler: ButtonHandler;
|
||||||
|
depositHandler: ButtonHandler;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getFeeForAmount(
|
||||||
|
p: PaytoUri,
|
||||||
|
a: AmountJson,
|
||||||
|
api: typeof wxApi,
|
||||||
|
): Promise<DepositGroupFees> {
|
||||||
|
const account = `payto://${p.targetType}/${p.targetPath}`;
|
||||||
|
const amount = Amounts.stringify(a);
|
||||||
|
return await api.getFeeForDeposit(account, amount);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function useComponentState(
|
export function useComponentState(
|
||||||
currency: string,
|
currency: string,
|
||||||
accounts: PaytoUri[],
|
onCancel: (currency: string) => void,
|
||||||
balances: Balance[],
|
onSuccess: (currency: string) => void,
|
||||||
onCalculateFee: (
|
api: typeof wxApi,
|
||||||
account: PaytoUri,
|
|
||||||
amount: AmountJson,
|
|
||||||
) => Promise<DepositGroupFees>,
|
|
||||||
): State {
|
): State {
|
||||||
const accountMap = createLabelsForBankAccount(accounts);
|
const hook = useAsyncAsHook(async () => {
|
||||||
|
const { balances } = await api.getBalance();
|
||||||
|
const { accounts } = await api.listKnownBankAccounts(currency);
|
||||||
|
const defaultSelectedAccount =
|
||||||
|
accounts.length > 0 ? accounts[0] : undefined;
|
||||||
|
return { accounts, balances, defaultSelectedAccount };
|
||||||
|
});
|
||||||
|
|
||||||
const [accountIdx, setAccountIdx] = useState(0);
|
const [accountIdx, setAccountIdx] = useState(0);
|
||||||
const [amount, setAmount] = useState<number | undefined>(undefined);
|
const [amount, setAmount] = useState<number>(0);
|
||||||
|
|
||||||
|
const [selectedAccount, setSelectedAccount] = useState<
|
||||||
|
PaytoUri | undefined
|
||||||
|
>();
|
||||||
|
|
||||||
|
const parsedAmount = Amounts.parse(`${currency}:${amount}`);
|
||||||
|
|
||||||
const [fee, setFee] = useState<DepositGroupFees | undefined>(undefined);
|
const [fee, setFee] = useState<DepositGroupFees | undefined>(undefined);
|
||||||
function updateAmount(num: number | undefined): void {
|
|
||||||
setAmount(num);
|
// const hookResponse = !hook || hook.hasError ? undefined : hook.response;
|
||||||
setFee(undefined);
|
|
||||||
|
// useEffect(() => {}, [hookResponse]);
|
||||||
|
|
||||||
|
if (!hook || hook.hasError) {
|
||||||
|
return {
|
||||||
|
status: "loading",
|
||||||
|
hook,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const selectedAmountSTR: AmountString = `${currency}:${amount}`;
|
const { accounts, balances, defaultSelectedAccount } = hook.response;
|
||||||
const totalFee =
|
const currentAccount = selectedAccount ?? defaultSelectedAccount;
|
||||||
fee !== undefined
|
|
||||||
? Amounts.sum([fee.wire, fee.coin, fee.refresh]).amount
|
|
||||||
: Amounts.getZero(currency);
|
|
||||||
|
|
||||||
const selectedAccount = accounts.length ? accounts[accountIdx] : undefined;
|
|
||||||
|
|
||||||
const parsedAmount =
|
|
||||||
amount === undefined ? undefined : Amounts.parse(selectedAmountSTR);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (selectedAccount === undefined || parsedAmount === undefined) return;
|
|
||||||
onCalculateFee(selectedAccount, parsedAmount).then((result) => {
|
|
||||||
setFee(result);
|
|
||||||
});
|
|
||||||
}, [amount, selectedAccount, parsedAmount, onCalculateFee]);
|
|
||||||
|
|
||||||
const bs = balances.filter((b) => b.available.startsWith(currency));
|
const bs = balances.filter((b) => b.available.startsWith(currency));
|
||||||
const balance =
|
const balance =
|
||||||
@ -171,6 +137,63 @@ export function useComponentState(
|
|||||||
? Amounts.parseOrThrow(bs[0].available)
|
? Amounts.parseOrThrow(bs[0].available)
|
||||||
: Amounts.getZero(currency);
|
: Amounts.getZero(currency);
|
||||||
|
|
||||||
|
if (Amounts.isZero(balance)) {
|
||||||
|
return {
|
||||||
|
status: "no-balance",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!currentAccount) {
|
||||||
|
return {
|
||||||
|
status: "no-accounts",
|
||||||
|
cancelHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
onCancel(currency);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
const accountMap = createLabelsForBankAccount(accounts);
|
||||||
|
|
||||||
|
async function updateAccount(accountStr: string): Promise<void> {
|
||||||
|
const idx = parseInt(accountStr, 10);
|
||||||
|
const newSelected = accounts.length > idx ? accounts[idx] : undefined;
|
||||||
|
if (accountIdx === idx || !newSelected) return;
|
||||||
|
|
||||||
|
if (!parsedAmount) {
|
||||||
|
setAccountIdx(idx);
|
||||||
|
setSelectedAccount(newSelected);
|
||||||
|
} else {
|
||||||
|
const result = await getFeeForAmount(newSelected, parsedAmount, api);
|
||||||
|
setAccountIdx(idx);
|
||||||
|
setSelectedAccount(newSelected);
|
||||||
|
setFee(result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function updateAmount(numStr: string): Promise<void> {
|
||||||
|
const num = parseFloat(numStr);
|
||||||
|
const newAmount = Number.isNaN(num) ? 0 : num;
|
||||||
|
if (amount === newAmount || !currentAccount) return;
|
||||||
|
const parsed = Amounts.parse(`${currency}:${newAmount}`);
|
||||||
|
if (!parsed) {
|
||||||
|
setAmount(newAmount);
|
||||||
|
} else {
|
||||||
|
const result = await getFeeForAmount(currentAccount, parsed, api);
|
||||||
|
setAmount(newAmount);
|
||||||
|
setFee(result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const totalFee =
|
||||||
|
fee !== undefined
|
||||||
|
? Amounts.sum([fee.wire, fee.coin, fee.refresh]).amount
|
||||||
|
: Amounts.getZero(currency);
|
||||||
|
|
||||||
|
const totalToDeposit = parsedAmount
|
||||||
|
? Amounts.sub(parsedAmount, totalFee).amount
|
||||||
|
: Amounts.getZero(currency);
|
||||||
|
|
||||||
const isDirty = amount !== 0;
|
const isDirty = amount !== 0;
|
||||||
const amountError = !isDirty
|
const amountError = !isDirty
|
||||||
? undefined
|
? undefined
|
||||||
@ -180,65 +203,63 @@ export function useComponentState(
|
|||||||
? `Too much, your current balance is ${Amounts.stringifyValue(balance)}`
|
? `Too much, your current balance is ${Amounts.stringifyValue(balance)}`
|
||||||
: undefined;
|
: undefined;
|
||||||
|
|
||||||
const totalToDeposit = parsedAmount
|
|
||||||
? Amounts.sub(parsedAmount, totalFee).amount
|
|
||||||
: Amounts.getZero(currency);
|
|
||||||
|
|
||||||
const unableToDeposit =
|
const unableToDeposit =
|
||||||
|
!parsedAmount ||
|
||||||
Amounts.isZero(totalToDeposit) ||
|
Amounts.isZero(totalToDeposit) ||
|
||||||
fee === undefined ||
|
fee === undefined ||
|
||||||
amountError !== undefined;
|
amountError !== undefined;
|
||||||
|
|
||||||
if (Amounts.isZero(balance)) {
|
async function doSend(): Promise<void> {
|
||||||
return {
|
if (!currentAccount || !parsedAmount) return;
|
||||||
status: "no-balance",
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!accounts || !accounts.length || !selectedAccount) {
|
const account = `payto://${currentAccount.targetType}/${currentAccount.targetPath}`;
|
||||||
return {
|
const amount = Amounts.stringify(parsedAmount);
|
||||||
status: "no-accounts",
|
await api.createDepositGroup(account, amount);
|
||||||
};
|
onSuccess(currency);
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
status: "deposit",
|
status: "ready",
|
||||||
|
currency,
|
||||||
amount: {
|
amount: {
|
||||||
value: String(amount),
|
value: String(amount),
|
||||||
onInput: (e) => {
|
onInput: updateAmount,
|
||||||
const num = parseFloat(e);
|
|
||||||
if (!Number.isNaN(num)) {
|
|
||||||
updateAmount(num);
|
|
||||||
} else {
|
|
||||||
updateAmount(undefined);
|
|
||||||
setFee(undefined);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
error: amountError,
|
error: amountError,
|
||||||
},
|
},
|
||||||
account: {
|
account: {
|
||||||
list: accountMap,
|
list: accountMap,
|
||||||
value: String(accountIdx),
|
value: String(accountIdx),
|
||||||
onChange: (s) => setAccountIdx(parseInt(s, 10)),
|
onChange: updateAccount,
|
||||||
|
},
|
||||||
|
cancelHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
onCancel(currency);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
depositHandler: {
|
||||||
|
onClick: unableToDeposit ? undefined : doSend,
|
||||||
},
|
},
|
||||||
totalFee,
|
totalFee,
|
||||||
totalToDeposit,
|
totalToDeposit,
|
||||||
unableToDeposit,
|
// currentAccount,
|
||||||
selectedAccount,
|
// parsedAmount,
|
||||||
parsedAmount,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function View({
|
export function View({ state }: ViewProps): VNode {
|
||||||
onCancel,
|
|
||||||
currency,
|
|
||||||
accounts,
|
|
||||||
balances,
|
|
||||||
onSend,
|
|
||||||
onCalculateFee,
|
|
||||||
}: ViewProps): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
const { i18n } = useTranslationContext();
|
||||||
const state = useComponentState(currency, accounts, balances, onCalculateFee);
|
|
||||||
|
if (state === undefined) return <Loading />;
|
||||||
|
|
||||||
|
if (state.status === "loading") {
|
||||||
|
if (!state.hook) return <Loading />;
|
||||||
|
return (
|
||||||
|
<LoadingError
|
||||||
|
title={<i18n.Translate>Could not load deposit balance</i18n.Translate>}
|
||||||
|
error={state.hook}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (state.status === "no-balance") {
|
if (state.status === "no-balance") {
|
||||||
return (
|
return (
|
||||||
@ -258,7 +279,7 @@ export function View({
|
|||||||
</p>
|
</p>
|
||||||
</WarningBox>
|
</WarningBox>
|
||||||
<footer>
|
<footer>
|
||||||
<Button onClick={onCancel}>
|
<Button onClick={state.cancelHandler.onClick}>
|
||||||
<i18n.Translate>Cancel</i18n.Translate>
|
<i18n.Translate>Cancel</i18n.Translate>
|
||||||
</Button>
|
</Button>
|
||||||
</footer>
|
</footer>
|
||||||
@ -269,7 +290,7 @@ export function View({
|
|||||||
return (
|
return (
|
||||||
<Fragment>
|
<Fragment>
|
||||||
<SubTitle>
|
<SubTitle>
|
||||||
<i18n.Translate>Send {currency} to your account</i18n.Translate>
|
<i18n.Translate>Send {state.currency} to your account</i18n.Translate>
|
||||||
</SubTitle>
|
</SubTitle>
|
||||||
<section>
|
<section>
|
||||||
<Input>
|
<Input>
|
||||||
@ -286,7 +307,7 @@ export function View({
|
|||||||
<i18n.Translate>Amount</i18n.Translate>
|
<i18n.Translate>Amount</i18n.Translate>
|
||||||
</label>
|
</label>
|
||||||
<div>
|
<div>
|
||||||
<span>{currency}</span>
|
<span>{state.currency}</span>
|
||||||
<input
|
<input
|
||||||
type="number"
|
type="number"
|
||||||
value={state.amount.value}
|
value={state.amount.value}
|
||||||
@ -302,7 +323,7 @@ export function View({
|
|||||||
<i18n.Translate>Deposit fee</i18n.Translate>
|
<i18n.Translate>Deposit fee</i18n.Translate>
|
||||||
</label>
|
</label>
|
||||||
<div>
|
<div>
|
||||||
<span>{currency}</span>
|
<span>{state.currency}</span>
|
||||||
<input
|
<input
|
||||||
type="number"
|
type="number"
|
||||||
disabled
|
disabled
|
||||||
@ -316,7 +337,7 @@ export function View({
|
|||||||
<i18n.Translate>Total deposit</i18n.Translate>
|
<i18n.Translate>Total deposit</i18n.Translate>
|
||||||
</label>
|
</label>
|
||||||
<div>
|
<div>
|
||||||
<span>{currency}</span>
|
<span>{state.currency}</span>
|
||||||
<input
|
<input
|
||||||
type="number"
|
type="number"
|
||||||
disabled
|
disabled
|
||||||
@ -328,19 +349,18 @@ export function View({
|
|||||||
}
|
}
|
||||||
</section>
|
</section>
|
||||||
<footer>
|
<footer>
|
||||||
<Button onClick={onCancel}>
|
<Button onClick={state.cancelHandler.onClick}>
|
||||||
<i18n.Translate>Cancel</i18n.Translate>
|
<i18n.Translate>Cancel</i18n.Translate>
|
||||||
</Button>
|
</Button>
|
||||||
{state.unableToDeposit ? (
|
{!state.depositHandler.onClick ? (
|
||||||
<ButtonPrimary disabled>
|
<ButtonPrimary disabled>
|
||||||
<i18n.Translate>Deposit</i18n.Translate>
|
<i18n.Translate>Deposit</i18n.Translate>
|
||||||
</ButtonPrimary>
|
</ButtonPrimary>
|
||||||
) : (
|
) : (
|
||||||
<ButtonPrimary
|
<ButtonPrimary onClick={state.depositHandler.onClick}>
|
||||||
onClick={() => onSend(state.selectedAccount, state.parsedAmount!)}
|
|
||||||
>
|
|
||||||
<i18n.Translate>
|
<i18n.Translate>
|
||||||
Deposit {Amounts.stringifyValue(state.totalToDeposit)} {currency}
|
Deposit {Amounts.stringifyValue(state.totalToDeposit)}{" "}
|
||||||
|
{state.currency}
|
||||||
</i18n.Translate>
|
</i18n.Translate>
|
||||||
</ButtonPrimary>
|
</ButtonPrimary>
|
||||||
)}
|
)}
|
||||||
@ -349,7 +369,9 @@ export function View({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function createLabelsForBankAccount(knownBankAccounts: Array<PaytoUri>): {
|
export function createLabelsForBankAccount(
|
||||||
|
knownBankAccounts: Array<PaytoUri>,
|
||||||
|
): {
|
||||||
[label: number]: string;
|
[label: number]: string;
|
||||||
} {
|
} {
|
||||||
if (!knownBankAccounts) return {};
|
if (!knownBankAccounts) return {};
|
||||||
|
@ -349,6 +349,7 @@ importers:
|
|||||||
babel-loader: ^8.2.3
|
babel-loader: ^8.2.3
|
||||||
babel-plugin-transform-react-jsx: ^6.24.1
|
babel-plugin-transform-react-jsx: ^6.24.1
|
||||||
chai: ^4.3.6
|
chai: ^4.3.6
|
||||||
|
chokidar: ^3.5.3
|
||||||
date-fns: ^2.28.0
|
date-fns: ^2.28.0
|
||||||
history: 4.10.1
|
history: 4.10.1
|
||||||
mocha: ^9.2.0
|
mocha: ^9.2.0
|
||||||
@ -367,6 +368,7 @@ importers:
|
|||||||
rollup-plugin-terser: ^7.0.2
|
rollup-plugin-terser: ^7.0.2
|
||||||
tslib: ^2.3.1
|
tslib: ^2.3.1
|
||||||
typescript: ^4.5.5
|
typescript: ^4.5.5
|
||||||
|
ws: 7.4.5
|
||||||
dependencies:
|
dependencies:
|
||||||
'@gnu-taler/taler-util': link:../taler-util
|
'@gnu-taler/taler-util': link:../taler-util
|
||||||
'@gnu-taler/taler-wallet-core': link:../taler-wallet-core
|
'@gnu-taler/taler-wallet-core': link:../taler-wallet-core
|
||||||
@ -376,6 +378,7 @@ importers:
|
|||||||
preact-router: 3.2.1_preact@10.6.5
|
preact-router: 3.2.1_preact@10.6.5
|
||||||
qrcode-generator: 1.4.4
|
qrcode-generator: 1.4.4
|
||||||
tslib: 2.3.1
|
tslib: 2.3.1
|
||||||
|
ws: 7.4.5
|
||||||
devDependencies:
|
devDependencies:
|
||||||
'@babel/core': 7.13.16
|
'@babel/core': 7.13.16
|
||||||
'@babel/plugin-transform-react-jsx-source': 7.14.5_@babel+core@7.13.16
|
'@babel/plugin-transform-react-jsx-source': 7.14.5_@babel+core@7.13.16
|
||||||
@ -404,6 +407,7 @@ importers:
|
|||||||
babel-loader: 8.2.3_@babel+core@7.13.16
|
babel-loader: 8.2.3_@babel+core@7.13.16
|
||||||
babel-plugin-transform-react-jsx: 6.24.1
|
babel-plugin-transform-react-jsx: 6.24.1
|
||||||
chai: 4.3.6
|
chai: 4.3.6
|
||||||
|
chokidar: 3.5.3
|
||||||
mocha: 9.2.0
|
mocha: 9.2.0
|
||||||
nyc: 15.1.0
|
nyc: 15.1.0
|
||||||
polished: 4.1.4
|
polished: 4.1.4
|
||||||
@ -19088,6 +19092,19 @@ packages:
|
|||||||
async-limiter: 1.0.1
|
async-limiter: 1.0.1
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
|
/ws/7.4.5:
|
||||||
|
resolution: {integrity: sha512-xzyu3hFvomRfXKH8vOFMU3OguG6oOvhXMo3xsGy3xWExqaM2dxBbVxuD99O7m3ZUFMvvscsZDqxfgMaRr/Nr1g==}
|
||||||
|
engines: {node: '>=8.3.0'}
|
||||||
|
peerDependencies:
|
||||||
|
bufferutil: ^4.0.1
|
||||||
|
utf-8-validate: ^5.0.2
|
||||||
|
peerDependenciesMeta:
|
||||||
|
bufferutil:
|
||||||
|
optional: true
|
||||||
|
utf-8-validate:
|
||||||
|
optional: true
|
||||||
|
dev: false
|
||||||
|
|
||||||
/ws/7.5.7:
|
/ws/7.5.7:
|
||||||
resolution: {integrity: sha512-KMvVuFzpKBuiIXW3E4u3mySRO2/mCHSyZDJQM5NQ9Q9KHWHWh0NHgfbRMLLrceUK5qAL4ytALJbpRMjixFZh8A==}
|
resolution: {integrity: sha512-KMvVuFzpKBuiIXW3E4u3mySRO2/mCHSyZDJQM5NQ9Q9KHWHWh0NHgfbRMLLrceUK5qAL4ytALJbpRMjixFZh8A==}
|
||||||
engines: {node: '>=8.3.0'}
|
engines: {node: '>=8.3.0'}
|
||||||
|
Loading…
Reference in New Issue
Block a user