feat: Initialize core application structure including authentication, role-based dashboards, service task management, and integration workflows.

This commit is contained in:
jeremy bayse
2026-02-16 09:30:23 +01:00
commit af060a8847
208 changed files with 26822 additions and 0 deletions

View File

@@ -0,0 +1,103 @@
<script setup>
import AuthenticatedLayout from '@/Layouts/AuthenticatedLayout.vue';
import { Head, useForm } from '@inertiajs/vue3';
import InputError from '@/Components/InputError.vue';
import InputLabel from '@/Components/InputLabel.vue';
import PrimaryButton from '@/Components/PrimaryButton.vue';
import TextInput from '@/Components/TextInput.vue';
const props = defineProps({
templates: Array,
});
const form = useForm({
first_name: '',
last_name: '',
email: '',
position: '',
department: '',
arrival_date: '',
template_id: props.templates.length > 0 ? props.templates[0].id : '',
});
const submit = () => {
form.post(route('integrations.store'));
};
</script>
<template>
<Head title="Nouvelle Intégration" />
<AuthenticatedLayout>
<template #header>
<h2 class="text-xl font-semibold leading-tight text-gray-800 dark:text-gray-200">
Nouvelle fiche agent
</h2>
</template>
<div class="py-12">
<div class="mx-auto max-w-7xl sm:px-6 lg:px-8">
<div class="bg-white overflow-hidden shadow-sm sm:rounded-lg dark:bg-gray-800 p-8 max-w-2xl mx-auto">
<form @submit.prevent="submit" class="space-y-6">
<div class="grid grid-cols-2 gap-4">
<div>
<InputLabel for="first_name" value="Prénom" />
<TextInput id="first_name" type="text" class="mt-1 block w-full" v-model="form.first_name" required autofocus />
<InputError class="mt-2" :message="form.errors.first_name" />
</div>
<div>
<InputLabel for="last_name" value="Nom" />
<TextInput id="last_name" type="text" class="mt-1 block w-full" v-model="form.last_name" required />
<InputError class="mt-2" :message="form.errors.last_name" />
</div>
</div>
<div>
<InputLabel for="email" value="Email" />
<TextInput id="email" type="email" class="mt-1 block w-full" v-model="form.email" required />
<InputError class="mt-2" :message="form.errors.email" />
</div>
<div class="grid grid-cols-2 gap-4">
<div>
<InputLabel for="position" value="Poste" />
<TextInput id="position" type="text" class="mt-1 block w-full" v-model="form.position" required />
<InputError class="mt-2" :message="form.errors.position" />
</div>
<div>
<InputLabel for="department" value="Service de destination" />
<TextInput id="department" type="text" class="mt-1 block w-full" v-model="form.department" required />
<InputError class="mt-2" :message="form.errors.department" />
</div>
</div>
<div>
<InputLabel for="arrival_date" value="Date d'arrivée" />
<TextInput id="arrival_date" type="date" class="mt-1 block w-full" v-model="form.arrival_date" required />
<InputError class="mt-2" :message="form.errors.arrival_date" />
</div>
<div class="warning-box bg-red-500 p-4 rounded-md text-center">
<span class="text-white">Attention, si vous selectionner l'option <b><u>AVEC TELEPHONE PORTABLE</u></b> une validation de la direction générale est nécessaire.</span>
<br>
<span class="text-white"><b><u><i>C'est à vous d'initier celle-ci auprès de votre Direction Générale Adjointe</i></u></b></span>
</div>
<div>
<InputLabel for="template_id" value="Template de parcours" />
<select id="template_id" v-model="form.template_id" class="mt-1 block w-full border-gray-300 dark:border-gray-700 dark:bg-gray-900 dark:text-gray-300 focus:border-indigo-500 dark:focus:border-indigo-600 focus:ring-indigo-500 dark:focus:ring-indigo-600 rounded-md shadow-sm">
<option value="">Aucun (Pas de tâches)</option>
<option v-for="t in templates" :key="t.id" :value="t.id">{{ t.name }}</option>
</select>
<InputError class="mt-2" :message="form.errors.template_id" />
</div>
<div class="flex items-center justify-end mt-4">
<PrimaryButton class="ml-4" :class="{ 'opacity-25': form.processing }" :disabled="form.processing">
Créer la fiche agent
</PrimaryButton>
</div>
</form>
</div>
</div>
</div>
</AuthenticatedLayout>
</template>

View File

