wallet-core/packages/merchant-backoffice-ui/src/InstanceRoutes.tsx

781 lines
26 KiB
TypeScript
Raw Normal View History

/*
This file is part of GNU Taler
2022-12-16 20:59:37 +01:00
(C) 2021-2023 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)
*/
2023-02-08 21:39:39 +01:00
import {
useTranslationContext,
HttpError,
ErrorType,
2023-05-05 13:38:28 +02:00
} from "@gnu-taler/web-util/browser";
import { format } from "date-fns";
import { Fragment, FunctionComponent, h, VNode } from "preact";
import { Route, route, Router } from "preact-router";
import { useCallback, useEffect, useMemo, useState } from "preact/hooks";
2022-11-04 14:24:29 +01:00
import { Loading } from "./components/exception/loading.js";
import { Menu, NotificationCard } from "./components/menu/index.js";
import { useBackendContext } from "./context/backend.js";
import { InstanceContextProvider } from "./context/instance.js";
import {
useBackendDefaultToken,
useBackendInstanceToken,
useLocalStorage,
2022-11-04 14:24:29 +01:00
} from "./hooks/index.js";
import { useInstanceKYCDetails } from "./hooks/instance.js";
2022-11-04 14:24:29 +01:00
import InstanceCreatePage from "./paths/admin/create/index.js";
import InstanceListPage from "./paths/admin/list/index.js";
2023-09-04 19:17:55 +02:00
import TokenPage from "./paths/instance/token/index.js";
import ListKYCPage from "./paths/instance/kyc/list/index.js";
2022-11-04 14:24:29 +01:00
import OrderCreatePage from "./paths/instance/orders/create/index.js";
import OrderDetailsPage from "./paths/instance/orders/details/index.js";
import OrderListPage from "./paths/instance/orders/list/index.js";
import ProductCreatePage from "./paths/instance/products/create/index.js";
import ProductListPage from "./paths/instance/products/list/index.js";
import ProductUpdatePage from "./paths/instance/products/update/index.js";
2023-09-04 19:17:55 +02:00
import BankAccountCreatePage from "./paths/instance/accounts/create/index.js";
import BankAccountListPage from "./paths/instance/accounts/list/index.js";
import BankAccountUpdatePage from "./paths/instance/accounts/update/index.js";
2022-11-04 14:24:29 +01:00
import ReservesCreatePage from "./paths/instance/reserves/create/index.js";
import ReservesDetailsPage from "./paths/instance/reserves/details/index.js";
import ReservesListPage from "./paths/instance/reserves/list/index.js";
import TemplateCreatePage from "./paths/instance/templates/create/index.js";
2023-01-27 16:35:10 +01:00
import TemplateUsePage from "./paths/instance/templates/use/index.js";
import TemplateQrPage from "./paths/instance/templates/qr/index.js";
import TemplateListPage from "./paths/instance/templates/list/index.js";
import TemplateUpdatePage from "./paths/instance/templates/update/index.js";
2023-01-27 19:08:03 +01:00
import WebhookCreatePage from "./paths/instance/webhooks/create/index.js";
import WebhookListPage from "./paths/instance/webhooks/list/index.js";
import WebhookUpdatePage from "./paths/instance/webhooks/update/index.js";
2023-09-04 19:17:55 +02:00
import ValidatorCreatePage from "./paths/instance/validators/create/index.js";
import ValidatorListPage from "./paths/instance/validators/list/index.js";
import ValidatorUpdatePage from "./paths/instance/validators/update/index.js";
import TransferCreatePage from "./paths/instance/transfers/create/index.js";
import TransferListPage from "./paths/instance/transfers/list/index.js";
import InstanceUpdatePage, {
AdminUpdate as InstanceAdminUpdatePage,
Props as InstanceUpdatePageProps,
2022-11-04 14:24:29 +01:00
} from "./paths/instance/update/index.js";
import LoginPage from "./paths/login/index.js";
import NotFoundPage from "./paths/notfound/index.js";
import { Notification } from "./utils/types.js";
2023-02-08 21:39:39 +01:00
import { MerchantBackend } from "./declaration.js";
2023-08-07 11:51:10 +02:00
import { Settings } from "./paths/settings/index.js";
2023-09-04 19:17:55 +02:00
import { dateFormatForSettings, useSettings } from "./hooks/useSettings.js";
export enum InstancePaths {
error = "/error",
2023-09-04 19:17:55 +02:00
server = "/server",
token = "/token",
bank_list = "/bank",
bank_update = "/bank/:bid/update",
bank_new = "/bank/new",
product_list = "/products",
product_update = "/product/:pid/update",
product_new = "/product/new",
order_list = "/orders",
order_new = "/order/new",
order_details = "/order/:oid/details",
reserves_list = "/reserves",
reserves_details = "/reserves/:rid/details",
reserves_new = "/reserves/new",
kyc = "/kyc",
transfers_list = "/transfers",
transfers_new = "/transfer/new",
2022-12-19 20:25:09 +01:00
templates_list = "/templates",
templates_update = "/templates/:tid/update",
templates_new = "/templates/new",
2023-01-27 16:35:10 +01:00
templates_use = "/templates/:tid/use",
templates_qr = "/templates/:tid/qr",
2023-01-27 19:08:03 +01:00
webhooks_list = "/webhooks",
webhooks_update = "/webhooks/:tid/update",
webhooks_new = "/webhooks/new",
2023-08-07 11:51:10 +02:00
2023-09-04 19:17:55 +02:00
validators_list = "/validators",
validators_update = "/validators/:vid/update",
validators_new = "/validators/new",
settings = "/inteface",
}
// eslint-disable-next-line @typescript-eslint/no-empty-function
2023-09-04 19:17:55 +02:00
const noop = () => { };
export enum AdminPaths {
list_instances = "/instances",
new_instance = "/instance/new",
update_instance = "/instance/:id/update",
}
export interface Props {
id: string;
admin?: boolean;
path: string;
2023-09-04 19:17:55 +02:00
onUnauthorized: () => void;
onLoginPass: () => void;
setInstanceName: (s: string) => void;
}
export function InstanceRoutes({
id,
admin,
path,
2023-09-04 19:17:55 +02:00
onUnauthorized,
onLoginPass,
setInstanceName,
}: Props): VNode {
2023-09-04 19:17:55 +02:00
const [defaultToken, updateDefaultToken] = useBackendDefaultToken();
const [token, updateToken] = useBackendInstanceToken(id);
const { i18n } = useTranslationContext();
type GlobalNotifState = (Notification & { to: string }) | undefined;
const [globalNotification, setGlobalNotification] =
useState<GlobalNotifState>(undefined);
const changeToken = (token?: string) => {
if (admin) {
updateToken(token);
} else {
updateDefaultToken(token);
}
2023-09-04 19:17:55 +02:00
onLoginPass()
};
2023-09-04 19:17:55 +02:00
// const updateLoginStatus = (url: string, token?: string) => {
// changeToken(token);
// };
const value = useMemo(
() => ({ id, token, admin, changeToken }),
2022-12-19 16:23:39 +01:00
[id, token, admin],
);
function ServerErrorRedirectTo(to: InstancePaths | AdminPaths) {
2023-02-08 21:39:39 +01:00
return function ServerErrorRedirectToImpl(
error: HttpError<MerchantBackend.ErrorDetail>,
) {
if (error.type === ErrorType.TIMEOUT) {
setGlobalNotification({
message: i18n.str`The request to the backend take too long and was cancelled`,
description: i18n.str`Diagnostic from ${error.info.url} is "${error.message}"`,
type: "ERROR",
to,
});
} else {
setGlobalNotification({
message: i18n.str`The backend reported a problem: HTTP status #${error.status}`,
description: i18n.str`Diagnostic from ${error.info.url} is '${error.message}'`,
details:
error.type === ErrorType.CLIENT || error.type === ErrorType.SERVER
? error.payload.detail
: undefined,
type: "ERROR",
to,
});
}
return <Redirect to={to} />;
};
}
2023-09-04 19:17:55 +02:00
// const LoginPageAccessDeniend = onUnauthorized
const LoginPageAccessDenied = () => {
onUnauthorized()
return <NotificationCard
notification={{
message: i18n.str`Access denied`,
description: i18n.str`Redirecting to login page.`,
type: "ERROR",
}}
/>
}
function IfAdminCreateDefaultOr<T>(Next: FunctionComponent<any>) {
return function IfAdminCreateDefaultOrImpl(props?: T) {
if (admin && id === "default") {
return (
<Fragment>
<NotificationCard
notification={{
message: i18n.str`No 'default' instance configured yet.`,
description: i18n.str`Create a 'default' instance to begin using the merchant backoffice.`,
type: "INFO",
}}
/>
<InstanceCreatePage
forceId="default"
onConfirm={() => {
route(AdminPaths.list_instances);
}}
/>
</Fragment>
);
}
if (props) {
return <Next {...props} />;
}
return <Next />;
};
}
const clearTokenAndGoToRoot = () => {
route("/");
2023-09-04 19:17:55 +02:00
// clear all tokens
updateToken(undefined)
updateDefaultToken(undefined)
};
return (
<InstanceContextProvider value={value}>
<Menu
instance={id}
admin={admin}
2023-08-07 11:51:10 +02:00
onShowSettings={() => {
2023-09-04 19:17:55 +02:00
route("/inteface")
2023-08-07 11:51:10 +02:00
}}
path={path}
onLogout={clearTokenAndGoToRoot}
setInstanceName={setInstanceName}
2023-09-04 19:17:55 +02:00
isPasswordOk={defaultToken !== undefined}
/>
<KycBanner />
<NotificationCard notification={globalNotification} />
<Router
onChange={(e) => {
const movingOutFromNotification =
globalNotification && e.url !== globalNotification.to;
if (movingOutFromNotification) {
setGlobalNotification(undefined);
}
}}
>
<Route path="/" component={Redirect} to={InstancePaths.order_list} />
{/**
* Admin pages
*/}
{admin && (
<Route
path={AdminPaths.list_instances}
component={InstanceListPage}
onCreate={() => {
route(AdminPaths.new_instance);
}}
onUpdate={(id: string): void => {
route(`/instance/${id}/update`);
}}
setInstanceName={setInstanceName}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.error)}
/>
)}
{admin && (
<Route
path={AdminPaths.new_instance}
component={InstanceCreatePage}
onBack={() => route(AdminPaths.list_instances)}
onConfirm={() => {
route(AdminPaths.list_instances);
}}
/>
)}
{admin && (
<Route
path={AdminPaths.update_instance}
component={AdminInstanceUpdatePage}
onBack={() => route(AdminPaths.list_instances)}
onConfirm={() => {
route(AdminPaths.list_instances);
}}
onUpdateError={ServerErrorRedirectTo(AdminPaths.list_instances)}
onLoadError={ServerErrorRedirectTo(AdminPaths.list_instances)}
onNotFound={NotFoundPage}
/>
)}
{/**
* Update instance page
*/}
<Route
2023-09-04 19:17:55 +02:00
path={InstancePaths.server}
component={InstanceUpdatePage}
onBack={() => {
route(`/`);
}}
onConfirm={() => {
route(`/`);
}}
onUpdateError={noop}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.error)}
/>
2023-09-04 19:17:55 +02:00
{/**
* Update instance page
*/}
<Route
path={InstancePaths.token}
component={TokenPage}
onChange={() => {
route(`/`);
}}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.error)}
/>
{/**
* Product pages
*/}
<Route
path={InstancePaths.product_list}
component={ProductListPage}
onUnauthorized={LoginPageAccessDenied}
2023-09-04 19:17:55 +02:00
onLoadError={ServerErrorRedirectTo(InstancePaths.server)}
onCreate={() => {
route(InstancePaths.product_new);
}}
onSelect={(id: string) => {
route(InstancePaths.product_update.replace(":pid", id));
}}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
/>
<Route
path={InstancePaths.product_update}
component={ProductUpdatePage}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.product_list)}
onConfirm={() => {
route(InstancePaths.product_list);
}}
onBack={() => {
route(InstancePaths.product_list);
}}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
/>
<Route
path={InstancePaths.product_new}
component={ProductCreatePage}
onConfirm={() => {
route(InstancePaths.product_list);
}}
onBack={() => {
route(InstancePaths.product_list);
}}
/>
2023-09-04 19:17:55 +02:00
{/**
* Bank pages
*/}
<Route
path={InstancePaths.bank_list}
component={BankAccountListPage}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.server)}
onCreate={() => {
route(InstancePaths.bank_new);
}}
onSelect={(id: string) => {
route(InstancePaths.bank_update.replace(":bid", id));
}}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
/>
<Route
path={InstancePaths.bank_update}
component={BankAccountUpdatePage}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.product_list)}
onConfirm={() => {
route(InstancePaths.bank_list);
}}
onBack={() => {
route(InstancePaths.bank_list);
}}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
/>
<Route
path={InstancePaths.bank_new}
component={BankAccountCreatePage}
onConfirm={() => {
route(InstancePaths.bank_list);
}}
onBack={() => {
route(InstancePaths.bank_list);
}}
/>
{/**
* Order pages
*/}
<Route
path={InstancePaths.order_list}
component={OrderListPage}
onCreate={() => {
route(InstancePaths.order_new);
}}
onSelect={(id: string) => {
route(InstancePaths.order_details.replace(":oid", id));
}}
onUnauthorized={LoginPageAccessDenied}
2023-09-04 19:17:55 +02:00
onLoadError={ServerErrorRedirectTo(InstancePaths.server)}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
/>
<Route
path={InstancePaths.order_details}
component={OrderDetailsPage}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.order_list)}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
onBack={() => {
route(InstancePaths.order_list);
}}
/>
<Route
path={InstancePaths.order_new}
component={OrderCreatePage}
2023-09-04 19:17:55 +02:00
onConfirm={(orderId: string) => {
route(InstancePaths.order_details.replace(":oid", orderId));
}}
onBack={() => {
route(InstancePaths.order_list);
}}
/>
{/**
* Transfer pages
*/}
<Route
path={InstancePaths.transfers_list}
component={TransferListPage}
onUnauthorized={LoginPageAccessDenied}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
2023-09-04 19:17:55 +02:00
onLoadError={ServerErrorRedirectTo(InstancePaths.server)}
onCreate={() => {
route(InstancePaths.transfers_new);
}}
/>
<Route
path={InstancePaths.transfers_new}
component={TransferCreatePage}
onConfirm={() => {
route(InstancePaths.transfers_list);
}}
onBack={() => {
route(InstancePaths.transfers_list);
}}
/>
2023-01-27 19:08:03 +01:00
{/**
* Webhooks pages
*/}
<Route
path={InstancePaths.webhooks_list}
component={WebhookListPage}
onUnauthorized={LoginPageAccessDenied}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
2023-09-04 19:17:55 +02:00
onLoadError={ServerErrorRedirectTo(InstancePaths.server)}
2023-01-27 19:08:03 +01:00
onCreate={() => {
route(InstancePaths.webhooks_new);
}}
onSelect={(id: string) => {
route(InstancePaths.webhooks_update.replace(":tid", id));
}}
/>
<Route
path={InstancePaths.webhooks_update}
component={WebhookUpdatePage}
onConfirm={() => {
route(InstancePaths.webhooks_list);
}}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.webhooks_list)}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
onBack={() => {
route(InstancePaths.webhooks_list);
}}
/>
<Route
path={InstancePaths.webhooks_new}
component={WebhookCreatePage}
onConfirm={() => {
route(InstancePaths.webhooks_list);
}}
onBack={() => {
route(InstancePaths.webhooks_list);
}}
/>
2023-09-04 19:17:55 +02:00
{/**
* Validator pages
*/}
<Route
path={InstancePaths.validators_list}
component={ValidatorListPage}
onUnauthorized={LoginPageAccessDenied}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
onLoadError={ServerErrorRedirectTo(InstancePaths.server)}
onCreate={() => {
route(InstancePaths.validators_new);
}}
onSelect={(id: string) => {
route(InstancePaths.validators_update.replace(":vid", id));
}}
/>
<Route
path={InstancePaths.validators_update}
component={ValidatorUpdatePage}
onConfirm={() => {
route(InstancePaths.validators_list);
}}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.validators_list)}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
onBack={() => {
route(InstancePaths.validators_list);
}}
/>
<Route
path={InstancePaths.validators_new}
component={ValidatorCreatePage}
onConfirm={() => {
route(InstancePaths.validators_list);
}}
onBack={() => {
route(InstancePaths.validators_list);
}}
/>
2022-12-19 20:25:09 +01:00
{/**
* Templates pages
*/}
<Route
path={InstancePaths.templates_list}
component={TemplateListPage}
onUnauthorized={LoginPageAccessDenied}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
2023-09-04 19:17:55 +02:00
onLoadError={ServerErrorRedirectTo(InstancePaths.server)}
2022-12-19 20:25:09 +01:00
onCreate={() => {
route(InstancePaths.templates_new);
}}
2023-01-27 16:35:10 +01:00
onNewOrder={(id: string) => {
route(InstancePaths.templates_use.replace(":tid", id));
}}
onQR={(id: string) => {
route(InstancePaths.templates_qr.replace(":tid", id));
}}
2022-12-19 20:25:09 +01:00
onSelect={(id: string) => {
route(InstancePaths.templates_update.replace(":tid", id));
}}
/>
<Route
path={InstancePaths.templates_update}
component={TemplateUpdatePage}
onConfirm={() => {
route(InstancePaths.templates_list);
}}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.templates_list)}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
onBack={() => {
route(InstancePaths.templates_list);
}}
/>
<Route
path={InstancePaths.templates_new}
component={TemplateCreatePage}
onConfirm={() => {
route(InstancePaths.templates_list);
}}
onBack={() => {
route(InstancePaths.templates_list);
}}
/>
2023-01-27 16:35:10 +01:00
<Route
path={InstancePaths.templates_use}
component={TemplateUsePage}
onOrderCreated={(id: string) => {
route(InstancePaths.order_details.replace(":oid", id));
}}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.templates_list)}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
onBack={() => {
route(InstancePaths.templates_list);
}}
/>
<Route
path={InstancePaths.templates_qr}
component={TemplateQrPage}
onUnauthorized={LoginPageAccessDenied}
2023-01-27 16:35:10 +01:00
onLoadError={ServerErrorRedirectTo(InstancePaths.templates_list)}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
onBack={() => {
route(InstancePaths.templates_list);
}}
/>
{/**
* reserves pages
*/}
<Route
path={InstancePaths.reserves_list}
component={ReservesListPage}
onUnauthorized={LoginPageAccessDenied}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
2023-09-04 19:17:55 +02:00
onLoadError={ServerErrorRedirectTo(InstancePaths.server)}
onSelect={(id: string) => {
route(InstancePaths.reserves_details.replace(":rid", id));
}}
onCreate={() => {
route(InstancePaths.reserves_new);
}}
/>
<Route
path={InstancePaths.reserves_details}
component={ReservesDetailsPage}
onUnauthorized={LoginPageAccessDenied}
onLoadError={ServerErrorRedirectTo(InstancePaths.reserves_list)}
onNotFound={IfAdminCreateDefaultOr(NotFoundPage)}
onBack={() => {
route(InstancePaths.reserves_list);
}}
/>
<Route
path={InstancePaths.reserves_new}
component={ReservesCreatePage}
onConfirm={() => {
route(InstancePaths.reserves_list);
}}
onBack={() => {
route(InstancePaths.reserves_list);
}}
/>
<Route path={InstancePaths.kyc} component={ListKYCPage} />
2023-08-07 11:51:10 +02:00
<Route path={InstancePaths.settings} component={Settings} />
{/**
* Example pages
*/}
<Route path="/loading" component={Loading} />
<Route default component={NotFoundPage} />
</Router>
</InstanceContextProvider>
);
}
export function Redirect({ to }: { to: string }): null {
useEffect(() => {
route(to, true);
});
return null;
}
function AdminInstanceUpdatePage({
id,
...rest
}: { id: string } & InstanceUpdatePageProps): VNode {
const [token, changeToken] = useBackendInstanceToken(id);
const { updateLoginStatus: changeBackend } = useBackendContext();
2023-02-08 21:39:39 +01:00
const updateLoginStatus = (url: string, token?: string): void => {
changeBackend(url);
2023-09-04 19:17:55 +02:00
changeToken(token);
};
const value = useMemo(
() => ({ id, token, admin: true, changeToken }),
2022-12-19 16:23:39 +01:00
[id, token],
);
const { i18n } = useTranslationContext();
return (
<InstanceContextProvider value={value}>
<InstanceAdminUpdatePage
{...rest}
instanceId={id}
2023-02-08 21:39:39 +01:00
onLoadError={(error: HttpError<MerchantBackend.ErrorDetail>) => {
const notif =
error.type === ErrorType.TIMEOUT
? {
2023-09-04 19:17:55 +02:00
message: i18n.str`The request to the backend take too long and was cancelled`,
description: i18n.str`Diagnostic from ${error.info.url} is '${error.message}'`,
type: "ERROR" as const,
}
: {
2023-09-04 19:17:55 +02:00
message: i18n.str`The backend reported a problem: HTTP status #${error.status}`,
description: i18n.str`Diagnostic from ${error.info.url} is '${error.message}'`,
details:
error.type === ErrorType.CLIENT ||
error.type === ErrorType.SERVER
2023-09-04 19:17:55 +02:00
? error.payload.detail
: undefined,
type: "ERROR" as const,
};
return (
<Fragment>
<NotificationCard notification={notif} />
<LoginPage onConfirm={updateLoginStatus} />
</Fragment>
);
}}
onUnauthorized={() => {
return (
<Fragment>
<NotificationCard
notification={{
message: i18n.str`Access denied`,
description: i18n.str`The access token provided is invalid`,
type: "ERROR",
}}
/>
<LoginPage onConfirm={updateLoginStatus} />
</Fragment>
);
}}
/>
</InstanceContextProvider>
);
}
function KycBanner(): VNode {
const kycStatus = useInstanceKYCDetails();
const { i18n } = useTranslationContext();
2023-09-04 19:17:55 +02:00
const [settings] = useSettings();
const today = format(new Date(), dateFormatForSettings(settings));
const [lastHide, setLastHide] = useLocalStorage("kyc-last-hide");
const hasBeenHidden = today === lastHide;
const needsToBeShown = kycStatus.ok && kycStatus.data.type === "redirect";
if (hasBeenHidden || !needsToBeShown) return <Fragment />;
return (
<NotificationCard
notification={{
type: "WARN",
message: "KYC verification needed",
description: (
<div>
<p>
Some transfer are on hold until a KYC process is completed. Go to
the KYC section in the left panel for more information
</p>
<div class="buttons is-right">
<button class="button" onClick={() => setLastHide(today)}>
<i18n.Translate>Hide for today</i18n.Translate>
</button>
</div>
</div>
),
}}
/>
);
}