standarizing components
This commit is contained in:
parent
979cd2daf2
commit
614a3e3c87
@ -9,7 +9,7 @@
|
|||||||
"private": false,
|
"private": false,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"clean": "rimraf dist lib tsconfig.tsbuildinfo",
|
"clean": "rimraf dist lib tsconfig.tsbuildinfo",
|
||||||
"test": "pnpm compile && mocha --enable-source-maps 'dist/**/*.test.js'",
|
"test": "pnpm compile && mocha --enable-source-maps 'dist/**/*.test.js' 'dist/**/test.js'",
|
||||||
"test:coverage": "nyc pnpm test",
|
"test:coverage": "nyc pnpm test",
|
||||||
"compile": "tsc && ./build-fast-with-linaria.mjs",
|
"compile": "tsc && ./build-fast-with-linaria.mjs",
|
||||||
"prepare": "pnpm compile",
|
"prepare": "pnpm compile",
|
||||||
|
@ -1,221 +0,0 @@
|
|||||||
/*
|
|
||||||
This file is part of GNU Taler
|
|
||||||
(C) 2022 Taler Systems S.A.
|
|
||||||
|
|
||||||
GNU 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.
|
|
||||||
|
|
||||||
GNU 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
|
|
||||||
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Page shown to the user to confirm entering
|
|
||||||
* a contract.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Imports.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import {
|
|
||||||
AmountJson,
|
|
||||||
Amounts,
|
|
||||||
AmountString,
|
|
||||||
CreateDepositGroupResponse,
|
|
||||||
} from "@gnu-taler/taler-util";
|
|
||||||
import { Fragment, h, VNode } from "preact";
|
|
||||||
import { useState } from "preact/hooks";
|
|
||||||
import { Amount } from "../components/Amount.js";
|
|
||||||
import { Loading } from "../components/Loading.js";
|
|
||||||
import { LoadingError } from "../components/LoadingError.js";
|
|
||||||
import { LogoHeader } from "../components/LogoHeader.js";
|
|
||||||
import { Part } from "../components/Part.js";
|
|
||||||
import {
|
|
||||||
ButtonSuccess,
|
|
||||||
SubTitle,
|
|
||||||
WalletAction,
|
|
||||||
} from "../components/styled/index.js";
|
|
||||||
import { useTranslationContext } from "../context/translation.js";
|
|
||||||
import { HookError, useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
|
||||||
import { Button } from "../mui/Button.js";
|
|
||||||
import { ButtonHandler } from "../mui/handlers.js";
|
|
||||||
import * as wxApi from "../wxApi.js";
|
|
||||||
|
|
||||||
interface Props {
|
|
||||||
talerDepositUri?: string;
|
|
||||||
amount: AmountString;
|
|
||||||
goBack: () => Promise<void>;
|
|
||||||
}
|
|
||||||
|
|
||||||
type State = Loading | Ready | Completed;
|
|
||||||
interface Loading {
|
|
||||||
status: "loading";
|
|
||||||
hook: HookError | undefined;
|
|
||||||
}
|
|
||||||
interface Ready {
|
|
||||||
status: "ready";
|
|
||||||
hook: undefined;
|
|
||||||
fee: AmountJson;
|
|
||||||
cost: AmountJson;
|
|
||||||
effective: AmountJson;
|
|
||||||
confirm: ButtonHandler;
|
|
||||||
}
|
|
||||||
interface Completed {
|
|
||||||
status: "completed";
|
|
||||||
hook: undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useComponentState(
|
|
||||||
talerDepositUri: string | undefined,
|
|
||||||
amountStr: AmountString | undefined,
|
|
||||||
api: typeof wxApi,
|
|
||||||
): State {
|
|
||||||
const [result, setResult] = useState<CreateDepositGroupResponse | undefined>(
|
|
||||||
undefined,
|
|
||||||
);
|
|
||||||
|
|
||||||
const info = useAsyncAsHook(async () => {
|
|
||||||
if (!talerDepositUri) throw Error("ERROR_NO-URI-FOR-DEPOSIT");
|
|
||||||
if (!amountStr) throw Error("ERROR_NO-AMOUNT-FOR-DEPOSIT");
|
|
||||||
const amount = Amounts.parse(amountStr);
|
|
||||||
if (!amount) throw Error("ERROR_INVALID-AMOUNT-FOR-DEPOSIT");
|
|
||||||
const deposit = await api.prepareDeposit(
|
|
||||||
talerDepositUri,
|
|
||||||
Amounts.stringify(amount),
|
|
||||||
);
|
|
||||||
return { deposit, uri: talerDepositUri, amount };
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!info || info.hasError) {
|
|
||||||
return {
|
|
||||||
status: "loading",
|
|
||||||
hook: info,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const { deposit, uri, amount } = info.response;
|
|
||||||
async function doDeposit(): Promise<void> {
|
|
||||||
const resp = await api.createDepositGroup(uri, Amounts.stringify(amount));
|
|
||||||
setResult(resp);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (result !== undefined) {
|
|
||||||
return {
|
|
||||||
status: "completed",
|
|
||||||
hook: undefined,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
confirm: {
|
|
||||||
onClick: doDeposit,
|
|
||||||
},
|
|
||||||
fee: Amounts.sub(deposit.totalDepositCost, deposit.effectiveDepositAmount)
|
|
||||||
.amount,
|
|
||||||
cost: deposit.totalDepositCost,
|
|
||||||
effective: deposit.effectiveDepositAmount,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function DepositPage({ talerDepositUri, amount, goBack }: Props): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
|
||||||
|
|
||||||
const state = useComponentState(talerDepositUri, amount, wxApi);
|
|
||||||
|
|
||||||
if (!talerDepositUri) {
|
|
||||||
return (
|
|
||||||
<span>
|
|
||||||
<i18n.Translate>missing taler deposit uri</i18n.Translate>
|
|
||||||
</span>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return <View state={state} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface ViewProps {
|
|
||||||
state: State;
|
|
||||||
}
|
|
||||||
export function View({ state }: ViewProps): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
|
||||||
|
|
||||||
if (state.status === "loading") {
|
|
||||||
if (!state.hook) return <Loading />;
|
|
||||||
return (
|
|
||||||
<LoadingError
|
|
||||||
title={<i18n.Translate>Could not load deposit status</i18n.Translate>}
|
|
||||||
error={state.hook}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state.status === "completed") {
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash deposit</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<section>
|
|
||||||
<p>
|
|
||||||
<i18n.Translate>deposit completed</i18n.Translate>
|
|
||||||
</p>
|
|
||||||
</section>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash deposit</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<section>
|
|
||||||
{Amounts.isNonZero(state.cost) && (
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Cost</i18n.Translate>}
|
|
||||||
text={<Amount value={state.cost} />}
|
|
||||||
kind="negative"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
{Amounts.isNonZero(state.fee) && (
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Fee</i18n.Translate>}
|
|
||||||
text={<Amount value={state.fee} />}
|
|
||||||
kind="negative"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>To be received</i18n.Translate>}
|
|
||||||
text={<Amount value={state.effective} />}
|
|
||||||
kind="positive"
|
|
||||||
/>
|
|
||||||
</section>
|
|
||||||
<section>
|
|
||||||
<Button
|
|
||||||
variant="contained"
|
|
||||||
color="success"
|
|
||||||
onClick={state.confirm.onClick}
|
|
||||||
>
|
|
||||||
<i18n.Translate>
|
|
||||||
Deposit {<Amount value={state.effective} />}
|
|
||||||
</i18n.Translate>
|
|
||||||
</Button>
|
|
||||||
</section>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
70
packages/taler-wallet-webextension/src/cta/Deposit/index.ts
Normal file
70
packages/taler-wallet-webextension/src/cta/Deposit/index.ts
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { AmountJson, AmountString } from "@gnu-taler/taler-util";
|
||||||
|
import { Loading } from "../../components/Loading.js";
|
||||||
|
import { HookError } from "../../hooks/useAsyncAsHook.js";
|
||||||
|
import { ButtonHandler } from "../../mui/handlers.js";
|
||||||
|
import { compose, StateViewMap } from "../../utils/index.js";
|
||||||
|
import * as wxApi from "../../wxApi.js";
|
||||||
|
import { useComponentState } from "./state.js";
|
||||||
|
import { CompletedView, LoadingUriView, ReadyView } from "./views.js";
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
export interface Props {
|
||||||
|
talerDepositUri: string | undefined,
|
||||||
|
amountStr: AmountString | undefined,
|
||||||
|
}
|
||||||
|
|
||||||
|
export type State =
|
||||||
|
| State.Loading
|
||||||
|
| State.LoadingUriError
|
||||||
|
| State.Ready
|
||||||
|
| State.Completed;
|
||||||
|
|
||||||
|
export namespace State {
|
||||||
|
|
||||||
|
export interface Loading {
|
||||||
|
status: "loading";
|
||||||
|
error: undefined;
|
||||||
|
}
|
||||||
|
export interface LoadingUriError {
|
||||||
|
status: "loading-uri";
|
||||||
|
error: HookError;
|
||||||
|
}
|
||||||
|
export interface Ready {
|
||||||
|
status: "ready";
|
||||||
|
error: undefined;
|
||||||
|
fee: AmountJson;
|
||||||
|
cost: AmountJson;
|
||||||
|
effective: AmountJson;
|
||||||
|
confirm: ButtonHandler;
|
||||||
|
}
|
||||||
|
export interface Completed {
|
||||||
|
status: "completed";
|
||||||
|
error: undefined;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const viewMapping: StateViewMap<State> = {
|
||||||
|
"loading": Loading,
|
||||||
|
"loading-uri": LoadingUriView,
|
||||||
|
completed: CompletedView,
|
||||||
|
ready: ReadyView,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const DepositPage = compose("Deposit", (p: Props) => useComponentState(p, wxApi), viewMapping)
|
76
packages/taler-wallet-webextension/src/cta/Deposit/state.ts
Normal file
76
packages/taler-wallet-webextension/src/cta/Deposit/state.ts
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
import { Amounts, CreateDepositGroupResponse } from "@gnu-taler/taler-util";
|
||||||
|
import { useState } from "preact/hooks";
|
||||||
|
import { useAsyncAsHook } from "../../hooks/useAsyncAsHook.js";
|
||||||
|
import * as wxApi from "../../wxApi.js";
|
||||||
|
import { Props, State } from "./index.js";
|
||||||
|
|
||||||
|
export function useComponentState(
|
||||||
|
{ talerDepositUri, amountStr }: Props,
|
||||||
|
api: typeof wxApi,
|
||||||
|
): State {
|
||||||
|
const [result, setResult] = useState<CreateDepositGroupResponse | undefined>(
|
||||||
|
undefined,
|
||||||
|
);
|
||||||
|
|
||||||
|
const info = useAsyncAsHook(async () => {
|
||||||
|
if (!talerDepositUri) throw Error("ERROR_NO-URI-FOR-DEPOSIT");
|
||||||
|
if (!amountStr) throw Error("ERROR_NO-AMOUNT-FOR-DEPOSIT");
|
||||||
|
const amount = Amounts.parse(amountStr);
|
||||||
|
if (!amount) throw Error("ERROR_INVALID-AMOUNT-FOR-DEPOSIT");
|
||||||
|
const deposit = await api.prepareDeposit(
|
||||||
|
talerDepositUri,
|
||||||
|
Amounts.stringify(amount),
|
||||||
|
);
|
||||||
|
return { deposit, uri: talerDepositUri, amount };
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!info) return { status: "loading", error: undefined }
|
||||||
|
if (info.hasError) {
|
||||||
|
return {
|
||||||
|
status: "loading-uri",
|
||||||
|
error: info,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const { deposit, uri, amount } = info.response;
|
||||||
|
async function doDeposit(): Promise<void> {
|
||||||
|
const resp = await api.createDepositGroup(uri, Amounts.stringify(amount));
|
||||||
|
setResult(resp);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result !== undefined) {
|
||||||
|
return {
|
||||||
|
status: "completed",
|
||||||
|
error: undefined,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
confirm: {
|
||||||
|
onClick: doDeposit,
|
||||||
|
},
|
||||||
|
fee: Amounts.sub(deposit.totalDepositCost, deposit.effectiveDepositAmount)
|
||||||
|
.amount,
|
||||||
|
cost: deposit.totalDepositCost,
|
||||||
|
effective: deposit.effectiveDepositAmount,
|
||||||
|
};
|
||||||
|
}
|
@ -20,22 +20,18 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Amounts } from "@gnu-taler/taler-util";
|
import { Amounts } from "@gnu-taler/taler-util";
|
||||||
import { createExample } from "../test-utils.js";
|
import { createExample } from "../../test-utils.js";
|
||||||
import { View as TestedComponent } from "./Deposit.js";
|
import { ReadyView } from "./views.js";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
title: "cta/deposit",
|
title: "cta/deposit",
|
||||||
component: TestedComponent,
|
|
||||||
argTypes: {},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Ready = createExample(TestedComponent, {
|
export const Ready = createExample(ReadyView, {
|
||||||
state: {
|
|
||||||
status: "ready",
|
status: "ready",
|
||||||
confirm: {},
|
confirm: {},
|
||||||
cost: Amounts.parseOrThrow("EUR:1.2"),
|
cost: Amounts.parseOrThrow("EUR:1.2"),
|
||||||
effective: Amounts.parseOrThrow("EUR:1"),
|
effective: Amounts.parseOrThrow("EUR:1"),
|
||||||
fee: Amounts.parseOrThrow("EUR:0.2"),
|
fee: Amounts.parseOrThrow("EUR:0.2"),
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
},
|
|
||||||
});
|
});
|
@ -19,16 +19,18 @@
|
|||||||
* @author Sebastian Javier Marchano (sebasjm)
|
* @author Sebastian Javier Marchano (sebasjm)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { Amounts, PrepareDepositResponse } from "@gnu-taler/taler-util";
|
import {
|
||||||
|
Amounts, PrepareDepositResponse
|
||||||
|
} from "@gnu-taler/taler-util";
|
||||||
import { expect } from "chai";
|
import { expect } from "chai";
|
||||||
import { mountHook } from "../test-utils.js";
|
import { mountHook } from "../../test-utils.js";
|
||||||
import { useComponentState } from "./Deposit.jsx";
|
import { useComponentState } from "./state.js";
|
||||||
|
|
||||||
describe("Deposit CTA states", () => {
|
describe("Deposit CTA states", () => {
|
||||||
it("should tell the user that the URI is missing", async () => {
|
it("should tell the user that the URI is missing", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState(undefined, undefined, {
|
useComponentState({ talerDepositUri: undefined, amountStr: undefined }, {
|
||||||
prepareRefund: async () => ({}),
|
prepareRefund: async () => ({}),
|
||||||
applyRefund: async () => ({}),
|
applyRefund: async () => ({}),
|
||||||
onUpdateNotification: async () => ({}),
|
onUpdateNotification: async () => ({}),
|
||||||
@ -36,21 +38,21 @@ describe("Deposit CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
|
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading-uri");
|
||||||
if (!hook) expect.fail();
|
|
||||||
if (!hook.hasError) expect.fail();
|
if (!error) expect.fail();
|
||||||
if (hook.operational) expect.fail();
|
if (!error.hasError) expect.fail();
|
||||||
expect(hook.message).eq("ERROR_NO-URI-FOR-DEPOSIT");
|
if (error.operational) expect.fail();
|
||||||
|
expect(error.message).eq("ERROR_NO-URI-FOR-DEPOSIT");
|
||||||
}
|
}
|
||||||
|
|
||||||
await assertNoPendingUpdate();
|
await assertNoPendingUpdate();
|
||||||
@ -59,7 +61,7 @@ describe("Deposit CTA states", () => {
|
|||||||
it("should be ready after loading", async () => {
|
it("should be ready after loading", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("payto://refund/asdasdas", "EUR:1", {
|
useComponentState({ talerDepositUri: "payto://refund/asdasdas", amountStr: "EUR:1" }, {
|
||||||
prepareDeposit: async () =>
|
prepareDeposit: async () =>
|
||||||
({
|
({
|
||||||
effectiveDepositAmount: Amounts.parseOrThrow("EUR:1"),
|
effectiveDepositAmount: Amounts.parseOrThrow("EUR:1"),
|
||||||
@ -70,9 +72,8 @@ describe("Deposit CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -81,7 +82,7 @@ describe("Deposit CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "ready") expect.fail();
|
if (state.status !== "ready") expect.fail();
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.confirm.onClick).not.undefined;
|
expect(state.confirm.onClick).not.undefined;
|
||||||
expect(state.cost).deep.eq(Amounts.parseOrThrow("EUR:1.2"));
|
expect(state.cost).deep.eq(Amounts.parseOrThrow("EUR:1.2"));
|
||||||
expect(state.fee).deep.eq(Amounts.parseOrThrow("EUR:0.2"));
|
expect(state.fee).deep.eq(Amounts.parseOrThrow("EUR:0.2"));
|
109
packages/taler-wallet-webextension/src/cta/Deposit/views.tsx
Normal file
109
packages/taler-wallet-webextension/src/cta/Deposit/views.tsx
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Amounts } from "@gnu-taler/taler-util";
|
||||||
|
import { Fragment, h, VNode } from "preact";
|
||||||
|
import { Amount } from "../../components/Amount.js";
|
||||||
|
import { LoadingError } from "../../components/LoadingError.js";
|
||||||
|
import { LogoHeader } from "../../components/LogoHeader.js";
|
||||||
|
import { Part } from "../../components/Part.js";
|
||||||
|
import { SubTitle, WalletAction } from "../../components/styled/index.js";
|
||||||
|
import { useTranslationContext } from "../../context/translation.js";
|
||||||
|
import { Button } from "../../mui/Button.js";
|
||||||
|
import { State } from "./index.js";
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author sebasjm
|
||||||
|
*/
|
||||||
|
|
||||||
|
export function LoadingUriView({ error }: State.LoadingUriError): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<LoadingError
|
||||||
|
title={<i18n.Translate>Could not load deposit status</i18n.Translate>}
|
||||||
|
error={error}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
export function CompletedView(state: State.Completed): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<WalletAction>
|
||||||
|
<LogoHeader />
|
||||||
|
|
||||||
|
<SubTitle>
|
||||||
|
<i18n.Translate>Digital cash deposit</i18n.Translate>
|
||||||
|
</SubTitle>
|
||||||
|
<section>
|
||||||
|
<p>
|
||||||
|
<i18n.Translate>deposit completed</i18n.Translate>
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
</WalletAction>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ReadyView(state: State.Ready): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<WalletAction>
|
||||||
|
<LogoHeader />
|
||||||
|
|
||||||
|
<SubTitle>
|
||||||
|
<i18n.Translate>Digital cash deposit</i18n.Translate>
|
||||||
|
</SubTitle>
|
||||||
|
<section>
|
||||||
|
{Amounts.isNonZero(state.cost) && (
|
||||||
|
<Part
|
||||||
|
big
|
||||||
|
title={<i18n.Translate>Cost</i18n.Translate>}
|
||||||
|
text={<Amount value={state.cost} />}
|
||||||
|
kind="negative"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{Amounts.isNonZero(state.fee) && (
|
||||||
|
<Part
|
||||||
|
big
|
||||||
|
title={<i18n.Translate>Fee</i18n.Translate>}
|
||||||
|
text={<Amount value={state.fee} />}
|
||||||
|
kind="negative"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<Part
|
||||||
|
big
|
||||||
|
title={<i18n.Translate>To be received</i18n.Translate>}
|
||||||
|
text={<Amount value={state.effective} />}
|
||||||
|
kind="positive"
|
||||||
|
/>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<Button
|
||||||
|
variant="contained"
|
||||||
|
color="success"
|
||||||
|
onClick={state.confirm.onClick}
|
||||||
|
>
|
||||||
|
<i18n.Translate>
|
||||||
|
Deposit {<Amount value={state.effective} />}
|
||||||
|
</i18n.Translate>
|
||||||
|
</Button>
|
||||||
|
</section>
|
||||||
|
</WalletAction>
|
||||||
|
);
|
||||||
|
}
|
@ -1,396 +0,0 @@
|
|||||||
/*
|
|
||||||
This file is part of GNU Taler
|
|
||||||
(C) 2022 Taler Systems S.A.
|
|
||||||
|
|
||||||
GNU 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.
|
|
||||||
|
|
||||||
GNU 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
|
|
||||||
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @author Sebastian Javier Marchano (sebasjm)
|
|
||||||
*/
|
|
||||||
|
|
||||||
import {
|
|
||||||
Amounts,
|
|
||||||
ContractTerms,
|
|
||||||
PreparePayResultType,
|
|
||||||
} from "@gnu-taler/taler-util";
|
|
||||||
import { createExample } from "../test-utils.js";
|
|
||||||
import { View as TestedComponent } from "./Pay.js";
|
|
||||||
|
|
||||||
export default {
|
|
||||||
title: "cta/pay",
|
|
||||||
component: TestedComponent,
|
|
||||||
argTypes: {},
|
|
||||||
};
|
|
||||||
|
|
||||||
const noop = async (): Promise<void> => {
|
|
||||||
return;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const NoBalance = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow("USD:10"),
|
|
||||||
balance: undefined,
|
|
||||||
payHandler: {
|
|
||||||
onClick: async () => {
|
|
||||||
null;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
totalFees: Amounts.parseOrThrow("USD:0"),
|
|
||||||
payResult: undefined,
|
|
||||||
uri: "",
|
|
||||||
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",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
goBack: noop,
|
|
||||||
goToWalletManualWithdraw: noop,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const NoEnoughBalance = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow("USD:10"),
|
|
||||||
balance: {
|
|
||||||
currency: "USD",
|
|
||||||
fraction: 40000000,
|
|
||||||
value: 9,
|
|
||||||
},
|
|
||||||
payHandler: {
|
|
||||||
onClick: async () => {
|
|
||||||
null;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
totalFees: Amounts.parseOrThrow("USD:0"),
|
|
||||||
payResult: undefined,
|
|
||||||
uri: "",
|
|
||||||
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",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
goBack: noop,
|
|
||||||
goToWalletManualWithdraw: noop,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const EnoughBalanceButRestricted = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow("USD:10"),
|
|
||||||
balance: {
|
|
||||||
currency: "USD",
|
|
||||||
fraction: 40000000,
|
|
||||||
value: 19,
|
|
||||||
},
|
|
||||||
payHandler: {
|
|
||||||
onClick: async () => {
|
|
||||||
null;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
totalFees: Amounts.parseOrThrow("USD:0"),
|
|
||||||
payResult: undefined,
|
|
||||||
uri: "",
|
|
||||||
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",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
goBack: noop,
|
|
||||||
goToWalletManualWithdraw: noop,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const PaymentPossible = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow("USD:10"),
|
|
||||||
balance: {
|
|
||||||
currency: "USD",
|
|
||||||
fraction: 40000000,
|
|
||||||
value: 11,
|
|
||||||
},
|
|
||||||
payHandler: {
|
|
||||||
onClick: async () => {
|
|
||||||
null;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
totalFees: Amounts.parseOrThrow("USD:0"),
|
|
||||||
payResult: undefined,
|
|
||||||
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",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
goBack: noop,
|
|
||||||
goToWalletManualWithdraw: noop,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const PaymentPossibleWithFee = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow("USD:10"),
|
|
||||||
balance: {
|
|
||||||
currency: "USD",
|
|
||||||
fraction: 40000000,
|
|
||||||
value: 11,
|
|
||||||
},
|
|
||||||
payHandler: {
|
|
||||||
onClick: async () => {
|
|
||||||
null;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
totalFees: Amounts.parseOrThrow("USD:0.20"),
|
|
||||||
payResult: undefined,
|
|
||||||
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",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
goBack: noop,
|
|
||||||
goToWalletManualWithdraw: noop,
|
|
||||||
});
|
|
||||||
|
|
||||||
import beer from "../../static-dev/beer.png";
|
|
||||||
|
|
||||||
export const TicketWithAProductList = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow("USD:10"),
|
|
||||||
balance: {
|
|
||||||
currency: "USD",
|
|
||||||
fraction: 40000000,
|
|
||||||
value: 11,
|
|
||||||
},
|
|
||||||
payHandler: {
|
|
||||||
onClick: async () => {
|
|
||||||
null;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
totalFees: Amounts.parseOrThrow("USD:0.20"),
|
|
||||||
payResult: undefined,
|
|
||||||
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",
|
|
||||||
products: [
|
|
||||||
{
|
|
||||||
description: "ten beers",
|
|
||||||
price: "USD:1",
|
|
||||||
quantity: 10,
|
|
||||||
image: beer,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
description: "beer without image",
|
|
||||||
price: "USD:1",
|
|
||||||
quantity: 10,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
description: "one brown beer",
|
|
||||||
price: "USD:2",
|
|
||||||
quantity: 1,
|
|
||||||
image: beer,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
} as Partial<ContractTerms> as any,
|
|
||||||
contractTermsHash: "123456",
|
|
||||||
proposalId: "proposal1234",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
goBack: noop,
|
|
||||||
goToWalletManualWithdraw: noop,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const AlreadyConfirmedByOther = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow("USD:10"),
|
|
||||||
balance: {
|
|
||||||
currency: "USD",
|
|
||||||
fraction: 40000000,
|
|
||||||
value: 11,
|
|
||||||
},
|
|
||||||
payHandler: {
|
|
||||||
onClick: async () => {
|
|
||||||
null;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
totalFees: Amounts.parseOrThrow("USD:0.20"),
|
|
||||||
payResult: undefined,
|
|
||||||
uri: "taler://pay/merchant-backend.taler/2021.242-01G2X4275RBWG/?c=66BE594PDZR24744J6EQK52XM0",
|
|
||||||
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,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
goBack: noop,
|
|
||||||
goToWalletManualWithdraw: noop,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const AlreadyPaidWithoutFulfillment = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow("USD:10"),
|
|
||||||
balance: {
|
|
||||||
currency: "USD",
|
|
||||||
fraction: 40000000,
|
|
||||||
value: 11,
|
|
||||||
},
|
|
||||||
payHandler: {
|
|
||||||
onClick: async () => {
|
|
||||||
null;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
totalFees: Amounts.parseOrThrow("USD:0.20"),
|
|
||||||
payResult: undefined,
|
|
||||||
uri: "taler://pay/merchant-backend.taler/2021.242-01G2X4275RBWG/?c=66BE594PDZR24744J6EQK52XM0",
|
|
||||||
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: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
goBack: noop,
|
|
||||||
goToWalletManualWithdraw: noop,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const AlreadyPaidWithFulfillment = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow("USD:10"),
|
|
||||||
balance: {
|
|
||||||
currency: "USD",
|
|
||||||
fraction: 40000000,
|
|
||||||
value: 11,
|
|
||||||
},
|
|
||||||
payHandler: {
|
|
||||||
onClick: async () => {
|
|
||||||
null;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
totalFees: Amounts.parseOrThrow("USD:0.20"),
|
|
||||||
payResult: undefined,
|
|
||||||
uri: "taler://pay/merchant-backend.taler/2021.242-01G2X4275RBWG/?c=66BE594PDZR24744J6EQK52XM0",
|
|
||||||
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,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
goBack: noop,
|
|
||||||
goToWalletManualWithdraw: noop,
|
|
||||||
});
|
|
93
packages/taler-wallet-webextension/src/cta/Payment/index.ts
Normal file
93
packages/taler-wallet-webextension/src/cta/Payment/index.ts
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { AmountJson, ConfirmPayResult, PreparePayResult } from "@gnu-taler/taler-util";
|
||||||
|
import { Loading } from "../../components/Loading.js";
|
||||||
|
import { HookError } from "../../hooks/useAsyncAsHook.js";
|
||||||
|
import { ButtonHandler } from "../../mui/handlers.js";
|
||||||
|
import { compose, StateViewMap } from "../../utils/index.js";
|
||||||
|
import * as wxApi from "../../wxApi.js";
|
||||||
|
import { useComponentState } from "./state.js";
|
||||||
|
import { LoadingUriView, BaseView } from "./views.js";
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
export interface Props {
|
||||||
|
talerPayUri?: string;
|
||||||
|
goToWalletManualWithdraw: (currency?: string) => Promise<void>;
|
||||||
|
goBack: () => Promise<void>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type State =
|
||||||
|
| State.Loading
|
||||||
|
| State.LoadingUriError
|
||||||
|
| State.Ready
|
||||||
|
| State.NoEnoughBalance
|
||||||
|
| State.NoBalanceForCurrency
|
||||||
|
| State.Confirmed;
|
||||||
|
|
||||||
|
export namespace State {
|
||||||
|
|
||||||
|
export interface Loading {
|
||||||
|
status: "loading";
|
||||||
|
error: undefined;
|
||||||
|
}
|
||||||
|
export interface LoadingUriError {
|
||||||
|
status: "loading-uri";
|
||||||
|
error: HookError;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface BaseInfo {
|
||||||
|
amount: AmountJson;
|
||||||
|
totalFees: AmountJson;
|
||||||
|
payStatus: PreparePayResult;
|
||||||
|
uri: string;
|
||||||
|
error: undefined;
|
||||||
|
goToWalletManualWithdraw: (currency?: string) => Promise<void>;
|
||||||
|
goBack: () => Promise<void>;
|
||||||
|
}
|
||||||
|
export interface NoBalanceForCurrency extends BaseInfo {
|
||||||
|
status: "no-balance-for-currency"
|
||||||
|
balance: undefined;
|
||||||
|
}
|
||||||
|
export interface NoEnoughBalance extends BaseInfo {
|
||||||
|
status: "no-enough-balance"
|
||||||
|
balance: AmountJson;
|
||||||
|
}
|
||||||
|
export interface Ready extends BaseInfo {
|
||||||
|
status: "ready";
|
||||||
|
payHandler: ButtonHandler;
|
||||||
|
balance: AmountJson;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Confirmed extends BaseInfo {
|
||||||
|
status: "confirmed";
|
||||||
|
payResult: ConfirmPayResult;
|
||||||
|
payHandler: ButtonHandler;
|
||||||
|
balance: AmountJson;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const viewMapping: StateViewMap<State> = {
|
||||||
|
loading: Loading,
|
||||||
|
"loading-uri": LoadingUriView,
|
||||||
|
"no-balance-for-currency": BaseView,
|
||||||
|
"no-enough-balance": BaseView,
|
||||||
|
confirmed: BaseView,
|
||||||
|
ready: BaseView,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const PaymentPage = compose("Payment", (p: Props) => useComponentState(p, wxApi), viewMapping)
|
171
packages/taler-wallet-webextension/src/cta/Payment/state.ts
Normal file
171
packages/taler-wallet-webextension/src/cta/Payment/state.ts
Normal file
@ -0,0 +1,171 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
import { AmountJson, Amounts, ConfirmPayResult, ConfirmPayResultType, NotificationType, PreparePayResultType, TalerErrorCode } from "@gnu-taler/taler-util";
|
||||||
|
import { TalerError } from "@gnu-taler/taler-wallet-core";
|
||||||
|
import { useEffect, useState } from "preact/hooks";
|
||||||
|
import { useAsyncAsHook } from "../../hooks/useAsyncAsHook.js";
|
||||||
|
import { ButtonHandler } from "../../mui/handlers.js";
|
||||||
|
import * as wxApi from "../../wxApi.js";
|
||||||
|
import { Props, State } from "./index.js";
|
||||||
|
|
||||||
|
export function useComponentState(
|
||||||
|
{ talerPayUri, goBack, goToWalletManualWithdraw }: Props,
|
||||||
|
api: typeof wxApi,
|
||||||
|
): State {
|
||||||
|
const [payResult, setPayResult] = useState<ConfirmPayResult | undefined>(
|
||||||
|
undefined,
|
||||||
|
);
|
||||||
|
const [payErrMsg, setPayErrMsg] = useState<TalerError | undefined>(undefined);
|
||||||
|
|
||||||
|
const hook = useAsyncAsHook(async () => {
|
||||||
|
if (!talerPayUri) throw Error("ERROR_NO-URI-FOR-PAYMENT");
|
||||||
|
const payStatus = await api.preparePay(talerPayUri);
|
||||||
|
const balance = await api.getBalance();
|
||||||
|
return { payStatus, balance, uri: talerPayUri };
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
api.onUpdateNotification([NotificationType.CoinWithdrawn], () => {
|
||||||
|
hook?.retry();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const hookResponse = !hook || hook.hasError ? undefined : hook.response;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!hookResponse) return;
|
||||||
|
const { payStatus } = hookResponse;
|
||||||
|
if (
|
||||||
|
payStatus &&
|
||||||
|
payStatus.status === PreparePayResultType.AlreadyConfirmed &&
|
||||||
|
payStatus.paid
|
||||||
|
) {
|
||||||
|
const fu = payStatus.contractTerms.fulfillment_url;
|
||||||
|
if (fu) {
|
||||||
|
setTimeout(() => {
|
||||||
|
document.location.href = fu;
|
||||||
|
}, 3000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [hookResponse]);
|
||||||
|
|
||||||
|
if (!hook) return { status: "loading", error: undefined };
|
||||||
|
if (hook.hasError) {
|
||||||
|
return {
|
||||||
|
status: "loading-uri",
|
||||||
|
error: hook,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
const { payStatus } = hook.response;
|
||||||
|
const amount = Amounts.parseOrThrow(payStatus.amountRaw);
|
||||||
|
|
||||||
|
const foundBalance = hook.response.balance.balances.find(
|
||||||
|
(b) => Amounts.parseOrThrow(b.available).currency === amount.currency,
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
let totalFees = Amounts.getZero(amount.currency);
|
||||||
|
if (payStatus.status === PreparePayResultType.PaymentPossible) {
|
||||||
|
const amountEffective: AmountJson = Amounts.parseOrThrow(
|
||||||
|
payStatus.amountEffective,
|
||||||
|
);
|
||||||
|
totalFees = Amounts.sub(amountEffective, amount).amount;
|
||||||
|
}
|
||||||
|
|
||||||
|
const baseResult = {
|
||||||
|
uri: hook.response.uri,
|
||||||
|
amount,
|
||||||
|
totalFees,
|
||||||
|
payStatus,
|
||||||
|
error: undefined,
|
||||||
|
goBack, goToWalletManualWithdraw
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!foundBalance) {
|
||||||
|
return {
|
||||||
|
status: "no-balance-for-currency",
|
||||||
|
balance: undefined,
|
||||||
|
...baseResult,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const foundAmount = Amounts.parseOrThrow(foundBalance.available);
|
||||||
|
|
||||||
|
async function doPayment(): Promise<void> {
|
||||||
|
try {
|
||||||
|
if (payStatus.status !== "payment-possible") {
|
||||||
|
throw TalerError.fromUncheckedDetail({
|
||||||
|
code: TalerErrorCode.GENERIC_CLIENT_INTERNAL_ERROR,
|
||||||
|
hint: `payment is not possible: ${payStatus.status}`,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
const res = await api.confirmPay(payStatus.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) {
|
||||||
|
if (typeof window !== "undefined") {
|
||||||
|
document.location.href = fu;
|
||||||
|
} else {
|
||||||
|
console.log(`should d to ${fu}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
setPayResult(res);
|
||||||
|
} catch (e) {
|
||||||
|
if (e instanceof TalerError) {
|
||||||
|
setPayErrMsg(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (payStatus.status === PreparePayResultType.InsufficientBalance) {
|
||||||
|
return {
|
||||||
|
status: 'no-enough-balance',
|
||||||
|
balance: foundAmount,
|
||||||
|
...baseResult,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const payHandler: ButtonHandler = {
|
||||||
|
onClick: payErrMsg ? undefined : doPayment,
|
||||||
|
error: payErrMsg,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!payResult) {
|
||||||
|
return {
|
||||||
|
status: "ready",
|
||||||
|
payHandler,
|
||||||
|
...baseResult,
|
||||||
|
balance: foundAmount
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
status: "confirmed",
|
||||||
|
balance: foundAmount,
|
||||||
|
payResult,
|
||||||
|
payHandler: {},
|
||||||
|
...baseResult,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
356
packages/taler-wallet-webextension/src/cta/Payment/stories.tsx
Normal file
356
packages/taler-wallet-webextension/src/cta/Payment/stories.tsx
Normal file
@ -0,0 +1,356 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Sebastian Javier Marchano (sebasjm)
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {
|
||||||
|
Amounts,
|
||||||
|
ContractTerms,
|
||||||
|
PreparePayResultType,
|
||||||
|
} from "@gnu-taler/taler-util";
|
||||||
|
import { createExample } from "../../test-utils.js";
|
||||||
|
import { BaseView } from "./views.js";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
title: "cta/payment",
|
||||||
|
component: BaseView,
|
||||||
|
argTypes: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const NoBalance = createExample(BaseView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
amount: Amounts.parseOrThrow("USD:10"),
|
||||||
|
balance: undefined,
|
||||||
|
payHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
totalFees: Amounts.parseOrThrow("USD:0"),
|
||||||
|
|
||||||
|
uri: "",
|
||||||
|
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",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const NoEnoughBalance = createExample(BaseView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
amount: Amounts.parseOrThrow("USD:10"),
|
||||||
|
balance: {
|
||||||
|
currency: "USD",
|
||||||
|
fraction: 40000000,
|
||||||
|
value: 9,
|
||||||
|
},
|
||||||
|
payHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
totalFees: Amounts.parseOrThrow("USD:0"),
|
||||||
|
|
||||||
|
uri: "",
|
||||||
|
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",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const EnoughBalanceButRestricted = createExample(BaseView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
amount: Amounts.parseOrThrow("USD:10"),
|
||||||
|
balance: {
|
||||||
|
currency: "USD",
|
||||||
|
fraction: 40000000,
|
||||||
|
value: 19,
|
||||||
|
},
|
||||||
|
payHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
totalFees: Amounts.parseOrThrow("USD:0"),
|
||||||
|
|
||||||
|
uri: "",
|
||||||
|
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",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const PaymentPossible = createExample(BaseView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
amount: Amounts.parseOrThrow("USD:10"),
|
||||||
|
balance: {
|
||||||
|
currency: "USD",
|
||||||
|
fraction: 40000000,
|
||||||
|
value: 11,
|
||||||
|
},
|
||||||
|
payHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
totalFees: Amounts.parseOrThrow("USD:0"),
|
||||||
|
|
||||||
|
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(BaseView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
amount: Amounts.parseOrThrow("USD:10"),
|
||||||
|
balance: {
|
||||||
|
currency: "USD",
|
||||||
|
fraction: 40000000,
|
||||||
|
value: 11,
|
||||||
|
},
|
||||||
|
payHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
totalFees: Amounts.parseOrThrow("USD:0.20"),
|
||||||
|
|
||||||
|
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",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
import beer from "../../../static-dev/beer.png";
|
||||||
|
|
||||||
|
export const TicketWithAProductList = createExample(BaseView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
amount: Amounts.parseOrThrow("USD:10"),
|
||||||
|
balance: {
|
||||||
|
currency: "USD",
|
||||||
|
fraction: 40000000,
|
||||||
|
value: 11,
|
||||||
|
},
|
||||||
|
payHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
totalFees: Amounts.parseOrThrow("USD:0.20"),
|
||||||
|
|
||||||
|
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",
|
||||||
|
products: [
|
||||||
|
{
|
||||||
|
description: "ten beers",
|
||||||
|
price: "USD:1",
|
||||||
|
quantity: 10,
|
||||||
|
image: beer,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
description: "beer without image",
|
||||||
|
price: "USD:1",
|
||||||
|
quantity: 10,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
description: "one brown beer",
|
||||||
|
price: "USD:2",
|
||||||
|
quantity: 1,
|
||||||
|
image: beer,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
} as Partial<ContractTerms> as any,
|
||||||
|
contractTermsHash: "123456",
|
||||||
|
proposalId: "proposal1234",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const AlreadyConfirmedByOther = createExample(BaseView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
amount: Amounts.parseOrThrow("USD:10"),
|
||||||
|
balance: {
|
||||||
|
currency: "USD",
|
||||||
|
fraction: 40000000,
|
||||||
|
value: 11,
|
||||||
|
},
|
||||||
|
payHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
totalFees: Amounts.parseOrThrow("USD:0.20"),
|
||||||
|
|
||||||
|
uri: "taler://pay/merchant-backend.taler/2021.242-01G2X4275RBWG/?c=66BE594PDZR24744J6EQK52XM0",
|
||||||
|
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 AlreadyPaidWithoutFulfillment = createExample(BaseView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
amount: Amounts.parseOrThrow("USD:10"),
|
||||||
|
balance: {
|
||||||
|
currency: "USD",
|
||||||
|
fraction: 40000000,
|
||||||
|
value: 11,
|
||||||
|
},
|
||||||
|
payHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
totalFees: Amounts.parseOrThrow("USD:0.20"),
|
||||||
|
|
||||||
|
uri: "taler://pay/merchant-backend.taler/2021.242-01G2X4275RBWG/?c=66BE594PDZR24744J6EQK52XM0",
|
||||||
|
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: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const AlreadyPaidWithFulfillment = createExample(BaseView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
amount: Amounts.parseOrThrow("USD:10"),
|
||||||
|
balance: {
|
||||||
|
currency: "USD",
|
||||||
|
fraction: 40000000,
|
||||||
|
value: 11,
|
||||||
|
},
|
||||||
|
payHandler: {
|
||||||
|
onClick: async () => {
|
||||||
|
null;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
totalFees: Amounts.parseOrThrow("USD:0.20"),
|
||||||
|
|
||||||
|
uri: "taler://pay/merchant-backend.taler/2021.242-01G2X4275RBWG/?c=66BE594PDZR24744J6EQK52XM0",
|
||||||
|
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,
|
||||||
|
},
|
||||||
|
});
|
@ -30,9 +30,9 @@ import {
|
|||||||
PreparePayResultType,
|
PreparePayResultType,
|
||||||
} from "@gnu-taler/taler-util";
|
} from "@gnu-taler/taler-util";
|
||||||
import { expect } from "chai";
|
import { expect } from "chai";
|
||||||
import { mountHook } from "../test-utils.js";
|
import { mountHook } from "../../test-utils.js";
|
||||||
import * as wxApi from "../wxApi.js";
|
import { useComponentState } from "./state.js";
|
||||||
import { useComponentState } from "./Pay.jsx";
|
import * as wxApi from "../../wxApi.js";
|
||||||
|
|
||||||
const nullFunction: any = () => null;
|
const nullFunction: any = () => null;
|
||||||
type VoidFunction = () => void;
|
type VoidFunction = () => void;
|
||||||
@ -66,30 +66,30 @@ export class SubsHandler {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
describe("Pay CTA states", () => {
|
describe("Payment CTA states", () => {
|
||||||
it("should tell the user that the URI is missing", async () => {
|
it("should tell the user that the URI is missing", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState(undefined, {
|
useComponentState({ talerPayUri: undefined, goBack: nullFunction, goToWalletManualWithdraw: nullFunction }, {
|
||||||
onUpdateNotification: nullFunction,
|
onUpdateNotification: nullFunction,
|
||||||
} as Partial<typeof wxApi> as any),
|
} as Partial<typeof wxApi> as any),
|
||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
|
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading-uri");
|
||||||
if (hook === undefined) expect.fail();
|
if (error === undefined) expect.fail();
|
||||||
expect(hook.hasError).true;
|
expect(error.hasError).true;
|
||||||
expect(hook.operational).false;
|
expect(error.operational).false;
|
||||||
}
|
}
|
||||||
|
|
||||||
await assertNoPendingUpdate();
|
await assertNoPendingUpdate();
|
||||||
@ -98,7 +98,7 @@ describe("Pay CTA states", () => {
|
|||||||
it("should response with no balance", async () => {
|
it("should response with no balance", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taller://pay", {
|
useComponentState({ talerPayUri: "taller://pay", goBack: nullFunction, goToWalletManualWithdraw: nullFunction }, {
|
||||||
onUpdateNotification: nullFunction,
|
onUpdateNotification: nullFunction,
|
||||||
preparePay: async () =>
|
preparePay: async () =>
|
||||||
({
|
({
|
||||||
@ -113,19 +113,18 @@ describe("Pay CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
|
||||||
{
|
{
|
||||||
const r = getLastResultOrThrow();
|
const r = getLastResultOrThrow();
|
||||||
if (r.status !== "ready") expect.fail();
|
if (r.status !== "no-balance-for-currency") expect.fail();
|
||||||
expect(r.balance).undefined;
|
expect(r.balance).undefined;
|
||||||
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:10"));
|
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:10"));
|
||||||
expect(r.payHandler.onClick).undefined;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await assertNoPendingUpdate();
|
await assertNoPendingUpdate();
|
||||||
@ -134,7 +133,7 @@ describe("Pay CTA states", () => {
|
|||||||
it("should not be able to pay if there is no enough balance", async () => {
|
it("should not be able to pay if there is no enough balance", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taller://pay", {
|
useComponentState({ talerPayUri: "taller://pay", goBack: nullFunction, goToWalletManualWithdraw: nullFunction }, {
|
||||||
onUpdateNotification: nullFunction,
|
onUpdateNotification: nullFunction,
|
||||||
preparePay: async () =>
|
preparePay: async () =>
|
||||||
({
|
({
|
||||||
@ -153,19 +152,18 @@ describe("Pay CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
|
||||||
{
|
{
|
||||||
const r = getLastResultOrThrow();
|
const r = getLastResultOrThrow();
|
||||||
if (r.status !== "ready") expect.fail();
|
if (r.status !== "no-enough-balance") expect.fail();
|
||||||
expect(r.balance).deep.equal(Amounts.parseOrThrow("USD:5"));
|
expect(r.balance).deep.equal(Amounts.parseOrThrow("USD:5"));
|
||||||
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:10"));
|
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:10"));
|
||||||
expect(r.payHandler.onClick).undefined;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await assertNoPendingUpdate();
|
await assertNoPendingUpdate();
|
||||||
@ -174,7 +172,7 @@ describe("Pay CTA states", () => {
|
|||||||
it("should be able to pay (without fee)", async () => {
|
it("should be able to pay (without fee)", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taller://pay", {
|
useComponentState({ talerPayUri: "taller://pay", goBack: nullFunction, goToWalletManualWithdraw: nullFunction }, {
|
||||||
onUpdateNotification: nullFunction,
|
onUpdateNotification: nullFunction,
|
||||||
preparePay: async () =>
|
preparePay: async () =>
|
||||||
({
|
({
|
||||||
@ -194,9 +192,9 @@ describe("Pay CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -216,7 +214,7 @@ describe("Pay CTA states", () => {
|
|||||||
it("should be able to pay (with fee)", async () => {
|
it("should be able to pay (with fee)", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taller://pay", {
|
useComponentState({ talerPayUri: "taller://pay", goBack: nullFunction, goToWalletManualWithdraw: nullFunction }, {
|
||||||
onUpdateNotification: nullFunction,
|
onUpdateNotification: nullFunction,
|
||||||
preparePay: async () =>
|
preparePay: async () =>
|
||||||
({
|
({
|
||||||
@ -236,9 +234,9 @@ describe("Pay CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -258,7 +256,7 @@ describe("Pay CTA states", () => {
|
|||||||
it("should get confirmation done after pay successfully", async () => {
|
it("should get confirmation done after pay successfully", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taller://pay", {
|
useComponentState({ talerPayUri: "taller://pay", goBack: nullFunction, goToWalletManualWithdraw: nullFunction }, {
|
||||||
onUpdateNotification: nullFunction,
|
onUpdateNotification: nullFunction,
|
||||||
preparePay: async () =>
|
preparePay: async () =>
|
||||||
({
|
({
|
||||||
@ -283,9 +281,9 @@ describe("Pay CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -319,7 +317,7 @@ describe("Pay CTA states", () => {
|
|||||||
it("should not stay in ready state after pay with error", async () => {
|
it("should not stay in ready state after pay with error", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taller://pay", {
|
useComponentState({ talerPayUri: "taller://pay", goBack: nullFunction, goToWalletManualWithdraw: nullFunction }, {
|
||||||
onUpdateNotification: nullFunction,
|
onUpdateNotification: nullFunction,
|
||||||
preparePay: async () =>
|
preparePay: async () =>
|
||||||
({
|
({
|
||||||
@ -344,9 +342,9 @@ describe("Pay CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -395,7 +393,7 @@ describe("Pay CTA states", () => {
|
|||||||
|
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taller://pay", {
|
useComponentState({ talerPayUri: "taller://pay", goBack: nullFunction, goToWalletManualWithdraw: nullFunction }, {
|
||||||
onUpdateNotification: subscriptions.saveSubscription,
|
onUpdateNotification: subscriptions.saveSubscription,
|
||||||
preparePay: async () =>
|
preparePay: async () =>
|
||||||
({
|
({
|
||||||
@ -415,9 +413,9 @@ describe("Pay CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
@ -14,40 +14,22 @@
|
|||||||
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
|
||||||
* Page shown to the user to confirm entering
|
|
||||||
* a contract.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Imports.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import {
|
import {
|
||||||
AmountJson,
|
|
||||||
Amounts,
|
Amounts,
|
||||||
ConfirmPayResult,
|
|
||||||
ConfirmPayResultType,
|
ConfirmPayResultType,
|
||||||
ContractTerms,
|
ContractTerms,
|
||||||
NotificationType,
|
|
||||||
PreparePayResult,
|
|
||||||
PreparePayResultType,
|
PreparePayResultType,
|
||||||
Product,
|
Product,
|
||||||
TalerErrorCode,
|
|
||||||
} from "@gnu-taler/taler-util";
|
} from "@gnu-taler/taler-util";
|
||||||
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 { useState } from "preact/hooks";
|
||||||
import { Amount } from "../components/Amount.js";
|
import { Amount } from "../../components/Amount.js";
|
||||||
import { ErrorMessage } from "../components/ErrorMessage.js";
|
import { ErrorTalerOperation } from "../../components/ErrorTalerOperation.js";
|
||||||
import { ErrorTalerOperation } from "../components/ErrorTalerOperation.js";
|
import { LoadingError } from "../../components/LoadingError.js";
|
||||||
import { Loading } from "../components/Loading.js";
|
import { LogoHeader } from "../../components/LogoHeader.js";
|
||||||
import { LoadingError } from "../components/LoadingError.js";
|
import { Part } from "../../components/Part.js";
|
||||||
import { LogoHeader } from "../components/LogoHeader.js";
|
import { QR } from "../../components/QR.js";
|
||||||
import { Part } from "../components/Part.js";
|
|
||||||
import { QR } from "../components/QR.js";
|
|
||||||
import {
|
import {
|
||||||
ButtonSuccess,
|
|
||||||
Link,
|
Link,
|
||||||
LinkSuccess,
|
LinkSuccess,
|
||||||
SmallLightText,
|
SmallLightText,
|
||||||
@ -55,233 +37,32 @@ import {
|
|||||||
SuccessBox,
|
SuccessBox,
|
||||||
WalletAction,
|
WalletAction,
|
||||||
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 { HookError, useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
import { Button } from "../../mui/Button.js";
|
||||||
import { Button } from "../mui/Button.js";
|
import { State } from "./index.js";
|
||||||
import { ButtonHandler } from "../mui/handlers.js";
|
|
||||||
import * as wxApi from "../wxApi.js";
|
|
||||||
|
|
||||||
interface Props {
|
export function LoadingUriView({ error }: State.LoadingUriError): VNode {
|
||||||
talerPayUri?: string;
|
|
||||||
goToWalletManualWithdraw: (currency?: string) => Promise<void>;
|
|
||||||
goBack: () => Promise<void>;
|
|
||||||
}
|
|
||||||
|
|
||||||
type State = Loading | Ready | Confirmed;
|
|
||||||
interface Loading {
|
|
||||||
status: "loading";
|
|
||||||
hook: HookError | undefined;
|
|
||||||
}
|
|
||||||
interface Ready {
|
|
||||||
status: "ready";
|
|
||||||
hook: undefined;
|
|
||||||
uri: string;
|
|
||||||
amount: AmountJson;
|
|
||||||
totalFees: AmountJson;
|
|
||||||
payStatus: PreparePayResult;
|
|
||||||
balance: AmountJson | undefined;
|
|
||||||
payHandler: ButtonHandler;
|
|
||||||
payResult: undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface Confirmed {
|
|
||||||
status: "confirmed";
|
|
||||||
hook: undefined;
|
|
||||||
uri: string;
|
|
||||||
amount: AmountJson;
|
|
||||||
totalFees: AmountJson;
|
|
||||||
payStatus: PreparePayResult;
|
|
||||||
balance: AmountJson | undefined;
|
|
||||||
payResult: ConfirmPayResult;
|
|
||||||
payHandler: ButtonHandler;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useComponentState(
|
|
||||||
talerPayUri: string | undefined,
|
|
||||||
api: typeof wxApi,
|
|
||||||
): State {
|
|
||||||
const [payResult, setPayResult] = useState<ConfirmPayResult | undefined>(
|
|
||||||
undefined,
|
|
||||||
);
|
|
||||||
const [payErrMsg, setPayErrMsg] = useState<TalerError | undefined>(undefined);
|
|
||||||
|
|
||||||
const hook = useAsyncAsHook(async () => {
|
|
||||||
if (!talerPayUri) throw Error("ERROR_NO-URI-FOR-PAYMENT");
|
|
||||||
const payStatus = await api.preparePay(talerPayUri);
|
|
||||||
const balance = await api.getBalance();
|
|
||||||
return { payStatus, balance, uri: talerPayUri };
|
|
||||||
});
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
api.onUpdateNotification([NotificationType.CoinWithdrawn], () => {
|
|
||||||
hook?.retry();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
const hookResponse = !hook || hook.hasError ? undefined : hook.response;
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!hookResponse) return;
|
|
||||||
const { payStatus } = hookResponse;
|
|
||||||
if (
|
|
||||||
payStatus &&
|
|
||||||
payStatus.status === PreparePayResultType.AlreadyConfirmed &&
|
|
||||||
payStatus.paid
|
|
||||||
) {
|
|
||||||
const fu = payStatus.contractTerms.fulfillment_url;
|
|
||||||
if (fu) {
|
|
||||||
setTimeout(() => {
|
|
||||||
document.location.href = fu;
|
|
||||||
}, 3000);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, [hookResponse]);
|
|
||||||
|
|
||||||
if (!hook || hook.hasError) {
|
|
||||||
return {
|
|
||||||
status: "loading",
|
|
||||||
hook,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
const { payStatus } = hook.response;
|
|
||||||
const amount = Amounts.parseOrThrow(payStatus.amountRaw);
|
|
||||||
|
|
||||||
const foundBalance = hook.response.balance.balances.find(
|
|
||||||
(b) => Amounts.parseOrThrow(b.available).currency === amount.currency,
|
|
||||||
);
|
|
||||||
const foundAmount = foundBalance
|
|
||||||
? Amounts.parseOrThrow(foundBalance.available)
|
|
||||||
: undefined;
|
|
||||||
|
|
||||||
async function doPayment(): Promise<void> {
|
|
||||||
try {
|
|
||||||
if (payStatus.status !== "payment-possible") {
|
|
||||||
throw TalerError.fromUncheckedDetail({
|
|
||||||
code: TalerErrorCode.GENERIC_CLIENT_INTERNAL_ERROR,
|
|
||||||
hint: `payment is not possible: ${payStatus.status}`,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
const res = await api.confirmPay(payStatus.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) {
|
|
||||||
if (typeof window !== "undefined") {
|
|
||||||
document.location.href = fu;
|
|
||||||
} else {
|
|
||||||
console.log(`should d to ${fu}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
setPayResult(res);
|
|
||||||
} catch (e) {
|
|
||||||
if (e instanceof TalerError) {
|
|
||||||
setPayErrMsg(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const payDisabled =
|
|
||||||
payErrMsg ||
|
|
||||||
!foundAmount ||
|
|
||||||
payStatus.status === PreparePayResultType.InsufficientBalance;
|
|
||||||
|
|
||||||
const payHandler: ButtonHandler = {
|
|
||||||
onClick: payDisabled ? undefined : doPayment,
|
|
||||||
error: payErrMsg,
|
|
||||||
};
|
|
||||||
|
|
||||||
let totalFees = Amounts.getZero(amount.currency);
|
|
||||||
if (payStatus.status === PreparePayResultType.PaymentPossible) {
|
|
||||||
const amountEffective: AmountJson = Amounts.parseOrThrow(
|
|
||||||
payStatus.amountEffective,
|
|
||||||
);
|
|
||||||
totalFees = Amounts.sub(amountEffective, amount).amount;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!payResult) {
|
|
||||||
return {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
uri: hook.response.uri,
|
|
||||||
amount,
|
|
||||||
totalFees,
|
|
||||||
balance: foundAmount,
|
|
||||||
payHandler,
|
|
||||||
payStatus: hook.response.payStatus,
|
|
||||||
payResult,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
status: "confirmed",
|
|
||||||
hook: undefined,
|
|
||||||
uri: hook.response.uri,
|
|
||||||
amount,
|
|
||||||
totalFees,
|
|
||||||
balance: foundAmount,
|
|
||||||
payStatus: hook.response.payStatus,
|
|
||||||
payResult,
|
|
||||||
payHandler: {},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function PayPage({
|
|
||||||
talerPayUri,
|
|
||||||
goToWalletManualWithdraw,
|
|
||||||
goBack,
|
|
||||||
}: Props): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
const state = useComponentState(talerPayUri, wxApi);
|
|
||||||
|
|
||||||
if (state.status === "loading") {
|
|
||||||
if (!state.hook) return <Loading />;
|
|
||||||
return (
|
return (
|
||||||
<LoadingError
|
<LoadingError
|
||||||
title={<i18n.Translate>Could not load pay status</i18n.Translate>}
|
title={<i18n.Translate>Could not load pay status</i18n.Translate>}
|
||||||
error={state.hook}
|
error={error}
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<View
|
|
||||||
state={state}
|
|
||||||
goBack={goBack}
|
|
||||||
goToWalletManualWithdraw={goToWalletManualWithdraw}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function View({
|
type SupportedStates =
|
||||||
state,
|
| State.Ready
|
||||||
goBack,
|
| State.Confirmed
|
||||||
goToWalletManualWithdraw,
|
| State.NoBalanceForCurrency
|
||||||
}: {
|
| State.NoEnoughBalance;
|
||||||
state: Ready | Confirmed;
|
|
||||||
goToWalletManualWithdraw: (currency?: string) => Promise<void>;
|
export function BaseView(state: SupportedStates): VNode {
|
||||||
goBack: () => Promise<void>;
|
|
||||||
}): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
const { i18n } = useTranslationContext();
|
||||||
const contractTerms: ContractTerms = state.payStatus.contractTerms;
|
const contractTerms: ContractTerms = state.payStatus.contractTerms;
|
||||||
|
|
||||||
if (!contractTerms) {
|
|
||||||
return (
|
|
||||||
<ErrorMessage
|
|
||||||
title={
|
|
||||||
<i18n.Translate>
|
|
||||||
Could not load contract terms from merchant or wallet backend.
|
|
||||||
</i18n.Translate>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<WalletAction>
|
<WalletAction>
|
||||||
<LogoHeader />
|
<LogoHeader />
|
||||||
@ -341,10 +122,10 @@ export function View({
|
|||||||
</section>
|
</section>
|
||||||
<ButtonsSection
|
<ButtonsSection
|
||||||
state={state}
|
state={state}
|
||||||
goToWalletManualWithdraw={goToWalletManualWithdraw}
|
goToWalletManualWithdraw={state.goToWalletManualWithdraw}
|
||||||
/>
|
/>
|
||||||
<section>
|
<section>
|
||||||
<Link upperCased onClick={goBack}>
|
<Link upperCased onClick={state.goBack}>
|
||||||
<i18n.Translate>Cancel</i18n.Translate>
|
<i18n.Translate>Cancel</i18n.Translate>
|
||||||
</Link>
|
</Link>
|
||||||
</section>
|
</section>
|
||||||
@ -421,7 +202,7 @@ export function ProductList({ products }: { products: Product[] }): VNode {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function ShowImportantMessage({ state }: { state: Ready | Confirmed }): VNode {
|
function ShowImportantMessage({ state }: { state: SupportedStates }): VNode {
|
||||||
const { i18n } = useTranslationContext();
|
const { i18n } = useTranslationContext();
|
||||||
const { payStatus } = state;
|
const { payStatus } = state;
|
||||||
if (payStatus.status === PreparePayResultType.AlreadyConfirmed) {
|
if (payStatus.status === PreparePayResultType.AlreadyConfirmed) {
|
||||||
@ -483,7 +264,7 @@ function ShowImportantMessage({ state }: { state: Ready | Confirmed }): VNode {
|
|||||||
return <Fragment />;
|
return <Fragment />;
|
||||||
}
|
}
|
||||||
|
|
||||||
function PayWithMobile({ state }: { state: Ready }): VNode {
|
function PayWithMobile({ state }: { state: State.Ready }): VNode {
|
||||||
const { i18n } = useTranslationContext();
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
const [showQR, setShowQR] = useState<boolean>(false);
|
const [showQR, setShowQR] = useState<boolean>(false);
|
||||||
@ -520,7 +301,7 @@ function ButtonsSection({
|
|||||||
state,
|
state,
|
||||||
goToWalletManualWithdraw,
|
goToWalletManualWithdraw,
|
||||||
}: {
|
}: {
|
||||||
state: Ready | Confirmed;
|
state: SupportedStates;
|
||||||
goToWalletManualWithdraw: (currency: string) => Promise<void>;
|
goToWalletManualWithdraw: (currency: string) => Promise<void>;
|
||||||
}): VNode {
|
}): VNode {
|
||||||
const { i18n } = useTranslationContext();
|
const { i18n } = useTranslationContext();
|
@ -1,105 +0,0 @@
|
|||||||
/*
|
|
||||||
This file is part of GNU Taler
|
|
||||||
(C) 2022 Taler Systems S.A.
|
|
||||||
|
|
||||||
GNU 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.
|
|
||||||
|
|
||||||
GNU 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
|
|
||||||
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @author Sebastian Javier Marchano (sebasjm)
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { Amounts } from "@gnu-taler/taler-util";
|
|
||||||
import { createExample } from "../test-utils.js";
|
|
||||||
import { View as TestedComponent } from "./Refund.js";
|
|
||||||
|
|
||||||
export default {
|
|
||||||
title: "cta/refund",
|
|
||||||
component: TestedComponent,
|
|
||||||
argTypes: {},
|
|
||||||
};
|
|
||||||
|
|
||||||
export const Complete = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "completed",
|
|
||||||
amount: Amounts.parseOrThrow("USD:1"),
|
|
||||||
granted: Amounts.parseOrThrow("USD:1"),
|
|
||||||
hook: undefined,
|
|
||||||
merchantName: "the merchant",
|
|
||||||
products: undefined,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export const InProgress = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "in-progress",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow("USD:1"),
|
|
||||||
awaitingAmount: Amounts.parseOrThrow("USD:1"),
|
|
||||||
granted: Amounts.parseOrThrow("USD:0"),
|
|
||||||
merchantName: "the merchant",
|
|
||||||
products: undefined,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export const Ready = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
accept: {},
|
|
||||||
ignore: {},
|
|
||||||
|
|
||||||
amount: Amounts.parseOrThrow("USD:1"),
|
|
||||||
awaitingAmount: Amounts.parseOrThrow("USD:1"),
|
|
||||||
granted: Amounts.parseOrThrow("USD:0"),
|
|
||||||
merchantName: "the merchant",
|
|
||||||
products: [],
|
|
||||||
orderId: "abcdef",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
import beer from "../../static-dev/beer.png";
|
|
||||||
|
|
||||||
export const WithAProductList = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
accept: {},
|
|
||||||
ignore: {},
|
|
||||||
amount: Amounts.parseOrThrow("USD:1"),
|
|
||||||
awaitingAmount: Amounts.parseOrThrow("USD:1"),
|
|
||||||
granted: Amounts.parseOrThrow("USD:0"),
|
|
||||||
merchantName: "the merchant",
|
|
||||||
products: [
|
|
||||||
{
|
|
||||||
description: "beer",
|
|
||||||
image: beer,
|
|
||||||
quantity: 2,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
description: "t-shirt",
|
|
||||||
price: "EUR:1",
|
|
||||||
quantity: 5,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
orderId: "abcdef",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export const Ignored = createExample(TestedComponent, {
|
|
||||||
state: {
|
|
||||||
status: "ignored",
|
|
||||||
hook: undefined,
|
|
||||||
merchantName: "the merchant",
|
|
||||||
},
|
|
||||||
});
|
|
@ -1,364 +0,0 @@
|
|||||||
/*
|
|
||||||
This file is part of GNU Taler
|
|
||||||
(C) 2022 Taler Systems S.A.
|
|
||||||
|
|
||||||
GNU 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.
|
|
||||||
|
|
||||||
GNU 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
|
|
||||||
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Page that shows refund status for purchases.
|
|
||||||
*
|
|
||||||
* @author sebasjm
|
|
||||||
*/
|
|
||||||
|
|
||||||
import {
|
|
||||||
AmountJson,
|
|
||||||
Amounts,
|
|
||||||
NotificationType,
|
|
||||||
Product,
|
|
||||||
} from "@gnu-taler/taler-util";
|
|
||||||
import { h, VNode } from "preact";
|
|
||||||
import { useEffect, useState } from "preact/hooks";
|
|
||||||
import { Amount } from "../components/Amount.js";
|
|
||||||
import { Loading } from "../components/Loading.js";
|
|
||||||
import { LoadingError } from "../components/LoadingError.js";
|
|
||||||
import { LogoHeader } from "../components/LogoHeader.js";
|
|
||||||
import { Part } from "../components/Part.js";
|
|
||||||
import {
|
|
||||||
ButtonSuccess,
|
|
||||||
SubTitle,
|
|
||||||
WalletAction,
|
|
||||||
} from "../components/styled/index.js";
|
|
||||||
import { useTranslationContext } from "../context/translation.js";
|
|
||||||
import { HookError, useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
|
||||||
import { Button } from "../mui/Button.js";
|
|
||||||
import { ButtonHandler } from "../mui/handlers.js";
|
|
||||||
import * as wxApi from "../wxApi.js";
|
|
||||||
import { ProductList } from "./Pay.js";
|
|
||||||
|
|
||||||
interface Props {
|
|
||||||
talerRefundUri?: string;
|
|
||||||
}
|
|
||||||
export interface ViewProps {
|
|
||||||
state: State;
|
|
||||||
}
|
|
||||||
export function View({ state }: ViewProps): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
|
||||||
if (state.status === "loading") {
|
|
||||||
if (!state.hook) {
|
|
||||||
return <Loading />;
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<LoadingError
|
|
||||||
title={<i18n.Translate>Could not load refund status</i18n.Translate>}
|
|
||||||
error={state.hook}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state.status === "ignored") {
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash refund</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<section>
|
|
||||||
<p>
|
|
||||||
<i18n.Translate>You've ignored the tip.</i18n.Translate>
|
|
||||||
</p>
|
|
||||||
</section>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state.status === "in-progress") {
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash refund</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<section>
|
|
||||||
<p>
|
|
||||||
<i18n.Translate>The refund is in progress.</i18n.Translate>
|
|
||||||
</p>
|
|
||||||
</section>
|
|
||||||
<section>
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Total to refund</i18n.Translate>}
|
|
||||||
text={<Amount value={state.awaitingAmount} />}
|
|
||||||
kind="negative"
|
|
||||||
/>
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Refunded</i18n.Translate>}
|
|
||||||
text={<Amount value={state.amount} />}
|
|
||||||
kind="negative"
|
|
||||||
/>
|
|
||||||
</section>
|
|
||||||
{state.products && state.products.length ? (
|
|
||||||
<section>
|
|
||||||
<ProductList products={state.products} />
|
|
||||||
</section>
|
|
||||||
) : undefined}
|
|
||||||
{/* <section>
|
|
||||||
<ProgressBar value={state.progress} />
|
|
||||||
</section> */}
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state.status === "completed") {
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash refund</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<section>
|
|
||||||
<p>
|
|
||||||
<i18n.Translate>this refund is already accepted.</i18n.Translate>
|
|
||||||
</p>
|
|
||||||
</section>
|
|
||||||
<section>
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Total to refunded</i18n.Translate>}
|
|
||||||
text={<Amount value={state.granted} />}
|
|
||||||
kind="negative"
|
|
||||||
/>
|
|
||||||
</section>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash refund</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<section>
|
|
||||||
<p>
|
|
||||||
<i18n.Translate>
|
|
||||||
The merchant "<b>{state.merchantName}</b>" is offering you
|
|
||||||
a refund.
|
|
||||||
</i18n.Translate>
|
|
||||||
</p>
|
|
||||||
</section>
|
|
||||||
<section>
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Order amount</i18n.Translate>}
|
|
||||||
text={<Amount value={state.amount} />}
|
|
||||||
kind="neutral"
|
|
||||||
/>
|
|
||||||
{Amounts.isNonZero(state.granted) && (
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Already refunded</i18n.Translate>}
|
|
||||||
text={<Amount value={state.granted} />}
|
|
||||||
kind="neutral"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
<Part
|
|
||||||
big
|
|
||||||
title={<i18n.Translate>Refund offered</i18n.Translate>}
|
|
||||||
text={<Amount value={state.awaitingAmount} />}
|
|
||||||
kind="positive"
|
|
||||||
/>
|
|
||||||
</section>
|
|
||||||
{state.products && state.products.length ? (
|
|
||||||
<section>
|
|
||||||
<ProductList products={state.products} />
|
|
||||||
</section>
|
|
||||||
) : undefined}
|
|
||||||
<section>
|
|
||||||
<Button variant="contained" onClick={state.accept.onClick}>
|
|
||||||
<i18n.Translate>Confirm refund</i18n.Translate>
|
|
||||||
</Button>
|
|
||||||
</section>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
type State = Loading | Ready | Ignored | InProgress | Completed;
|
|
||||||
|
|
||||||
interface Loading {
|
|
||||||
status: "loading";
|
|
||||||
hook: HookError | undefined;
|
|
||||||
}
|
|
||||||
interface Ready {
|
|
||||||
status: "ready";
|
|
||||||
hook: undefined;
|
|
||||||
merchantName: string;
|
|
||||||
products: Product[] | undefined;
|
|
||||||
amount: AmountJson;
|
|
||||||
awaitingAmount: AmountJson;
|
|
||||||
granted: AmountJson;
|
|
||||||
accept: ButtonHandler;
|
|
||||||
ignore: ButtonHandler;
|
|
||||||
orderId: string;
|
|
||||||
}
|
|
||||||
interface Ignored {
|
|
||||||
status: "ignored";
|
|
||||||
hook: undefined;
|
|
||||||
merchantName: string;
|
|
||||||
}
|
|
||||||
interface InProgress {
|
|
||||||
status: "in-progress";
|
|
||||||
hook: undefined;
|
|
||||||
merchantName: string;
|
|
||||||
products: Product[] | undefined;
|
|
||||||
amount: AmountJson;
|
|
||||||
awaitingAmount: AmountJson;
|
|
||||||
granted: AmountJson;
|
|
||||||
}
|
|
||||||
interface Completed {
|
|
||||||
status: "completed";
|
|
||||||
hook: undefined;
|
|
||||||
merchantName: string;
|
|
||||||
products: Product[] | undefined;
|
|
||||||
amount: AmountJson;
|
|
||||||
granted: AmountJson;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useComponentState(
|
|
||||||
talerRefundUri: string | undefined,
|
|
||||||
api: typeof wxApi,
|
|
||||||
): State {
|
|
||||||
const [ignored, setIgnored] = useState(false);
|
|
||||||
|
|
||||||
const info = useAsyncAsHook(async () => {
|
|
||||||
if (!talerRefundUri) throw Error("ERROR_NO-URI-FOR-REFUND");
|
|
||||||
const refund = await api.prepareRefund({ talerRefundUri });
|
|
||||||
return { refund, uri: talerRefundUri };
|
|
||||||
});
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
api.onUpdateNotification([NotificationType.RefreshMelted], () => {
|
|
||||||
info?.retry();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!info || info.hasError) {
|
|
||||||
return {
|
|
||||||
status: "loading",
|
|
||||||
hook: info,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const { refund, uri } = info.response;
|
|
||||||
|
|
||||||
const doAccept = async (): Promise<void> => {
|
|
||||||
await api.applyRefund(uri);
|
|
||||||
info.retry();
|
|
||||||
};
|
|
||||||
|
|
||||||
const doIgnore = async (): Promise<void> => {
|
|
||||||
setIgnored(true);
|
|
||||||
};
|
|
||||||
|
|
||||||
if (ignored) {
|
|
||||||
return {
|
|
||||||
status: "ignored",
|
|
||||||
hook: undefined,
|
|
||||||
merchantName: info.response.refund.info.merchant.name,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const awaitingAmount = Amounts.parseOrThrow(refund.awaiting);
|
|
||||||
|
|
||||||
if (Amounts.isZero(awaitingAmount)) {
|
|
||||||
return {
|
|
||||||
status: "completed",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow(info.response.refund.effectivePaid),
|
|
||||||
granted: Amounts.parseOrThrow(info.response.refund.granted),
|
|
||||||
merchantName: info.response.refund.info.merchant.name,
|
|
||||||
products: info.response.refund.info.products,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (refund.pending) {
|
|
||||||
return {
|
|
||||||
status: "in-progress",
|
|
||||||
hook: undefined,
|
|
||||||
awaitingAmount,
|
|
||||||
amount: Amounts.parseOrThrow(info.response.refund.effectivePaid),
|
|
||||||
granted: Amounts.parseOrThrow(info.response.refund.granted),
|
|
||||||
|
|
||||||
merchantName: info.response.refund.info.merchant.name,
|
|
||||||
products: info.response.refund.info.products,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
amount: Amounts.parseOrThrow(info.response.refund.effectivePaid),
|
|
||||||
granted: Amounts.parseOrThrow(info.response.refund.granted),
|
|
||||||
awaitingAmount,
|
|
||||||
merchantName: info.response.refund.info.merchant.name,
|
|
||||||
products: info.response.refund.info.products,
|
|
||||||
orderId: info.response.refund.info.orderId,
|
|
||||||
accept: {
|
|
||||||
onClick: doAccept,
|
|
||||||
},
|
|
||||||
ignore: {
|
|
||||||
onClick: doIgnore,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function RefundPage({ talerRefundUri }: Props): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
|
||||||
|
|
||||||
const state = useComponentState(talerRefundUri, wxApi);
|
|
||||||
|
|
||||||
if (!talerRefundUri) {
|
|
||||||
return (
|
|
||||||
<span>
|
|
||||||
<i18n.Translate>missing taler refund uri</i18n.Translate>
|
|
||||||
</span>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return <View state={state} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
function ProgressBar({ value }: { value: number }): VNode {
|
|
||||||
return (
|
|
||||||
<div
|
|
||||||
style={{
|
|
||||||
width: 400,
|
|
||||||
height: 20,
|
|
||||||
backgroundColor: "white",
|
|
||||||
border: "solid black 1px",
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
style={{
|
|
||||||
width: `${value * 100}%`,
|
|
||||||
height: "100%",
|
|
||||||
backgroundColor: "lightgreen",
|
|
||||||
}}
|
|
||||||
></div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
94
packages/taler-wallet-webextension/src/cta/Refund/index.ts
Normal file
94
packages/taler-wallet-webextension/src/cta/Refund/index.ts
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { AmountJson, Product } from "@gnu-taler/taler-util";
|
||||||
|
import { Loading } from "../../components/Loading.js";
|
||||||
|
import { HookError } from "../../hooks/useAsyncAsHook.js";
|
||||||
|
import { ButtonHandler } from "../../mui/handlers.js";
|
||||||
|
import { compose, StateViewMap } from "../../utils/index.js";
|
||||||
|
import * as wxApi from "../../wxApi.js";
|
||||||
|
import { useComponentState } from "./state.js";
|
||||||
|
import { CompletedView, IgnoredView, InProgressView, LoadingUriView, ReadyView } from "./views.js";
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
export interface Props {
|
||||||
|
talerRefundUri?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type State =
|
||||||
|
| State.Loading
|
||||||
|
| State.LoadingUriError
|
||||||
|
| State.Ready
|
||||||
|
| State.Ignored
|
||||||
|
| State.InProgress
|
||||||
|
| State.Completed;
|
||||||
|
|
||||||
|
export namespace State {
|
||||||
|
|
||||||
|
export interface Loading {
|
||||||
|
status: "loading";
|
||||||
|
error: undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface LoadingUriError {
|
||||||
|
status: "loading-uri";
|
||||||
|
error: HookError;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface BaseInfo {
|
||||||
|
merchantName: string;
|
||||||
|
products: Product[] | undefined;
|
||||||
|
amount: AmountJson;
|
||||||
|
awaitingAmount: AmountJson;
|
||||||
|
granted: AmountJson;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Ready extends BaseInfo {
|
||||||
|
status: "ready";
|
||||||
|
error: undefined;
|
||||||
|
|
||||||
|
accept: ButtonHandler;
|
||||||
|
ignore: ButtonHandler;
|
||||||
|
orderId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Ignored extends BaseInfo {
|
||||||
|
status: "ignored";
|
||||||
|
error: undefined;
|
||||||
|
}
|
||||||
|
export interface InProgress extends BaseInfo {
|
||||||
|
status: "in-progress";
|
||||||
|
error: undefined;
|
||||||
|
|
||||||
|
}
|
||||||
|
export interface Completed extends BaseInfo {
|
||||||
|
status: "completed";
|
||||||
|
error: undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
const viewMapping: StateViewMap<State> = {
|
||||||
|
loading: Loading,
|
||||||
|
"loading-uri": LoadingUriView,
|
||||||
|
"in-progress": InProgressView,
|
||||||
|
completed: CompletedView,
|
||||||
|
ignored: IgnoredView,
|
||||||
|
ready: ReadyView,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const RefundPage = compose("Refund", (p: Props) => useComponentState(p, wxApi), viewMapping)
|
104
packages/taler-wallet-webextension/src/cta/Refund/state.ts
Normal file
104
packages/taler-wallet-webextension/src/cta/Refund/state.ts
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
import { Amounts, NotificationType } from "@gnu-taler/taler-util";
|
||||||
|
import { useEffect, useState } from "preact/hooks";
|
||||||
|
import { useAsyncAsHook } from "../../hooks/useAsyncAsHook.js";
|
||||||
|
import * as wxApi from "../../wxApi.js";
|
||||||
|
import { Props, State } from "./index.js";
|
||||||
|
|
||||||
|
export function useComponentState(
|
||||||
|
{ talerRefundUri }: Props,
|
||||||
|
api: typeof wxApi,
|
||||||
|
): State {
|
||||||
|
const [ignored, setIgnored] = useState(false);
|
||||||
|
|
||||||
|
const info = useAsyncAsHook(async () => {
|
||||||
|
if (!talerRefundUri) throw Error("ERROR_NO-URI-FOR-REFUND");
|
||||||
|
const refund = await api.prepareRefund({ talerRefundUri });
|
||||||
|
return { refund, uri: talerRefundUri };
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
api.onUpdateNotification([NotificationType.RefreshMelted], () => {
|
||||||
|
info?.retry();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!info) {
|
||||||
|
return { status: "loading", error: undefined }
|
||||||
|
}
|
||||||
|
if (info.hasError) {
|
||||||
|
return {
|
||||||
|
status: "loading-uri",
|
||||||
|
error: info,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const { refund, uri } = info.response;
|
||||||
|
|
||||||
|
const doAccept = async (): Promise<void> => {
|
||||||
|
await api.applyRefund(uri);
|
||||||
|
info.retry();
|
||||||
|
};
|
||||||
|
|
||||||
|
const doIgnore = async (): Promise<void> => {
|
||||||
|
setIgnored(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
const baseInfo = {
|
||||||
|
amount: Amounts.parseOrThrow(info.response.refund.effectivePaid),
|
||||||
|
granted: Amounts.parseOrThrow(info.response.refund.granted),
|
||||||
|
merchantName: info.response.refund.info.merchant.name,
|
||||||
|
products: info.response.refund.info.products,
|
||||||
|
awaitingAmount: Amounts.parseOrThrow(refund.awaiting),
|
||||||
|
error: undefined,
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ignored) {
|
||||||
|
return {
|
||||||
|
status: "ignored",
|
||||||
|
...baseInfo,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Amounts.isZero(baseInfo.awaitingAmount)) {
|
||||||
|
return {
|
||||||
|
status: "completed",
|
||||||
|
...baseInfo,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (refund.pending) {
|
||||||
|
return {
|
||||||
|
status: "in-progress",
|
||||||
|
...baseInfo,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
status: "ready",
|
||||||
|
...baseInfo,
|
||||||
|
orderId: info.response.refund.info.orderId,
|
||||||
|
accept: {
|
||||||
|
onClick: doAccept,
|
||||||
|
},
|
||||||
|
ignore: {
|
||||||
|
onClick: doIgnore,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
@ -0,0 +1,96 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Sebastian Javier Marchano (sebasjm)
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Amounts } from "@gnu-taler/taler-util";
|
||||||
|
import beer from "../../../static-dev/beer.png";
|
||||||
|
import { createExample } from "../../test-utils.js";
|
||||||
|
import {
|
||||||
|
CompletedView,
|
||||||
|
IgnoredView,
|
||||||
|
InProgressView,
|
||||||
|
ReadyView,
|
||||||
|
} from "./views.js";
|
||||||
|
export default {
|
||||||
|
title: "cta/refund",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Complete = createExample(CompletedView, {
|
||||||
|
status: "completed",
|
||||||
|
amount: Amounts.parseOrThrow("USD:1"),
|
||||||
|
granted: Amounts.parseOrThrow("USD:1"),
|
||||||
|
error: undefined,
|
||||||
|
merchantName: "the merchant",
|
||||||
|
products: undefined,
|
||||||
|
});
|
||||||
|
|
||||||
|
export const InProgress = createExample(InProgressView, {
|
||||||
|
status: "in-progress",
|
||||||
|
error: undefined,
|
||||||
|
amount: Amounts.parseOrThrow("USD:1"),
|
||||||
|
awaitingAmount: Amounts.parseOrThrow("USD:1"),
|
||||||
|
granted: Amounts.parseOrThrow("USD:0"),
|
||||||
|
merchantName: "the merchant",
|
||||||
|
products: undefined,
|
||||||
|
});
|
||||||
|
|
||||||
|
export const Ready = createExample(ReadyView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
accept: {},
|
||||||
|
ignore: {},
|
||||||
|
|
||||||
|
amount: Amounts.parseOrThrow("USD:1"),
|
||||||
|
awaitingAmount: Amounts.parseOrThrow("USD:1"),
|
||||||
|
granted: Amounts.parseOrThrow("USD:0"),
|
||||||
|
merchantName: "the merchant",
|
||||||
|
products: [],
|
||||||
|
orderId: "abcdef",
|
||||||
|
});
|
||||||
|
|
||||||
|
export const WithAProductList = createExample(ReadyView, {
|
||||||
|
status: "ready",
|
||||||
|
error: undefined,
|
||||||
|
accept: {},
|
||||||
|
ignore: {},
|
||||||
|
amount: Amounts.parseOrThrow("USD:1"),
|
||||||
|
awaitingAmount: Amounts.parseOrThrow("USD:1"),
|
||||||
|
granted: Amounts.parseOrThrow("USD:0"),
|
||||||
|
merchantName: "the merchant",
|
||||||
|
products: [
|
||||||
|
{
|
||||||
|
description: "beer",
|
||||||
|
image: beer,
|
||||||
|
quantity: 2,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
description: "t-shirt",
|
||||||
|
price: "EUR:1",
|
||||||
|
quantity: 5,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
orderId: "abcdef",
|
||||||
|
});
|
||||||
|
|
||||||
|
export const Ignored = createExample(IgnoredView, {
|
||||||
|
status: "ignored",
|
||||||
|
error: undefined,
|
||||||
|
merchantName: "the merchant",
|
||||||
|
});
|
@ -21,22 +21,19 @@
|
|||||||
|
|
||||||
import {
|
import {
|
||||||
AmountJson,
|
AmountJson,
|
||||||
Amounts,
|
Amounts, NotificationType,
|
||||||
NotificationType,
|
PrepareRefundResult
|
||||||
PrepareRefundResult,
|
|
||||||
} from "@gnu-taler/taler-util";
|
} from "@gnu-taler/taler-util";
|
||||||
import { expect } from "chai";
|
import { expect } from "chai";
|
||||||
import { mountHook } from "../test-utils.js";
|
import { mountHook } from "../../test-utils.js";
|
||||||
import { SubsHandler } from "./Pay.test.js";
|
import { SubsHandler } from "../Payment/test.js";
|
||||||
import { useComponentState } from "./Refund.jsx";
|
import { useComponentState } from "./state.js";
|
||||||
|
|
||||||
// onUpdateNotification: subscriptions.saveSubscription,
|
|
||||||
|
|
||||||
describe("Refund CTA states", () => {
|
describe("Refund CTA states", () => {
|
||||||
it("should tell the user that the URI is missing", async () => {
|
it("should tell the user that the URI is missing", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState(undefined, {
|
useComponentState({ talerRefundUri: undefined }, {
|
||||||
prepareRefund: async () => ({}),
|
prepareRefund: async () => ({}),
|
||||||
applyRefund: async () => ({}),
|
applyRefund: async () => ({}),
|
||||||
onUpdateNotification: async () => ({}),
|
onUpdateNotification: async () => ({}),
|
||||||
@ -44,21 +41,21 @@ describe("Refund CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
|
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading-uri");
|
||||||
if (!hook) expect.fail();
|
if (!error) expect.fail();
|
||||||
if (!hook.hasError) expect.fail();
|
if (!error.hasError) expect.fail();
|
||||||
if (hook.operational) expect.fail();
|
if (error.operational) expect.fail();
|
||||||
expect(hook.message).eq("ERROR_NO-URI-FOR-REFUND");
|
expect(error.message).eq("ERROR_NO-URI-FOR-REFUND");
|
||||||
}
|
}
|
||||||
|
|
||||||
await assertNoPendingUpdate();
|
await assertNoPendingUpdate();
|
||||||
@ -67,7 +64,7 @@ describe("Refund CTA states", () => {
|
|||||||
it("should be ready after loading", async () => {
|
it("should be ready after loading", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taler://refund/asdasdas", {
|
useComponentState({ talerRefundUri: "taler://refund/asdasdas" }, {
|
||||||
prepareRefund: async () =>
|
prepareRefund: async () =>
|
||||||
({
|
({
|
||||||
effectivePaid: "EUR:2",
|
effectivePaid: "EUR:2",
|
||||||
@ -91,9 +88,9 @@ describe("Refund CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -102,7 +99,7 @@ describe("Refund CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "ready") expect.fail();
|
if (state.status !== "ready") expect.fail();
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.accept.onClick).not.undefined;
|
expect(state.accept.onClick).not.undefined;
|
||||||
expect(state.ignore.onClick).not.undefined;
|
expect(state.ignore.onClick).not.undefined;
|
||||||
expect(state.merchantName).eq("the merchant name");
|
expect(state.merchantName).eq("the merchant name");
|
||||||
@ -116,7 +113,7 @@ describe("Refund CTA states", () => {
|
|||||||
it("should be ignored after clicking the ignore button", async () => {
|
it("should be ignored after clicking the ignore button", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taler://refund/asdasdas", {
|
useComponentState({ talerRefundUri: "taler://refund/asdasdas" }, {
|
||||||
prepareRefund: async () =>
|
prepareRefund: async () =>
|
||||||
({
|
({
|
||||||
effectivePaid: "EUR:2",
|
effectivePaid: "EUR:2",
|
||||||
@ -140,9 +137,9 @@ describe("Refund CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -151,7 +148,7 @@ describe("Refund CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "ready") expect.fail();
|
if (state.status !== "ready") expect.fail();
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.accept.onClick).not.undefined;
|
expect(state.accept.onClick).not.undefined;
|
||||||
expect(state.merchantName).eq("the merchant name");
|
expect(state.merchantName).eq("the merchant name");
|
||||||
expect(state.orderId).eq("orderId1");
|
expect(state.orderId).eq("orderId1");
|
||||||
@ -167,7 +164,7 @@ describe("Refund CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "ignored") expect.fail();
|
if (state.status !== "ignored") expect.fail();
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.merchantName).eq("the merchant name");
|
expect(state.merchantName).eq("the merchant name");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -192,7 +189,7 @@ describe("Refund CTA states", () => {
|
|||||||
|
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taler://refund/asdasdas", {
|
useComponentState({ talerRefundUri: "taler://refund/asdasdas" }, {
|
||||||
prepareRefund: async () =>
|
prepareRefund: async () =>
|
||||||
({
|
({
|
||||||
awaiting: Amounts.stringify(awaiting),
|
awaiting: Amounts.stringify(awaiting),
|
||||||
@ -216,9 +213,9 @@ describe("Refund CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -227,7 +224,7 @@ describe("Refund CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "in-progress") expect.fail("1");
|
if (state.status !== "in-progress") expect.fail("1");
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.merchantName).eq("the merchant name");
|
expect(state.merchantName).eq("the merchant name");
|
||||||
expect(state.products).undefined;
|
expect(state.products).undefined;
|
||||||
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:2"));
|
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:2"));
|
||||||
@ -242,7 +239,7 @@ describe("Refund CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "in-progress") expect.fail("2");
|
if (state.status !== "in-progress") expect.fail("2");
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.merchantName).eq("the merchant name");
|
expect(state.merchantName).eq("the merchant name");
|
||||||
expect(state.products).undefined;
|
expect(state.products).undefined;
|
||||||
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:2"));
|
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:2"));
|
||||||
@ -257,7 +254,7 @@ describe("Refund CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "completed") expect.fail("3");
|
if (state.status !== "completed") expect.fail("3");
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.merchantName).eq("the merchant name");
|
expect(state.merchantName).eq("the merchant name");
|
||||||
expect(state.products).undefined;
|
expect(state.products).undefined;
|
||||||
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:2"));
|
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:2"));
|
172
packages/taler-wallet-webextension/src/cta/Refund/views.tsx
Normal file
172
packages/taler-wallet-webextension/src/cta/Refund/views.tsx
Normal file
@ -0,0 +1,172 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Amounts } from "@gnu-taler/taler-util";
|
||||||
|
import { Fragment, h, VNode } from "preact";
|
||||||
|
import { Amount } from "../../components/Amount.js";
|
||||||
|
import { LoadingError } from "../../components/LoadingError.js";
|
||||||
|
import { LogoHeader } from "../../components/LogoHeader.js";
|
||||||
|
import { Part } from "../../components/Part.js";
|
||||||
|
import { SubTitle, WalletAction } from "../../components/styled/index.js";
|
||||||
|
import { useTranslationContext } from "../../context/translation.js";
|
||||||
|
import { Button } from "../../mui/Button.js";
|
||||||
|
import { ProductList } from "../Payment/views.js";
|
||||||
|
import { State } from "./index.js";
|
||||||
|
|
||||||
|
export function LoadingUriView({ error }: State.LoadingUriError): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<LoadingError
|
||||||
|
title={<i18n.Translate>Could not load refund status</i18n.Translate>}
|
||||||
|
error={error}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function IgnoredView(state: State.Ignored): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<WalletAction>
|
||||||
|
<LogoHeader />
|
||||||
|
|
||||||
|
<SubTitle>
|
||||||
|
<i18n.Translate>Digital cash refund</i18n.Translate>
|
||||||
|
</SubTitle>
|
||||||
|
<section>
|
||||||
|
<p>
|
||||||
|
<i18n.Translate>You've ignored the tip.</i18n.Translate>
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
</WalletAction>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
export function InProgressView(state: State.InProgress): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<WalletAction>
|
||||||
|
<LogoHeader />
|
||||||
|
|
||||||
|
<SubTitle>
|
||||||
|
<i18n.Translate>Digital cash refund</i18n.Translate>
|
||||||
|
</SubTitle>
|
||||||
|
<section>
|
||||||
|
<p>
|
||||||
|
<i18n.Translate>The refund is in progress.</i18n.Translate>
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<Part
|
||||||
|
big
|
||||||
|
title={<i18n.Translate>Total to refund</i18n.Translate>}
|
||||||
|
text={<Amount value={state.awaitingAmount} />}
|
||||||
|
kind="negative"
|
||||||
|
/>
|
||||||
|
<Part
|
||||||
|
big
|
||||||
|
title={<i18n.Translate>Refunded</i18n.Translate>}
|
||||||
|
text={<Amount value={state.amount} />}
|
||||||
|
kind="negative"
|
||||||
|
/>
|
||||||
|
</section>
|
||||||
|
{state.products && state.products.length ? (
|
||||||
|
<section>
|
||||||
|
<ProductList products={state.products} />
|
||||||
|
</section>
|
||||||
|
) : undefined}
|
||||||
|
</WalletAction>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
export function CompletedView(state: State.Completed): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<WalletAction>
|
||||||
|
<LogoHeader />
|
||||||
|
|
||||||
|
<SubTitle>
|
||||||
|
<i18n.Translate>Digital cash refund</i18n.Translate>
|
||||||
|
</SubTitle>
|
||||||
|
<section>
|
||||||
|
<p>
|
||||||
|
<i18n.Translate>this refund is already accepted.</i18n.Translate>
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<Part
|
||||||
|
big
|
||||||
|
title={<i18n.Translate>Total to refunded</i18n.Translate>}
|
||||||
|
text={<Amount value={state.granted} />}
|
||||||
|
kind="negative"
|
||||||
|
/>
|
||||||
|
</section>
|
||||||
|
</WalletAction>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
export function ReadyView(state: State.Ready): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
return (
|
||||||
|
<WalletAction>
|
||||||
|
<LogoHeader />
|
||||||
|
|
||||||
|
<SubTitle>
|
||||||
|
<i18n.Translate>Digital cash refund</i18n.Translate>
|
||||||
|
</SubTitle>
|
||||||
|
<section>
|
||||||
|
<p>
|
||||||
|
<i18n.Translate>
|
||||||
|
The merchant "<b>{state.merchantName}</b>" is offering you
|
||||||
|
a refund.
|
||||||
|
</i18n.Translate>
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<Part
|
||||||
|
big
|
||||||
|
title={<i18n.Translate>Order amount</i18n.Translate>}
|
||||||
|
text={<Amount value={state.amount} />}
|
||||||
|
kind="neutral"
|
||||||
|
/>
|
||||||
|
{Amounts.isNonZero(state.granted) && (
|
||||||
|
<Part
|
||||||
|
big
|
||||||
|
title={<i18n.Translate>Already refunded</i18n.Translate>}
|
||||||
|
text={<Amount value={state.granted} />}
|
||||||
|
kind="neutral"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<Part
|
||||||
|
big
|
||||||
|
title={<i18n.Translate>Refund offered</i18n.Translate>}
|
||||||
|
text={<Amount value={state.awaitingAmount} />}
|
||||||
|
kind="positive"
|
||||||
|
/>
|
||||||
|
</section>
|
||||||
|
{state.products && state.products.length ? (
|
||||||
|
<section>
|
||||||
|
<ProductList products={state.products} />
|
||||||
|
</section>
|
||||||
|
) : undefined}
|
||||||
|
<section>
|
||||||
|
<Button variant="contained" onClick={state.accept.onClick}>
|
||||||
|
<i18n.Translate>Confirm refund</i18n.Translate>
|
||||||
|
</Button>
|
||||||
|
</section>
|
||||||
|
</WalletAction>
|
||||||
|
);
|
||||||
|
}
|
@ -1,241 +0,0 @@
|
|||||||
/*
|
|
||||||
This file is part of GNU Taler
|
|
||||||
(C) 2022 Taler Systems S.A.
|
|
||||||
|
|
||||||
GNU 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.
|
|
||||||
|
|
||||||
GNU 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
|
|
||||||
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Page shown to the user to accept or ignore a tip from a merchant.
|
|
||||||
*
|
|
||||||
* @author sebasjm
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { AmountJson, Amounts, PrepareTipResult } from "@gnu-taler/taler-util";
|
|
||||||
import { h, VNode } from "preact";
|
|
||||||
import { useEffect, useState } from "preact/hooks";
|
|
||||||
import { Amount } from "../components/Amount.js";
|
|
||||||
import { Loading } from "../components/Loading.js";
|
|
||||||
import { LoadingError } from "../components/LoadingError.js";
|
|
||||||
import { LogoHeader } from "../components/LogoHeader.js";
|
|
||||||
import { Part } from "../components/Part.js";
|
|
||||||
import {
|
|
||||||
Button,
|
|
||||||
ButtonSuccess,
|
|
||||||
SubTitle,
|
|
||||||
WalletAction,
|
|
||||||
} from "../components/styled/index.js";
|
|
||||||
import { useTranslationContext } from "../context/translation.js";
|
|
||||||
import { HookError, useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
|
||||||
import { ButtonHandler } from "../mui/handlers.js";
|
|
||||||
import * as wxApi from "../wxApi.js";
|
|
||||||
|
|
||||||
interface Props {
|
|
||||||
talerTipUri?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
type State = Loading | Ready | Accepted | Ignored;
|
|
||||||
|
|
||||||
interface Loading {
|
|
||||||
status: "loading";
|
|
||||||
hook: HookError | undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface Ignored {
|
|
||||||
status: "ignored";
|
|
||||||
hook: undefined;
|
|
||||||
}
|
|
||||||
interface Accepted {
|
|
||||||
status: "accepted";
|
|
||||||
hook: undefined;
|
|
||||||
merchantBaseUrl: string;
|
|
||||||
amount: AmountJson;
|
|
||||||
exchangeBaseUrl: string;
|
|
||||||
}
|
|
||||||
interface Ready {
|
|
||||||
status: "ready";
|
|
||||||
hook: undefined;
|
|
||||||
merchantBaseUrl: string;
|
|
||||||
amount: AmountJson;
|
|
||||||
exchangeBaseUrl: string;
|
|
||||||
accept: ButtonHandler;
|
|
||||||
ignore: ButtonHandler;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useComponentState(
|
|
||||||
talerTipUri: string | undefined,
|
|
||||||
api: typeof wxApi,
|
|
||||||
): State {
|
|
||||||
const [tipIgnored, setTipIgnored] = useState(false);
|
|
||||||
|
|
||||||
const tipInfo = useAsyncAsHook(async () => {
|
|
||||||
if (!talerTipUri) throw Error("ERROR_NO-URI-FOR-TIP");
|
|
||||||
const tip = await api.prepareTip({ talerTipUri });
|
|
||||||
return { tip };
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!tipInfo || tipInfo.hasError) {
|
|
||||||
return {
|
|
||||||
status: "loading",
|
|
||||||
hook: tipInfo,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const { tip } = tipInfo.response;
|
|
||||||
|
|
||||||
const doAccept = async (): Promise<void> => {
|
|
||||||
await api.acceptTip({ walletTipId: tip.walletTipId });
|
|
||||||
tipInfo.retry();
|
|
||||||
};
|
|
||||||
|
|
||||||
const doIgnore = async (): Promise<void> => {
|
|
||||||
setTipIgnored(true);
|
|
||||||
};
|
|
||||||
|
|
||||||
if (tipIgnored) {
|
|
||||||
return {
|
|
||||||
status: "ignored",
|
|
||||||
hook: undefined,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tip.accepted) {
|
|
||||||
return {
|
|
||||||
status: "accepted",
|
|
||||||
hook: undefined,
|
|
||||||
merchantBaseUrl: tip.merchantBaseUrl,
|
|
||||||
exchangeBaseUrl: tip.exchangeBaseUrl,
|
|
||||||
amount: Amounts.parseOrThrow(tip.tipAmountEffective),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
status: "ready",
|
|
||||||
hook: undefined,
|
|
||||||
merchantBaseUrl: tip.merchantBaseUrl,
|
|
||||||
exchangeBaseUrl: tip.exchangeBaseUrl,
|
|
||||||
accept: {
|
|
||||||
onClick: doAccept,
|
|
||||||
},
|
|
||||||
ignore: {
|
|
||||||
onClick: doIgnore,
|
|
||||||
},
|
|
||||||
amount: Amounts.parseOrThrow(tip.tipAmountEffective),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function View({ state }: { state: State }): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
|
||||||
if (state.status === "loading") {
|
|
||||||
if (!state.hook) {
|
|
||||||
return <Loading />;
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<LoadingError
|
|
||||||
title={<i18n.Translate>Could not load tip status</i18n.Translate>}
|
|
||||||
error={state.hook}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state.status === "ignored") {
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash tip</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<span>
|
|
||||||
<i18n.Translate>You've ignored the tip.</i18n.Translate>
|
|
||||||
</span>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state.status === "accepted") {
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash tip</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<section>
|
|
||||||
<i18n.Translate>
|
|
||||||
Tip from <code>{state.merchantBaseUrl}</code> accepted. Check your
|
|
||||||
transactions list for more details.
|
|
||||||
</i18n.Translate>
|
|
||||||
</section>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash tip</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
|
|
||||||
<section>
|
|
||||||
<p>
|
|
||||||
<i18n.Translate>The merchant is offering you a tip</i18n.Translate>
|
|
||||||
</p>
|
|
||||||
<Part
|
|
||||||
title={<i18n.Translate>Amount</i18n.Translate>}
|
|
||||||
text={<Amount value={state.amount} />}
|
|
||||||
kind="positive"
|
|
||||||
big
|
|
||||||
/>
|
|
||||||
<Part
|
|
||||||
title={<i18n.Translate>Merchant URL</i18n.Translate>}
|
|
||||||
text={state.merchantBaseUrl}
|
|
||||||
kind="neutral"
|
|
||||||
/>
|
|
||||||
<Part
|
|
||||||
title={<i18n.Translate>Exchange</i18n.Translate>}
|
|
||||||
text={state.exchangeBaseUrl}
|
|
||||||
kind="neutral"
|
|
||||||
/>
|
|
||||||
</section>
|
|
||||||
<section>
|
|
||||||
<Button
|
|
||||||
variant="contained"
|
|
||||||
color="success"
|
|
||||||
onClick={state.accept.onClick}
|
|
||||||
>
|
|
||||||
<i18n.Translate>Accept tip</i18n.Translate>
|
|
||||||
</Button>
|
|
||||||
<Button onClick={state.ignore.onClick}>
|
|
||||||
<i18n.Translate>Ignore</i18n.Translate>
|
|
||||||
</Button>
|
|
||||||
</section>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function TipPage({ talerTipUri }: Props): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
|
||||||
const state = useComponentState(talerTipUri, wxApi);
|
|
||||||
|
|
||||||
if (!talerTipUri) {
|
|
||||||
return (
|
|
||||||
<span>
|
|
||||||
<i18n.Translate>missing tip uri</i18n.Translate>
|
|
||||||
</span>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return <View state={state} />;
|
|
||||||
}
|
|
84
packages/taler-wallet-webextension/src/cta/Tip/index.ts
Normal file
84
packages/taler-wallet-webextension/src/cta/Tip/index.ts
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { AmountJson } from "@gnu-taler/taler-util";
|
||||||
|
import { Loading } from "../../components/Loading.js";
|
||||||
|
import { HookError } from "../../hooks/useAsyncAsHook.js";
|
||||||
|
import { ButtonHandler, SelectFieldHandler } from "../../mui/handlers.js";
|
||||||
|
import { compose, StateViewMap } from "../../utils/index.js";
|
||||||
|
import * as wxApi from "../../wxApi.js";
|
||||||
|
import {
|
||||||
|
Props as TermsOfServiceSectionProps
|
||||||
|
} from "../TermsOfServiceSection.js";
|
||||||
|
import { useComponentState } from "./state.js";
|
||||||
|
import { AcceptedView, IgnoredView, LoadingUriView, ReadyView } from "./views.js";
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
export interface Props {
|
||||||
|
talerTipUri?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type State =
|
||||||
|
| State.Loading
|
||||||
|
| State.LoadingUriError
|
||||||
|
| State.Ignored
|
||||||
|
| State.Accepted
|
||||||
|
| State.Ready
|
||||||
|
| State.Ignored;
|
||||||
|
|
||||||
|
export namespace State {
|
||||||
|
|
||||||
|
export interface Loading {
|
||||||
|
status: "loading";
|
||||||
|
error: undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface LoadingUriError {
|
||||||
|
status: "loading-uri";
|
||||||
|
error: HookError;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface BaseInfo {
|
||||||
|
merchantBaseUrl: string;
|
||||||
|
amount: AmountJson;
|
||||||
|
exchangeBaseUrl: string;
|
||||||
|
error: undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Ignored extends BaseInfo {
|
||||||
|
status: "ignored";
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Accepted extends BaseInfo {
|
||||||
|
status: "accepted";
|
||||||
|
}
|
||||||
|
export interface Ready extends BaseInfo {
|
||||||
|
status: "ready";
|
||||||
|
accept: ButtonHandler;
|
||||||
|
ignore: ButtonHandler;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const viewMapping: StateViewMap<State> = {
|
||||||
|
loading: Loading,
|
||||||
|
"loading-uri": LoadingUriView,
|
||||||
|
"accepted": AcceptedView,
|
||||||
|
"ignored": IgnoredView,
|
||||||
|
"ready": ReadyView,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const TipPage = compose("Tip", (p: Props) => useComponentState(p, wxApi), viewMapping)
|
92
packages/taler-wallet-webextension/src/cta/Tip/state.ts
Normal file
92
packages/taler-wallet-webextension/src/cta/Tip/state.ts
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
import { Amounts } from "@gnu-taler/taler-util";
|
||||||
|
import { useState } from "preact/hooks";
|
||||||
|
import { useAsyncAsHook } from "../../hooks/useAsyncAsHook.js";
|
||||||
|
import * as wxApi from "../../wxApi.js";
|
||||||
|
import { Props, State } from "./index.js";
|
||||||
|
|
||||||
|
export function useComponentState(
|
||||||
|
{ talerTipUri }: Props,
|
||||||
|
api: typeof wxApi,
|
||||||
|
): State {
|
||||||
|
const [tipIgnored, setTipIgnored] = useState(false);
|
||||||
|
|
||||||
|
const tipInfo = useAsyncAsHook(async () => {
|
||||||
|
if (!talerTipUri) throw Error("ERROR_NO-URI-FOR-TIP");
|
||||||
|
const tip = await api.prepareTip({ talerTipUri });
|
||||||
|
return { tip };
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!tipInfo) {
|
||||||
|
return {
|
||||||
|
status: "loading",
|
||||||
|
error: undefined,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (tipInfo.hasError) {
|
||||||
|
return {
|
||||||
|
status: "loading-uri",
|
||||||
|
error: tipInfo,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const { tip } = tipInfo.response;
|
||||||
|
|
||||||
|
const doAccept = async (): Promise<void> => {
|
||||||
|
await api.acceptTip({ walletTipId: tip.walletTipId });
|
||||||
|
tipInfo.retry();
|
||||||
|
};
|
||||||
|
|
||||||
|
const doIgnore = async (): Promise<void> => {
|
||||||
|
setTipIgnored(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
const baseInfo = {
|
||||||
|
merchantBaseUrl: tip.merchantBaseUrl,
|
||||||
|
exchangeBaseUrl: tip.exchangeBaseUrl,
|
||||||
|
amount: Amounts.parseOrThrow(tip.tipAmountEffective),
|
||||||
|
error: undefined,
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tipIgnored) {
|
||||||
|
return {
|
||||||
|
status: "ignored",
|
||||||
|
...baseInfo,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tip.accepted) {
|
||||||
|
return {
|
||||||
|
status: "accepted",
|
||||||
|
...baseInfo,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
status: "ready",
|
||||||
|
...baseInfo,
|
||||||
|
accept: {
|
||||||
|
onClick: doAccept,
|
||||||
|
},
|
||||||
|
ignore: {
|
||||||
|
onClick: doIgnore,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
@ -20,33 +20,27 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Amounts } from "@gnu-taler/taler-util";
|
import { Amounts } from "@gnu-taler/taler-util";
|
||||||
import { createExample } from "../test-utils.js";
|
import { createExample } from "../../test-utils.js";
|
||||||
import { View as TestedComponent } from "./Tip.js";
|
import { AcceptedView, ReadyView } from "./views.js";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
title: "cta/tip",
|
title: "cta/tip",
|
||||||
component: TestedComponent,
|
|
||||||
argTypes: {},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Accepted = createExample(TestedComponent, {
|
export const Accepted = createExample(AcceptedView, {
|
||||||
state: {
|
|
||||||
status: "accepted",
|
status: "accepted",
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
amount: Amounts.parseOrThrow("EUR:1"),
|
amount: Amounts.parseOrThrow("EUR:1"),
|
||||||
exchangeBaseUrl: "",
|
exchangeBaseUrl: "",
|
||||||
merchantBaseUrl: "",
|
merchantBaseUrl: "",
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
export const Ready = createExample(TestedComponent, {
|
export const Ready = createExample(ReadyView, {
|
||||||
state: {
|
|
||||||
status: "ready",
|
status: "ready",
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
amount: Amounts.parseOrThrow("EUR:1"),
|
amount: Amounts.parseOrThrow("EUR:1"),
|
||||||
merchantBaseUrl: "http://merchant.url/",
|
merchantBaseUrl: "http://merchant.url/",
|
||||||
exchangeBaseUrl: "http://exchange.url/",
|
exchangeBaseUrl: "http://exchange.url/",
|
||||||
accept: {},
|
accept: {},
|
||||||
ignore: {},
|
ignore: {},
|
||||||
},
|
|
||||||
});
|
});
|
@ -19,37 +19,39 @@
|
|||||||
* @author Sebastian Javier Marchano (sebasjm)
|
* @author Sebastian Javier Marchano (sebasjm)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { Amounts, PrepareTipResult } from "@gnu-taler/taler-util";
|
import {
|
||||||
|
Amounts, PrepareTipResult
|
||||||
|
} from "@gnu-taler/taler-util";
|
||||||
import { expect } from "chai";
|
import { expect } from "chai";
|
||||||
import { mountHook } from "../test-utils.js";
|
import { mountHook } from "../../test-utils.js";
|
||||||
import { useComponentState } from "./Tip.jsx";
|
import { useComponentState } from "./state.js";
|
||||||
|
|
||||||
describe("Tip CTA states", () => {
|
describe("Tip CTA states", () => {
|
||||||
it("should tell the user that the URI is missing", async () => {
|
it("should tell the user that the URI is missing", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState(undefined, {
|
useComponentState({ talerTipUri: undefined }, {
|
||||||
prepareTip: async () => ({}),
|
prepareTip: async () => ({}),
|
||||||
acceptTip: async () => ({}),
|
acceptTip: async () => ({}),
|
||||||
} as any),
|
} as any),
|
||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
|
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading-uri");
|
||||||
if (!hook) expect.fail();
|
if (!error) expect.fail();
|
||||||
if (!hook.hasError) expect.fail();
|
if (!error.hasError) expect.fail();
|
||||||
if (hook.operational) expect.fail();
|
if (error.operational) expect.fail();
|
||||||
expect(hook.message).eq("ERROR_NO-URI-FOR-TIP");
|
expect(error.message).eq("ERROR_NO-URI-FOR-TIP");
|
||||||
}
|
}
|
||||||
|
|
||||||
await assertNoPendingUpdate();
|
await assertNoPendingUpdate();
|
||||||
@ -60,7 +62,7 @@ describe("Tip CTA states", () => {
|
|||||||
|
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taler://tip/asd", {
|
useComponentState({ talerTipUri: "taler://tip/asd" }, {
|
||||||
prepareTip: async () =>
|
prepareTip: async () =>
|
||||||
({
|
({
|
||||||
accepted: tipAccepted,
|
accepted: tipAccepted,
|
||||||
@ -76,9 +78,9 @@ describe("Tip CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -87,7 +89,7 @@ describe("Tip CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "ready") expect.fail();
|
if (state.status !== "ready") expect.fail();
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:1"));
|
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:1"));
|
||||||
expect(state.merchantBaseUrl).eq("merchant url");
|
expect(state.merchantBaseUrl).eq("merchant url");
|
||||||
expect(state.exchangeBaseUrl).eq("exchange url");
|
expect(state.exchangeBaseUrl).eq("exchange url");
|
||||||
@ -101,7 +103,7 @@ describe("Tip CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "accepted") expect.fail();
|
if (state.status !== "accepted") expect.fail();
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:1"));
|
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:1"));
|
||||||
expect(state.merchantBaseUrl).eq("merchant url");
|
expect(state.merchantBaseUrl).eq("merchant url");
|
||||||
expect(state.exchangeBaseUrl).eq("exchange url");
|
expect(state.exchangeBaseUrl).eq("exchange url");
|
||||||
@ -112,7 +114,7 @@ describe("Tip CTA states", () => {
|
|||||||
it("should be ignored after clicking the ignore button", async () => {
|
it("should be ignored after clicking the ignore button", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taler://tip/asd", {
|
useComponentState({ talerTipUri: "taler://tip/asd" }, {
|
||||||
prepareTip: async () =>
|
prepareTip: async () =>
|
||||||
({
|
({
|
||||||
exchangeBaseUrl: "exchange url",
|
exchangeBaseUrl: "exchange url",
|
||||||
@ -125,9 +127,9 @@ describe("Tip CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -136,7 +138,7 @@ describe("Tip CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "ready") expect.fail();
|
if (state.status !== "ready") expect.fail();
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:1"));
|
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:1"));
|
||||||
expect(state.merchantBaseUrl).eq("merchant url");
|
expect(state.merchantBaseUrl).eq("merchant url");
|
||||||
expect(state.exchangeBaseUrl).eq("exchange url");
|
expect(state.exchangeBaseUrl).eq("exchange url");
|
||||||
@ -150,7 +152,7 @@ describe("Tip CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "ignored") expect.fail();
|
if (state.status !== "ignored") expect.fail();
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
}
|
}
|
||||||
await assertNoPendingUpdate();
|
await assertNoPendingUpdate();
|
||||||
});
|
});
|
||||||
@ -158,7 +160,7 @@ describe("Tip CTA states", () => {
|
|||||||
it("should render accepted if the tip has been used previously", async () => {
|
it("should render accepted if the tip has been used previously", async () => {
|
||||||
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
const { getLastResultOrThrow, waitNextUpdate, assertNoPendingUpdate } =
|
||||||
mountHook(() =>
|
mountHook(() =>
|
||||||
useComponentState("taler://tip/asd", {
|
useComponentState({ talerTipUri: "taler://tip/asd" }, {
|
||||||
prepareTip: async () =>
|
prepareTip: async () =>
|
||||||
({
|
({
|
||||||
accepted: true,
|
accepted: true,
|
||||||
@ -172,9 +174,9 @@ describe("Tip CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -183,7 +185,7 @@ describe("Tip CTA states", () => {
|
|||||||
const state = getLastResultOrThrow();
|
const state = getLastResultOrThrow();
|
||||||
|
|
||||||
if (state.status !== "accepted") expect.fail();
|
if (state.status !== "accepted") expect.fail();
|
||||||
if (state.hook) expect.fail();
|
if (state.error) expect.fail();
|
||||||
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:1"));
|
expect(state.amount).deep.eq(Amounts.parseOrThrow("EUR:1"));
|
||||||
expect(state.merchantBaseUrl).eq("merchant url");
|
expect(state.merchantBaseUrl).eq("merchant url");
|
||||||
expect(state.exchangeBaseUrl).eq("exchange url");
|
expect(state.exchangeBaseUrl).eq("exchange url");
|
118
packages/taler-wallet-webextension/src/cta/Tip/views.tsx
Normal file
118
packages/taler-wallet-webextension/src/cta/Tip/views.tsx
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
/*
|
||||||
|
This file is part of GNU Taler
|
||||||
|
(C) 2022 Taler Systems S.A.
|
||||||
|
|
||||||
|
GNU 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.
|
||||||
|
|
||||||
|
GNU 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
|
||||||
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Fragment, h, VNode } from "preact";
|
||||||
|
import { Amount } from "../../components/Amount.js";
|
||||||
|
import { LoadingError } from "../../components/LoadingError.js";
|
||||||
|
import { LogoHeader } from "../../components/LogoHeader.js";
|
||||||
|
import { Part } from "../../components/Part.js";
|
||||||
|
import { SubTitle, WalletAction } from "../../components/styled/index.js";
|
||||||
|
import { useTranslationContext } from "../../context/translation.js";
|
||||||
|
import { Button } from "../../mui/Button.js";
|
||||||
|
import { State } from "./index.js";
|
||||||
|
|
||||||
|
export function LoadingUriView({ error }: State.LoadingUriError): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<LoadingError
|
||||||
|
title={<i18n.Translate>Could not load tip status</i18n.Translate>}
|
||||||
|
error={error}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function IgnoredView(state: State.Ignored): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
return (
|
||||||
|
<WalletAction>
|
||||||
|
<LogoHeader />
|
||||||
|
|
||||||
|
<SubTitle>
|
||||||
|
<i18n.Translate>Digital cash tip</i18n.Translate>
|
||||||
|
</SubTitle>
|
||||||
|
<span>
|
||||||
|
<i18n.Translate>You've ignored the tip.</i18n.Translate>
|
||||||
|
</span>
|
||||||
|
</WalletAction>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ReadyView(state: State.Ready): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
return (
|
||||||
|
<WalletAction>
|
||||||
|
<LogoHeader />
|
||||||
|
|
||||||
|
<SubTitle>
|
||||||
|
<i18n.Translate>Digital cash tip</i18n.Translate>
|
||||||
|
</SubTitle>
|
||||||
|
|
||||||
|
<section>
|
||||||
|
<p>
|
||||||
|
<i18n.Translate>The merchant is offering you a tip</i18n.Translate>
|
||||||
|
</p>
|
||||||
|
<Part
|
||||||
|
title={<i18n.Translate>Amount</i18n.Translate>}
|
||||||
|
text={<Amount value={state.amount} />}
|
||||||
|
kind="positive"
|
||||||
|
big
|
||||||
|
/>
|
||||||
|
<Part
|
||||||
|
title={<i18n.Translate>Merchant URL</i18n.Translate>}
|
||||||
|
text={state.merchantBaseUrl}
|
||||||
|
kind="neutral"
|
||||||
|
/>
|
||||||
|
<Part
|
||||||
|
title={<i18n.Translate>Exchange</i18n.Translate>}
|
||||||
|
text={state.exchangeBaseUrl}
|
||||||
|
kind="neutral"
|
||||||
|
/>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<Button
|
||||||
|
variant="contained"
|
||||||
|
color="success"
|
||||||
|
onClick={state.accept.onClick}
|
||||||
|
>
|
||||||
|
<i18n.Translate>Accept tip</i18n.Translate>
|
||||||
|
</Button>
|
||||||
|
<Button onClick={state.ignore.onClick}>
|
||||||
|
<i18n.Translate>Ignore</i18n.Translate>
|
||||||
|
</Button>
|
||||||
|
</section>
|
||||||
|
</WalletAction>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function AcceptedView(state: State.Accepted): VNode {
|
||||||
|
const { i18n } = useTranslationContext();
|
||||||
|
return (
|
||||||
|
<WalletAction>
|
||||||
|
<LogoHeader />
|
||||||
|
|
||||||
|
<SubTitle>
|
||||||
|
<i18n.Translate>Digital cash tip</i18n.Translate>
|
||||||
|
</SubTitle>
|
||||||
|
<section>
|
||||||
|
<i18n.Translate>
|
||||||
|
Tip from <code>{state.merchantBaseUrl}</code> accepted. Check your
|
||||||
|
transactions list for more details.
|
||||||
|
</i18n.Translate>
|
||||||
|
</section>
|
||||||
|
</WalletAction>
|
||||||
|
);
|
||||||
|
}
|
@ -1,570 +0,0 @@
|
|||||||
/*
|
|
||||||
This file is part of GNU Taler
|
|
||||||
(C) 2022 Taler Systems S.A.
|
|
||||||
|
|
||||||
GNU 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.
|
|
||||||
|
|
||||||
GNU 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
|
|
||||||
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Page shown to the user to confirm creation
|
|
||||||
* of a reserve, usually requested by the bank.
|
|
||||||
*
|
|
||||||
* @author sebasjm
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { AmountJson, Amounts } from "@gnu-taler/taler-util";
|
|
||||||
import { TalerError } from "@gnu-taler/taler-wallet-core";
|
|
||||||
import { Fragment, h, VNode } from "preact";
|
|
||||||
import { useMemo, useState } from "preact/hooks";
|
|
||||||
import { Amount } from "../components/Amount.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 { Part } from "../components/Part.js";
|
|
||||||
import { SelectList } from "../components/SelectList.js";
|
|
||||||
import {
|
|
||||||
Input,
|
|
||||||
LinkSuccess,
|
|
||||||
SubTitle,
|
|
||||||
SuccessBox,
|
|
||||||
WalletAction,
|
|
||||||
} from "../components/styled/index.js";
|
|
||||||
import { useTranslationContext } from "../context/translation.js";
|
|
||||||
import { HookError, useAsyncAsHook } from "../hooks/useAsyncAsHook.js";
|
|
||||||
import { Button } from "../mui/Button.js";
|
|
||||||
import { ButtonHandler, SelectFieldHandler } from "../mui/handlers.js";
|
|
||||||
import { buildTermsOfServiceState } from "../utils/index.js";
|
|
||||||
import * as wxApi from "../wxApi.js";
|
|
||||||
import {
|
|
||||||
Props as TermsOfServiceSectionProps,
|
|
||||||
TermsOfServiceSection,
|
|
||||||
} from "./TermsOfServiceSection.js";
|
|
||||||
|
|
||||||
interface Props {
|
|
||||||
talerWithdrawUri?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
type State =
|
|
||||||
| LoadingUri
|
|
||||||
| LoadingExchange
|
|
||||||
| LoadingInfoError
|
|
||||||
| Success
|
|
||||||
| Completed;
|
|
||||||
|
|
||||||
interface LoadingUri {
|
|
||||||
status: "loading-uri";
|
|
||||||
hook: HookError | undefined;
|
|
||||||
}
|
|
||||||
interface LoadingExchange {
|
|
||||||
status: "loading-exchange";
|
|
||||||
hook: HookError | undefined;
|
|
||||||
}
|
|
||||||
interface LoadingInfoError {
|
|
||||||
status: "loading-info";
|
|
||||||
hook: HookError | undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
type Completed = {
|
|
||||||
status: "completed";
|
|
||||||
hook: undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
type Success = {
|
|
||||||
status: "success";
|
|
||||||
hook: undefined;
|
|
||||||
|
|
||||||
exchange: SelectFieldHandler;
|
|
||||||
|
|
||||||
editExchange: ButtonHandler;
|
|
||||||
cancelEditExchange: ButtonHandler;
|
|
||||||
confirmEditExchange: ButtonHandler;
|
|
||||||
|
|
||||||
showExchangeSelection: boolean;
|
|
||||||
chosenAmount: AmountJson;
|
|
||||||
withdrawalFee: AmountJson;
|
|
||||||
toBeReceived: AmountJson;
|
|
||||||
|
|
||||||
doWithdrawal: ButtonHandler;
|
|
||||||
tosProps?: TermsOfServiceSectionProps;
|
|
||||||
mustAcceptFirst: boolean;
|
|
||||||
|
|
||||||
ageRestriction: SelectFieldHandler;
|
|
||||||
};
|
|
||||||
|
|
||||||
export function useComponentState(
|
|
||||||
talerWithdrawUri: string | undefined,
|
|
||||||
api: typeof wxApi,
|
|
||||||
): State {
|
|
||||||
const [customExchange, setCustomExchange] = useState<string | undefined>(
|
|
||||||
undefined,
|
|
||||||
);
|
|
||||||
const [ageRestricted, setAgeRestricted] = useState(0);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Ask the wallet about the withdraw URI
|
|
||||||
*/
|
|
||||||
const uriInfoHook = useAsyncAsHook(async () => {
|
|
||||||
if (!talerWithdrawUri) throw Error("ERROR_NO-URI-FOR-WITHDRAWAL");
|
|
||||||
|
|
||||||
const uriInfo = await api.getWithdrawalDetailsForUri({
|
|
||||||
talerWithdrawUri,
|
|
||||||
});
|
|
||||||
const { exchanges: knownExchanges } = await api.listExchanges();
|
|
||||||
|
|
||||||
return { uriInfo, knownExchanges };
|
|
||||||
});
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the amount and select one exchange
|
|
||||||
*/
|
|
||||||
const uriHookDep =
|
|
||||||
!uriInfoHook || uriInfoHook.hasError || !uriInfoHook.response
|
|
||||||
? undefined
|
|
||||||
: uriInfoHook.response;
|
|
||||||
|
|
||||||
const { amount, thisExchange, thisCurrencyExchanges } = useMemo(() => {
|
|
||||||
if (!uriHookDep)
|
|
||||||
return {
|
|
||||||
amount: undefined,
|
|
||||||
thisExchange: undefined,
|
|
||||||
thisCurrencyExchanges: [],
|
|
||||||
};
|
|
||||||
|
|
||||||
const { uriInfo, knownExchanges } = uriHookDep;
|
|
||||||
|
|
||||||
const amount = uriInfo ? Amounts.parseOrThrow(uriInfo.amount) : undefined;
|
|
||||||
const thisCurrencyExchanges =
|
|
||||||
!amount || !knownExchanges
|
|
||||||
? []
|
|
||||||
: knownExchanges.filter((ex) => ex.currency === amount.currency);
|
|
||||||
|
|
||||||
const thisExchange: string | undefined =
|
|
||||||
customExchange ??
|
|
||||||
uriInfo?.defaultExchangeBaseUrl ??
|
|
||||||
(thisCurrencyExchanges && thisCurrencyExchanges[0]
|
|
||||||
? thisCurrencyExchanges[0].exchangeBaseUrl
|
|
||||||
: undefined);
|
|
||||||
|
|
||||||
return { amount, thisExchange, thisCurrencyExchanges };
|
|
||||||
}, [uriHookDep, customExchange]);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* For the exchange selected, bring the status of the terms of service
|
|
||||||
*/
|
|
||||||
const terms = useAsyncAsHook(async () => {
|
|
||||||
if (!thisExchange) return false;
|
|
||||||
|
|
||||||
const exchangeTos = await api.getExchangeTos(thisExchange, ["text/xml"]);
|
|
||||||
|
|
||||||
const state = buildTermsOfServiceState(exchangeTos);
|
|
||||||
|
|
||||||
return { state };
|
|
||||||
}, [thisExchange]);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* With the exchange and amount, ask the wallet the information
|
|
||||||
* about the withdrawal
|
|
||||||
*/
|
|
||||||
const info = useAsyncAsHook(async () => {
|
|
||||||
if (!thisExchange || !amount) return false;
|
|
||||||
|
|
||||||
const info = await api.getExchangeWithdrawalInfo({
|
|
||||||
exchangeBaseUrl: thisExchange,
|
|
||||||
amount,
|
|
||||||
tosAcceptedFormat: ["text/xml"],
|
|
||||||
});
|
|
||||||
|
|
||||||
const withdrawalFee = Amounts.sub(
|
|
||||||
Amounts.parseOrThrow(info.withdrawalAmountRaw),
|
|
||||||
Amounts.parseOrThrow(info.withdrawalAmountEffective),
|
|
||||||
).amount;
|
|
||||||
|
|
||||||
return { info, withdrawalFee };
|
|
||||||
}, [thisExchange, amount]);
|
|
||||||
|
|
||||||
const [reviewing, setReviewing] = useState<boolean>(false);
|
|
||||||
const [reviewed, setReviewed] = useState<boolean>(false);
|
|
||||||
|
|
||||||
const [withdrawError, setWithdrawError] = useState<TalerError | undefined>(
|
|
||||||
undefined,
|
|
||||||
);
|
|
||||||
const [doingWithdraw, setDoingWithdraw] = useState<boolean>(false);
|
|
||||||
const [withdrawCompleted, setWithdrawCompleted] = useState<boolean>(false);
|
|
||||||
|
|
||||||
const [showExchangeSelection, setShowExchangeSelection] = useState(false);
|
|
||||||
const [nextExchange, setNextExchange] = useState<string | undefined>();
|
|
||||||
|
|
||||||
if (!uriInfoHook || uriInfoHook.hasError) {
|
|
||||||
return {
|
|
||||||
status: "loading-uri",
|
|
||||||
hook: uriInfoHook,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!thisExchange || !amount) {
|
|
||||||
return {
|
|
||||||
status: "loading-exchange",
|
|
||||||
hook: {
|
|
||||||
hasError: true,
|
|
||||||
operational: false,
|
|
||||||
message: "ERROR_NO-DEFAULT-EXCHANGE",
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const selectedExchange = thisExchange;
|
|
||||||
|
|
||||||
async function doWithdrawAndCheckError(): Promise<void> {
|
|
||||||
try {
|
|
||||||
setDoingWithdraw(true);
|
|
||||||
if (!talerWithdrawUri) return;
|
|
||||||
const res = await api.acceptWithdrawal(
|
|
||||||
talerWithdrawUri,
|
|
||||||
selectedExchange,
|
|
||||||
!ageRestricted ? undefined : ageRestricted,
|
|
||||||
);
|
|
||||||
if (res.confirmTransferUrl) {
|
|
||||||
document.location.href = res.confirmTransferUrl;
|
|
||||||
}
|
|
||||||
setWithdrawCompleted(true);
|
|
||||||
} catch (e) {
|
|
||||||
if (e instanceof TalerError) {
|
|
||||||
setWithdrawError(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
setDoingWithdraw(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
const exchanges = thisCurrencyExchanges.reduce(
|
|
||||||
(prev, ex) => ({ ...prev, [ex.exchangeBaseUrl]: ex.exchangeBaseUrl }),
|
|
||||||
{},
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!info || info.hasError) {
|
|
||||||
return {
|
|
||||||
status: "loading-info",
|
|
||||||
hook: info,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
if (!info.response) {
|
|
||||||
return {
|
|
||||||
status: "loading-info",
|
|
||||||
hook: undefined,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
if (withdrawCompleted) {
|
|
||||||
return {
|
|
||||||
status: "completed",
|
|
||||||
hook: undefined,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const exchangeHandler: SelectFieldHandler = {
|
|
||||||
onChange: async (e) => setNextExchange(e),
|
|
||||||
value: nextExchange ?? thisExchange,
|
|
||||||
list: exchanges,
|
|
||||||
isDirty: nextExchange !== undefined,
|
|
||||||
};
|
|
||||||
|
|
||||||
const editExchange: ButtonHandler = {
|
|
||||||
onClick: async () => {
|
|
||||||
setShowExchangeSelection(true);
|
|
||||||
},
|
|
||||||
};
|
|
||||||
const cancelEditExchange: ButtonHandler = {
|
|
||||||
onClick: async () => {
|
|
||||||
setShowExchangeSelection(false);
|
|
||||||
},
|
|
||||||
};
|
|
||||||
const confirmEditExchange: ButtonHandler = {
|
|
||||||
onClick: async () => {
|
|
||||||
setCustomExchange(exchangeHandler.value);
|
|
||||||
setShowExchangeSelection(false);
|
|
||||||
setNextExchange(undefined);
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const { withdrawalFee } = info.response;
|
|
||||||
const toBeReceived = Amounts.sub(amount, withdrawalFee).amount;
|
|
||||||
|
|
||||||
const { state: termsState } = (!terms
|
|
||||||
? undefined
|
|
||||||
: terms.hasError
|
|
||||||
? undefined
|
|
||||||
: terms.response) || { state: undefined };
|
|
||||||
|
|
||||||
async function onAccept(accepted: boolean): Promise<void> {
|
|
||||||
if (!termsState) return;
|
|
||||||
|
|
||||||
try {
|
|
||||||
await api.setExchangeTosAccepted(
|
|
||||||
selectedExchange,
|
|
||||||
accepted ? termsState.version : undefined,
|
|
||||||
);
|
|
||||||
setReviewed(accepted);
|
|
||||||
} catch (e) {
|
|
||||||
if (e instanceof Error) {
|
|
||||||
//FIXME: uncomment this and display error
|
|
||||||
// setErrorAccepting(e.message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const mustAcceptFirst =
|
|
||||||
termsState !== undefined &&
|
|
||||||
(termsState.status === "changed" || termsState.status === "new");
|
|
||||||
|
|
||||||
const ageRestrictionOptions: Record<string, string> | undefined = "6:12:18"
|
|
||||||
.split(":")
|
|
||||||
.reduce((p, c) => ({ ...p, [c]: `under ${c}` }), {});
|
|
||||||
|
|
||||||
if (ageRestrictionOptions) {
|
|
||||||
ageRestrictionOptions["0"] = "Not restricted";
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
status: "success",
|
|
||||||
hook: undefined,
|
|
||||||
exchange: exchangeHandler,
|
|
||||||
editExchange,
|
|
||||||
cancelEditExchange,
|
|
||||||
confirmEditExchange,
|
|
||||||
showExchangeSelection,
|
|
||||||
toBeReceived,
|
|
||||||
withdrawalFee,
|
|
||||||
chosenAmount: amount,
|
|
||||||
ageRestriction: {
|
|
||||||
list: ageRestrictionOptions,
|
|
||||||
value: String(ageRestricted),
|
|
||||||
onChange: async (v) => setAgeRestricted(parseInt(v, 10)),
|
|
||||||
},
|
|
||||||
doWithdrawal: {
|
|
||||||
onClick:
|
|
||||||
doingWithdraw || (mustAcceptFirst && !reviewed)
|
|
||||||
? undefined
|
|
||||||
: doWithdrawAndCheckError,
|
|
||||||
error: withdrawError,
|
|
||||||
},
|
|
||||||
tosProps: !termsState
|
|
||||||
? undefined
|
|
||||||
: {
|
|
||||||
onAccept,
|
|
||||||
onReview: setReviewing,
|
|
||||||
reviewed: reviewed,
|
|
||||||
reviewing: reviewing,
|
|
||||||
terms: termsState,
|
|
||||||
},
|
|
||||||
mustAcceptFirst,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function View({ state }: { state: State }): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
|
||||||
if (state.status === "loading-uri") {
|
|
||||||
if (!state.hook) return <Loading />;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<LoadingError
|
|
||||||
title={
|
|
||||||
<i18n.Translate>Could not get the info from the URI</i18n.Translate>
|
|
||||||
}
|
|
||||||
error={state.hook}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (state.status === "loading-exchange") {
|
|
||||||
if (!state.hook) return <Loading />;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<LoadingError
|
|
||||||
title={<i18n.Translate>Could not get exchange</i18n.Translate>}
|
|
||||||
error={state.hook}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (state.status === "loading-info") {
|
|
||||||
if (!state.hook) return <Loading />;
|
|
||||||
return (
|
|
||||||
<LoadingError
|
|
||||||
title={
|
|
||||||
<i18n.Translate>Could not get info of withdrawal</i18n.Translate>
|
|
||||||
}
|
|
||||||
error={state.hook}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state.status === "completed") {
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash withdrawal</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
<SuccessBox>
|
|
||||||
<h3>
|
|
||||||
<i18n.Translate>Withdrawal in process...</i18n.Translate>
|
|
||||||
</h3>
|
|
||||||
<p>
|
|
||||||
<i18n.Translate>
|
|
||||||
You can close the page now. Check your bank if the transaction
|
|
||||||
need a confirmation step to be completed
|
|
||||||
</i18n.Translate>
|
|
||||||
</p>
|
|
||||||
</SuccessBox>
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<WalletAction>
|
|
||||||
<LogoHeader />
|
|
||||||
<SubTitle>
|
|
||||||
<i18n.Translate>Digital cash withdrawal</i18n.Translate>
|
|
||||||
</SubTitle>
|
|
||||||
|
|
||||||
{state.doWithdrawal.error && (
|
|
||||||
<ErrorTalerOperation
|
|
||||||
title={
|
|
||||||
<i18n.Translate>
|
|
||||||
Could not finish the withdrawal operation
|
|
||||||
</i18n.Translate>
|
|
||||||
}
|
|
||||||
error={state.doWithdrawal.error.errorDetail}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<section>
|
|
||||||
<Part
|
|
||||||
title={<i18n.Translate>Total to withdraw</i18n.Translate>}
|
|
||||||
text={<Amount value={state.toBeReceived} />}
|
|
||||||
kind="positive"
|
|
||||||
/>
|
|
||||||
{Amounts.isNonZero(state.withdrawalFee) && (
|
|
||||||
<Fragment>
|
|
||||||
<Part
|
|
||||||
title={<i18n.Translate>Chosen amount</i18n.Translate>}
|
|
||||||
text={<Amount value={state.chosenAmount} />}
|
|
||||||
kind="neutral"
|
|
||||||
/>
|
|
||||||
<Part
|
|
||||||
title={<i18n.Translate>Exchange fee</i18n.Translate>}
|
|
||||||
text={<Amount value={state.withdrawalFee} />}
|
|
||||||
kind="negative"
|
|
||||||
/>
|
|
||||||
</Fragment>
|
|
||||||
)}
|
|
||||||
<Part
|
|
||||||
title={<i18n.Translate>Exchange</i18n.Translate>}
|
|
||||||
text={state.exchange.value}
|
|
||||||
kind="neutral"
|
|
||||||
big
|
|
||||||
/>
|
|
||||||
{state.showExchangeSelection ? (
|
|
||||||
<Fragment>
|
|
||||||
<div>
|
|
||||||
<SelectList
|
|
||||||
label={<i18n.Translate>Known exchanges</i18n.Translate>}
|
|
||||||
list={state.exchange.list}
|
|
||||||
value={state.exchange.value}
|
|
||||||
name="switchingExchange"
|
|
||||||
onChange={state.exchange.onChange}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<LinkSuccess
|
|
||||||
upperCased
|
|
||||||
style={{ fontSize: "small" }}
|
|
||||||
onClick={state.confirmEditExchange.onClick}
|
|
||||||
>
|
|
||||||
{state.exchange.isDirty ? (
|
|
||||||
<i18n.Translate>Confirm exchange selection</i18n.Translate>
|
|
||||||
) : (
|
|
||||||
<i18n.Translate>Cancel exchange selection</i18n.Translate>
|
|
||||||
)}
|
|
||||||
</LinkSuccess>
|
|
||||||
</Fragment>
|
|
||||||
) : (
|
|
||||||
<LinkSuccess
|
|
||||||
style={{ fontSize: "small" }}
|
|
||||||
upperCased
|
|
||||||
onClick={state.editExchange.onClick}
|
|
||||||
>
|
|
||||||
<i18n.Translate>Edit exchange</i18n.Translate>
|
|
||||||
</LinkSuccess>
|
|
||||||
)}
|
|
||||||
</section>
|
|
||||||
<section>
|
|
||||||
<Input>
|
|
||||||
<SelectList
|
|
||||||
label={<i18n.Translate>Age restriction</i18n.Translate>}
|
|
||||||
list={state.ageRestriction.list}
|
|
||||||
name="age"
|
|
||||||
maxWidth
|
|
||||||
value={state.ageRestriction.value}
|
|
||||||
onChange={state.ageRestriction.onChange}
|
|
||||||
/>
|
|
||||||
</Input>
|
|
||||||
</section>
|
|
||||||
{state.tosProps && <TermsOfServiceSection {...state.tosProps} />}
|
|
||||||
{state.tosProps ? (
|
|
||||||
<section>
|
|
||||||
{(state.tosProps.terms.status === "accepted" ||
|
|
||||||
(state.mustAcceptFirst && state.tosProps.reviewed)) && (
|
|
||||||
<Button
|
|
||||||
variant="contained"
|
|
||||||
color="success"
|
|
||||||
disabled={!state.doWithdrawal.onClick}
|
|
||||||
onClick={state.doWithdrawal.onClick}
|
|
||||||
>
|
|
||||||
<i18n.Translate>Confirm withdrawal</i18n.Translate>
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
{state.tosProps.terms.status === "notfound" && (
|
|
||||||
<Button
|
|
||||||
variant="contained"
|
|
||||||
color="warning"
|
|
||||||
disabled={!state.doWithdrawal.onClick}
|
|
||||||
onClick={state.doWithdrawal.onClick}
|
|
||||||
>
|
|
||||||
<i18n.Translate>Withdraw anyway</i18n.Translate>
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</section>
|
|
||||||
) : (
|
|
||||||
<section>
|
|
||||||
<i18n.Translate>Loading terms of service...</i18n.Translate>
|
|
||||||
</section>
|
|
||||||
)}
|
|
||||||
</WalletAction>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function WithdrawPage({ talerWithdrawUri }: Props): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
|
||||||
|
|
||||||
const state = useComponentState(talerWithdrawUri, wxApi);
|
|
||||||
|
|
||||||
if (!talerWithdrawUri) {
|
|
||||||
return (
|
|
||||||
<span>
|
|
||||||
<i18n.Translate>missing withdraw uri</i18n.Translate>
|
|
||||||
</span>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!state) {
|
|
||||||
return <Loading />;
|
|
||||||
}
|
|
||||||
|
|
||||||
return <View state={state} />;
|
|
||||||
}
|
|
@ -15,56 +15,57 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { AmountJson } from "@gnu-taler/taler-util";
|
import { AmountJson } from "@gnu-taler/taler-util";
|
||||||
import { compose, StateViewMap } from "../../utils/index.js";
|
import { Loading } from "../../components/Loading.js";
|
||||||
import { HookError } from "../../hooks/useAsyncAsHook.js";
|
import { HookError } from "../../hooks/useAsyncAsHook.js";
|
||||||
import { ButtonHandler, SelectFieldHandler } from "../../mui/handlers.js";
|
import { ButtonHandler, SelectFieldHandler } from "../../mui/handlers.js";
|
||||||
|
import { compose, StateViewMap } from "../../utils/index.js";
|
||||||
|
import * as wxApi from "../../wxApi.js";
|
||||||
import {
|
import {
|
||||||
Props as TermsOfServiceSectionProps
|
Props as TermsOfServiceSectionProps
|
||||||
} from "../TermsOfServiceSection.js";
|
} from "../TermsOfServiceSection.js";
|
||||||
import { CompletedView, LoadingExchangeView, LoadingInfoView, LoadingUriView, SuccessView } from "./views.js";
|
|
||||||
import { useComponentState } from "./state.js";
|
import { useComponentState } from "./state.js";
|
||||||
|
import { CompletedView, LoadingExchangeView, LoadingInfoView, LoadingUriView, SuccessView } from "./views.js";
|
||||||
|
|
||||||
/**
|
|
||||||
* Page shown to the user to confirm creation
|
|
||||||
* of a reserve, usually requested by the bank.
|
|
||||||
*
|
|
||||||
* @author sebasjm
|
|
||||||
*/
|
|
||||||
|
|
||||||
export interface Props {
|
export interface Props {
|
||||||
talerWithdrawUri: string | undefined;
|
talerWithdrawUri: string | undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type State =
|
export type State =
|
||||||
| State.LoadingUri
|
| State.Loading
|
||||||
| State.LoadingExchange
|
| State.LoadingUriError
|
||||||
|
| State.LoadingExchangeError
|
||||||
| State.LoadingInfoError
|
| State.LoadingInfoError
|
||||||
| State.Success
|
| State.Success
|
||||||
| State.Completed;
|
| State.Completed;
|
||||||
|
|
||||||
export namespace State {
|
export namespace State {
|
||||||
|
|
||||||
export interface LoadingUri {
|
export interface Loading {
|
||||||
status: "loading-uri";
|
status: "loading";
|
||||||
hook: HookError | undefined;
|
error: undefined;
|
||||||
}
|
}
|
||||||
export interface LoadingExchange {
|
export interface LoadingUriError {
|
||||||
|
status: "loading-uri";
|
||||||
|
error: HookError;
|
||||||
|
}
|
||||||
|
export interface LoadingExchangeError {
|
||||||
status: "loading-exchange";
|
status: "loading-exchange";
|
||||||
hook: HookError | undefined;
|
error: HookError;
|
||||||
}
|
}
|
||||||
export interface LoadingInfoError {
|
export interface LoadingInfoError {
|
||||||
status: "loading-info";
|
status: "loading-info";
|
||||||
hook: HookError | undefined;
|
error: HookError;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Completed = {
|
export type Completed = {
|
||||||
status: "completed";
|
status: "completed";
|
||||||
hook: undefined;
|
error: undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type Success = {
|
export type Success = {
|
||||||
status: "success";
|
status: "success";
|
||||||
hook: undefined;
|
error: undefined;
|
||||||
|
|
||||||
exchange: SelectFieldHandler;
|
exchange: SelectFieldHandler;
|
||||||
|
|
||||||
@ -86,6 +87,7 @@ export namespace State {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const viewMapping: StateViewMap<State> = {
|
const viewMapping: StateViewMap<State> = {
|
||||||
|
loading: Loading,
|
||||||
"loading-uri": LoadingUriView,
|
"loading-uri": LoadingUriView,
|
||||||
"loading-exchange": LoadingExchangeView,
|
"loading-exchange": LoadingExchangeView,
|
||||||
"loading-info": LoadingInfoView,
|
"loading-info": LoadingInfoView,
|
||||||
@ -93,6 +95,4 @@ const viewMapping: StateViewMap<State> = {
|
|||||||
success: SuccessView,
|
success: SuccessView,
|
||||||
};
|
};
|
||||||
|
|
||||||
import * as wxApi from "../../wxApi.js";
|
|
||||||
|
|
||||||
export const WithdrawPage = compose("Withdraw", (p: Props) => useComponentState(p, wxApi), viewMapping)
|
export const WithdrawPage = compose("Withdraw", (p: Props) => useComponentState(p, wxApi), viewMapping)
|
||||||
|
@ -14,12 +14,6 @@
|
|||||||
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
|
||||||
* Page shown to the user to confirm creation
|
|
||||||
* of a reserve, usually requested by the bank.
|
|
||||||
*
|
|
||||||
* @author sebasjm
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { Amounts } from "@gnu-taler/taler-util";
|
import { Amounts } from "@gnu-taler/taler-util";
|
||||||
import { TalerError } from "@gnu-taler/taler-wallet-core";
|
import { TalerError } from "@gnu-taler/taler-wallet-core";
|
||||||
@ -133,17 +127,18 @@ export function useComponentState(
|
|||||||
const [showExchangeSelection, setShowExchangeSelection] = useState(false);
|
const [showExchangeSelection, setShowExchangeSelection] = useState(false);
|
||||||
const [nextExchange, setNextExchange] = useState<string | undefined>();
|
const [nextExchange, setNextExchange] = useState<string | undefined>();
|
||||||
|
|
||||||
if (!uriInfoHook || uriInfoHook.hasError) {
|
if (!uriInfoHook) return { status: "loading", error: undefined }
|
||||||
|
if (uriInfoHook.hasError) {
|
||||||
return {
|
return {
|
||||||
status: "loading-uri",
|
status: "loading-uri",
|
||||||
hook: uriInfoHook,
|
error: uriInfoHook,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!thisExchange || !amount) {
|
if (!thisExchange || !amount) {
|
||||||
return {
|
return {
|
||||||
status: "loading-exchange",
|
status: "loading-exchange",
|
||||||
hook: {
|
error: {
|
||||||
hasError: true,
|
hasError: true,
|
||||||
operational: false,
|
operational: false,
|
||||||
message: "ERROR_NO-DEFAULT-EXCHANGE",
|
message: "ERROR_NO-DEFAULT-EXCHANGE",
|
||||||
@ -179,23 +174,20 @@ export function useComponentState(
|
|||||||
{},
|
{},
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!info || info.hasError) {
|
if (!info) {
|
||||||
|
return { status: "loading", error: undefined }
|
||||||
|
}
|
||||||
|
if (info.hasError) {
|
||||||
return {
|
return {
|
||||||
status: "loading-info",
|
status: "loading-info",
|
||||||
hook: info,
|
error: info,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
if (!info.response) {
|
if (!info.response) {
|
||||||
return {
|
return { status: "loading", error: undefined };
|
||||||
status: "loading-info",
|
|
||||||
hook: undefined,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
if (withdrawCompleted) {
|
if (withdrawCompleted) {
|
||||||
return {
|
return { status: "completed", error: undefined };
|
||||||
status: "completed",
|
|
||||||
hook: undefined,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const exchangeHandler: SelectFieldHandler = {
|
const exchangeHandler: SelectFieldHandler = {
|
||||||
@ -263,7 +255,7 @@ export function useComponentState(
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
status: "success",
|
status: "success",
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
exchange: exchangeHandler,
|
exchange: exchangeHandler,
|
||||||
editExchange,
|
editExchange,
|
||||||
cancelEditExchange,
|
cancelEditExchange,
|
||||||
|
@ -61,7 +61,7 @@ const ageRestrictionSelectField = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const TermsOfServiceNotYetLoaded = createExample(SuccessView, {
|
export const TermsOfServiceNotYetLoaded = createExample(SuccessView, {
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
status: "success",
|
status: "success",
|
||||||
cancelEditExchange: nullHandler,
|
cancelEditExchange: nullHandler,
|
||||||
confirmEditExchange: nullHandler,
|
confirmEditExchange: nullHandler,
|
||||||
@ -95,7 +95,7 @@ export const TermsOfServiceNotYetLoaded = createExample(SuccessView, {
|
|||||||
});
|
});
|
||||||
|
|
||||||
export const WithSomeFee = createExample(SuccessView, {
|
export const WithSomeFee = createExample(SuccessView, {
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
status: "success",
|
status: "success",
|
||||||
cancelEditExchange: nullHandler,
|
cancelEditExchange: nullHandler,
|
||||||
confirmEditExchange: nullHandler,
|
confirmEditExchange: nullHandler,
|
||||||
@ -130,7 +130,7 @@ export const WithSomeFee = createExample(SuccessView, {
|
|||||||
});
|
});
|
||||||
|
|
||||||
export const WithoutFee = createExample(SuccessView, {
|
export const WithoutFee = createExample(SuccessView, {
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
status: "success",
|
status: "success",
|
||||||
cancelEditExchange: nullHandler,
|
cancelEditExchange: nullHandler,
|
||||||
confirmEditExchange: nullHandler,
|
confirmEditExchange: nullHandler,
|
||||||
@ -165,7 +165,7 @@ export const WithoutFee = createExample(SuccessView, {
|
|||||||
});
|
});
|
||||||
|
|
||||||
export const EditExchangeUntouched = createExample(SuccessView, {
|
export const EditExchangeUntouched = createExample(SuccessView, {
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
status: "success",
|
status: "success",
|
||||||
cancelEditExchange: nullHandler,
|
cancelEditExchange: nullHandler,
|
||||||
confirmEditExchange: nullHandler,
|
confirmEditExchange: nullHandler,
|
||||||
@ -200,7 +200,7 @@ export const EditExchangeUntouched = createExample(SuccessView, {
|
|||||||
});
|
});
|
||||||
|
|
||||||
export const EditExchangeModified = createExample(SuccessView, {
|
export const EditExchangeModified = createExample(SuccessView, {
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
status: "success",
|
status: "success",
|
||||||
cancelEditExchange: nullHandler,
|
cancelEditExchange: nullHandler,
|
||||||
confirmEditExchange: nullHandler,
|
confirmEditExchange: nullHandler,
|
||||||
@ -237,11 +237,11 @@ export const EditExchangeModified = createExample(SuccessView, {
|
|||||||
|
|
||||||
export const CompletedWithoutBankURL = createExample(CompletedView, {
|
export const CompletedWithoutBankURL = createExample(CompletedView, {
|
||||||
status: "completed",
|
status: "completed",
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
});
|
});
|
||||||
|
|
||||||
export const WithAgeRestrictionSelected = createExample(SuccessView, {
|
export const WithAgeRestrictionSelected = createExample(SuccessView, {
|
||||||
hook: undefined,
|
error: undefined,
|
||||||
status: "success",
|
status: "success",
|
||||||
cancelEditExchange: nullHandler,
|
cancelEditExchange: nullHandler,
|
||||||
confirmEditExchange: nullHandler,
|
confirmEditExchange: nullHandler,
|
||||||
|
@ -54,21 +54,20 @@ describe("Withdraw CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading-uri");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
|
|
||||||
expect(status).equals("loading-uri");
|
if (status != "loading-uri") expect.fail();
|
||||||
if (!hook) expect.fail();
|
if (!error) expect.fail();
|
||||||
if (!hook.hasError) expect.fail();
|
if (!error.hasError) expect.fail();
|
||||||
if (hook.operational) expect.fail();
|
if (error.operational) expect.fail();
|
||||||
expect(hook.message).eq("ERROR_NO-URI-FOR-WITHDRAWAL");
|
expect(error.message).eq("ERROR_NO-URI-FOR-WITHDRAWAL");
|
||||||
}
|
}
|
||||||
|
|
||||||
await assertNoPendingUpdate();
|
await assertNoPendingUpdate();
|
||||||
@ -87,19 +86,18 @@ describe("Withdraw CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading-uri");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
|
|
||||||
expect(status).equals("loading-exchange");
|
expect(status).equals("loading-exchange");
|
||||||
|
|
||||||
expect(hook).deep.equals({
|
expect(error).deep.equals({
|
||||||
hasError: true,
|
hasError: true,
|
||||||
operational: false,
|
operational: false,
|
||||||
message: "ERROR_NO-DEFAULT-EXCHANGE",
|
message: "ERROR_NO-DEFAULT-EXCHANGE",
|
||||||
@ -134,19 +132,19 @@ describe("Withdraw CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading-uri");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
|
|
||||||
expect(status).equals("loading-info");
|
expect(status).equals("loading");
|
||||||
|
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
@ -200,19 +198,19 @@ describe("Withdraw CTA states", () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
expect(status).equals("loading-uri");
|
expect(status).equals("loading");
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
|
||||||
{
|
{
|
||||||
const { status, hook } = getLastResultOrThrow();
|
const { status, error } = getLastResultOrThrow();
|
||||||
|
|
||||||
expect(status).equals("loading-info");
|
expect(status).equals("loading");
|
||||||
|
|
||||||
expect(hook).undefined;
|
expect(error).undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
await waitNextUpdate();
|
await waitNextUpdate();
|
||||||
|
@ -35,46 +35,39 @@ import { Amounts } from "@gnu-taler/taler-util";
|
|||||||
import { TermsOfServiceSection } from "../TermsOfServiceSection.js";
|
import { TermsOfServiceSection } from "../TermsOfServiceSection.js";
|
||||||
import { Button } from "../../mui/Button.js";
|
import { Button } from "../../mui/Button.js";
|
||||||
|
|
||||||
/**
|
export function LoadingUriView({ error }: State.LoadingUriError): VNode {
|
||||||
* Page shown to the user to confirm creation
|
|
||||||
* of a reserve, usually requested by the bank.
|
|
||||||
*
|
|
||||||
* @author sebasjm
|
|
||||||
*/
|
|
||||||
|
|
||||||
export function LoadingUriView(state: State.LoadingUri): VNode {
|
|
||||||
const { i18n } = useTranslationContext();
|
const { i18n } = useTranslationContext();
|
||||||
if (!state.hook) return <Loading />;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<LoadingError
|
<LoadingError
|
||||||
title={
|
title={
|
||||||
<i18n.Translate>Could not get the info from the URI</i18n.Translate>
|
<i18n.Translate>Could not get the info from the URI</i18n.Translate>
|
||||||
}
|
}
|
||||||
error={state.hook}
|
error={error}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function LoadingExchangeView(state: State.LoadingExchange): VNode {
|
export function LoadingExchangeView({
|
||||||
|
error,
|
||||||
|
}: State.LoadingExchangeError): VNode {
|
||||||
const { i18n } = useTranslationContext();
|
const { i18n } = useTranslationContext();
|
||||||
if (!state.hook) return <Loading />;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<LoadingError
|
<LoadingError
|
||||||
title={<i18n.Translate>Could not get exchange</i18n.Translate>}
|
title={<i18n.Translate>Could not get exchange</i18n.Translate>}
|
||||||
error={state.hook}
|
error={error}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function LoadingInfoView(state: State.LoadingInfoError): VNode {
|
export function LoadingInfoView({ error }: State.LoadingInfoError): VNode {
|
||||||
const { i18n } = useTranslationContext();
|
const { i18n } = useTranslationContext();
|
||||||
if (!state.hook) return <Loading />;
|
|
||||||
return (
|
return (
|
||||||
<LoadingError
|
<LoadingError
|
||||||
title={<i18n.Translate>Could not get info of withdrawal</i18n.Translate>}
|
title={<i18n.Translate>Could not get info of withdrawal</i18n.Translate>}
|
||||||
error={state.hook}
|
error={error}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -19,10 +19,10 @@
|
|||||||
* @author Sebastian Javier Marchano (sebasjm)
|
* @author Sebastian Javier Marchano (sebasjm)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import * as a1 from "./Deposit.stories.jsx";
|
import * as a1 from "./Deposit/stories.jsx";
|
||||||
import * as a3 from "./Pay.stories.jsx";
|
import * as a3 from "./Payment/stories.jsx";
|
||||||
import * as a4 from "./Refund.stories.jsx";
|
import * as a4 from "./Refund/stories.jsx";
|
||||||
import * as a5 from "./Tip.stories.jsx";
|
import * as a5 from "./Tip/stories.jsx";
|
||||||
import * as a6 from "./Withdraw/stories.jsx";
|
import * as a6 from "./Withdraw/stories.jsx";
|
||||||
import * as a7 from "./TermsOfServiceSection.stories.js";
|
import * as a7 from "./TermsOfServiceSection.stories.js";
|
||||||
|
|
||||||
|
@ -34,11 +34,11 @@ import {
|
|||||||
TranslationProvider,
|
TranslationProvider,
|
||||||
useTranslationContext,
|
useTranslationContext,
|
||||||
} from "../context/translation.js";
|
} from "../context/translation.js";
|
||||||
import { PayPage } from "../cta/Pay.js";
|
import { PaymentPage } from "../cta/Payment/index.js";
|
||||||
import { RefundPage } from "../cta/Refund.js";
|
import { RefundPage } from "../cta/Refund/index.js";
|
||||||
import { TipPage } from "../cta/Tip.js";
|
import { TipPage } from "../cta/Tip/index.js";
|
||||||
import { WithdrawPage } from "../cta/Withdraw/index.js";
|
import { WithdrawPage } from "../cta/Withdraw/index.js";
|
||||||
import { DepositPage as DepositPageCTA } from "../cta/Deposit.js";
|
import { DepositPage as DepositPageCTA } from "../cta/Deposit/index.js";
|
||||||
import { Pages, WalletNavBar } from "../NavigationBar.js";
|
import { Pages, WalletNavBar } from "../NavigationBar.js";
|
||||||
import { DeveloperPage } from "./DeveloperPage.js";
|
import { DeveloperPage } from "./DeveloperPage.js";
|
||||||
import { BackupPage } from "./BackupPage.js";
|
import { BackupPage } from "./BackupPage.js";
|
||||||
@ -202,7 +202,7 @@ export function Application(): VNode {
|
|||||||
*/}
|
*/}
|
||||||
<Route
|
<Route
|
||||||
path={Pages.ctaPay}
|
path={Pages.ctaPay}
|
||||||
component={PayPage}
|
component={PaymentPage}
|
||||||
goToWalletManualWithdraw={(currency?: string) =>
|
goToWalletManualWithdraw={(currency?: string) =>
|
||||||
redirectTo(Pages.balanceManualWithdraw({ currency }))
|
redirectTo(Pages.balanceManualWithdraw({ currency }))
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user