/* 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, AmountJson, Amounts, ConfirmPayResultType, ContractTerms, PreparePayResult, PreparePayResultPaymentPossible, 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 { ButtonHandler } from "../../mui/handlers.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.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 ); } return ; } export function PayWithMobile({ uri }: { uri: string }): VNode { const { i18n } = useTranslationContext(); const [showQR, setShowQR] = useState(false); return (
setShowQR((qr) => !qr)}> {!showQR ? ( Pay with a mobile phone ) : ( Hide QR )} {showQR && (
Scan the QR code or   click here
)}
); } interface ButtonSectionProps { payStatus: PreparePayResult; payHandler: ButtonHandler | undefined; balance: AmountJson | undefined; uri: string; amount: AmountJson; goToWalletManualWithdraw: (currency: string) => Promise; } export function ButtonsSection({ payStatus, uri, payHandler, balance, amount, goToWalletManualWithdraw, }: ButtonSectionProps): VNode { const { i18n } = useTranslationContext(); if (payStatus.status === PreparePayResultType.PaymentPossible) { const privateUri = `${uri}&n=${payStatus.noncePriv}`; return (
); } if (payStatus.status === PreparePayResultType.InsufficientBalance) { let BalanceMessage = ""; if (!balance) { BalanceMessage = i18n.str`You have no balance for this currency. Withdraw digital cash first.`; } else { const balanceShouldBeEnough = Amounts.cmp(balance, amount) !== -1; if (balanceShouldBeEnough) { BalanceMessage = i18n.str`Could not find enough coins to pay. Even if you have enough ${balance.currency} some restriction may apply.`; } else { BalanceMessage = i18n.str`Your current balance is not enough.`; } } const uriPrivate = `${uri}&n=${payStatus.noncePriv}`; return (
{BalanceMessage}
); } if (payStatus.status === PreparePayResultType.AlreadyConfirmed) { return (
{payStatus.paid && payStatus.contractTerms.fulfillment_message && ( Merchant message} text={payStatus.contractTerms.fulfillment_message} kind="neutral" /> )}
{!payStatus.paid && }
); } const error: never = payStatus; return ; }