/* eslint-disable @typescript-eslint/no-unused-vars */ /* eslint-disable @typescript-eslint/no-non-null-assertion */ /* This file is part of TALER (C) 2015 GNUnet e.V. TALER is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version. TALER is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with TALER; see the file COPYING. If not, see */ /** * Page shown to the user to confirm entering * a contract. */ /** * Imports. */ import { AmountJson, Amounts, ConfirmPayResult, ConfirmPayResultType, ContractTerms, NotificationType, PreparePayResult, PreparePayResultType, Product, TalerErrorCode, } from "@gnu-taler/taler-util"; import { TalerError } from "@gnu-taler/taler-wallet-core"; import { Fragment, h, VNode } from "preact"; import { useEffect, useState } from "preact/hooks"; import { Amount } from "../components/Amount.js"; import { ErrorMessage } from "../components/ErrorMessage.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 { QR } from "../components/QR.js"; import { ButtonSuccess, Link, LinkSuccess, SmallLightText, SubTitle, SuccessBox, WalletAction, WarningBox, } 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 { talerPayUri?: string; goToWalletManualWithdraw: (currency?: string) => void; goBack: () => 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( undefined, ); const [payErrMsg, setPayErrMsg] = useState(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 { 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 state = useComponentState(talerPayUri, wxApi); if (state.status === "loading") { if (!state.hook) return ; return ( Could not load pay status} error={state.hook} /> ); } return ( ); } export function View({ state, goBack, goToWalletManualWithdraw, }: { state: Ready | Confirmed; goToWalletManualWithdraw: (currency?: string) => void; goBack: () => void; }): VNode { const { i18n } = useTranslationContext(); const contractTerms: ContractTerms = state.payStatus.contractTerms; if (!contractTerms) { return ( Could not load contract terms from merchant or wallet backend. } /> ); } return ( Digital cash payment
{state.payStatus.status !== PreparePayResultType.InsufficientBalance && Amounts.isNonZero(state.totalFees) && ( Total to pay} text={} kind="negative" /> )} Purchase amount} text={} kind="neutral" /> {Amounts.isNonZero(state.totalFees) && ( Fee} text={} kind="negative" /> )} Merchant} text={contractTerms.merchant.name} kind="neutral" /> Purchase} text={contractTerms.summary} kind="neutral" /> {contractTerms.order_id && ( Receipt} text={`#${contractTerms.order_id}`} kind="neutral" /> )} {contractTerms.products && contractTerms.products.length > 0 && ( )}
Cancel
); } export function ProductList({ products }: { products: Product[] }): VNode { const { i18n } = useTranslationContext(); return ( List of products
{products.map((p, i) => { if (p.price) { const pPrice = Amounts.parseOrThrow(p.price); return (
{p.quantity ?? 1} x {p.description}{" "} {Amounts.stringify(pPrice)}
{Amounts.stringify( Amounts.mult(pPrice, p.quantity ?? 1).amount, )}
); } return (
{p.quantity ?? 1} x {p.description}
Total {` `} {p.price ? ( `${Amounts.stringifyValue( Amounts.mult( Amounts.parseOrThrow(p.price), p.quantity ?? 1, ).amount, )} ${p}` ) : ( free )}
); })}
); } function ShowImportantMessage({ state }: { state: Ready | Confirmed }): VNode { const { i18n } = useTranslationContext(); const { payStatus } = state; if (payStatus.status === PreparePayResultType.AlreadyConfirmed) { if (payStatus.paid) { if (payStatus.contractTerms.fulfillment_url) { return ( Already paid, you are going to be redirected to{" "} {payStatus.contractTerms.fulfillment_url} ); } return ( Already paid ); } return ( Already claimed ); } if (state.status == "confirmed") { const { payResult, payHandler } = state; if (payHandler.error) { return ; } if (payResult.type === ConfirmPayResultType.Done) { return (

Payment complete

{!payResult.contractTerms.fulfillment_message ? ( payResult.contractTerms.fulfillment_url ? ( You are going to be redirected to $ {payResult.contractTerms.fulfillment_url} ) : ( You can close this page. ) ) : ( payResult.contractTerms.fulfillment_message )}

); } } return ; } function PayWithMobile({ state }: { state: Ready }): VNode { const { i18n } = useTranslationContext(); const [showQR, setShowQR] = useState(false); const privateUri = state.payStatus.status !== PreparePayResultType.AlreadyConfirmed ? `${state.uri}&n=${state.payStatus.noncePriv}` : state.uri; return (
setShowQR((qr) => !qr)}> {!showQR ? ( Pay with a mobile phone ) : ( Hide QR )} {showQR && (
Scan the QR code or click here
)}
); } function ButtonsSection({ state, goToWalletManualWithdraw, }: { state: Ready | Confirmed; goToWalletManualWithdraw: (currency: string) => void; }): VNode { const { i18n } = useTranslationContext(); if (state.status === "ready") { const { payStatus } = state; if (payStatus.status === PreparePayResultType.PaymentPossible) { return (
Pay {}
); } if (payStatus.status === PreparePayResultType.InsufficientBalance) { let BalanceMessage = ""; if (!state.balance) { BalanceMessage = i18n.str`You have no balance for this currency. Withdraw digital cash first.`; } else { const balanceShouldBeEnough = Amounts.cmp(state.balance, state.amount) !== -1; if (balanceShouldBeEnough) { BalanceMessage = i18n.str`Could not find enough coins to pay this order. Even if you have enough ${state.balance.currency} some restriction may apply.`; } else { BalanceMessage = i18n.str`Your current balance is not enough for this order.`; } } return (
{BalanceMessage}
goToWalletManualWithdraw(state.amount.currency)} > Withdraw digital cash
); } if (payStatus.status === PreparePayResultType.AlreadyConfirmed) { return (
{payStatus.paid && state.payStatus.contractTerms.fulfillment_message && ( Merchant message} text={state.payStatus.contractTerms.fulfillment_message} kind="neutral" /> )}
{!payStatus.paid && }
); } } if (state.status === "confirmed") { if (state.payResult.type === ConfirmPayResultType.Pending) { return (

Processing...

); } } return ; }