/*
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
);
}
/**
* 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);
}
}