Skip to content

Commit

Permalink
WIP: journey form
Browse files Browse the repository at this point in the history
  • Loading branch information
alexandre-kakal-akarah committed Jun 25, 2024
1 parent a76b622 commit 76a1efc
Show file tree
Hide file tree
Showing 12 changed files with 250 additions and 98 deletions.
33 changes: 33 additions & 0 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
"@radix-ui/react-popover": "^1.1.1",
"@radix-ui/react-radio-group": "^1.2.0",
"@radix-ui/react-select": "^2.1.1",
"@radix-ui/react-slider": "^1.2.0",
"@radix-ui/react-slot": "^1.1.0",
"bcrypt": "^5.1.1",
"class-variance-authority": "^0.7.0",
Expand Down
2 changes: 2 additions & 0 deletions src/app/(app)/layout.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import React from "react";
import Nav from "@/components/Nav";
import { ReactParallaxProvider } from "@/providers/Providers";
import EventForm from "@/components/form/event/EventForm";

export default function AppLayout({
children,
Expand All @@ -10,6 +11,7 @@ export default function AppLayout({
return (
<ReactParallaxProvider>
{children}
<EventForm />
<Nav />
</ReactParallaxProvider>
);
Expand Down
16 changes: 14 additions & 2 deletions src/app/(app)/parcours/[id]/page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,15 +2,27 @@ import React from "react";
import ParallaxImage from "@/components/clients/ParallaxImage";
import Rating from "@/components/Rating";
import { Icons } from "@/components/Icons";
import { Button } from "@/components/ui/button";
import { buttonAction } from "@/types/enums/button";

type Params = { id: string };

const JourneyDetail = ({ params }: { params: Params }) => {
// get journey by id
return (
<main className="flex min-h-screen flex-col bg-gray">
<ParallaxImage />
<div className="relative flex-1 -translate-y-4 rounded-t-2xl px-5 pb-40 pt-14">
<div className="relative">
<ParallaxImage />
<Button
action={buttonAction.SET_JOURNEY_ID}
ressourceId={parseInt(params.id)}
className="absolute right-5 top-5 border-orange bg-gray"
>
<span>Créer un évènement</span>
<Icons.arrowLink fill="#d8552b" className="ml-2" />
</Button>
</div>
<div className="relative flex-1 rounded-t-2xl px-5 pb-40 pt-14">
<div className="absolute left-4 top-0 flex gap-[6px]">
<div className="flex items-center rounded-b-md bg-white px-2 py-[6px]">
<Icons.dumbbel />
Expand Down
3 changes: 2 additions & 1 deletion src/app/(app)/profil/page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import TopBar from "@/components/TopBar";
import Link from "next/link";
import { Icons } from "@/components/Icons";
import { Button } from "@/components/ui/button";
import { buttonAction } from "@/types/enums/button";

const Profil = () => {
return (
Expand Down Expand Up @@ -55,7 +56,7 @@ const Profil = () => {
<span className="font-semibold">Mentions légales</span>
</Link>
</section>
<Button isSignOut variant="danger">
<Button action={buttonAction.SIGN_OUT} variant="danger">
Déconnexion
</Button>
<Button variant="dangerFilled">Supprimer mon compte</Button>
Expand Down
163 changes: 114 additions & 49 deletions src/components/form/event/EventForm.tsx
Original file line number Diff line number Diff line change
@@ -1,40 +1,31 @@
"use client";
import React, { useState } from "react";
import {
eventJourneySchema,
eventDataSchema,
eventFormSchema,
} from "@/validators/EventFormSchema";
import { eventFormSchema } from "@/validators/EventFormSchema";
import { z } from "zod";
import { useEventFormStore } from "@/store/eventFormStore";
import { useForm, SubmitHandler } from "react-hook-form";
import { Form } from "@/components/ui/form";
import {
Form,
FormControl,
FormField,
FormItem,
FormMessage,
} from "@/components/ui/form";
import { Button } from "@/components/ui/button";
import { zodResolver } from "@hookform/resolvers/zod";
import StepCounter from "../StepCounter";
import EventFormSteps from "./steps/EventFormSteps";
import { Icons } from "@/components/Icons";
import { Input } from "@/components/ui/input";
import { Label } from "@radix-ui/react-label";
import { Event } from "@prisma/client";
import { Textarea } from "@/components/ui/textarea";

type EventRequestBody = Omit<Event, "id" | "createdAt" | "updatedAt">;
export type EventFormValues = z.infer<typeof eventFormSchema>;
type FieldName = keyof EventFormValues;

const firstFormStepFields = Object.keys(eventJourneySchema.shape);
const secondFormStepFields = Object.keys(eventDataSchema.shape);

const formSteps = [
{
fields: firstFormStepFields,
},
{
fields: secondFormStepFields,
},
];

const EventForm = () => {
const { isVisible, hideModal } = useEventFormStore();
const { isVisible, hideModal, journeyIdValue, setJourneyIdValue } =
useEventFormStore();
const [formStatus, setFormStatus] = useState<"idle" | "errored">("idle");
const [currentStep, setCurrentStep] = useState(0);
const [dir, setDir] = useState<"ltr" | "rtl">("ltr");

const form = useForm<EventFormValues>({
resolver: zodResolver(eventFormSchema),
Expand All @@ -51,31 +42,45 @@ const EventForm = () => {
if (!isVisible) return null;

const processForm: SubmitHandler<EventFormValues> = async (data) => {
console.log(data);
};

const next = async () => {
const fields = formSteps[currentStep].fields;
const output = await form.trigger(fields as FieldName[]);
if (!output) return;
if (currentStep < formSteps.length - 1) {
setCurrentStep((step) => step + 1);
setDir("ltr");
const parsedData = eventFormSchema.safeParse(data);
if (!parsedData.success || !journeyIdValue) {
setFormStatus("errored");
return;
}
};
try {
const body: EventRequestBody = {
journeyId: journeyIdValue,
authorId: 1,
title: data.title,
description: data.description,
numberPlayerMin: data.numberPlayerMin,
numberPlayerMax: data.numberPlayerMax,
isPrivate: data.isPrivate,
accessCode: data.accessCode || null,
endAt: data.endAt,
startAt: data.startAt,
image: "",
};
const response = await fetch("/api/events", {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify(body),
});

const prev = () => {
if (currentStep > 0) {
setCurrentStep((step) => step - 1);
setDir("rtl");
console.log(response, "create event response");
} catch (error) {
console.log(error);
}
console.log(data);
};

const dismissModal = () => {
hideModal();
setCurrentStep(0);
form.reset();
setFormStatus("idle");
setJourneyIdValue(null);
};

return (
Expand All @@ -89,21 +94,80 @@ const EventForm = () => {
<Form {...form}>
{!form.formState.isSubmitSuccessful && formStatus === "idle" && (
<form onSubmit={form.handleSubmit(processForm)}>
<StepCounter step={currentStep} totalSteps={4} />
<EventFormSteps
form={form}
step={currentStep}
next={next}
prev={prev}
dir={dir}
<FormField
control={form.control}
name="title"
render={({ field }) => (
<FormItem>
<FormControl>
<>
<Label htmlFor="title">Titre*</Label>
<Input {...field} id="title" />
</>
</FormControl>
<FormMessage className="text-red-500" />
</FormItem>
)}
/>

<FormField
control={form.control}
name="description"
render={({ field }) => (
<FormItem>
<FormControl>
<>
<Label htmlFor="description">Description*</Label>
<Textarea {...field} id="description" />
</>
</FormControl>
<FormMessage className="text-red-500" />
</FormItem>
)}
/>
<div className="flex gap-5">
<FormField
control={form.control}
name="numberPlayerMin"
render={({ field }) => (
<FormItem>
<FormControl>
<>
<Label htmlFor="numberPlayerMin">
Nombre mininum de participants*
</Label>
<Input {...field} type="number" id="numberPlayerMin" />
</>
</FormControl>
<FormMessage className="text-red-500" />
</FormItem>
)}
/>
<FormField
control={form.control}
name="numberPlayerMax"
render={({ field }) => (
<FormItem>
<FormControl>
<>
<Label htmlFor="numberPlayerMax">
Nombre maximum de participants*
</Label>
<Input {...field} type="number" id="numberPlayerMax" />
</>
</FormControl>
<FormMessage className="text-red-500" />
</FormItem>
)}
/>
</div>
</form>
)}

{form.formState.isSubmitSuccessful && formStatus !== "errored" && (
<div className="mt-[10vh]">
<h2 className="mb-8 text-4xl font-bold">
Votre parcours a bien été enregistré !
Votre évènement a bien été enregistré !
</h2>
<p className="mb-6">
Lorem, ipsum dolor sit amet consectetur adipisicing elit. Quasi
Expand All @@ -127,7 +191,8 @@ const EventForm = () => {
Oups, une erreur est survenue...
</h2>
<p className="mb-6">
Votre parcours n'a pas pu être créé, veuillez réessayer plus tard.
Votre évènement n'a pas pu être créé, veuillez réessayer plus
tard.
</p>
<Button type="button" onClick={dismissModal}>
<span>Fermer</span>
Expand Down
23 changes: 0 additions & 23 deletions src/components/form/event/steps/EventFormSteps.tsx

This file was deleted.

Loading

0 comments on commit 76a1efc

Please sign in to comment.