Enhance candidates management: add city field, drag-and-drop ranking persistence, AI analysis popover, and CV preview
This commit is contained in:
302
resources/js/Pages/Admin/Candidates/Selected.vue
Normal file
302
resources/js/Pages/Admin/Candidates/Selected.vue
Normal file
@@ -0,0 +1,302 @@
|
||||
<script setup>
|
||||
import AdminLayout from '@/Layouts/AdminLayout.vue';
|
||||
import { Head, Link, router } from '@inertiajs/vue3';
|
||||
import { ref, computed, watch } from 'vue';
|
||||
import draggable from 'vuedraggable';
|
||||
import Modal from '@/Components/Modal.vue';
|
||||
import SecondaryButton from '@/Components/SecondaryButton.vue';
|
||||
|
||||
const props = defineProps({
|
||||
candidates: Array
|
||||
});
|
||||
|
||||
const searchQuery = ref('');
|
||||
|
||||
// Maintain local state for dragging
|
||||
const localCandidates = ref([]);
|
||||
|
||||
// We initialize sorting based on the server data
|
||||
watch(() => props.candidates, (newVal) => {
|
||||
localCandidates.value = [...newVal];
|
||||
}, { immediate: true });
|
||||
|
||||
const filteredCandidates = computed({
|
||||
get() {
|
||||
const query = searchQuery.value.toLowerCase();
|
||||
if (!query) return localCandidates.value;
|
||||
|
||||
return localCandidates.value.filter(c =>
|
||||
c.user.name.toLowerCase().includes(query) ||
|
||||
c.user.email.toLowerCase().includes(query) ||
|
||||
(c.job_position && c.job_position.title.toLowerCase().includes(query))
|
||||
);
|
||||
},
|
||||
set(val) {
|
||||
if (!searchQuery.value) {
|
||||
localCandidates.value = val;
|
||||
saveOrder();
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
const saveOrder = () => {
|
||||
router.post(route('admin.candidates.update-order'), {
|
||||
ids: localCandidates.value.map(c => c.id)
|
||||
}, {
|
||||
preserveScroll: true,
|
||||
onSuccess: () => {
|
||||
// Optional: Show a toast or notification
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const hoveredCandidateId = ref(null);
|
||||
const popoverPosition = ref({ top: 0, left: 0 });
|
||||
|
||||
const handleMouseEnter = (event, id) => {
|
||||
hoveredCandidateId.value = id;
|
||||
const rect = event.currentTarget.getBoundingClientRect();
|
||||
popoverPosition.value = {
|
||||
top: rect.bottom + window.scrollY + 10,
|
||||
left: rect.left + window.scrollX
|
||||
};
|
||||
};
|
||||
|
||||
const handleMouseLeave = () => {
|
||||
hoveredCandidateId.value = null;
|
||||
};
|
||||
|
||||
const selectedDocument = ref(null);
|
||||
const openCvPreview = (candidate) => {
|
||||
const cv = (candidate.documents || []).find(d => d.type === 'cv');
|
||||
if (cv) {
|
||||
selectedDocument.value = cv;
|
||||
} else {
|
||||
alert("Aucun CV n'a été trouvé pour ce candidat.");
|
||||
}
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Head title="Candidats Sélectionnés" />
|
||||
|
||||
<AdminLayout>
|
||||
<template #header>
|
||||
Candidats Sélectionnés (Comparateur)
|
||||
</template>
|
||||
|
||||
<div class="mb-8 flex flex-col sm:flex-row justify-between items-start sm:items-center gap-4">
|
||||
<div>
|
||||
<h3 class="text-2xl font-bold">Listes de Candidats retenus</h3>
|
||||
<p class="text-sm text-slate-500 mt-1">
|
||||
Faites glisser les candidats sur la poignée (⋮⋮) pour modifier manuellement le classement.
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div class="relative w-full sm:w-64">
|
||||
<span class="absolute inset-y-0 left-0 pl-3 flex items-center text-slate-400">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z" />
|
||||
</svg>
|
||||
</span>
|
||||
<input
|
||||
v-model="searchQuery"
|
||||
type="text"
|
||||
placeholder="Rechercher..."
|
||||
class="block w-full pl-10 pr-3 py-2 border border-slate-300 dark:border-slate-700 rounded-lg leading-5 bg-white dark:bg-slate-800 placeholder-slate-500 focus:outline-none focus:ring-1 focus:ring-indigo-500 focus:border-indigo-500 sm:text-sm transition-all"
|
||||
>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="bg-white dark:bg-slate-800 rounded-2xl shadow-sm border border-slate-200 dark:border-slate-700 overflow-hidden text-sm">
|
||||
<table class="w-full text-left">
|
||||
<thead class="bg-slate-50 dark:bg-slate-700/50 border-b border-slate-200 dark:border-slate-700">
|
||||
<tr>
|
||||
<th class="px-4 py-3 font-bold text-slate-700 dark:text-slate-300 uppercase tracking-wider text-xs">Ordre</th>
|
||||
<th class="px-4 py-3 font-bold text-slate-700 dark:text-slate-300 uppercase tracking-wider text-xs">Candidat</th>
|
||||
<th class="px-4 py-3 font-bold text-slate-700 dark:text-slate-300 uppercase tracking-wider text-xs">Poste</th>
|
||||
<th class="px-4 py-3 font-bold text-slate-700 dark:text-slate-300 uppercase tracking-wider text-xs text-center">Ville</th>
|
||||
<th class="px-4 py-3 font-bold text-slate-700 dark:text-slate-300 uppercase tracking-wider text-xs text-center">Score Global (/20)</th>
|
||||
<th class="px-4 py-3 font-bold text-slate-700 dark:text-slate-300 uppercase tracking-wider text-xs text-center">IA Match</th>
|
||||
<th class="px-4 py-3 font-bold text-slate-700 dark:text-slate-300 uppercase tracking-wider text-xs text-center">CV (/20)</th>
|
||||
<th class="px-4 py-3 font-bold text-slate-700 dark:text-slate-300 uppercase tracking-wider text-xs text-center">Motiv (/10)</th>
|
||||
<th class="px-4 py-3 font-bold text-slate-700 dark:text-slate-300 uppercase tracking-wider text-xs text-center">Test QCM (/20)</th>
|
||||
<th class="px-4 py-3 font-bold text-slate-700 dark:text-slate-300 uppercase tracking-wider text-xs text-center">Entretien (/30)</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<draggable
|
||||
v-model="filteredCandidates"
|
||||
tag="tbody"
|
||||
item-key="id"
|
||||
class="divide-y divide-slate-200 dark:divide-slate-700"
|
||||
handle=".drag-handle"
|
||||
:animation="200"
|
||||
:disabled="searchQuery.length > 0"
|
||||
>
|
||||
<template #item="{ element: candidate, index }">
|
||||
<tr class="hover:bg-slate-50/50 dark:hover:bg-slate-700/30 transition-colors group bg-white dark:bg-slate-800">
|
||||
<td class="px-4 py-3">
|
||||
<div class="flex items-center gap-3">
|
||||
<div
|
||||
class="drag-handle text-slate-400 transition-colors"
|
||||
:class="searchQuery.length > 0 ? 'opacity-30 cursor-not-allowed' : 'cursor-grab active:cursor-grabbing hover:text-slate-600'"
|
||||
:title="searchQuery.length > 0 ? 'Désactivé pendant la recherche' : 'Glisser pour réordonner'"
|
||||
>
|
||||
<svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||
<circle cx="9" cy="12" r="1"/><circle cx="9" cy="5" r="1"/><circle cx="9" cy="19" r="1"/>
|
||||
<circle cx="15" cy="12" r="1"/><circle cx="15" cy="5" r="1"/><circle cx="15" cy="19" r="1"/>
|
||||
</svg>
|
||||
</div>
|
||||
<div
|
||||
class="w-8 h-8 rounded-full flex items-center justify-center font-bold text-sm"
|
||||
:class="[
|
||||
index === 0 && !searchQuery ? 'bg-yellow-100 text-yellow-700 shadow-sm' :
|
||||
index === 1 && !searchQuery ? 'bg-slate-200 text-slate-700 shadow-sm' :
|
||||
index === 2 && !searchQuery ? 'bg-orange-100 text-orange-700 shadow-sm' : 'bg-slate-50 text-slate-500'
|
||||
]"
|
||||
>
|
||||
{{ index + 1 }}
|
||||
</div>
|
||||
</div>
|
||||
</td>
|
||||
<td class="px-4 py-3 relative">
|
||||
<div class="flex items-center gap-2">
|
||||
<div
|
||||
class="font-bold text-slate-900 dark:text-slate-100"
|
||||
@mouseenter="handleMouseEnter($event, candidate.id)"
|
||||
@mouseleave="handleMouseLeave"
|
||||
>
|
||||
<Link :href="route('admin.candidates.show', candidate.id)" class="hover:text-indigo-600 transition-colors">
|
||||
{{ candidate.user.name }}
|
||||
</Link>
|
||||
</div>
|
||||
<button
|
||||
v-if="(candidate.documents || []).some(d => d.type === 'cv')"
|
||||
@click="openCvPreview(candidate)"
|
||||
class="p-1 text-slate-400 hover:text-indigo-600 transition-colors"
|
||||
title="Voir le CV"
|
||||
>
|
||||
<svg xmlns="http://www.w3.org/2000/svg" class="h-4 w-4" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z" />
|
||||
</svg>
|
||||
</button>
|
||||
</div>
|
||||
<div class="text-xs text-slate-500">{{ candidate.user.email }}</div>
|
||||
|
||||
<!-- AI Analysis Popover -->
|
||||
<div
|
||||
v-if="hoveredCandidateId === candidate.id && candidate.ai_analysis"
|
||||
class="fixed z-[100] w-80 bg-white dark:bg-slate-900 border border-slate-200 dark:border-slate-700 rounded-xl shadow-2xl p-4 pointer-events-none transition-all animate-in fade-in zoom-in-95 duration-200"
|
||||
:style="{ top: popoverPosition.top + 'px', left: popoverPosition.left + 'px' }"
|
||||
>
|
||||
<div class="flex items-center justify-between mb-3">
|
||||
<span class="text-[10px] font-black uppercase tracking-widest text-slate-400">Synthèse IA</span>
|
||||
<div
|
||||
class="px-2 py-0.5 rounded-lg text-[10px] font-black"
|
||||
:class="[
|
||||
candidate.ai_analysis.match_score >= 90 ? 'bg-emerald-50 text-emerald-700' :
|
||||
candidate.ai_analysis.match_score >= 80 ? 'bg-amber-50 text-amber-700' :
|
||||
'bg-rose-50 text-rose-700'
|
||||
]"
|
||||
>
|
||||
{{ candidate.ai_analysis.match_score }}% Match
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<p class="text-xs text-slate-600 dark:text-slate-300 italic mb-4 leading-relaxed">
|
||||
"{{ candidate.ai_analysis.synthese || candidate.ai_analysis.summary }}"
|
||||
</p>
|
||||
|
||||
<div class="grid grid-cols-2 gap-3">
|
||||
<div v-if="candidate.ai_analysis.points_forts || candidate.ai_analysis.strengths">
|
||||
<p class="text-[9px] font-bold uppercase text-emerald-600 mb-1">Forces</p>
|
||||
<ul class="text-[10px] text-slate-500 space-y-0.5">
|
||||
<li v-for="s in (candidate.ai_analysis.points_forts || candidate.ai_analysis.strengths).slice(0,3)" :key="s">• {{ s }}</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div v-if="candidate.ai_analysis.points_faibles || candidate.ai_analysis.gaps">
|
||||
<p class="text-[9px] font-bold uppercase text-rose-600 mb-1">Points d'attention</p>
|
||||
<ul class="text-[10px] text-slate-500 space-y-0.5">
|
||||
<li v-for="g in (candidate.ai_analysis.points_faibles || candidate.ai_analysis.gaps).slice(0,3)" :key="g">• {{ g }}</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div v-if="candidate.ai_analysis.verdict" class="mt-3 pt-3 border-t border-slate-100 dark:border-slate-800">
|
||||
<span class="text-[9px] font-bold uppercase text-slate-400">Verdict final :</span>
|
||||
<span class="ml-2 text-[10px] font-black text-indigo-600">{{ candidate.ai_analysis.verdict }}</span>
|
||||
</div>
|
||||
</div>
|
||||
</td>
|
||||
<td class="px-4 py-3 text-slate-600">
|
||||
{{ candidate.job_position ? candidate.job_position.title : '--' }}
|
||||
</td>
|
||||
<td class="px-4 py-3 text-center text-slate-500 font-medium">
|
||||
{{ candidate.city ?? '--' }}
|
||||
</td>
|
||||
<td class="px-4 py-3">
|
||||
<div class="flex flex-col items-center">
|
||||
<span class="text-lg font-black text-indigo-600 dark:text-indigo-400">{{ candidate.weighted_score }}</span>
|
||||
</div>
|
||||
</td>
|
||||
<td class="px-4 py-3">
|
||||
<div v-if="candidate.ai_analysis" class="flex flex-col items-center">
|
||||
<div
|
||||
class="px-2 py-0.5 rounded-lg text-[10px] font-black shadow-sm"
|
||||
:class="[
|
||||
candidate.ai_analysis.match_score >= 90 ? 'bg-emerald-50 text-emerald-700 border border-emerald-200' :
|
||||
candidate.ai_analysis.match_score >= 80 ? 'bg-highlight/10 text-[#3a2800] border border-highlight/30' :
|
||||
'bg-accent/10 text-accent border border-accent/20'
|
||||
]"
|
||||
>
|
||||
{{ candidate.ai_analysis.match_score }}%
|
||||
</div>
|
||||
</div>
|
||||
<div v-else class="text-center text-[10px] text-slate-300 italic">--</div>
|
||||
</td>
|
||||
<td class="px-4 py-3 text-center font-medium text-slate-600">
|
||||
{{ candidate.cv_score ?? '--' }}
|
||||
</td>
|
||||
<td class="px-4 py-3 text-center font-medium text-slate-600">
|
||||
{{ candidate.motivation_score ?? '--' }}
|
||||
</td>
|
||||
<td class="px-4 py-3 text-center font-medium text-slate-600">
|
||||
{{ (() => {
|
||||
const bestAttempt = (candidate.attempts || []).filter(a => a.finished_at).map(a => a.max_score > 0 ? (a.score / a.max_score) * 20 : 0);
|
||||
return bestAttempt.length ? Math.max(...bestAttempt).toFixed(1) : '--';
|
||||
})() }}
|
||||
</td>
|
||||
<td class="px-4 py-3 text-center font-medium text-slate-600">
|
||||
{{ candidate.interview_score ?? '--' }}
|
||||
</td>
|
||||
</tr>
|
||||
</template>
|
||||
<template #footer>
|
||||
<tr v-if="filteredCandidates.length === 0">
|
||||
<td colspan="10" class="px-6 py-12 text-center text-slate-500 italic">
|
||||
Aucun candidat sélectionné ne correspond.
|
||||
</td>
|
||||
</tr>
|
||||
</template>
|
||||
</draggable>
|
||||
</table>
|
||||
</div>
|
||||
|
||||
<!-- Document Preview Modal -->
|
||||
<Modal :show="!!selectedDocument" @close="selectedDocument = null" max-width="4xl">
|
||||
<div class="p-6 h-[80vh] flex flex-col">
|
||||
<div class="flex justify-between items-center mb-4">
|
||||
<h3 class="text-xl font-bold">Aperçu : {{ selectedDocument.original_name }}</h3>
|
||||
<SecondaryButton @click="selectedDocument = null">Fermer</SecondaryButton>
|
||||
</div>
|
||||
<div class="flex-1 bg-slate-100 dark:bg-slate-900 rounded-lg overflow-hidden border border-slate-200 dark:border-slate-700">
|
||||
<iframe
|
||||
v-if="selectedDocument"
|
||||
:src="route('admin.documents.show', selectedDocument.id)"
|
||||
class="w-full h-full"
|
||||
></iframe>
|
||||
</div>
|
||||
</div>
|
||||
</Modal>
|
||||
</AdminLayout>
|
||||
</template>
|
||||
Reference in New Issue
Block a user