feat: add CRUD forms with Server Actions for establishments, users, classes, students

This commit is contained in:
root
2026-06-06 20:08:17 +00:00
parent 0a73a70820
commit a1883080d3
26 changed files with 1206 additions and 16 deletions
@@ -0,0 +1,89 @@
"use client";
import { useState } from "react";
import { useRouter } from "next/navigation";
import { Card, CardContent } from "@/components/ui/card";
import { Input } from "@/components/ui/input";
import { Select } from "@/components/ui/select";
import { Button } from "@/components/ui/button";
import { createUser } from "../actions";
export default function NewUserForm({
establishments,
isSuperadmin,
}: {
establishments: any[];
isSuperadmin: boolean;
}) {
const [error, setError] = useState<string | null>(null);
const [loading, setLoading] = useState(false);
const router = useRouter();
async function handleSubmit(formData: FormData) {
setLoading(true);
setError(null);
try {
await createUser(formData);
} catch (err: any) {
setError(err.message || "Une erreur est survenue");
setLoading(false);
}
}
return (
<Card>
<CardContent className="pt-6">
<form action={handleSubmit} className="space-y-4">
{error && (
<div className="rounded-md bg-destructive/10 p-3 text-sm text-destructive">
{error}
</div>
)}
<div>
<label className="block text-sm font-medium mb-1">Email</label>
<Input type="email" name="email" required />
</div>
<div>
<label className="block text-sm font-medium mb-1">Mot de passe</label>
<Input type="password" name="password" minLength={8} required />
<p className="text-xs text-muted-foreground mt-1">Minimum 8 caractères</p>
</div>
<div>
<label className="block text-sm font-medium mb-1">Rôle</label>
<Select name="role" required>
<option value="">Choisir un rôle</option>
<option value="admin">Admin</option>
<option value="teacher">Teacher</option>
</Select>
</div>
{isSuperadmin && (
<div>
<label className="block text-sm font-medium mb-1">Établissement</label>
<Select name="establishmentId">
<option value="">Aucun</option>
{establishments.map((e) => (
<option key={e.id} value={e.id}>
{e.name}
</option>
))}
</Select>
</div>
)}
<div className="flex gap-2 pt-2">
<Button type="submit" disabled={loading}>
{loading ? "Création..." : "Créer"}
</Button>
<Button
type="button"
variant="outline"
onClick={() => router.push("/dashboard/users")}
disabled={loading}
>
Annuler
</Button>
</div>
</form>
</CardContent>
</Card>
);
}
+26
View File
@@ -0,0 +1,26 @@
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/lib/auth-config";
import { redirect } from "next/navigation";
import { prisma } from "@/lib/prisma";
import NewUserForm from "./NewUserForm";
export const dynamic = "force-dynamic";
export default async function NewUserPage() {
const session = await getServerSession(authOptions);
if (!session?.user) redirect("/login");
const isSuperadmin = session.user.role === "superadmin";
if (!isSuperadmin && session.user.role !== "admin") redirect("/dashboard");
const establishments = isSuperadmin
? await prisma.establishment.findMany({ orderBy: { name: "asc" } })
: [];
return (
<div className="space-y-6 max-w-xl">
<h1 className="text-3xl font-bold">Nouvel utilisateur</h1>
<NewUserForm establishments={establishments} isSuperadmin={isSuperadmin} />
</div>
);
}