✨ Nouvelles fonctionnalités - Page de gestion des avenants (/staff/avenants) - Page de détail d'un avenant (/staff/avenants/[id]) - Création d'avenants (objet, durée, rémunération) - Génération automatique de PDF d'avenant - Signature électronique via DocuSeal (employeur puis salarié) - Changement manuel du statut d'un avenant - Suppression d'avenants 🔧 Routes API - POST /api/staff/amendments/create - Créer un avenant - POST /api/staff/amendments/generate-pdf - Générer le PDF - POST /api/staff/amendments/[id]/send-signature - Envoyer en signature - POST /api/staff/amendments/[id]/change-status - Changer le statut - POST /api/webhooks/docuseal-amendment - Webhook après signature employeur - GET /api/signatures-electroniques/avenants - Liste des avenants en signature 📧 Système email universel v2 - Migration vers le système universel v2 pour les emails d'avenants - Template 'signature-request-employee-amendment' pour salariés - Insertion automatique dans DynamoDB pour la Lambda - Mise à jour automatique du statut dans Supabase 🗄️ Base de données - Table 'avenants' avec tous les champs (objet, durée, rémunération) - Colonnes de notification (last_employer_notification_at, last_employee_notification_at) - Liaison avec cddu_contracts 🎨 Composants - AvenantDetailPageClient - Détail complet d'un avenant - ChangeStatusModal - Changement de statut manuel - SendSignatureModal - Envoi en signature - DeleteAvenantModal - Suppression avec confirmation - AvenantSuccessModal - Confirmation de création 📚 Documentation - AVENANT_EMAIL_SYSTEM_MIGRATION.md - Guide complet de migration 🐛 Corrections - Fix parsing défensif dans Lambda AWS - Fix récupération des données depuis DynamoDB - Fix statut MFA !== 'verified' au lieu de === 'unverified'
93 lines
2.5 KiB
TypeScript
93 lines
2.5 KiB
TypeScript
// app/(app)/staff/avenants/page.tsx
|
|
import { createSbServer } from "@/lib/supabaseServer";
|
|
import NextDynamic from "next/dynamic";
|
|
|
|
export const dynamic = "force-dynamic";
|
|
|
|
const StaffAvenantsPageClient = NextDynamic<any>(
|
|
() => import("../../../../components/staff/StaffAvenantsPageClient"),
|
|
{ ssr: false }
|
|
);
|
|
|
|
export default async function StaffAvenantsPage() {
|
|
const sb = createSbServer();
|
|
const { data: { user } } = await sb.auth.getUser();
|
|
|
|
if (!user) {
|
|
return (
|
|
<main className="p-6">
|
|
<h1 className="text-lg font-semibold">Accès refusé</h1>
|
|
<p className="text-sm text-slate-600">Vous devez être connecté.</p>
|
|
</main>
|
|
);
|
|
}
|
|
|
|
const { data: me } = await sb
|
|
.from("staff_users")
|
|
.select("is_staff")
|
|
.eq("user_id", user.id)
|
|
.maybeSingle();
|
|
|
|
const isStaff = !!me?.is_staff;
|
|
if (!isStaff) {
|
|
return (
|
|
<main className="p-6">
|
|
<h1 className="text-lg font-semibold">Accès refusé</h1>
|
|
<p className="text-sm text-slate-600">Cette page est réservée au Staff.</p>
|
|
</main>
|
|
);
|
|
}
|
|
|
|
// Récupérer les avenants depuis la base de données
|
|
const { data: avenants, error } = await sb
|
|
.from("avenants")
|
|
.select(`
|
|
id,
|
|
numero_avenant,
|
|
date_avenant,
|
|
date_effet,
|
|
type_avenant,
|
|
motif_avenant,
|
|
elements_avenantes,
|
|
statut,
|
|
contract_id,
|
|
created_at,
|
|
signature_status,
|
|
last_employer_notification_at,
|
|
last_employee_notification_at,
|
|
cddu_contracts!inner(
|
|
contract_number,
|
|
employee_name,
|
|
structure,
|
|
org_id
|
|
)
|
|
`)
|
|
.order("created_at", { ascending: false });
|
|
|
|
if (error) {
|
|
console.error("Erreur récupération avenants:", error);
|
|
}
|
|
|
|
// Transformer les données pour le format attendu
|
|
const formattedAvenants = (avenants || []).map((a: any) => ({
|
|
id: a.id,
|
|
contract_id: a.contract_id,
|
|
contract_number: a.cddu_contracts?.contract_number,
|
|
employee_name: a.cddu_contracts?.employee_name,
|
|
organization_name: a.cddu_contracts?.structure,
|
|
date_effet: a.date_effet,
|
|
date_signature: a.date_avenant,
|
|
status: a.statut,
|
|
elements: a.elements_avenantes || [],
|
|
created_at: a.created_at,
|
|
signature_status: a.signature_status,
|
|
last_employer_notification_at: a.last_employer_notification_at,
|
|
last_employee_notification_at: a.last_employee_notification_at,
|
|
}));
|
|
|
|
return (
|
|
<main className="p-6">
|
|
<StaffAvenantsPageClient initialData={formattedAvenants} />
|
|
</main>
|
|
);
|
|
}
|