Files
protocol/sites/dev0x-portal/app/routes/create-account._index.tsx
2023-04-06 20:56:53 +02:00

165 lines
6.0 KiB
TypeScript

import { Form, Link, useActionData, useLoaderData, useNavigation } from '@remix-run/react';
import { json, redirect } from '@remix-run/server-runtime';
import { getSignedInUser, sessionStorage } from '../auth.server';
import { TextInput } from '../components/TextInput';
import { validateFormData } from '../utils/utils';
import { z } from 'zod';
import { Button } from '../components/Button';
import type { ActionArgs, LoaderArgs } from '@remix-run/server-runtime';
import { getUserByEmail } from '../data/zippo.server';
const firstPageFormModel = z.object({
firstName: z.string().min(1, 'First name is required'),
lastName: z.string().min(1, 'Last name is required'),
email: z.string().email('Please enter a valid email address'),
});
type ActionInput = z.TypeOf<typeof firstPageFormModel>;
type Errors = Partial<{ general: string } & Record<keyof ActionInput, string>>;
export async function loader({ request }: LoaderArgs) {
const [user, headers] = await getSignedInUser(request);
if (user) {
throw redirect('/apps', { headers });
}
const session = await sessionStorage.getSession(request.headers.get('Cookie'));
const createAccountSession = session.get('createAccount');
return json(
createAccountSession
? {
firstName: createAccountSession.firstName as string,
lastName: createAccountSession.lastName as string,
email: createAccountSession.email as string,
}
: null,
200,
);
}
export async function action({ request }: ActionArgs) {
const [user, headers] = await getSignedInUser(request);
if (user) {
throw redirect('/apps', { headers });
}
const formData = await request.formData();
const { body, errors } = validateFormData<ActionInput>({
formData: formData,
schema: firstPageFormModel,
});
if (errors) {
return json({ errors: errors as Errors, values: body });
}
const lowerCasedEmail = body.email.toLowerCase();
const userResult = await getUserByEmail({ email: lowerCasedEmail });
if (userResult.result === 'SUCCESS') {
return json({
errors: {
email: 'This email is already in use',
} as Errors,
values: body,
});
} else if (userResult.result === 'ERROR' && userResult.error.message !== 'User not found') {
return json({
errors: {
email: 'An error occurred while creating an account',
} as Errors,
values: body,
});
}
const session = await sessionStorage.getSession(request.headers.get('Cookie'));
session.set('createAccount', {
firstName: body.firstName,
lastName: body.lastName,
email: lowerCasedEmail,
});
throw redirect('/create-account/set-password', {
headers: {
'Set-Cookie': await sessionStorage.commitSession(session),
},
});
}
export const handle = {
showNavSwitch: true,
};
export default function CreateAccount() {
const actionData = useActionData<typeof action>();
const maybeSessionInfo = useLoaderData<typeof loader>();
const navigation = useNavigation();
return (
<main className="min-w-screen flex h-full min-h-full w-full flex-col bg-white">
<div className="flex h-full w-full justify-center">
<div className="mt-40 w-full max-w-[456px]">
<h1 className="text-2.5xl mb-6 text-black">Create an account</h1>
<Form method="post" className="flex w-full flex-col gap-x-[18px] gap-y-4">
<div className="grid w-full max-w-full grid-cols-2 gap-[18px]">
<TextInput
placeholder="Enter first name"
name="firstName"
id="firstName"
label="First name"
hiddenLabel
error={actionData?.errors.firstName}
className="w-full"
initialValue={actionData?.values?.firstName || maybeSessionInfo?.firstName}
/>
<TextInput
placeholder="Enter last name"
name="lastName"
id="lastName"
label="Last name"
hiddenLabel
error={actionData?.errors.lastName}
className="w-full"
initialValue={actionData?.values?.lastName || maybeSessionInfo?.lastName}
/>
</div>
<TextInput
placeholder="What's your work email"
name="email"
id="email"
label="E-Mail address"
hiddenLabel
className="w-full"
error={actionData?.errors.email}
initialValue={actionData?.values?.email || maybeSessionInfo?.email}
/>
<Button
type="submit"
className="col-span-2 justify-center"
size="md"
disabled={navigation.state !== 'idle'}
>
Continue
</Button>
</Form>
<p className="text-grey-400 mt-20 text-center">
Do you have an account?{' '}
<Link className="text-black" to="/login">
Login
</Link>
</p>
</div>
</div>
</main>
);
}