@@ -0,0 +1,66 @@
<script setup>
import AuthenticatedLayout from '@/Layouts/AuthenticatedLayout.vue';
import { Head, Link } from '@inertiajs/vue3';
import StatusBadge from '@/Components/App/StatusBadge.vue';
defineProps({
integrations: Array,
});
</script>
<template>
<Head title="Intégrations" />
<AuthenticatedLayout>
<template #header>
<div class="flex justify-between items-center">
<h2 class="text-xl font-semibold leading-tight text-gray-800 dark:text-gray-200">
Processus d'intégration
</h2>
<Link :href="route('integrations.create')" class="inline-flex items-center px-4 py-2 bg-blue-600 border border-transparent rounded-md font-semibold text-xs text-white uppercase tracking-widest hover:bg-blue-700 focus:bg-blue-700 active:bg-blue-900 focus:outline-none focus:ring-2 focus:ring-indigo-500 focus:ring-offset-2 transition ease-in-out duration-150">
Déclencher une arrivée
</Link>
</div>
</template>
<div class="py-12">
<div class="mx-auto max-w-7xl sm:px-6 lg:px-8">
<div class="bg-white overflow-hidden shadow-sm sm:rounded-lg dark:bg-gray-800">
<div class="overflow-x-auto">
<table class="w-full text-left border-collapse">
<thead>
<tr class="border-b dark:border-gray-700 bg-gray-50 dark:bg-gray-700/50">
<th class="px-6 py-3 text-xs font-bold uppercase text-gray-500 dark:text-gray-400">Agent</th>
<th class="px-6 py-3 text-xs font-bold uppercase text-gray-500 dark:text-gray-400">Status</th>
<th class="px-6 py-3 text-xs font-bold uppercase text-gray-500 dark:text-gray-400">Date d'arrivée</th>
<th class="px-6 py-3 text-xs font-bold uppercase text-gray-500 dark:text-gray-400">Template</th>
<th class="px-6 py-3 text-xs font-bold uppercase text-gray-500 dark:text-gray-400">Actions</th>
</tr>
</thead>
<tbody class="divide-y divide-gray-200 dark:divide-gray-700">
<tr v-for="integration in integrations" :key="integration.id" class="hover:bg-gray-50 dark:hover:bg-gray-700/30">
<td class="px-6 py-4">
<div class="text-sm font-medium text-gray-900 dark:text-white">{{ integration.agent.first_name }} {{ integration.agent.last_name }}</div>
<div class="text-xs text-gray-500 dark:text-gray-400">{{ integration.agent.position }}</div>
</td>
<td class="px-6 py-4">
<StatusBadge :status="integration.status" />
</td>
<td class="px-6 py-4 text-sm text-gray-500 dark:text-gray-400">
{{ new Date(integration.agent.arrival_date).toLocaleDateString() }}
</td>
<td class="px-6 py-4 text-sm text-gray-500 dark:text-gray-400">
{{ integration.template?.name || 'Standard' }}
</td>
<td class="px-6 py-4 text-sm">
<Link :href="route('integrations.show', integration.id)" class="text-blue-600 hover:text-blue-900 dark:text-blue-400 dark:hover:text-blue-300">Gérer</Link>
</td>
</tr>
</tbody>
</table>
</div>
</div>
</div>
</div>
</AuthenticatedLayout>
</template>

View File

@@ -0,0 +1,86 @@
<script setup>
import AuthenticatedLayout from '@/Layouts/AuthenticatedLayout.vue';
import { Head, Link } from '@inertiajs/vue3';
import StatusBadge from '@/Components/App/StatusBadge.vue';
import ServiceTaskCard from '@/Components/App/ServiceTaskCard.vue';
import ActivityTimeline from '@/Components/App/ActivityTimeline.vue';
import { router } from '@inertiajs/vue3';
const props = defineProps({
integration: {
type: Object,
required: true,
},
activities: {
type: Array,
default: () => [],
},
});
const validateRH = () => {
if (confirm('Souhaitez-vous valider ce dossier et notifier les services ?')) {
router.post(route('integrations.validate-rh', props.integration.id));
}
};
</script>
<template>
<Head title="Détails Intégration" />
<AuthenticatedLayout>
<template #header>
<div class="flex items-center justify-between">
<h2 class="text-xl font-semibold leading-tight text-gray-800 dark:text-gray-200">
Integration: {{ integration.agent.first_name }} {{ integration.agent.last_name }}
</h2>
<div class="flex items-center space-x-4">
<button
v-if="integration.status === 'pending_rh_validation' || integration.status === 'draft'"
@click="validateRH"
class="px-4 py-2 bg-green-600 text-white rounded-md text-sm font-bold hover:bg-green-700"
>
Valider Dossier RH
</button>
<StatusBadge :status="integration.status" />
</div>
</div>
</template>
<div class="py-12">
<div class="mx-auto max-w-7xl sm:px-6 lg:px-8">
<div class="grid grid-cols-1 gap-6 md:grid-cols-3 items-start">
<!-- Right/Side Column: Agent Info & Activity -->
<div class="md:col-span-1 space-y-6">
<!-- Agent Info -->
<div class="overflow-hidden bg-white shadow-sm sm:rounded-lg dark:bg-gray-800 p-6 border border-gray-200 dark:border-gray-700">
<h3 class="text-lg font-medium text-gray-900 dark:text-gray-100 mb-4">Informations Agent</h3>
<dl class="space-y-4">
<div>
<dt class="text-sm font-medium text-gray-500 dark:text-gray-400">Poste</dt>
<dd class="mt-1 text-sm text-gray-900 dark:text-gray-100">{{ integration.agent.position }}</dd>
</div>
<div>
<dt class="text-sm font-medium text-gray-500 dark:text-gray-400">Service</dt>
<dd class="mt-1 text-sm text-gray-900 dark:text-gray-100">{{ integration.agent.department }}</dd>
</div>
<div>
<dt class="text-sm font-medium text-gray-500 dark:text-gray-400">Date d'arrivée</dt>
<dd class="mt-1 text-sm text-gray-900 dark:text-gray-100">{{ new Date(integration.agent.arrival_date).toLocaleDateString() }}</dd>
</div>
</dl>
</div>
<ActivityTimeline :activities="activities" />
</div>
<!-- Main Column: Tasks -->
<div class="md:col-span-2 space-y-6">
<div v-for="task in integration.service_tasks" :key="task.id">
<ServiceTaskCard :task="task" />
</div>
</div>
</div>
</div>
</div>
</AuthenticatedLayout>
</template>