/* 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 */ /** * Popup shown to the user when they click * the Taler browser action button. * * @author sebasjm */ /** * Imports. */ import { WalletApiOperation } from "@gnu-taler/taler-wallet-core"; import { Fragment, h, VNode } from "preact"; import { JustInDevMode } from "./components/JustInDevMode.js"; import { NavigationHeader, NavigationHeaderHolder, SvgIcon, } from "./components/styled/index.js"; import { useBackendContext } from "./context/backend.js"; import { useTranslationContext } from "./context/translation.js"; import { useAsyncAsHook } from "./hooks/useAsyncAsHook.js"; import qrIcon from "./svg/qr_code_24px.svg"; import settingsIcon from "./svg/settings_black_24dp.svg"; import warningIcon from "./svg/warning_24px.svg"; /** * List of pages used by the wallet * * @author sebasjm */ type PageLocation = { pattern: string; (params: DynamicPart): string; }; function replaceAll( pattern: string, vars: Record, values: Record, ): string { let result = pattern; for (const v in vars) { result = result.replace(vars[v], !values[v] ? "" : values[v]); } return result; } function pageDefinition(pattern: string): PageLocation { const patternParams = pattern.match(/(:[\w?]*)/g); if (!patternParams) throw Error( `page definition pattern ${pattern} doesn't have any parameter`, ); const vars = patternParams.reduce((prev, cur) => { const pName = cur.match(/(\w+)/g); //skip things like :? in the path pattern if (!pName || !pName[0]) return prev; const name = pName[0]; return { ...prev, [name]: cur }; }, {} as Record); const f = (values: T): string => replaceAll(pattern, vars, values); f.pattern = pattern; return f; } export const Pages = { welcome: "/welcome", balance: "/balance", balanceHistory: pageDefinition<{ currency?: string }>( "/balance/history/:currency?", ), balanceDeposit: pageDefinition<{ amount: string }>( "/balance/deposit/:amount", ), balanceTransaction: pageDefinition<{ tid: string }>( "/balance/transaction/:tid", ), sendCash: pageDefinition<{ amount?: string }>("/destination/send/:amount"), receiveCash: pageDefinition<{ amount?: string }>("/destination/get/:amount?"), dev: "/dev", exchanges: "/exchanges", backup: "/backup", backupProviderDetail: pageDefinition<{ pid: string }>( "/backup/provider/:pid", ), backupProviderAdd: "/backup/provider/add", qr: "/qr", notifications: "/notifications", settings: "/settings", settingsExchangeAdd: pageDefinition<{ currency?: string }>( "/settings/exchange/add/:currency?", ), cta: pageDefinition<{ action: string }>("/cta/:action"), ctaPay: "/cta/pay", ctaRecovery: "/cta/recovery", ctaRefund: "/cta/refund", ctaTips: "/cta/tip", ctaWithdraw: "/cta/withdraw", ctaDeposit: "/cta/deposit", ctaInvoiceCreate: pageDefinition<{ amount?: string }>( "/cta/invoice/create/:amount?", ), ctaTransferCreate: pageDefinition<{ amount?: string }>( "/cta/transfer/create/:amount?", ), ctaInvoicePay: "/cta/invoice/pay", ctaTransferPickup: "/cta/transfer/pickup", ctaWithdrawManual: pageDefinition<{ amount?: string }>( "/cta/manual-withdraw/:amount?", ), }; export function PopupNavBar({ path = "" }: { path?: string }): VNode { const api = useBackendContext(); const hook = useAsyncAsHook(async () => { return await api.wallet.call( WalletApiOperation.GetUserAttentionUnreadCount, {}, ); }); const attentionCount = !hook || hook.hasError ? 0 : hook.response.total; const { i18n } = useTranslationContext(); return ( Balance Backup
{attentionCount > 0 ? ( ) : ( )}
); } export function WalletNavBar({ path = "" }: { path?: string }): VNode { const { i18n } = useTranslationContext(); const api = useBackendContext(); const hook = useAsyncAsHook(async () => { return await api.wallet.call( WalletApiOperation.GetUserAttentionUnreadCount, {}, ); }); const attentionCount = !hook || hook.hasError ? 0 : hook.response.total; return ( Balance Backup {attentionCount > 0 ? ( Notifications ) : ( )} Dev ); }