/* This file is part of GNU Taler (C) 2022-2024 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 { Button, InputLine, InternationalizationAPI, LocalNotificationBanner, UIHandlerId, useLocalNotificationHandler, useTranslationContext, } from "@gnu-taler/web-util/browser"; import { VNode, h } from "preact"; import { FormErrors, FormStatus, FormValues, RecursivePartial, useFormState, } from "../hooks/form.js"; import { useOfficer } from "../hooks/officer.js"; import { usePreferences } from "../hooks/preferences.js"; type FormType = { password: string; repeat: string; }; function createFormValidator( i18n: InternationalizationAPI, allowInsecurePassword: boolean, ) { return function check( state: RecursivePartial>, ): FormStatus { const errors = undefinedIfEmpty>({ password: !state.password ? i18n.str`required` : allowInsecurePassword ? undefined : state.password.length < 8 ? i18n.str`should have at least 8 characters` : !state.password.match(/[a-z]/) && state.password.match(/[A-Z]/) ? i18n.str`should have lowercase and uppercase characters` : !state.password.match(/\d/) ? i18n.str`should have numbers` : !state.password.match(/[^a-zA-Z\d]/) ? i18n.str`should have at least one character which is not a number or letter` : undefined, repeat: !state.repeat ? i18n.str`required` : state.password !== state.repeat ? i18n.str`doesn't match` : undefined, }); if (errors === undefined) { const result: FormType = { password: state.password!, repeat: state.repeat!, }; return { status: "ok", result, errors, }; } const result: RecursivePartial = { password: state.password, repeat: state.repeat, }; return { status: "fail", result, errors, }; }; } export function undefinedIfEmpty(obj: T): T | undefined { if (obj === undefined) return undefined; return Object.keys(obj).some( (k) => (obj as Record)[k] !== undefined, ) ? obj : undefined; } export function CreateAccount(): VNode { const { i18n } = useTranslationContext(); const [settings] = usePreferences(); const officer = useOfficer(); const [notification, withErrorHandler] = useLocalNotificationHandler(); const [form, status] = useFormState( [".password", ".repeat"] as Array, { password: undefined, repeat: undefined, }, createFormValidator(i18n, settings.allowInsecurePassword), ); const createAccountHandler = status.status === "fail" || officer.state !== "not-found" ? undefined : withErrorHandler( async () => officer.create(form.password!.value!), () => {}, ); return (

Create account

{ e.preventDefault(); }} autoCapitalize="none" autoCorrect="off" >
label={i18n.str`Password`} name="password" type="password" required handler={form.password} />
label={i18n.str`Repeat password`} name="repeat" type="password" required handler={form.repeat} />
); } /** * Show the element when the load ended * @param element */ export function doAutoFocus(element: HTMLElement | null) { if (element) { setTimeout(() => { element.focus({ preventScroll: true }); element.scrollIntoView({ behavior: "smooth", block: "center", inline: "center", }); }, 100); } }