/* 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 */ import { AbsoluteTime, Amounts, ConfirmPayResultType, ContractTerms, PreparePayResultType, Product, } from "@gnu-taler/taler-util"; import { Fragment, h, VNode } from "preact"; import { useState } from "preact/hooks"; import { Amount } from "../../components/Amount.js"; import { ErrorTalerOperation } from "../../components/ErrorTalerOperation.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 { Link, LinkSuccess, SmallLightText, SubTitle, SuccessBox, WalletAction, WarningBox, } from "../../components/styled/index.js"; import { Time } from "../../components/Time.js"; import { useTranslationContext } from "../../context/translation.js"; import { Button } from "../../mui/Button.js"; import { MerchantDetails, PurchaseDetails } from "../../wallet/Transaction.js"; import { State } from "./index.js"; export function LoadingUriView({ error }: State.LoadingUriError): VNode { const { i18n } = useTranslationContext(); return ( Could not load pay status} error={error} /> ); } type SupportedStates = | State.Ready | State.Confirmed | State.Completed | State.NoBalanceForCurrency | State.NoEnoughBalance; export function BaseView(state: SupportedStates): VNode { const { i18n } = useTranslationContext(); const contractTerms: ContractTerms = state.payStatus.contractTerms; const price = { raw: state.amount, effective: "amountEffective" in state.payStatus ? Amounts.parseOrThrow(state.payStatus.amountEffective) : state.amount, }; const totalFees = Amounts.sub(price.effective, price.raw).amount; return ( Digital cash payment
{/* {state.payStatus.status !== PreparePayResultType.InsufficientBalance && Amounts.isNonZero(totalFees) && ( Total to pay} text={} kind="negative" /> )} Purchase amount} text={} kind="neutral" /> {Amounts.isNonZero(totalFees) && ( Fee} text={} kind="negative" /> )} */} Purchase} text={contractTerms.summary} kind="neutral" /> Merchant} text={} kind="neutral" /> {/*
{JSON.stringify(price)}

{JSON.stringify(state.payStatus, undefined, 2)}
*/} Details} text={ } kind="neutral" /> {contractTerms.order_id && ( Receipt} text={`#${contractTerms.order_id}`} kind="neutral" /> )} {contractTerms.pay_deadline && ( Valid until} text={
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: SupportedStates }): 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 == "completed") { 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: SupportedStates }): 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: SupportedStates; goToWalletManualWithdraw: (currency: string) => Promise; }): VNode { const { i18n } = useTranslationContext(); if (state.status === "ready") { return (
); } if ( state.status === "no-enough-balance" || state.status === "no-balance-for-currency" ) { // if (state.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}
); // } } if (state.status === "confirmed") { if (state.payStatus.status === PreparePayResultType.AlreadyConfirmed) { return (
{state.payStatus.paid && state.payStatus.contractTerms.fulfillment_message && ( Merchant message} text={state.payStatus.contractTerms.fulfillment_message} kind="neutral" /> )}
{!state.payStatus.paid && }
); } } if (state.status === "completed") { if (state.payResult.type === ConfirmPayResultType.Pending) { return (

Processing...

); } } return ; }