chore: update frontend configuration

This commit is contained in:
Matthieu
2025-09-26 11:29:47 +02:00
parent b7caa4f552
commit a78938a4d1
64 changed files with 5790 additions and 5129 deletions

View File

@@ -13,8 +13,8 @@
> >
<li class="pt-1 pb-2 lg:hidden"> <li class="pt-1 pb-2 lg:hidden">
<button <button
@click="openDisplaySettings"
class="w-full flex items-center gap-2 rounded-md px-2 py-1 transition-colors text-base-content hover:bg-primary/10 hover:text-primary" class="w-full flex items-center gap-2 rounded-md px-2 py-1 transition-colors text-base-content hover:bg-primary/10 hover:text-primary"
@click="openDisplaySettings"
> >
<IconLucideSettings class="w-4 h-4" aria-hidden="true" /> <IconLucideSettings class="w-4 h-4" aria-hidden="true" />
Paramètres d'affichage Paramètres d'affichage
@@ -125,7 +125,7 @@
class="rounded-md px-2 py-1 transition-colors cursor-pointer" class="rounded-md px-2 py-1 transition-colors cursor-pointer"
:class=" :class="
isActive('/component-catalog') || isActive('/component-catalog') ||
isActive('/component-category') isActive('/component-category')
? 'bg-primary text-primary-content font-semibold shadow-sm' ? 'bg-primary text-primary-content font-semibold shadow-sm'
: 'text-base-content hover:bg-primary/10 hover:text-primary' : 'text-base-content hover:bg-primary/10 hover:text-primary'
" "
@@ -178,8 +178,8 @@
class="rounded-md px-2 py-1 transition-colors cursor-pointer" class="rounded-md px-2 py-1 transition-colors cursor-pointer"
:class=" :class="
isActive('/sites') || isActive('/sites') ||
isActive('/documents') || isActive('/documents') ||
isActive('/constructeurs') isActive('/constructeurs')
? 'bg-primary text-primary-content font-semibold shadow-sm' ? 'bg-primary text-primary-content font-semibold shadow-sm'
: 'text-base-content hover:bg-primary/10 hover:text-primary' : 'text-base-content hover:bg-primary/10 hover:text-primary'
" "
@@ -241,9 +241,12 @@
<IconLucideBoxes class="w-6 h-6" aria-hidden="true" /> <IconLucideBoxes class="w-6 h-6" aria-hidden="true" />
</div> </div>
</div> </div>
<NuxtLink to="/" class="btn btn-ghost text-xl" <NuxtLink
>Inventaire Pro</NuxtLink to="/"
class="btn btn-ghost text-xl"
> >
Inventaire Pro
</NuxtLink>
</div> </div>
</div> </div>
<div class="navbar-center hidden lg:flex"> <div class="navbar-center hidden lg:flex">
@@ -353,7 +356,7 @@
class="transition-colors px-3 py-2 rounded-md inline-flex items-center gap-1 cursor-pointer" class="transition-colors px-3 py-2 rounded-md inline-flex items-center gap-1 cursor-pointer"
:class=" :class="
isActive('/component-category') || isActive('/component-category') ||
isActive('/component-catalog') isActive('/component-catalog')
? 'bg-primary text-primary-content font-semibold shadow-sm' ? 'bg-primary text-primary-content font-semibold shadow-sm'
: 'text-base-content hover:bg-primary/10 hover:text-primary' : 'text-base-content hover:bg-primary/10 hover:text-primary'
" "
@@ -406,8 +409,8 @@
class="transition-colors px-3 py-2 rounded-md inline-flex items-center gap-1 cursor-pointer" class="transition-colors px-3 py-2 rounded-md inline-flex items-center gap-1 cursor-pointer"
:class=" :class="
isActive('/sites') || isActive('/sites') ||
isActive('/documents') || isActive('/documents') ||
isActive('/constructeurs') isActive('/constructeurs')
? 'bg-primary text-primary-content font-semibold shadow-sm' ? 'bg-primary text-primary-content font-semibold shadow-sm'
: 'text-base-content hover:bg-primary/10 hover:text-primary' : 'text-base-content hover:bg-primary/10 hover:text-primary'
" "
@@ -465,9 +468,9 @@
<div class="flex items-center gap-2"> <div class="flex items-center gap-2">
<!-- Bouton paramètres d'affichage --> <!-- Bouton paramètres d'affichage -->
<button <button
@click="openDisplaySettings"
class="btn btn-ghost btn-circle hidden lg:inline-flex" class="btn btn-ghost btn-circle hidden lg:inline-flex"
title="Paramètres d'affichage" title="Paramètres d'affichage"
@click="openDisplaySettings"
> >
<IconLucideSettings class="w-5 h-5" aria-hidden="true" /> <IconLucideSettings class="w-5 h-5" aria-hidden="true" />
</button> </button>
@@ -507,7 +510,7 @@
</div> </div>
<ClientOnly> <ClientOnly>
<div class="dropdown dropdown-end" v-if="activeProfile"> <div v-if="activeProfile" class="dropdown dropdown-end">
<div <div
tabindex="0" tabindex="0"
role="button" role="button"
@@ -528,12 +531,12 @@
class="menu dropdown-content mt-3 p-2 shadow bg-base-100 rounded-box w-64" class="menu dropdown-content mt-3 p-2 shadow bg-base-100 rounded-box w-64"
> >
<li class="px-2 py-1 text-sm text-base-content/70"> <li class="px-2 py-1 text-sm text-base-content/70">
Connecté en tant que<br /> Connecté en tant que<br>
<span class="font-semibold text-base-content">{{ <span class="font-semibold text-base-content">{{
activeProfileLabel activeProfileLabel
}}</span> }}</span>
</li> </li>
<li><hr class="my-1" /></li> <li><hr class="my-1"></li>
<li> <li>
<NuxtLink to="/profiles/manage" class="justify-between"> <NuxtLink to="/profiles/manage" class="justify-between">
Gestion des profils Gestion des profils
@@ -583,99 +586,99 @@
</template> </template>
<script setup> <script setup>
import { ref, computed, onMounted, onUnmounted } from "vue"; import { ref, computed, onMounted, onUnmounted } from 'vue'
import { useRoute, navigateTo } from "#imports"; import { useRoute, navigateTo } from '#imports'
import { useProfileSession } from "~/composables/useProfileSession"; import { useProfileSession } from '~/composables/useProfileSession'
import IconLucideMenu from "~icons/lucide/menu"; import IconLucideMenu from '~icons/lucide/menu'
import IconLucideSettings from "~icons/lucide/settings"; import IconLucideSettings from '~icons/lucide/settings'
import IconLucideBoxes from "~icons/lucide/boxes"; import IconLucideBoxes from '~icons/lucide/boxes'
import IconLucidePlus from "~icons/lucide/plus"; import IconLucidePlus from '~icons/lucide/plus'
import IconLucideCpu from "~icons/lucide/cpu"; import IconLucideCpu from '~icons/lucide/cpu'
import IconLucideFilePlus from "~icons/lucide/file-plus"; import IconLucideFilePlus from '~icons/lucide/file-plus'
import IconLucideMapPin from "~icons/lucide/map-pin"; import IconLucideMapPin from '~icons/lucide/map-pin'
import IconLucideChevronRight from "~icons/lucide/chevron-right"; import IconLucideChevronRight from '~icons/lucide/chevron-right'
import IconLucideLogOut from "~icons/lucide/log-out"; import IconLucideLogOut from '~icons/lucide/log-out'
// État du modal des paramètres d'affichage // État du modal des paramètres d'affichage
const displaySettingsOpen = ref(false); const displaySettingsOpen = ref(false)
const { activeProfile, ensureSession, logout } = useProfileSession(); const { activeProfile, ensureSession, logout } = useProfileSession()
// Route active pour souligner l'onglet sélectionné dans la navbar // Route active pour souligner l'onglet sélectionné dans la navbar
const route = useRoute(); const route = useRoute()
const isActive = (path) => { const isActive = (path) => {
if (path === "/") { if (path === '/') {
return route.path === "/"; return route.path === '/'
} }
return route.path.startsWith(path); return route.path.startsWith(path)
}; }
// Ouvrir les paramètres d'affichage // Ouvrir les paramètres d'affichage
const openDisplaySettings = () => { const openDisplaySettings = () => {
displaySettingsOpen.value = true; displaySettingsOpen.value = true
}; }
// Fermer les paramètres d'affichage // Fermer les paramètres d'affichage
const closeDisplaySettings = () => { const closeDisplaySettings = () => {
displaySettingsOpen.value = false; displaySettingsOpen.value = false
}; }
// Gérer les mises à jour des paramètres // Gérer les mises à jour des paramètres
const handleSettingsUpdate = (settings) => { const handleSettingsUpdate = (settings) => {
console.log("Paramètres d'affichage mis à jour:", settings); console.log("Paramètres d'affichage mis à jour:", settings)
}; }
const handleLogout = async () => { const handleLogout = async () => {
await logout(); await logout()
await navigateTo("/profiles"); await navigateTo('/profiles')
}; }
const openDropdown = ref(null); const openDropdown = ref(null)
let dropdownCloseTimer = null; let dropdownCloseTimer = null
const setDropdown = (name) => { const setDropdown = (name) => {
if (dropdownCloseTimer) { if (dropdownCloseTimer) {
clearTimeout(dropdownCloseTimer); clearTimeout(dropdownCloseTimer)
dropdownCloseTimer = null; dropdownCloseTimer = null
} }
if (openDropdown.value !== name) { if (openDropdown.value !== name) {
openDropdown.value = name; openDropdown.value = name
} }
}; }
const scheduleDropdownClose = (name) => { const scheduleDropdownClose = (name) => {
if (dropdownCloseTimer) { if (dropdownCloseTimer) {
clearTimeout(dropdownCloseTimer); clearTimeout(dropdownCloseTimer)
} }
dropdownCloseTimer = setTimeout(() => { dropdownCloseTimer = setTimeout(() => {
if (openDropdown.value === name) { if (openDropdown.value === name) {
openDropdown.value = null; openDropdown.value = null
} }
dropdownCloseTimer = null; dropdownCloseTimer = null
}, 200); }, 200)
}; }
const activeProfileLabel = computed(() => { const activeProfileLabel = computed(() => {
if (!activeProfile.value) return "Profil inconnu"; if (!activeProfile.value) { return 'Profil inconnu' }
return `${activeProfile.value.firstName} ${activeProfile.value.lastName}`; return `${activeProfile.value.firstName} ${activeProfile.value.lastName}`
}); })
const activeProfileInitials = computed(() => { const activeProfileInitials = computed(() => {
if (!activeProfile.value) return "??"; if (!activeProfile.value) { return '??' }
const { firstName = "", lastName = "" } = activeProfile.value; const { firstName = '', lastName = '' } = activeProfile.value
return ( return (
`${firstName.charAt(0) || ""}${lastName.charAt(0) || ""}`.toUpperCase() || `${firstName.charAt(0) || ''}${lastName.charAt(0) || ''}`.toUpperCase() ||
"??" '??'
); )
}); })
onMounted(async () => { onMounted(async () => {
await ensureSession(); await ensureSession()
}); })
onUnmounted(() => { onUnmounted(() => {
if (dropdownCloseTimer) { if (dropdownCloseTimer) {
clearTimeout(dropdownCloseTimer); clearTimeout(dropdownCloseTimer)
dropdownCloseTimer = null; dropdownCloseTimer = null
} }
}); })
</script> </script>

View File

@@ -2,7 +2,7 @@
<div class="space-y-4"> <div class="space-y-4">
<!-- Root Components --> <!-- Root Components -->
<div v-for="component in components" :key="component.id" class="border border-gray-200 rounded-lg p-4"> <div v-for="component in components" :key="component.id" class="border border-gray-200 rounded-lg p-4">
<ComponentItem <ComponentItem
:component="component" :component="component"
:is-edit-mode="isEditMode" :is-edit-mode="isEditMode"
:collapse-all="collapseAll" :collapse-all="collapseAll"
@@ -43,13 +43,13 @@ defineProps({
}, },
componentModelOptionsProvider: { componentModelOptionsProvider: {
type: Function, type: Function,
default: () => [], default: () => []
}, },
pieceModelOptionsProvider: { pieceModelOptionsProvider: {
type: Function, type: Function,
default: () => [], default: () => []
}, }
}) })
defineEmits(['update', 'edit-piece', 'assign-model', 'assign-piece-model', 'custom-field-update', 'create-model-from-component']) defineEmits(['update', 'edit-piece', 'assign-model', 'assign-piece-model', 'custom-field-update', 'create-model-from-component'])
</script> </script>

View File

@@ -13,15 +13,17 @@
type="button" type="button"
class="btn btn-ghost btn-sm btn-circle shrink-0 transition-transform" class="btn btn-ghost btn-sm btn-circle shrink-0 transition-transform"
:class="{ 'rotate-90': !isCollapsed }" :class="{ 'rotate-90': !isCollapsed }"
@click="toggleCollapse"
:aria-expanded="!isCollapsed" :aria-expanded="!isCollapsed"
:title="isCollapsed ? 'Déplier les détails du composant' : 'Replier les détails du composant'" :title="isCollapsed ? 'Déplier les détails du composant' : 'Replier les détails du composant'"
@click="toggleCollapse"
> >
<IconLucideChevronRight class="w-5 h-5 transition-transform" aria-hidden="true" /> <IconLucideChevronRight class="w-5 h-5 transition-transform" aria-hidden="true" />
<span class="sr-only">{{ isCollapsed ? 'Déplier' : 'Replier' }} le composant</span> <span class="sr-only">{{ isCollapsed ? 'Déplier' : 'Replier' }} le composant</span>
</button> </button>
<div class="flex-1"> <div class="flex-1">
<h3 class="text-lg font-semibold">{{ component.name }}</h3> <h3 class="text-lg font-semibold">
{{ component.name }}
</h3>
<div class="flex flex-wrap gap-2 mt-2"> <div class="flex flex-wrap gap-2 mt-2">
<span v-if="component.reference" class="badge badge-outline badge-sm">{{ component.reference }}</span> <span v-if="component.reference" class="badge badge-outline badge-sm">{{ component.reference }}</span>
<span v-if="component.constructeur" class="badge badge-outline badge-sm">{{ component.constructeur?.name }}</span> <span v-if="component.constructeur" class="badge badge-outline badge-sm">{{ component.constructeur?.name }}</span>
@@ -55,8 +57,10 @@
type="text" type="text"
class="input input-bordered input-sm" class="input input-bordered input-sm"
@blur="updateComponent" @blur="updateComponent"
/> >
<div v-else class="input input-bordered input-sm bg-base-200">{{ component.name }}</div> <div v-else class="input input-bordered input-sm bg-base-200">
{{ component.name }}
</div>
</div> </div>
<div class="form-control"> <div class="form-control">
<label class="label"><span class="label-text font-medium">Référence</span></label> <label class="label"><span class="label-text font-medium">Référence</span></label>
@@ -66,8 +70,10 @@
type="text" type="text"
class="input input-bordered input-sm" class="input input-bordered input-sm"
@blur="updateComponent" @blur="updateComponent"
/> >
<div v-else class="input input-bordered input-sm bg-base-200">{{ component.reference || 'Non définie' }}</div> <div v-else class="input input-bordered input-sm bg-base-200">
{{ component.reference || 'Non définie' }}
</div>
</div> </div>
<div class="form-control"> <div class="form-control">
<label class="label"><span class="label-text font-medium">Prix</span></label> <label class="label"><span class="label-text font-medium">Prix</span></label>
@@ -78,8 +84,10 @@
step="0.01" step="0.01"
class="input input-bordered input-sm" class="input input-bordered input-sm"
@blur="updateComponent" @blur="updateComponent"
/> >
<div v-else class="input input-bordered input-sm bg-base-200">{{ component.prix ? `${component.prix}` : 'Non défini' }}</div> <div v-else class="input input-bordered input-sm bg-base-200">
{{ component.prix ? `${component.prix}` : 'Non défini' }}
</div>
</div> </div>
<div class="form-control"> <div class="form-control">
<label class="label"><span class="label-text font-medium">Constructeur</span></label> <label class="label"><span class="label-text font-medium">Constructeur</span></label>
@@ -87,7 +95,7 @@
v-if="isEditMode" v-if="isEditMode"
class="w-full" class="w-full"
:model-value="component.constructeurId || component.constructeur?.id || null" :model-value="component.constructeurId || component.constructeur?.id || null"
@update:modelValue="handleConstructeurChange" @update:model-value="handleConstructeurChange"
/> />
<div v-else class="input input-bordered input-sm bg-base-200"> <div v-else class="input input-bordered input-sm bg-base-200">
<div class="flex flex-col"> <div class="flex flex-col">
@@ -117,7 +125,9 @@
class="select select-bordered select-sm" class="select select-bordered select-sm"
@change="assignComponentModel($event.target.value)" @change="assignComponentModel($event.target.value)"
> >
<option value="">Définir manuellement</option> <option value="">
Définir manuellement
</option>
<option <option
v-for="model in componentModelOptionsList" v-for="model in componentModelOptionsList"
:key="model.id" :key="model.id"
@@ -141,7 +151,9 @@
<!-- Custom Fields Display - Editable or Read-only --> <!-- Custom Fields Display - Editable or Read-only -->
<div v-if="component.customFields && component.customFields.length > 0" class="mt-4 pt-4 border-t border-gray-200"> <div v-if="component.customFields && component.customFields.length > 0" class="mt-4 pt-4 border-t border-gray-200">
<h4 class="font-semibold text-sm text-gray-700 mb-3">Champs personnalisés</h4> <h4 class="font-semibold text-sm text-gray-700 mb-3">
Champs personnalisés
</h4>
<div class="grid grid-cols-1 md:grid-cols-2 gap-4"> <div class="grid grid-cols-1 md:grid-cols-2 gap-4">
<div v-for="field in component.customFields" :key="field.id" class="form-control"> <div v-for="field in component.customFields" :key="field.id" class="form-control">
<label class="label"> <label class="label">
@@ -156,7 +168,7 @@
class="input input-bordered input-sm" class="input input-bordered input-sm"
:required="field.required" :required="field.required"
@blur="updateComponentCustomField(field)" @blur="updateComponentCustomField(field)"
/> >
<input <input
v-else-if="field.type === 'number'" v-else-if="field.type === 'number'"
v-model="field.value" v-model="field.value"
@@ -164,7 +176,7 @@
class="input input-bordered input-sm" class="input input-bordered input-sm"
:required="field.required" :required="field.required"
@blur="updateComponentCustomField(field)" @blur="updateComponentCustomField(field)"
/> >
<select <select
v-else-if="field.type === 'select'" v-else-if="field.type === 'select'"
v-model="field.value" v-model="field.value"
@@ -172,8 +184,12 @@
:required="field.required" :required="field.required"
@change="updateComponentCustomField(field)" @change="updateComponentCustomField(field)"
> >
<option value="">Sélectionner...</option> <option value="">
<option v-for="option in field.options" :key="option" :value="option">{{ option }}</option> Sélectionner...
</option>
<option v-for="option in field.options" :key="option" :value="option">
{{ option }}
</option>
</select> </select>
<div v-else-if="field.type === 'boolean'" class="flex items-center gap-2"> <div v-else-if="field.type === 'boolean'" class="flex items-center gap-2">
<input <input
@@ -183,7 +199,7 @@
true-value="true" true-value="true"
false-value="false" false-value="false"
@change="updateComponentCustomField(field)" @change="updateComponentCustomField(field)"
/> >
<span class="text-sm">{{ field.value === 'true' ? 'Oui' : 'Non' }}</span> <span class="text-sm">{{ field.value === 'true' ? 'Oui' : 'Non' }}</span>
</div> </div>
<input <input
@@ -193,10 +209,12 @@
class="input input-bordered input-sm" class="input input-bordered input-sm"
:required="field.required" :required="field.required"
@blur="updateComponentCustomField(field)" @blur="updateComponentCustomField(field)"
/> >
</template> </template>
<template v-else> <template v-else>
<div class="input input-bordered input-sm bg-base-200">{{ field.value || 'Non défini' }}</div> <div class="input input-bordered input-sm bg-base-200">
{{ field.value || 'Non défini' }}
</div>
</template> </template>
</div> </div>
</div> </div>
@@ -204,13 +222,17 @@
<div class="mt-4 pt-4 border-t border-gray-200 space-y-3"> <div class="mt-4 pt-4 border-t border-gray-200 space-y-3">
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<h4 class="font-semibold text-sm text-gray-700">Documents</h4> <h4 class="font-semibold text-sm text-gray-700">
Documents
</h4>
<span v-if="isEditMode && selectedFiles.length" class="badge badge-outline"> <span v-if="isEditMode && selectedFiles.length" class="badge badge-outline">
{{ selectedFiles.length }} fichier{{ selectedFiles.length > 1 ? 's' : '' }} sélectionné{{ selectedFiles.length > 1 ? 's' : '' }} {{ selectedFiles.length }} fichier{{ selectedFiles.length > 1 ? 's' : '' }} sélectionné{{ selectedFiles.length > 1 ? 's' : '' }}
</span> </span>
</div> </div>
<p v-if="loadingDocuments" class="text-xs text-gray-500">Chargement des documents...</p> <p v-if="loadingDocuments" class="text-xs text-gray-500">
Chargement des documents...
</p>
<DocumentUpload <DocumentUpload
v-if="isEditMode" v-if="isEditMode"
@@ -235,7 +257,9 @@
/> />
</span> </span>
<div> <div>
<div class="font-medium">{{ document.name }}</div> <div class="font-medium">
{{ document.name }}
</div>
<div class="text-xs text-gray-500"> <div class="text-xs text-gray-500">
{{ document.mimeType || 'Inconnu' }} {{ formatSize(document.size) }} {{ document.mimeType || 'Inconnu' }} {{ formatSize(document.size) }}
</div> </div>
@@ -266,12 +290,16 @@
</div> </div>
</div> </div>
</div> </div>
<p v-else-if="!loadingDocuments" class="text-xs text-gray-500">Aucun document lié à ce composant.</p> <p v-else-if="!loadingDocuments" class="text-xs text-gray-500">
Aucun document lié à ce composant.
</p>
</div> </div>
<!-- Component Pieces --> <!-- Component Pieces -->
<div v-if="component.pieces && component.pieces.length > 0" class="space-y-2"> <div v-if="component.pieces && component.pieces.length > 0" class="space-y-2">
<h4 class="font-semibold text-gray-700">Pièces du composant</h4> <h4 class="font-semibold text-gray-700">
Pièces du composant
</h4>
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-3"> <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-3">
<PieceItem <PieceItem
v-for="piece in component.pieces" v-for="piece in component.pieces"
@@ -289,7 +317,9 @@
<!-- Sub Components --> <!-- Sub Components -->
<div v-if="component.subComponents && component.subComponents.length > 0" class="space-y-3"> <div v-if="component.subComponents && component.subComponents.length > 0" class="space-y-3">
<h4 class="font-semibold text-gray-700">Sous-composants</h4> <h4 class="font-semibold text-gray-700">
Sous-composants
</h4>
<div class="space-y-3 pl-4 border-l-2 border-gray-200"> <div class="space-y-3 pl-4 border-l-2 border-gray-200">
<ComponentItem <ComponentItem
v-for="subComponent in component.subComponents" v-for="subComponent in component.subComponents"
@@ -326,32 +356,32 @@ import IconLucideChevronRight from '~icons/lucide/chevron-right'
const props = defineProps({ const props = defineProps({
component: { component: {
type: Object, type: Object,
required: true, required: true
}, },
isEditMode: { isEditMode: {
type: Boolean, type: Boolean,
default: false, default: false
}, },
collapseAll: { collapseAll: {
type: Boolean, type: Boolean,
default: true, default: true
}, },
toggleToken: { toggleToken: {
type: Number, type: Number,
default: 0, default: 0
}, },
componentModelOptions: { componentModelOptions: {
type: Array, type: Array,
default: () => [], default: () => []
}, },
componentModelOptionsProvider: { componentModelOptionsProvider: {
type: Function, type: Function,
default: () => [], default: () => []
}, },
pieceModelOptionsProvider: { pieceModelOptionsProvider: {
type: Function, type: Function,
default: () => [], default: () => []
}, }
}) })
const emit = defineEmits([ const emit = defineEmits([
@@ -360,7 +390,7 @@ const emit = defineEmits([
'custom-field-update', 'custom-field-update',
'assign-model', 'assign-model',
'assign-piece-model', 'assign-piece-model',
'create-model-from-component', 'create-model-from-component'
]) ])
const isCollapsed = ref(true) const isCollapsed = ref(true)
@@ -369,7 +399,7 @@ const uploadingDocuments = ref(false)
const loadingDocuments = ref(false) const loadingDocuments = ref(false)
const documentsLoaded = ref(!!(props.component.documents && props.component.documents.length)) const documentsLoaded = ref(!!(props.component.documents && props.component.documents.length))
const componentDocuments = computed(() => props.component.documents || []) const componentDocuments = computed(() => props.component.documents || [])
const documentIcon = (doc) => getFileIcon({ name: doc.filename || doc.name, mime: doc.mimeType }) const documentIcon = doc => getFileIcon({ name: doc.filename || doc.name, mime: doc.mimeType })
const previewDocument = ref(null) const previewDocument = ref(null)
const previewVisible = ref(false) const previewVisible = ref(false)
@@ -395,14 +425,14 @@ watch(
ensureDocumentsLoaded() ensureDocumentsLoaded()
} }
}, },
{ immediate: true }, { immediate: true }
) )
watch( watch(
() => props.component.documents, () => props.component.documents,
(docs) => { (docs) => {
documentsLoaded.value = !!(docs && docs.length) documentsLoaded.value = !!(docs && docs.length)
}, }
) )
const toggleCollapse = () => { const toggleCollapse = () => {
@@ -444,7 +474,7 @@ const assignComponentModel = (value) => {
componentId: props.component.id, componentId: props.component.id,
composantModelId: value || null, composantModelId: value || null,
previousModelId, previousModelId,
previousModel, previousModel
}) })
} }
@@ -453,7 +483,7 @@ const emitAssignPieceModel = (payload) => {
} }
const ensureDocumentsLoaded = async () => { const ensureDocumentsLoaded = async () => {
if (documentsLoaded.value || !props.component?.id) return if (documentsLoaded.value || !props.component?.id) { return }
await refreshDocuments() await refreshDocuments()
} }
@@ -471,15 +501,15 @@ const refreshDocuments = async () => {
} }
const handleFilesAdded = async (files) => { const handleFilesAdded = async (files) => {
if (!files.length || !props.component?.id) return if (!files.length || !props.component?.id) { return }
uploadingDocuments.value = true uploadingDocuments.value = true
try { try {
const result = await uploadDocuments( const result = await uploadDocuments(
{ {
files, files,
context: { composantId: props.component.id }, context: { composantId: props.component.id }
}, },
{ updateStore: false }, { updateStore: false }
) )
if (result.success) { if (result.success) {
@@ -494,15 +524,15 @@ const handleFilesAdded = async (files) => {
} }
const removeDocument = async (documentId) => { const removeDocument = async (documentId) => {
if (!documentId) return if (!documentId) { return }
const result = await deleteDocument(documentId, { updateStore: false }) const result = await deleteDocument(documentId, { updateStore: false })
if (result.success) { if (result.success) {
props.component.documents = (props.component.documents || []).filter((doc) => doc.id !== documentId) props.component.documents = (props.component.documents || []).filter(doc => doc.id !== documentId)
} }
} }
const downloadDocument = (doc) => { const downloadDocument = (doc) => {
if (!doc?.path) return if (!doc?.path) { return }
if (doc.path.startsWith('data:')) { if (doc.path.startsWith('data:')) {
const link = document.createElement('a') const link = document.createElement('a')
@@ -516,7 +546,7 @@ const downloadDocument = (doc) => {
} }
const openPreview = (doc) => { const openPreview = (doc) => {
if (!canPreviewDocument(doc)) return if (!canPreviewDocument(doc)) { return }
previewDocument.value = doc previewDocument.value = doc
previewVisible.value = true previewVisible.value = true
} }
@@ -527,8 +557,8 @@ const closePreview = () => {
} }
const formatSize = (size) => { const formatSize = (size) => {
if (size === undefined || size === null) return '—' if (size === undefined || size === null) { return '—' }
if (size === 0) return '0 B' if (size === 0) { return '0 B' }
const units = ['B', 'KB', 'MB', 'GB'] const units = ['B', 'KB', 'MB', 'GB']
const index = Math.min(units.length - 1, Math.floor(Math.log(size) / Math.log(1024))) const index = Math.min(units.length - 1, Math.floor(Math.log(size) / Math.log(1024)))
const formatted = size / Math.pow(1024, index) const formatted = size / Math.pow(1024, index)

View File

@@ -70,52 +70,43 @@
> >
<div class="flex items-start justify-between gap-2"> <div class="flex items-start justify-between gap-2">
<div class="flex-1 space-y-3"> <div class="flex-1 space-y-3">
<div class="grid grid-cols-1 md:grid-cols-3 gap-2"> <div class="grid grid-cols-1 md:grid-cols-2 gap-3">
<input <div class="form-control">
v-model="piece.name" <label class="label"><span class="label-text">Famille de pièce</span></label>
type="text" <div>
class="input input-bordered input-xs" <input
placeholder="Nom de la pièce" :list="`component-piece-type-options-${index}`"
/> v-model="piece.typePieceLabel"
<input type="search"
v-model="piece.reference" autocomplete="off"
type="text" class="input input-bordered input-xs"
class="input input-bordered input-xs" placeholder="Sélectionner une famille"
placeholder="Référence" @change="handlePieceTypeChange(piece)"
/> @blur="handlePieceTypeChange(piece)"
<input
v-model.number="piece.quantity"
type="number"
min="0"
step="1"
class="input input-bordered input-xs"
placeholder="Quantité"
/>
</div>
<div class="form-control">
<label class="label"><span class="label-text">Famille de pièce</span></label>
<div>
<input
:list="`component-piece-type-options-${index}`"
v-model="piece.typePieceLabel"
type="search"
autocomplete="off"
class="input input-bordered input-xs"
placeholder="Rechercher une famille"
@change="handlePieceTypeChange(piece)"
@blur="handlePieceTypeChange(piece)"
/>
<datalist :id="`component-piece-type-options-${index}`">
<option
v-for="type in availablePieceTypes"
:key="type.id"
:value="formatPieceTypeOption(type)"
/> />
</datalist> <datalist :id="`component-piece-type-options-${index}`">
<option
v-for="type in availablePieceTypes"
:key="type.id"
:value="formatPieceTypeOption(type)"
/>
</datalist>
</div>
<p class="mt-1 text-[11px] text-gray-500">
{{ piece.typePieceId ? `Sélection : ${getPieceTypeLabel(piece.typePieceId) || 'Inconnue'}` : 'Aucune famille sélectionnée' }}
</p>
</div>
<div class="form-control">
<label class="label"><span class="label-text">Quantité (optionnel)</span></label>
<input
v-model.number="piece.quantity"
type="number"
min="0"
step="1"
class="input input-bordered input-xs"
placeholder="Quantité"
/>
</div> </div>
<p class="mt-1 text-[11px] text-gray-500">
{{ piece.typePieceId ? `Sélection : ${getPieceTypeLabel(piece.typePieceId) || 'Inconnue'}` : 'Aucune famille sélectionnée' }}
</p>
</div> </div>
</div> </div>
<button type="button" class="btn btn-error btn-xs btn-square" @click="removePiece(index)"> <button type="button" class="btn btn-error btn-xs btn-square" @click="removePiece(index)">
@@ -144,6 +135,7 @@
:node="subComponent" :node="subComponent"
:depth="0" :depth="0"
:piece-types="availablePieceTypes" :piece-types="availablePieceTypes"
:component-types="availableComponentTypes"
@remove="removeSubComponent(index)" @remove="removeSubComponent(index)"
/> />
</div> </div>
@@ -162,6 +154,7 @@ import {
normalizeStructureForSave, normalizeStructureForSave,
} from '~/shared/modelUtils' } from '~/shared/modelUtils'
import { usePieceTypes } from '~/composables/usePieceTypes' import { usePieceTypes } from '~/composables/usePieceTypes'
import { useComponentTypes } from '~/composables/useComponentTypes'
defineOptions({ name: 'ComponentModelStructureEditor' }) defineOptions({ name: 'ComponentModelStructureEditor' })
@@ -207,18 +200,25 @@ watch(
{ deep: true } { deep: true }
) )
type PieceTypeOption = { type ModelTypeOption = {
id: string id: string
name: string name: string
code?: string | null code?: string | null
} }
const { pieceTypes, loadPieceTypes } = usePieceTypes() const formatModelTypeOption = (type: ModelTypeOption | undefined | null) => {
if (!type) return ''
return type.code ? `${type.name} (${type.code})` : type.name
}
const availablePieceTypes = computed<PieceTypeOption[]>(() => pieceTypes.value ?? []) const { pieceTypes, loadPieceTypes } = usePieceTypes()
const { componentTypes, loadComponentTypes } = useComponentTypes()
const availablePieceTypes = computed<ModelTypeOption[]>(() => pieceTypes.value ?? [])
const availableComponentTypes = computed<ModelTypeOption[]>(() => componentTypes.value ?? [])
const pieceTypeMap = computed(() => { const pieceTypeMap = computed(() => {
const map = new Map<string, PieceTypeOption>() const map = new Map<string, ModelTypeOption>()
availablePieceTypes.value.forEach((type) => { availablePieceTypes.value.forEach((type) => {
if (type && typeof type.id === 'string') { if (type && typeof type.id === 'string') {
map.set(type.id, type) map.set(type.id, type)
@@ -227,10 +227,18 @@ const pieceTypeMap = computed(() => {
return map return map
}) })
const formatPieceTypeOption = (type: PieceTypeOption | undefined | null) => { const componentTypeMap = computed(() => {
if (!type) return '' const map = new Map<string, ModelTypeOption>()
return type.code ? `${type.name} (${type.code})` : type.name availableComponentTypes.value.forEach((type) => {
} if (type && typeof type.id === 'string') {
map.set(type.id, type)
}
})
return map
})
const formatPieceTypeOption = (type: ModelTypeOption | undefined | null) => formatModelTypeOption(type)
const formatComponentTypeOption = (type: ModelTypeOption | undefined | null) => formatModelTypeOption(type)
const resolvePieceType = (input: string) => { const resolvePieceType = (input: string) => {
const normalized = input.trim().toLowerCase() const normalized = input.trim().toLowerCase()
@@ -251,6 +259,25 @@ const resolvePieceType = (input: string) => {
) )
} }
const resolveComponentType = (input: string) => {
const normalized = input.trim().toLowerCase()
if (!normalized) {
return null
}
return (
availableComponentTypes.value.find((type) => {
const formatted = formatComponentTypeOption(type).toLowerCase()
const name = (type?.name ?? '').toLowerCase()
const code = (type?.code ?? '').toLowerCase()
return (
formatted === normalized
|| name === normalized
|| (!!code && code === normalized)
)
}) ?? null
)
}
const getPieceTypeLabel = (id?: string) => { const getPieceTypeLabel = (id?: string) => {
if (!id) return '' if (!id) return ''
const option = pieceTypeMap.value.get(id) const option = pieceTypeMap.value.get(id)
@@ -263,12 +290,21 @@ const updatePieceTypeLabel = (piece: any) => {
} }
if (piece.typePieceId) { if (piece.typePieceId) {
const option = pieceTypeMap.value.get(piece.typePieceId) const option = pieceTypeMap.value.get(piece.typePieceId)
piece.typePieceLabel = option ? formatPieceTypeOption(option) : piece.typePieceLabel || '' if (option) {
piece.typePieceLabel = formatPieceTypeOption(option)
piece.name = option.name || formatPieceTypeOption(option)
} else if (!piece.typePieceLabel) {
piece.name = ''
}
} else if (piece.typePieceLabel) { } else if (piece.typePieceLabel) {
const match = resolvePieceType(piece.typePieceLabel) const match = resolvePieceType(piece.typePieceLabel)
if (match) { if (match) {
piece.typePieceId = match.id piece.typePieceId = match.id
piece.typePieceLabel = formatPieceTypeOption(match) piece.typePieceLabel = formatPieceTypeOption(match)
piece.name = match.name || formatPieceTypeOption(match)
} else {
piece.typePieceLabel = ''
piece.name = ''
} }
} }
} }
@@ -281,14 +317,18 @@ const handlePieceTypeChange = (piece: any) => {
if (!value) { if (!value) {
piece.typePieceId = '' piece.typePieceId = ''
piece.typePieceLabel = '' piece.typePieceLabel = ''
piece.name = ''
return return
} }
const match = resolvePieceType(value) const match = resolvePieceType(value)
if (match) { if (match) {
piece.typePieceId = match.id piece.typePieceId = match.id
piece.typePieceLabel = formatPieceTypeOption(match) piece.typePieceLabel = formatPieceTypeOption(match)
piece.name = match.name || formatPieceTypeOption(match)
} else { } else {
piece.typePieceId = '' piece.typePieceId = ''
piece.typePieceLabel = ''
piece.name = ''
} }
} }
@@ -304,37 +344,84 @@ const applyPieceLabels = (pieces?: any[]) => {
if (match) { if (match) {
piece.typePieceId = match.id piece.typePieceId = match.id
piece.typePieceLabel = formatPieceTypeOption(match) piece.typePieceLabel = formatPieceTypeOption(match)
piece.name = match.name || formatPieceTypeOption(match)
} else {
piece.typePieceLabel = ''
piece.name = ''
} }
} else if (!piece?.name) {
piece.name = ''
} }
}) })
} }
const applyComponentTypeLabel = (component: any) => {
if (!component) {
return
}
if (component.typeComposantId) {
const option = componentTypeMap.value.get(component.typeComposantId)
if (option) {
component.typeComposantLabel = formatComponentTypeOption(option)
component.name = option.name || formatComponentTypeOption(option)
} else if (!component.typeComposantLabel) {
component.name = ''
}
} else if (component.typeComposantLabel) {
const match = resolveComponentType(component.typeComposantLabel)
if (match) {
component.typeComposantId = match.id
component.typeComposantLabel = formatComponentTypeOption(match)
component.name = match.name || formatComponentTypeOption(match)
} else {
component.typeComposantLabel = ''
component.name = ''
}
}
}
const traverseSubComponents = (components?: any[]) => { const traverseSubComponents = (components?: any[]) => {
if (!Array.isArray(components)) { if (!Array.isArray(components)) {
return return
} }
components.forEach((component) => { components.forEach((component) => {
applyComponentTypeLabel(component)
applyPieceLabels(component?.pieces) applyPieceLabels(component?.pieces)
traverseSubComponents(component?.subComponents) traverseSubComponents(component?.subComponents)
}) })
} }
const syncAllPieceTypeLabels = () => { const syncAllTypeLabels = () => {
applyPieceLabels(localStructure.pieces) applyPieceLabels(localStructure.pieces)
traverseSubComponents(localStructure.subComponents) traverseSubComponents(localStructure.subComponents)
} }
onMounted(async () => { onMounted(async () => {
const loaders: Promise<any>[] = []
if (!availablePieceTypes.value.length) { if (!availablePieceTypes.value.length) {
await loadPieceTypes() loaders.push(loadPieceTypes())
} }
syncAllPieceTypeLabels() if (!availableComponentTypes.value.length) {
loaders.push(loadComponentTypes())
}
if (loaders.length) {
await Promise.all(loaders)
}
syncAllTypeLabels()
}) })
watch( watch(
() => availablePieceTypes.value, () => availablePieceTypes.value,
() => { () => {
syncAllPieceTypeLabels() syncAllTypeLabels()
},
{ deep: true }
)
watch(
() => availableComponentTypes.value,
() => {
syncAllTypeLabels()
}, },
{ deep: true } { deep: true }
) )
@@ -365,7 +452,6 @@ const addPiece = () => {
ensureArray('pieces') ensureArray('pieces')
localStructure.pieces.push({ localStructure.pieces.push({
name: '', name: '',
reference: '',
quantity: undefined, quantity: undefined,
typePieceId: '', typePieceId: '',
typePieceLabel: '', typePieceLabel: '',
@@ -383,6 +469,8 @@ const addSubComponent = () => {
name: '', name: '',
description: '', description: '',
quantity: undefined, quantity: undefined,
typeComposantId: '',
typeComposantLabel: '',
customFields: [], customFields: [],
pieces: [], pieces: [],
subComponents: [], subComponents: [],

View File

@@ -4,13 +4,13 @@
<div class="flex items-center gap-2"> <div class="flex items-center gap-2">
<div class="relative flex-1"> <div class="relative flex-1">
<input <input
type="text"
v-model="searchTerm" v-model="searchTerm"
type="text"
class="input input-bordered w-full pr-10" class="input input-bordered w-full pr-10"
:placeholder="placeholder" :placeholder="placeholder"
@focus="openDropdown = true; ensureOptionsLoaded()" @focus="openDropdown = true; ensureOptionsLoaded()"
@input="onSearch" @input="onSearch"
/> >
<button <button
type="button" type="button"
class="absolute right-2 top-1/2 -translate-y-1/2 btn btn-ghost btn-xs" class="absolute right-2 top-1/2 -translate-y-1/2 btn btn-ghost btn-xs"
@@ -57,11 +57,13 @@
<dialog class="modal" :class="{ 'modal-open': openCreateModal }"> <dialog class="modal" :class="{ 'modal-open': openCreateModal }">
<div class="modal-box"> <div class="modal-box">
<h3 class="font-bold text-lg mb-4">Nouveau constructeur</h3> <h3 class="font-bold text-lg mb-4">
Nouveau constructeur
</h3>
<form @submit.prevent="handleCreate"> <form @submit.prevent="handleCreate">
<div class="form-control mb-3"> <div class="form-control mb-3">
<label class="label"><span class="label-text">Nom</span></label> <label class="label"><span class="label-text">Nom</span></label>
<input v-model="createForm.name" type="text" class="input input-bordered" required /> <input v-model="createForm.name" type="text" class="input input-bordered" required>
</div> </div>
<FieldEmail <FieldEmail
v-model="createForm.email" v-model="createForm.email"
@@ -78,9 +80,11 @@
/> />
<div class="modal-action"> <div class="modal-action">
<button type="button" class="btn" @click="closeCreateModal">Annuler</button> <button type="button" class="btn" @click="closeCreateModal">
Annuler
</button>
<button type="submit" class="btn btn-primary" :disabled="creating"> <button type="submit" class="btn btn-primary" :disabled="creating">
<span v-if="creating" class="loading loading-spinner loading-xs mr-2"></span> <span v-if="creating" class="loading loading-spinner loading-xs mr-2" />
Créer Créer
</button> </button>
</div> </div>
@@ -100,16 +104,16 @@ import IconLucideChevronsUpDown from '~icons/lucide/chevrons-up-down'
const props = defineProps({ const props = defineProps({
modelValue: { modelValue: {
type: String, type: String,
default: null, default: null
}, },
label: { label: {
type: String, type: String,
default: '', default: ''
}, },
placeholder: { placeholder: {
type: String, type: String,
default: 'Sélectionner ou créer un constructeur...', default: 'Sélectionner ou créer un constructeur...'
}, }
}) })
const emit = defineEmits(['update:modelValue']) const emit = defineEmits(['update:modelValue'])
@@ -131,7 +135,7 @@ const applyOptions = (items = []) => {
if (selectedId && !limited.some(item => item.id === selectedId)) { if (selectedId && !limited.some(item => item.id === selectedId)) {
const selected = cloned.find(item => item.id === selectedId) const selected = cloned.find(item => item.id === selectedId)
if (selected) { if (selected) {
if (limited.length >= 10) limited.pop() if (limited.length >= 10) { limited.pop() }
limited.unshift(selected) limited.unshift(selected)
} }
} }
@@ -142,7 +146,7 @@ const applyOptions = (items = []) => {
const createForm = ref({ const createForm = ref({
name: '', name: '',
email: '', email: '',
phone: '', phone: ''
}) })
const selectedConstructeur = computed(() => const selectedConstructeur = computed(() =>
@@ -171,8 +175,8 @@ const ensureOptionsLoaded = async (force = false) => {
applyOptions(constructeurs.value) applyOptions(constructeurs.value)
return return
} }
if (!force && searchTerm.value === lastSearchTerm && options.value.length) return if (!force && searchTerm.value === lastSearchTerm && options.value.length) { return }
if (options.value.length && !force) return if (options.value.length && !force) { return }
const result = await searchConstructeurs(searchTerm.value) const result = await searchConstructeurs(searchTerm.value)
if (result.success) { if (result.success) {
applyOptions(result.data || []) applyOptions(result.data || [])
@@ -189,7 +193,7 @@ const onSearch = () => {
lastSearchTerm = '' lastSearchTerm = ''
return return
} }
if (searchTerm.value === lastSearchTerm) return if (searchTerm.value === lastSearchTerm) { return }
const result = await searchConstructeurs(searchTerm.value) const result = await searchConstructeurs(searchTerm.value)
if (result.success) { if (result.success) {
applyOptions(result.data || []) applyOptions(result.data || [])
@@ -212,8 +216,8 @@ const closeCreateModal = () => {
const handleCreate = async () => { const handleCreate = async () => {
creating.value = true creating.value = true
const payload = { ...createForm.value } const payload = { ...createForm.value }
if (!payload.phone) delete payload.phone if (!payload.phone) { delete payload.phone }
if (!payload.email) delete payload.email if (!payload.email) { delete payload.email }
const result = await createConstructeur(payload) const result = await createConstructeur(payload)
creating.value = false creating.value = false
if (result.success) { if (result.success) {

View File

@@ -1,9 +1,11 @@
<template> <template>
<div v-if="customFields && customFields.length > 0" class="space-y-4"> <div v-if="customFields && customFields.length > 0" class="space-y-4">
<h4 class="font-semibold text-gray-700 mb-3">Champs personnalisés</h4> <h4 class="font-semibold text-gray-700 mb-3">
Champs personnalisés
</h4>
<div class="grid grid-cols-1 md:grid-cols-2 gap-4"> <div class="grid grid-cols-1 md:grid-cols-2 gap-4">
<div <div
v-for="field in customFields" v-for="field in customFields"
:key="field.id" :key="field.id"
class="form-control" class="form-control"
> >
@@ -11,66 +13,68 @@
<span class="label-text">{{ field.name }}</span> <span class="label-text">{{ field.name }}</span>
<span v-if="field.required" class="label-text-alt text-error">*</span> <span v-if="field.required" class="label-text-alt text-error">*</span>
</label> </label>
<!-- Champ de type TEXT --> <!-- Champ de type TEXT -->
<input <input
v-if="field.type === 'text'" v-if="field.type === 'text'"
v-model="fieldValues[field.id]" v-model="fieldValues[field.id]"
type="text" type="text"
class="input input-bordered input-sm" class="input input-bordered input-sm"
:required="field.required" :required="field.required"
@blur="updateCustomFieldValue(field.id)" @blur="updateCustomFieldValue(field.id)"
/> >
<!-- Champ de type NUMBER --> <!-- Champ de type NUMBER -->
<input <input
v-else-if="field.type === 'number'" v-else-if="field.type === 'number'"
v-model="fieldValues[field.id]" v-model="fieldValues[field.id]"
type="number" type="number"
class="input input-bordered input-sm" class="input input-bordered input-sm"
:required="field.required" :required="field.required"
@blur="updateCustomFieldValue(field.id)" @blur="updateCustomFieldValue(field.id)"
/> >
<!-- Champ de type SELECT --> <!-- Champ de type SELECT -->
<select <select
v-else-if="field.type === 'select'" v-else-if="field.type === 'select'"
v-model="fieldValues[field.id]" v-model="fieldValues[field.id]"
class="select select-bordered select-sm" class="select select-bordered select-sm"
:required="field.required" :required="field.required"
@change="updateCustomFieldValue(field.id)" @change="updateCustomFieldValue(field.id)"
> >
<option value="">Sélectionner...</option> <option value="">
<option Sélectionner...
v-for="option in field.options" </option>
:key="option" <option
v-for="option in field.options"
:key="option"
:value="option" :value="option"
> >
{{ option }} {{ option }}
</option> </option>
</select> </select>
<!-- Champ de type BOOLEAN --> <!-- Champ de type BOOLEAN -->
<div v-else-if="field.type === 'boolean'" class="flex items-center gap-2"> <div v-else-if="field.type === 'boolean'" class="flex items-center gap-2">
<input <input
v-model="fieldValues[field.id]" v-model="fieldValues[field.id]"
type="checkbox" type="checkbox"
class="checkbox checkbox-sm" class="checkbox checkbox-sm"
:checked="fieldValues[field.id] === 'true'" :checked="fieldValues[field.id] === 'true'"
@change="updateCustomFieldValue(field.id)" @change="updateCustomFieldValue(field.id)"
/> >
<span class="text-sm">{{ fieldValues[field.id] === 'true' ? 'Oui' : 'Non' }}</span> <span class="text-sm">{{ fieldValues[field.id] === 'true' ? 'Oui' : 'Non' }}</span>
</div> </div>
<!-- Champ de type DATE --> <!-- Champ de type DATE -->
<input <input
v-else-if="field.type === 'date'" v-else-if="field.type === 'date'"
v-model="fieldValues[field.id]" v-model="fieldValues[field.id]"
type="date" type="date"
class="input input-bordered input-sm" class="input input-bordered input-sm"
:required="field.required" :required="field.required"
@blur="updateCustomFieldValue(field.id)" @blur="updateCustomFieldValue(field.id)"
/> >
</div> </div>
</div> </div>
</div> </div>
@@ -91,7 +95,7 @@ const props = defineProps({
entityType: { entityType: {
type: String, type: String,
required: true, // 'machine', 'composant', 'piece' required: true, // 'machine', 'composant', 'piece'
validator: (value) => ['machine', 'composant', 'piece'].includes(value) validator: value => ['machine', 'composant', 'piece'].includes(value)
} }
}) })
@@ -102,7 +106,7 @@ const fieldValues = reactive({})
// Initialiser les valeurs sans appliquer de valeur par défaut implicite // Initialiser les valeurs sans appliquer de valeur par défaut implicite
const initializeFieldValues = () => { const initializeFieldValues = () => {
props.customFields.forEach(field => { props.customFields.forEach((field) => {
if (!(field.id in fieldValues)) { if (!(field.id in fieldValues)) {
fieldValues[field.id] = field.value ?? '' fieldValues[field.id] = field.value ?? ''
} }
@@ -129,4 +133,4 @@ watch(() => props.customFields, () => {
onMounted(() => { onMounted(() => {
initializeFieldValues() initializeFieldValues()
}) })
</script> </script>

View File

@@ -1,23 +1,25 @@
<template> <template>
<div class="modal" :class="{ 'modal-open': isOpen }"> <div class="modal" :class="{ 'modal-open': isOpen }">
<div class="modal-box"> <div class="modal-box">
<h3 class="font-bold text-lg mb-4">Paramètres d'affichage</h3> <h3 class="font-bold text-lg mb-4">
Paramètres d'affichage
</h3>
<!-- Contrôle du zoom --> <!-- Contrôle du zoom -->
<div class="form-control mb-4"> <div class="form-control mb-4">
<label class="label"> <label class="label">
<span class="label-text">Taille du texte</span> <span class="label-text">Taille du texte</span>
<span class="label-text-alt">{{ zoomLevel }}%</span> <span class="label-text-alt">{{ zoomLevel }}%</span>
</label> </label>
<input <input
type="range" type="range"
min="80" min="80"
max="150" max="150"
step="10" step="10"
:value="zoomLevel" :value="zoomLevel"
class="range range-primary"
@input="updateZoom" @input="updateZoom"
class="range range-primary" >
/>
<div class="w-full flex justify-between text-xs px-2 mt-1"> <div class="w-full flex justify-between text-xs px-2 mt-1">
<span>80%</span> <span>80%</span>
<span>100%</span> <span>100%</span>
@@ -31,24 +33,24 @@
<span class="label-text">Densité de l'interface</span> <span class="label-text">Densité de l'interface</span>
</label> </label>
<div class="flex gap-2"> <div class="flex gap-2">
<button <button
@click="setDensity('compact')"
class="btn btn-sm" class="btn btn-sm"
:class="density === 'compact' ? 'btn-primary' : 'btn-outline'" :class="density === 'compact' ? 'btn-primary' : 'btn-outline'"
@click="setDensity('compact')"
> >
Compacte Compacte
</button> </button>
<button <button
@click="setDensity('comfortable')"
class="btn btn-sm" class="btn btn-sm"
:class="density === 'comfortable' ? 'btn-primary' : 'btn-outline'" :class="density === 'comfortable' ? 'btn-primary' : 'btn-outline'"
@click="setDensity('comfortable')"
> >
Confortable Confortable
</button> </button>
<button <button
@click="setDensity('spacious')"
class="btn btn-sm" class="btn btn-sm"
:class="density === 'spacious' ? 'btn-primary' : 'btn-outline'" :class="density === 'spacious' ? 'btn-primary' : 'btn-outline'"
@click="setDensity('spacious')"
> >
Espacée Espacée
</button> </button>
@@ -61,17 +63,17 @@
<span class="label-text">Contraste</span> <span class="label-text">Contraste</span>
</label> </label>
<div class="flex gap-2"> <div class="flex gap-2">
<button <button
@click="setContrast('normal')"
class="btn btn-sm" class="btn btn-sm"
:class="contrast === 'normal' ? 'btn-primary' : 'btn-outline'" :class="contrast === 'normal' ? 'btn-primary' : 'btn-outline'"
@click="setContrast('normal')"
> >
Normal Normal
</button> </button>
<button <button
@click="setContrast('high')"
class="btn btn-sm" class="btn btn-sm"
:class="contrast === 'high' ? 'btn-primary' : 'btn-outline'" :class="contrast === 'high' ? 'btn-primary' : 'btn-outline'"
@click="setContrast('high')"
> >
Élevé Élevé
</button> </button>
@@ -80,9 +82,9 @@
<!-- Réinitialiser --> <!-- Réinitialiser -->
<div class="form-control"> <div class="form-control">
<button <button
@click="resetSettings"
class="btn btn-outline btn-sm" class="btn btn-outline btn-sm"
@click="resetSettings"
> >
Réinitialiser les paramètres Réinitialiser les paramètres
</button> </button>
@@ -90,7 +92,9 @@
<!-- Actions --> <!-- Actions -->
<div class="modal-action"> <div class="modal-action">
<button @click="closeModal" class="btn btn-primary">Fermer</button> <button class="btn btn-primary" @click="closeModal">
Fermer
</button>
</div> </div>
</div> </div>
</div> </div>
@@ -118,11 +122,11 @@ onMounted(() => {
const savedZoom = localStorage.getItem('display-zoom') const savedZoom = localStorage.getItem('display-zoom')
const savedDensity = localStorage.getItem('display-density') const savedDensity = localStorage.getItem('display-density')
const savedContrast = localStorage.getItem('display-contrast') const savedContrast = localStorage.getItem('display-contrast')
if (savedZoom) zoomLevel.value = parseInt(savedZoom) if (savedZoom) { zoomLevel.value = parseInt(savedZoom) }
if (savedDensity) density.value = savedDensity if (savedDensity) { density.value = savedDensity }
if (savedContrast) contrast.value = savedContrast if (savedContrast) { contrast.value = savedContrast }
applySettings() applySettings()
}) })
@@ -146,10 +150,10 @@ const applySettings = () => {
localStorage.setItem('display-zoom', zoomLevel.value.toString()) localStorage.setItem('display-zoom', zoomLevel.value.toString())
localStorage.setItem('display-density', density.value) localStorage.setItem('display-density', density.value)
localStorage.setItem('display-contrast', contrast.value) localStorage.setItem('display-contrast', contrast.value)
// Appliquer les styles // Appliquer les styles
const root = document.documentElement const root = document.documentElement
// Zoom - exclure complètement le modal des paramètres // Zoom - exclure complètement le modal des paramètres
const modal = document.querySelector('.modal') const modal = document.querySelector('.modal')
if (modal) { if (modal) {
@@ -157,27 +161,27 @@ const applySettings = () => {
modal.style.fontSize = '100%' modal.style.fontSize = '100%'
modal.style.transform = 'none' modal.style.transform = 'none'
modal.style.scale = '1' modal.style.scale = '1'
// Appliquer aux enfants du modal // Appliquer aux enfants du modal
const modalElements = modal.querySelectorAll('*') const modalElements = modal.querySelectorAll('*')
modalElements.forEach(element => { modalElements.forEach((element) => {
element.style.fontSize = 'inherit' element.style.fontSize = 'inherit'
element.style.transform = 'none' element.style.transform = 'none'
element.style.scale = '1' element.style.scale = '1'
}) })
} }
// Appliquer le zoom au reste de la page (sauf le modal) // Appliquer le zoom au reste de la page (sauf le modal)
root.style.fontSize = `${zoomLevel.value}%` root.style.fontSize = `${zoomLevel.value}%`
// Densité - utiliser les classes DaisyUI // Densité - utiliser les classes DaisyUI
root.classList.remove('density-compact', 'density-comfortable', 'density-spacious') root.classList.remove('density-compact', 'density-comfortable', 'density-spacious')
root.classList.add(`density-${density.value}`) root.classList.add(`density-${density.value}`)
// Contraste - utiliser les classes DaisyUI // Contraste - utiliser les classes DaisyUI
root.classList.remove('contrast-normal', 'contrast-high') root.classList.remove('contrast-normal', 'contrast-high')
root.classList.add(`contrast-${contrast.value}`) root.classList.add(`contrast-${contrast.value}`)
// Émettre les changements // Émettre les changements
emit('update-settings', { emit('update-settings', {
zoom: zoomLevel.value, zoom: zoomLevel.value,
@@ -200,4 +204,4 @@ const closeModal = () => {
<style scoped> <style scoped>
/* Les styles sont maintenant gérés par DaisyUI et le CSS global */ /* Les styles sont maintenant gérés par DaisyUI et le CSS global */
</style> </style>

View File

@@ -8,7 +8,9 @@
<div class="w-full max-w-[1600px] h-full max-h-[94vh] bg-base-100 rounded-2xl shadow-2xl flex flex-col overflow-hidden"> <div class="w-full max-w-[1600px] h-full max-h-[94vh] bg-base-100 rounded-2xl shadow-2xl flex flex-col overflow-hidden">
<header class="flex items-start justify-between gap-4 p-6 border-b border-base-200"> <header class="flex items-start justify-between gap-4 p-6 border-b border-base-200">
<div class="min-w-0"> <div class="min-w-0">
<h3 class="font-bold text-xl truncate">Prévisualisation</h3> <h3 class="font-bold text-xl truncate">
Prévisualisation
</h3>
<p class="text-sm text-gray-500 truncate"> <p class="text-sm text-gray-500 truncate">
{{ document?.name || document?.filename }}<span v-if="documentDescription"> {{ documentDescription }}</span> {{ document?.name || document?.filename }}<span v-if="documentDescription"> {{ documentDescription }}</span>
</p> </p>
@@ -21,7 +23,7 @@
<section class="flex-1 bg-base-200/40 px-6 py-5 overflow-hidden"> <section class="flex-1 bg-base-200/40 px-6 py-5 overflow-hidden">
<div class="h-full w-full rounded-xl border border-base-300 bg-base-100 flex items-center justify-center overflow-hidden"> <div class="h-full w-full rounded-xl border border-base-300 bg-base-100 flex items-center justify-center overflow-hidden">
<template v-if="previewType === 'image'"> <template v-if="previewType === 'image'">
<img :src="document?.path" alt="preview" class="max-h-full max-w-full object-contain" /> <img :src="document?.path" alt="preview" class="max-h-full max-w-full object-contain">
</template> </template>
<template v-else-if="previewType === 'pdf'"> <template v-else-if="previewType === 'pdf'">
@@ -30,21 +32,21 @@
class="w-full h-full bg-white" class="w-full h-full bg-white"
frameborder="0" frameborder="0"
title="Aperçu PDF" title="Aperçu PDF"
></iframe> />
</template> </template>
<template v-else-if="previewType === 'audio'"> <template v-else-if="previewType === 'audio'">
<audio :src="document?.path" controls class="w-full"></audio> <audio :src="document?.path" controls class="w-full" />
</template> </template>
<template v-else-if="previewType === 'video'"> <template v-else-if="previewType === 'video'">
<video :src="document?.path" controls class="w-full h-full bg-black"></video> <video :src="document?.path" controls class="w-full h-full bg-black" />
</template> </template>
<template v-else-if="previewType === 'text'"> <template v-else-if="previewType === 'text'">
<div class="w-full h-full overflow-auto"> <div class="w-full h-full overflow-auto">
<div v-if="textLoading" class="flex items-center justify-center py-10 text-sm text-gray-500"> <div v-if="textLoading" class="flex items-center justify-center py-10 text-sm text-gray-500">
<span class="loading loading-spinner loading-md mr-2"></span> <span class="loading loading-spinner loading-md mr-2" />
Chargement du document... Chargement du document...
</div> </div>
<div v-else-if="textError" class="alert alert-error text-sm"> <div v-else-if="textError" class="alert alert-error text-sm">
@@ -65,7 +67,9 @@
</section> </section>
<footer class="border-t border-base-200 px-6 py-4 flex flex-wrap gap-2 justify-end bg-base-100"> <footer class="border-t border-base-200 px-6 py-4 flex flex-wrap gap-2 justify-end bg-base-100">
<button type="button" class="btn" @click="close">Fermer</button> <button type="button" class="btn" @click="close">
Fermer
</button>
<button type="button" class="btn btn-primary" @click="download"> <button type="button" class="btn btn-primary" @click="download">
Télécharger Télécharger
</button> </button>
@@ -82,12 +86,12 @@ import { getPreviewType, describeDocument } from '~/utils/documentPreview'
const props = defineProps({ const props = defineProps({
document: { document: {
type: Object, type: Object,
default: null, default: null
}, },
visible: { visible: {
type: Boolean, type: Boolean,
default: false, default: false
}, }
}) })
const emit = defineEmits(['close']) const emit = defineEmits(['close'])
@@ -106,8 +110,8 @@ watch(
textError.value = '' textError.value = ''
textLoading.value = false textLoading.value = false
if (!doc) return if (!doc) { return }
if (getPreviewType(doc) !== 'text') return if (getPreviewType(doc) !== 'text') { return }
try { try {
textLoading.value = true textLoading.value = true
@@ -142,7 +146,7 @@ const close = () => {
} }
const download = () => { const download = () => {
if (!props.document?.path) return if (!props.document?.path) { return }
const link = document.createElement('a') const link = document.createElement('a')
link.href = props.document.path link.href = props.document.path
link.download = props.document.filename || props.document.name || 'document' link.download = props.document.filename || props.document.name || 'document'

View File

@@ -10,8 +10,12 @@
<IconLucideCloudUpload class="w-10 h-10 text-primary" aria-hidden="true" /> <IconLucideCloudUpload class="w-10 h-10 text-primary" aria-hidden="true" />
<div> <div>
<h3 class="font-semibold">{{ title }}</h3> <h3 class="font-semibold">
<p class="text-sm text-gray-500">{{ subtitle }}</p> {{ title }}
</h3>
<p class="text-sm text-gray-500">
{{ subtitle }}
</p>
</div> </div>
<div class="flex flex-wrap justify-center gap-2"> <div class="flex flex-wrap justify-center gap-2">
@@ -28,7 +32,7 @@
:accept="accept" :accept="accept"
:multiple="multiple" :multiple="multiple"
@change="onFileChange" @change="onFileChange"
/> >
<ul v-if="selectedFiles.length" class="mt-4 w-full space-y-2 text-left"> <ul v-if="selectedFiles.length" class="mt-4 w-full space-y-2 text-left">
<li v-for="file in selectedFiles" :key="file.name" class="flex items-center justify-between text-sm"> <li v-for="file in selectedFiles" :key="file.name" class="flex items-center justify-between text-sm">
@@ -62,28 +66,28 @@ import IconLucideCloudUpload from '~icons/lucide/cloud-upload'
const props = defineProps({ const props = defineProps({
title: { title: {
type: String, type: String,
default: 'Ajouter des documents', default: 'Ajouter des documents'
}, },
subtitle: { subtitle: {
type: String, type: String,
default: 'Formats acceptés : PDF, images, textes…', default: 'Formats acceptés : PDF, images, textes…'
}, },
accept: { accept: {
type: String, type: String,
default: '', default: ''
}, },
multiple: { multiple: {
type: Boolean, type: Boolean,
default: true, default: true
}, },
modelValue: { modelValue: {
type: Array, type: Array,
default: () => [], default: () => []
}, },
maxFileSizeMb: { maxFileSizeMb: {
type: Number, type: Number,
default: 200, default: 200
}, }
}) })
const emit = defineEmits(['update:modelValue', 'files-added']) const emit = defineEmits(['update:modelValue', 'files-added'])
@@ -166,7 +170,7 @@ const removeFile = (fileToRemove) => {
} }
const formatSize = (size) => { const formatSize = (size) => {
if (!size) return '0 B' if (!size) { return '0 B' }
const units = ['B', 'KB', 'MB', 'GB'] const units = ['B', 'KB', 'MB', 'GB']
const index = Math.floor(Math.log(size) / Math.log(1024)) const index = Math.floor(Math.log(size) / Math.log(1024))
const formatted = size / Math.pow(1024, index) const formatted = size / Math.pow(1024, index)

View File

@@ -1,8 +1,10 @@
<template> <template>
<dialog class="modal" :class="{ 'modal-open': open }"> <dialog class="modal" :class="{ 'modal-open': open }">
<div class="modal-box max-w-3xl"> <div class="modal-box max-w-3xl">
<form method="dialog" class="modal-close" @submit.prevent></form> <form method="dialog" class="modal-close" @submit.prevent />
<h3 class="font-bold text-lg mb-2">Préparer l'impression</h3> <h3 class="font-bold text-lg mb-2">
Préparer l'impression
</h3>
<p class="text-sm text-base-content/70 mb-4"> <p class="text-sm text-base-content/70 mb-4">
Choisissez les sections à inclure avant de lancer l'impression. Choisissez les sections à inclure avant de lancer l'impression.
</p> </p>
@@ -23,10 +25,10 @@
</h4> </h4>
<label class="flex items-start gap-3"> <label class="flex items-start gap-3">
<input <input
v-model="selection.machine.info"
type="checkbox" type="checkbox"
class="checkbox checkbox-primary mt-1" class="checkbox checkbox-primary mt-1"
v-model="selection.machine.info" >
/>
<div> <div>
<p class="font-medium">Informations générales</p> <p class="font-medium">Informations générales</p>
<p class="text-xs text-base-content/60"> <p class="text-xs text-base-content/60">
@@ -36,10 +38,10 @@
</label> </label>
<label class="flex items-start gap-3"> <label class="flex items-start gap-3">
<input <input
v-model="selection.machine.customFields"
type="checkbox" type="checkbox"
class="checkbox checkbox-primary mt-1" class="checkbox checkbox-primary mt-1"
v-model="selection.machine.customFields" >
/>
<div> <div>
<p class="font-medium">Champs personnalisés</p> <p class="font-medium">Champs personnalisés</p>
<p class="text-xs text-base-content/60"> <p class="text-xs text-base-content/60">
@@ -49,10 +51,10 @@
</label> </label>
<label class="flex items-start gap-3"> <label class="flex items-start gap-3">
<input <input
v-model="selection.machine.documents"
type="checkbox" type="checkbox"
class="checkbox checkbox-primary mt-1" class="checkbox checkbox-primary mt-1"
v-model="selection.machine.documents" >
/>
<div> <div>
<p class="font-medium">Documents</p> <p class="font-medium">Documents</p>
<p class="text-xs text-base-content/60"> <p class="text-xs text-base-content/60">
@@ -62,7 +64,7 @@
</label> </label>
</section> </section>
<section class="bg-base-200/30 rounded-xl p-4 space-y-3" v-if="hasComponents"> <section v-if="hasComponents" class="bg-base-200/30 rounded-xl p-4 space-y-3">
<h4 class="font-semibold text-sm uppercase tracking-wide text-base-content/70"> <h4 class="font-semibold text-sm uppercase tracking-wide text-base-content/70">
Composants & pièces Composants & pièces
</h4> </h4>
@@ -76,7 +78,7 @@
</div> </div>
</section> </section>
<section class="bg-base-200/30 rounded-xl p-4 space-y-3" v-if="hasPieces"> <section v-if="hasPieces" class="bg-base-200/30 rounded-xl p-4 space-y-3">
<h4 class="font-semibold text-sm uppercase tracking-wide text-base-content/70"> <h4 class="font-semibold text-sm uppercase tracking-wide text-base-content/70">
Pièces indépendantes Pièces indépendantes
</h4> </h4>
@@ -87,10 +89,10 @@
class="flex items-start gap-3" class="flex items-start gap-3"
> >
<input <input
v-model="selection.pieces[piece.id]"
type="checkbox" type="checkbox"
class="checkbox checkbox-secondary mt-1" class="checkbox checkbox-secondary mt-1"
v-model="selection.pieces[piece.id]" >
/>
<div> <div>
<p class="font-medium">{{ piece.name }}</p> <p class="font-medium">{{ piece.name }}</p>
<p class="text-xs text-base-content/60"> <p class="text-xs text-base-content/60">
@@ -122,7 +124,7 @@ const props = defineProps({
open: { type: Boolean, default: false }, open: { type: Boolean, default: false },
selection: { type: Object, required: true }, selection: { type: Object, required: true },
components: { type: Array, default: () => [] }, components: { type: Array, default: () => [] },
pieces: { type: Array, default: () => [] }, pieces: { type: Array, default: () => [] }
}) })
const emit = defineEmits(['close', 'confirm', 'select-all', 'deselect-all']) const emit = defineEmits(['close', 'confirm', 'select-all', 'deselect-all'])

View File

@@ -2,10 +2,10 @@
<div class="rounded-lg border border-base-300 bg-base-100/80 p-3 space-y-3"> <div class="rounded-lg border border-base-300 bg-base-100/80 p-3 space-y-3">
<label class="flex items-start gap-3"> <label class="flex items-start gap-3">
<input <input
v-model="selection.components[component.id]"
type="checkbox" type="checkbox"
class="checkbox checkbox-primary mt-1" class="checkbox checkbox-primary mt-1"
v-model="selection.components[component.id]" >
/>
<div class="flex-1"> <div class="flex-1">
<p class="font-medium">{{ component.name }}</p> <p class="font-medium">{{ component.name }}</p>
<p v-if="component.reference" class="text-xs text-base-content/60"> <p v-if="component.reference" class="text-xs text-base-content/60">
@@ -24,10 +24,10 @@
class="flex items-start gap-3" class="flex items-start gap-3"
> >
<input <input
v-model="selection.pieces[piece.id]"
type="checkbox" type="checkbox"
class="checkbox checkbox-secondary mt-1" class="checkbox checkbox-secondary mt-1"
v-model="selection.pieces[piece.id]" >
/>
<div> <div>
<p class="font-medium">{{ piece.name }}</p> <p class="font-medium">{{ piece.name }}</p>
<p class="text-xs text-base-content/60"> <p class="text-xs text-base-content/60">
@@ -55,7 +55,7 @@ defineOptions({ name: 'MachinePrintSelectionNode' })
const props = defineProps({ const props = defineProps({
component: { type: Object, required: true }, component: { type: Object, required: true },
selection: { type: Object, required: true }, selection: { type: Object, required: true }
}) })
const childComponents = computed(() => props.component.subComponents || []) const childComponents = computed(() => props.component.subComponents || [])

View File

@@ -1,16 +1,18 @@
<template> <template>
<div class="space-y-4"> <div class="space-y-4">
<div class="flex flex-wrap items-center gap-2 text-sm text-gray-600"> <div class="flex flex-wrap items-center gap-2 text-sm text-gray-600">
<span class="badge badge-outline badge-sm" v-if="stats.customFields">{{ stats.customFields }} champ(s)</span> <span v-if="stats.customFields" class="badge badge-outline badge-sm">{{ stats.customFields }} champ(s)</span>
<span class="badge badge-outline badge-sm" v-if="stats.pieces">{{ stats.pieces }} pièce(s)</span> <span v-if="stats.pieces" class="badge badge-outline badge-sm">{{ stats.pieces }} pièce(s)</span>
<span class="badge badge-outline badge-sm" v-if="stats.subComponents">{{ stats.subComponents }} sous-composant(s)</span> <span v-if="stats.subComponents" class="badge badge-outline badge-sm">{{ stats.subComponents }} sous-composant(s)</span>
<span v-if="!stats.customFields && !stats.pieces && !stats.subComponents" class="text-xs text-gray-500"> <span v-if="!stats.customFields && !stats.pieces && !stats.subComponents" class="text-xs text-gray-500">
Structure vide Structure vide
</span> </span>
</div> </div>
<details class="collapse collapse-arrow bg-base-200"> <details class="collapse collapse-arrow bg-base-200">
<summary class="collapse-title text-sm font-medium">Voir la structure JSON</summary> <summary class="collapse-title text-sm font-medium">
Voir la structure JSON
</summary>
<div class="collapse-content"> <div class="collapse-content">
<pre class="mockup-code whitespace-pre-wrap text-xs bg-base-300 p-4 rounded"> <pre class="mockup-code whitespace-pre-wrap text-xs bg-base-300 p-4 rounded">
<code>{{ formatted }}</code> <code>{{ formatted }}</code>
@@ -27,8 +29,8 @@ import { computeStructureStats } from '~/shared/modelUtils'
const props = defineProps({ const props = defineProps({
structure: { structure: {
type: Object, type: Object,
default: () => ({}), default: () => ({})
}, }
}) })
const stats = computed(() => computeStructureStats(props.structure)) const stats = computed(() => computeStructureStats(props.structure))

View File

@@ -5,7 +5,9 @@
<component :is="headingTag" v-if="title" class="text-4xl font-bold"> <component :is="headingTag" v-if="title" class="text-4xl font-bold">
{{ title }} {{ title }}
</component> </component>
<p v-if="subtitle" class="text-sm opacity-90">{{ subtitle }}</p> <p v-if="subtitle" class="text-sm opacity-90">
{{ subtitle }}
</p>
<slot /> <slot />
</div> </div>
</div> </div>
@@ -18,41 +20,41 @@ import { computed } from 'vue'
const props = defineProps({ const props = defineProps({
title: { title: {
type: String, type: String,
default: '', default: ''
}, },
subtitle: { subtitle: {
type: String, type: String,
default: '', default: ''
}, },
gradientFrom: { gradientFrom: {
type: String, type: String,
default: 'from-primary', default: 'from-primary'
}, },
gradientTo: { gradientTo: {
type: String, type: String,
default: 'to-secondary', default: 'to-secondary'
}, },
minHeight: { minHeight: {
type: String, type: String,
default: 'min-h-[25vh]', default: 'min-h-[25vh]'
}, },
maxWidth: { maxWidth: {
type: String, type: String,
default: 'max-w-xl', default: 'max-w-xl'
}, },
rounded: { rounded: {
type: Boolean, type: Boolean,
default: false, default: false
}, },
alignment: { alignment: {
type: String, type: String,
default: 'center', default: 'center',
validator: (value) => ['center', 'start', 'end'].includes(value), validator: value => ['center', 'start', 'end'].includes(value)
}, },
headingTag: { headingTag: {
type: String, type: String,
default: 'h1', default: 'h1'
}, }
}) })
const sectionClasses = computed(() => { const sectionClasses = computed(() => {

View File

@@ -12,14 +12,14 @@
class="w-4 h-4 text-purple-500" class="w-4 h-4 text-purple-500"
aria-hidden="true" aria-hidden="true"
/> />
<input <input
v-if="isEditMode" v-if="isEditMode"
:id="`piece-name-${piece.id}`" :id="`piece-name-${piece.id}`"
v-model="pieceData.name" v-model="pieceData.name"
type="text" type="text"
class="font-semibold text-lg input input-sm input-bordered" class="font-semibold text-lg input input-sm input-bordered"
@blur="updatePiece" @blur="updatePiece"
/> >
<div v-else class="font-semibold text-lg input input-sm input-bordered bg-base-200"> <div v-else class="font-semibold text-lg input input-sm input-bordered bg-base-200">
{{ pieceData.name }} {{ pieceData.name }}
</div> </div>
@@ -45,19 +45,19 @@
<div class="space-y-2 text-sm"> <div class="space-y-2 text-sm">
<div> <div>
<span class="font-medium">Référence:</span> <span class="font-medium">Référence:</span>
<input <input
v-if="isEditMode" v-if="isEditMode"
:id="`piece-reference-${piece.id}`" :id="`piece-reference-${piece.id}`"
v-model="pieceData.reference" v-model="pieceData.reference"
type="text" type="text"
class="input input-sm input-bordered ml-2" class="input input-sm input-bordered ml-2"
@blur="updatePiece" @blur="updatePiece"
/> >
<span v-else class="ml-2">{{ pieceData.reference || 'Non définie' }}</span> <span v-else class="ml-2">{{ pieceData.reference || 'Non définie' }}</span>
</div> </div>
<div> <div>
<span class="font-medium">Constructeur:</span> <span class="font-medium">Constructeur:</span>
<span v-if="!isEditMode" class="ml-2"> <span v-if="!isEditMode" class="ml-2">
<span class="font-medium">{{ piece.constructeur?.name || 'Non défini' }}</span> <span class="font-medium">{{ piece.constructeur?.name || 'Non défini' }}</span>
<span v-if="piece.constructeur" class="block text-xs text-gray-500"> <span v-if="piece.constructeur" class="block text-xs text-gray-500">
@@ -68,20 +68,20 @@
v-else v-else
class="w-full" class="w-full"
:model-value="piece.constructeurId || piece.constructeur?.id || null" :model-value="piece.constructeurId || piece.constructeur?.id || null"
@update:modelValue="handleConstructeurChange" @update:model-value="handleConstructeurChange"
/> />
</div> </div>
<div> <div>
<span class="font-medium">Prix:</span> <span class="font-medium">Prix:</span>
<input <input
v-if="isEditMode" v-if="isEditMode"
:id="`piece-prix-${piece.id}`" :id="`piece-prix-${piece.id}`"
v-model="pieceData.prix" v-model="pieceData.prix"
type="number" type="number"
step="0.01" step="0.01"
class="input input-sm input-bordered ml-2" class="input input-sm input-bordered ml-2"
@blur="updatePiece" @blur="updatePiece"
/> >
<span v-else class="ml-2">{{ pieceData.prix ? `${pieceData.prix}` : 'Non défini' }}</span> <span v-else class="ml-2">{{ pieceData.prix ? `${pieceData.prix}` : 'Non défini' }}</span>
</div> </div>
</div> </div>
@@ -90,7 +90,7 @@
v-if="isEditMode && piece.typeMachinePieceRequirement" v-if="isEditMode && piece.typeMachinePieceRequirement"
class="mt-3" class="mt-3"
> >
<label class="label"> <label class="label">
<span class="label-text text-sm font-medium">Modèle de pièce</span> <span class="label-text text-sm font-medium">Modèle de pièce</span>
<span class="label-text-alt text-xs"> <span class="label-text-alt text-xs">
{{ piece.typeMachinePieceRequirement.label || piece.typeMachinePieceRequirement.typePiece?.name || 'Groupe' }} {{ piece.typeMachinePieceRequirement.label || piece.typeMachinePieceRequirement.typePiece?.name || 'Groupe' }}
@@ -101,7 +101,9 @@
class="select select-bordered select-sm w-full" class="select select-bordered select-sm w-full"
@change="assignPieceModel($event.target.value)" @change="assignPieceModel($event.target.value)"
> >
<option value="">Définir manuellement</option> <option value="">
Définir manuellement
</option>
<option <option
v-for="model in pieceModelOptions" v-for="model in pieceModelOptions"
:key="model.id" :key="model.id"
@@ -114,10 +116,12 @@
<!-- Champs personnalisés de la pièce --> <!-- Champs personnalisés de la pièce -->
<div v-if="piece.customFieldValues && piece.customFieldValues.length > 0" class="mt-4 pt-4 border-t border-gray-200"> <div v-if="piece.customFieldValues && piece.customFieldValues.length > 0" class="mt-4 pt-4 border-t border-gray-200">
<h5 class="text-sm font-medium text-gray-700 mb-3">Champs personnalisés</h5> <h5 class="text-sm font-medium text-gray-700 mb-3">
Champs personnalisés
</h5>
<div class="space-y-3"> <div class="space-y-3">
<div <div
v-for="fieldValue in piece.customFieldValues" v-for="fieldValue in piece.customFieldValues"
:key="fieldValue.id" :key="fieldValue.id"
class="form-control" class="form-control"
> >
@@ -125,75 +129,77 @@
<span class="label-text text-sm">{{ fieldValue.customField.name }}</span> <span class="label-text text-sm">{{ fieldValue.customField.name }}</span>
<span v-if="fieldValue.customField.required" class="label-text-alt text-error">*</span> <span v-if="fieldValue.customField.required" class="label-text-alt text-error">*</span>
</label> </label>
<!-- Mode édition --> <!-- Mode édition -->
<template v-if="isEditMode"> <template v-if="isEditMode">
<!-- Champ de type TEXT --> <!-- Champ de type TEXT -->
<input <input
v-if="fieldValue.customField.type === 'text'" v-if="fieldValue.customField.type === 'text'"
:value="fieldValue.value" :value="fieldValue.value"
@input="setCustomFieldValue(fieldValue.id, $event.target.value)"
type="text" type="text"
class="input input-bordered input-sm" class="input input-bordered input-sm"
:required="fieldValue.customField.required" :required="fieldValue.customField.required"
@input="setCustomFieldValue(fieldValue.id, $event.target.value)"
@blur="updateCustomFieldValue(fieldValue.id)" @blur="updateCustomFieldValue(fieldValue.id)"
/> >
<!-- Champ de type NUMBER --> <!-- Champ de type NUMBER -->
<input <input
v-else-if="fieldValue.customField.type === 'number'" v-else-if="fieldValue.customField.type === 'number'"
:value="fieldValue.value" :value="fieldValue.value"
@input="setCustomFieldValue(fieldValue.id, $event.target.value)"
type="number" type="number"
class="input input-bordered input-sm" class="input input-bordered input-sm"
:required="fieldValue.customField.required" :required="fieldValue.customField.required"
@blur="updateCustomFieldValue(fieldValue.id)" @input="setCustomFieldValue(fieldValue.id, $event.target.value)"
/>
<!-- Champ de type SELECT -->
<select
v-else-if="fieldValue.customField.type === 'select'"
:value="fieldValue.value"
@change="setCustomFieldValue(fieldValue.id, $event.target.value)"
class="select select-bordered select-sm"
:required="fieldValue.customField.required"
@blur="updateCustomFieldValue(fieldValue.id)" @blur="updateCustomFieldValue(fieldValue.id)"
> >
<option value="">Sélectionner...</option>
<option <!-- Champ de type SELECT -->
v-for="option in fieldValue.customField.options" <select
:key="option" v-else-if="fieldValue.customField.type === 'select'"
:value="fieldValue.value"
class="select select-bordered select-sm"
:required="fieldValue.customField.required"
@change="setCustomFieldValue(fieldValue.id, $event.target.value)"
@blur="updateCustomFieldValue(fieldValue.id)"
>
<option value="">
Sélectionner...
</option>
<option
v-for="option in fieldValue.customField.options"
:key="option"
:value="option" :value="option"
> >
{{ option }} {{ option }}
</option> </option>
</select> </select>
<!-- Champ de type BOOLEAN --> <!-- Champ de type BOOLEAN -->
<div v-else-if="fieldValue.customField.type === 'boolean'" class="flex items-center gap-2"> <div v-else-if="fieldValue.customField.type === 'boolean'" class="flex items-center gap-2">
<input <input
:value="fieldValue.value" :value="fieldValue.value"
@change="setCustomFieldValue(fieldValue.id, $event.target.checked ? 'true' : 'false')"
type="checkbox" type="checkbox"
class="checkbox checkbox-sm" class="checkbox checkbox-sm"
:checked="fieldValue.value === 'true'" :checked="fieldValue.value === 'true'"
@change="setCustomFieldValue(fieldValue.id, $event.target.checked ? 'true' : 'false')"
@blur="updateCustomFieldValue(fieldValue.id)" @blur="updateCustomFieldValue(fieldValue.id)"
/> >
<span class="text-sm">{{ fieldValue.value === 'true' ? 'Oui' : 'Non' }}</span> <span class="text-sm">{{ fieldValue.value === 'true' ? 'Oui' : 'Non' }}</span>
</div> </div>
<!-- Champ de type DATE --> <!-- Champ de type DATE -->
<input <input
v-else-if="fieldValue.customField.type === 'date'" v-else-if="fieldValue.customField.type === 'date'"
:value="fieldValue.value" :value="fieldValue.value"
@input="setCustomFieldValue(fieldValue.id, $event.target.value)"
type="date" type="date"
class="input input-bordered input-sm" class="input input-bordered input-sm"
:required="fieldValue.customField.required" :required="fieldValue.customField.required"
@input="setCustomFieldValue(fieldValue.id, $event.target.value)"
@blur="updateCustomFieldValue(fieldValue.id)" @blur="updateCustomFieldValue(fieldValue.id)"
/> >
</template> </template>
<!-- Mode lecture seule --> <!-- Mode lecture seule -->
<template v-else> <template v-else>
<div class="input input-bordered input-sm bg-base-200"> <div class="input input-bordered input-sm bg-base-200">
@@ -206,13 +212,17 @@
<div class="mt-4 pt-4 border-t border-gray-200 space-y-3"> <div class="mt-4 pt-4 border-t border-gray-200 space-y-3">
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<h5 class="text-sm font-medium text-gray-700">Documents</h5> <h5 class="text-sm font-medium text-gray-700">
Documents
</h5>
<span v-if="isEditMode && selectedFiles.length" class="badge badge-outline"> <span v-if="isEditMode && selectedFiles.length" class="badge badge-outline">
{{ selectedFiles.length }} fichier{{ selectedFiles.length > 1 ? 's' : '' }} sélectionné{{ selectedFiles.length > 1 ? 's' : '' }} {{ selectedFiles.length }} fichier{{ selectedFiles.length > 1 ? 's' : '' }} sélectionné{{ selectedFiles.length > 1 ? 's' : '' }}
</span> </span>
</div> </div>
<p v-if="loadingDocuments" class="text-xs text-gray-500">Chargement des documents...</p> <p v-if="loadingDocuments" class="text-xs text-gray-500">
Chargement des documents...
</p>
<DocumentUpload <DocumentUpload
v-if="isEditMode" v-if="isEditMode"
@@ -237,7 +247,9 @@
/> />
</span> </span>
<div> <div>
<div class="font-medium">{{ document.name }}</div> <div class="font-medium">
{{ document.name }}
</div>
<div class="text-xs text-gray-500"> <div class="text-xs text-gray-500">
{{ document.mimeType || 'Inconnu' }} {{ formatSize(document.size) }} {{ document.mimeType || 'Inconnu' }} {{ formatSize(document.size) }}
</div> </div>
@@ -268,13 +280,16 @@
</div> </div>
</div> </div>
</div> </div>
<p v-else-if="!loadingDocuments" class="text-xs text-gray-500">Aucun document lié à cette pièce.</p> <p v-else-if="!loadingDocuments" class="text-xs text-gray-500">
Aucun document lié à cette pièce.
</p>
</div> </div>
</div> </div>
</template> </template>
<script setup> <script setup>
import { reactive, onMounted, watch, ref, computed } from 'vue' import { reactive, onMounted, watch, ref, computed } from 'vue'
import ConstructeurSelect from './ConstructeurSelect.vue'
import { useCustomFields } from '~/composables/useCustomFields' import { useCustomFields } from '~/composables/useCustomFields'
import { useToast } from '~/composables/useToast' import { useToast } from '~/composables/useToast'
import { useDocuments } from '~/composables/useDocuments' import { useDocuments } from '~/composables/useDocuments'
@@ -282,22 +297,21 @@ import { getFileIcon } from '~/utils/fileIcons'
import { canPreviewDocument } from '~/utils/documentPreview' import { canPreviewDocument } from '~/utils/documentPreview'
import DocumentUpload from '~/components/DocumentUpload.vue' import DocumentUpload from '~/components/DocumentUpload.vue'
import DocumentPreviewModal from '~/components/DocumentPreviewModal.vue' import DocumentPreviewModal from '~/components/DocumentPreviewModal.vue'
import ConstructeurSelect from './ConstructeurSelect.vue'
import IconLucidePackage from '~icons/lucide/package' import IconLucidePackage from '~icons/lucide/package'
const props = defineProps({ const props = defineProps({
piece: { piece: {
type: Object, type: Object,
required: true, required: true
}, },
isEditMode: { isEditMode: {
type: Boolean, type: Boolean,
default: false, default: false
}, },
pieceModelOptions: { pieceModelOptions: {
type: Array, type: Array,
default: () => [], default: () => []
}, }
}) })
const emit = defineEmits(['update', 'edit', 'custom-field-update', 'assign-model']) const emit = defineEmits(['update', 'edit', 'custom-field-update', 'assign-model'])
@@ -314,7 +328,7 @@ const uploadingDocuments = ref(false)
const loadingDocuments = ref(false) const loadingDocuments = ref(false)
const documentsLoaded = ref(!!(props.piece.documents && props.piece.documents.length)) const documentsLoaded = ref(!!(props.piece.documents && props.piece.documents.length))
const pieceDocuments = computed(() => props.piece.documents || []) const pieceDocuments = computed(() => props.piece.documents || [])
const documentIcon = (doc) => getFileIcon({ name: doc.filename || doc.name, mime: doc.mimeType }) const documentIcon = doc => getFileIcon({ name: doc.filename || doc.name, mime: doc.mimeType })
const previewDocument = ref(null) const previewDocument = ref(null)
const previewVisible = ref(false) const previewVisible = ref(false)
const selectedPieceModelId = computed(() => props.piece.pieceModelId || props.piece.pieceModel?.id || '') const selectedPieceModelId = computed(() => props.piece.pieceModelId || props.piece.pieceModel?.id || '')
@@ -328,7 +342,7 @@ const handleConstructeurChange = (value) => {
const { uploadDocuments, deleteDocument, loadDocumentsByPiece } = useDocuments() const { uploadDocuments, deleteDocument, loadDocumentsByPiece } = useDocuments()
const refreshDocuments = async () => { const refreshDocuments = async () => {
if (!props.piece?.id) return if (!props.piece?.id) { return }
loadingDocuments.value = true loadingDocuments.value = true
try { try {
const result = await loadDocumentsByPiece(props.piece.id, { updateStore: false }) const result = await loadDocumentsByPiece(props.piece.id, { updateStore: false })
@@ -342,7 +356,7 @@ const refreshDocuments = async () => {
} }
const handleFilesAdded = async (files) => { const handleFilesAdded = async (files) => {
if (!files.length || !props.piece?.id) return if (!files.length || !props.piece?.id) { return }
uploadingDocuments.value = true uploadingDocuments.value = true
try { try {
const result = await uploadDocuments( const result = await uploadDocuments(
@@ -365,7 +379,7 @@ const handleFilesAdded = async (files) => {
} }
const removeDocument = async (documentId) => { const removeDocument = async (documentId) => {
if (!documentId) return if (!documentId) { return }
const result = await deleteDocument(documentId, { updateStore: false }) const result = await deleteDocument(documentId, { updateStore: false })
if (result.success) { if (result.success) {
props.piece.documents = (props.piece.documents || []).filter(doc => doc.id !== documentId) props.piece.documents = (props.piece.documents || []).filter(doc => doc.id !== documentId)
@@ -373,7 +387,7 @@ const removeDocument = async (documentId) => {
} }
const downloadDocument = (doc) => { const downloadDocument = (doc) => {
if (!doc?.path) return if (!doc?.path) { return }
if (doc.path.startsWith('data:')) { if (doc.path.startsWith('data:')) {
const link = document.createElement('a') const link = document.createElement('a')
@@ -387,7 +401,7 @@ const downloadDocument = (doc) => {
} }
const openPreview = (doc) => { const openPreview = (doc) => {
if (!canPreviewDocument(doc)) return if (!canPreviewDocument(doc)) { return }
previewDocument.value = doc previewDocument.value = doc
previewVisible.value = true previewVisible.value = true
} }
@@ -398,8 +412,8 @@ const closePreview = () => {
} }
const formatSize = (size) => { const formatSize = (size) => {
if (size === undefined || size === null) return '—' if (size === undefined || size === null) { return '—' }
if (size === 0) return '0 B' if (size === 0) { return '0 B' }
const units = ['B', 'KB', 'MB', 'GB'] const units = ['B', 'KB', 'MB', 'GB']
const index = Math.min(units.length - 1, Math.floor(Math.log(size) / Math.log(1024))) const index = Math.min(units.length - 1, Math.floor(Math.log(size) / Math.log(1024)))
const formatted = size / Math.pow(1024, index) const formatted = size / Math.pow(1024, index)
@@ -413,7 +427,6 @@ watch(
} }
) )
// Méthodes pour gérer les champs personnalisés // Méthodes pour gérer les champs personnalisés
const setCustomFieldValue = (fieldValueId, value) => { const setCustomFieldValue = (fieldValueId, value) => {
const fieldValue = props.piece.customFieldValues?.find(fv => fv.id === fieldValueId) const fieldValue = props.piece.customFieldValues?.find(fv => fv.id === fieldValueId)
@@ -428,7 +441,7 @@ const updatePiece = () => {
...props.piece, ...props.piece,
...pieceData, ...pieceData,
prix: prixValue && prixValue !== '' ? parseFloat(prixValue) : null, prix: prixValue && prixValue !== '' ? parseFloat(prixValue) : null,
constructeurId: props.piece.constructeurId || null, constructeurId: props.piece.constructeurId || null
}) })
} }
@@ -443,7 +456,7 @@ const assignPieceModel = (value) => {
pieceId: props.piece.id, pieceId: props.piece.id,
pieceModelId: value || null, pieceModelId: value || null,
previousModelId, previousModelId,
previousModel, previousModel
}) })
} }
@@ -452,7 +465,7 @@ const updateCustomFieldValue = async (fieldValueId) => {
if (fieldValue) { if (fieldValue) {
const { updateCustomFieldValue } = useCustomFields() const { updateCustomFieldValue } = useCustomFields()
const { showSuccess, showError } = useToast() const { showSuccess, showError } = useToast()
const result = await updateCustomFieldValue(fieldValueId, { value: fieldValue.value }) const result = await updateCustomFieldValue(fieldValueId, { value: fieldValue.value })
if (result.success) { if (result.success) {
showSuccess(`Champ "${fieldValue.customField.name}" mis à jour avec succès`) showSuccess(`Champ "${fieldValue.customField.name}" mis à jour avec succès`)
@@ -473,7 +486,7 @@ watch(
pieceData.name = props.piece.name || '' pieceData.name = props.piece.name || ''
pieceData.reference = props.piece.reference || '' pieceData.reference = props.piece.reference || ''
pieceData.prix = props.piece.prix || '' pieceData.prix = props.piece.prix || ''
}, }
) )
onMounted(() => { onMounted(() => {
@@ -481,7 +494,7 @@ onMounted(() => {
pieceData.name = props.piece.name || '' pieceData.name = props.piece.name || ''
pieceData.reference = props.piece.reference || '' pieceData.reference = props.piece.reference || ''
pieceData.prix = props.piece.prix || '' pieceData.prix = props.piece.prix || ''
// Debug: vérifier si les champs personnalisés sont présents // Debug: vérifier si les champs personnalisés sont présents
console.log('PieceItem - piece:', props.piece) console.log('PieceItem - piece:', props.piece)
console.log('PieceItem - customFieldValues:', props.piece.customFieldValues) console.log('PieceItem - customFieldValues:', props.piece.customFieldValues)
@@ -490,4 +503,4 @@ onMounted(() => {
refreshDocuments() refreshDocuments()
} }
}) })
</script> </script>

View File

@@ -2,7 +2,9 @@
<div class="space-y-4"> <div class="space-y-4">
<section class="space-y-3"> <section class="space-y-3">
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<h3 class="text-sm font-semibold">Champs personnalisés</h3> <h3 class="text-sm font-semibold">
Champs personnalisés
</h3>
<button type="button" class="btn btn-outline btn-xs" @click="addField"> <button type="button" class="btn btn-outline btn-xs" @click="addField">
<IconLucidePlus class="w-3 h-3 mr-2" aria-hidden="true" /> <IconLucidePlus class="w-3 h-3 mr-2" aria-hidden="true" />
Ajouter Ajouter
@@ -27,18 +29,28 @@
type="text" type="text"
class="input input-bordered input-xs" class="input input-bordered input-xs"
placeholder="Nom du champ" placeholder="Nom du champ"
/> >
<select v-model="field.type" class="select select-bordered select-xs"> <select v-model="field.type" class="select select-bordered select-xs">
<option value="text">Texte</option> <option value="text">
<option value="number">Nombre</option> Texte
<option value="select">Liste</option> </option>
<option value="boolean">Oui/Non</option> <option value="number">
<option value="date">Date</option> Nombre
</option>
<option value="select">
Liste
</option>
<option value="boolean">
Oui/Non
</option>
<option value="date">
Date
</option>
</select> </select>
</div> </div>
<div class="flex items-center gap-2 text-xs"> <div class="flex items-center gap-2 text-xs">
<input v-model="field.required" type="checkbox" class="checkbox checkbox-xs" /> <input v-model="field.required" type="checkbox" class="checkbox checkbox-xs">
Obligatoire Obligatoire
</div> </div>
@@ -47,7 +59,7 @@
v-model="field.optionsText" v-model="field.optionsText"
class="textarea textarea-bordered textarea-xs h-20" class="textarea textarea-bordered textarea-xs h-20"
placeholder="Option 1&#10;Option 2" placeholder="Option 1&#10;Option 2"
></textarea> />
</div> </div>
<button <button
type="button" type="button"
@@ -71,13 +83,13 @@ import IconLucideTrash from '~icons/lucide/trash'
const props = defineProps({ const props = defineProps({
modelValue: { modelValue: {
type: Object, type: Object,
default: () => ({ customFields: [] }), default: () => ({ customFields: [] })
}, }
}) })
const emit = defineEmits(['update:modelValue']) const emit = defineEmits(['update:modelValue'])
const ensureArray = (value) => (Array.isArray(value) ? value : []) const ensureArray = value => (Array.isArray(value) ? value : [])
const clone = (input, fallback = {}) => { const clone = (input, fallback = {}) => {
try { try {
@@ -104,24 +116,24 @@ const toEditorField = (input = {}) => ({
? input.options.join('\n') ? input.options.join('\n')
: typeof input.optionsText === 'string' : typeof input.optionsText === 'string'
? input.optionsText ? input.optionsText
: '', : ''
}) })
const hydrateFields = (structure = {}) => ensureArray(structure.customFields).map(toEditorField) const hydrateFields = (structure = {}) => ensureArray(structure.customFields).map(toEditorField)
const localState = reactive({ const localState = reactive({
fields: hydrateFields(props.modelValue), fields: hydrateFields(props.modelValue)
}) })
const extraState = reactive({ const extraState = reactive({
rest: clone(extractRest(props.modelValue)), rest: clone(extractRest(props.modelValue))
}) })
const localFields = computed({ const localFields = computed({
get: () => localState.fields, get: () => localState.fields,
set: (value) => { set: (value) => {
localState.fields = ensureArray(value).map(toEditorField) localState.fields = ensureArray(value).map(toEditorField)
}, }
}) })
const normalizeFields = (fields) => { const normalizeFields = (fields) => {
@@ -139,8 +151,8 @@ const normalizeFields = (fields) => {
const raw = typeof field.optionsText === 'string' ? field.optionsText : '' const raw = typeof field.optionsText === 'string' ? field.optionsText : ''
const parsed = raw const parsed = raw
.split(/\r?\n/) .split(/\r?\n/)
.map((option) => option.trim()) .map(option => option.trim())
.filter((option) => option.length > 0) .filter(option => option.length > 0)
options = parsed.length > 0 ? parsed : undefined options = parsed.length > 0 ? parsed : undefined
} }
@@ -155,14 +167,14 @@ const normalizeFields = (fields) => {
let lastEmitted = JSON.stringify({ let lastEmitted = JSON.stringify({
...clone(extraState.rest, {}), ...clone(extraState.rest, {}),
customFields: normalizeFields(props.modelValue?.customFields), customFields: normalizeFields(props.modelValue?.customFields)
}) })
const emitUpdate = () => { const emitUpdate = () => {
const customFields = normalizeFields(localFields.value) const customFields = normalizeFields(localFields.value)
const payload = { const payload = {
...clone(extraState.rest, {}), ...clone(extraState.rest, {}),
customFields, customFields
} }
const serialized = JSON.stringify(payload) const serialized = JSON.stringify(payload)
if (serialized !== lastEmitted) { if (serialized !== lastEmitted) {
@@ -178,7 +190,7 @@ watch(
extraState.rest = clone(extractRest(value), {}) extraState.rest = clone(extractRest(value), {})
lastEmitted = JSON.stringify({ lastEmitted = JSON.stringify({
...clone(extraState.rest, {}), ...clone(extraState.rest, {}),
customFields: normalizeFields(value?.customFields), customFields: normalizeFields(value?.customFields)
}) })
}, },
{ deep: true } { deep: true }

View File

@@ -10,12 +10,28 @@
aria-hidden="true" aria-hidden="true"
/> />
</button> </button>
<input <div class="flex-1">
v-model="node.name" <input
type="text" :list="componentTypeListId"
class="input input-sm input-bordered w-full" v-model="node.typeComposantLabel"
placeholder="Nom du sous-composant" type="search"
/> autocomplete="off"
class="input input-sm input-bordered w-full"
placeholder="Sélectionner une famille de composant"
@change="handleComponentTypeChange(node)"
@blur="handleComponentTypeChange(node)"
/>
<datalist :id="componentTypeListId">
<option
v-for="type in componentTypes"
:key="type.id"
:value="formatComponentTypeOption(type)"
/>
</datalist>
<p class="mt-1 text-[11px] text-gray-500">
{{ node.typeComposantId ? `Sélection : ${getComponentTypeLabel(node.typeComposantId) || 'Inconnue'}` : 'Aucune famille sélectionnée' }}
</p>
</div>
</div> </div>
<span v-if="!expanded && node.description" class="text-xs text-gray-500 truncate"> <span v-if="!expanded && node.description" class="text-xs text-gray-500 truncate">
{{ node.description }} {{ node.description }}
@@ -112,52 +128,43 @@
> >
<div class="flex items-start justify-between gap-2"> <div class="flex items-start justify-between gap-2">
<div class="flex-1 space-y-3"> <div class="flex-1 space-y-3">
<div class="grid grid-cols-1 md:grid-cols-3 gap-2"> <div class="grid grid-cols-1 md:grid-cols-2 gap-3">
<input <div class="form-control">
v-model="piece.name" <label class="label"><span class="label-text">Famille de pièce</span></label>
type="text" <div>
class="input input-bordered input-xs" <input
placeholder="Nom de la pièce" :list="getPieceTypeListId(pieceIndex)"
/> v-model="piece.typePieceLabel"
<input type="search"
v-model="piece.reference" autocomplete="off"
type="text" class="input input-bordered input-xs"
class="input input-bordered input-xs" placeholder="Sélectionner une famille"
placeholder="Référence" @change="handlePieceTypeChange(piece)"
/> @blur="handlePieceTypeChange(piece)"
<input
v-model.number="piece.quantity"
type="number"
min="0"
step="1"
class="input input-bordered input-xs"
placeholder="Quantité"
/>
</div>
<div class="form-control">
<label class="label"><span class="label-text">Famille de pièce</span></label>
<div>
<input
:list="getPieceTypeListId(pieceIndex)"
v-model="piece.typePieceLabel"
type="search"
autocomplete="off"
class="input input-bordered input-xs"
placeholder="Rechercher une famille"
@change="handlePieceTypeChange(piece)"
@blur="handlePieceTypeChange(piece)"
/>
<datalist :id="getPieceTypeListId(pieceIndex)">
<option
v-for="type in pieceTypes"
:key="type.id"
:value="formatPieceTypeOption(type)"
/> />
</datalist> <datalist :id="getPieceTypeListId(pieceIndex)">
<option
v-for="type in pieceTypes"
:key="type.id"
:value="formatPieceTypeOption(type)"
/>
</datalist>
</div>
<p class="mt-1 text-[11px] text-gray-500">
{{ piece.typePieceId ? `Sélection : ${getPieceTypeLabel(piece.typePieceId) || 'Inconnue'}` : 'Aucune famille sélectionnée' }}
</p>
</div>
<div class="form-control">
<label class="label"><span class="label-text">Quantité (optionnel)</span></label>
<input
v-model.number="piece.quantity"
type="number"
min="0"
step="1"
class="input input-bordered input-xs"
placeholder="Quantité"
/>
</div> </div>
<p class="mt-1 text-[11px] text-gray-500">
{{ piece.typePieceId ? `Sélection : ${getPieceTypeLabel(piece.typePieceId) || 'Inconnue'}` : 'Aucune famille sélectionnée' }}
</p>
</div> </div>
</div> </div>
<button type="button" class="btn btn-error btn-xs btn-square" @click="removePiece(pieceIndex)"> <button type="button" class="btn btn-error btn-xs btn-square" @click="removePiece(pieceIndex)">
@@ -188,6 +195,7 @@
:node="sub" :node="sub"
:depth="depth + 1" :depth="depth + 1"
:piece-types="pieceTypes" :piece-types="pieceTypes"
:component-types="componentTypes"
@remove="removeSubComponent(index)" @remove="removeSubComponent(index)"
/> />
</div> </div>
@@ -197,14 +205,14 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { computed, ref, watch } from 'vue' import { computed, ref, watch, getCurrentInstance } from 'vue'
import IconLucideChevronRight from '~icons/lucide/chevron-right' import IconLucideChevronRight from '~icons/lucide/chevron-right'
import IconLucidePlus from '~icons/lucide/plus' import IconLucidePlus from '~icons/lucide/plus'
import IconLucideTrash from '~icons/lucide/trash' import IconLucideTrash from '~icons/lucide/trash'
defineOptions({ name: 'StructureSubComponentEditor' }) defineOptions({ name: 'StructureSubComponentEditor' })
type PieceTypeOption = { type ModelTypeOption = {
id: string id: string
name: string name: string
code?: string | null code?: string | null
@@ -213,17 +221,30 @@ type PieceTypeOption = {
const props = withDefaults(defineProps<{ const props = withDefaults(defineProps<{
node: Record<string, any> node: Record<string, any>
depth?: number depth?: number
pieceTypes?: PieceTypeOption[] pieceTypes?: ModelTypeOption[]
componentTypes?: ModelTypeOption[]
}>(), { }>(), {
depth: 0, depth: 0,
pieceTypes: () => [], pieceTypes: () => [],
componentTypes: () => [],
}) })
const emit = defineEmits(['remove']) const emit = defineEmits(['remove'])
const pieceTypes = computed(() => props.pieceTypes ?? [])
const componentTypes = computed(() => props.componentTypes ?? [])
const instance = getCurrentInstance()
const componentTypeListId = `sub-component-type-options-${instance?.uid ?? 0}`
const formatModelTypeOption = (type: ModelTypeOption | undefined | null) => {
if (!type) return ''
return type.code ? `${type.name} (${type.code})` : type.name
}
const pieceTypeMap = computed(() => { const pieceTypeMap = computed(() => {
const map = new Map<string, PieceTypeOption>() const map = new Map<string, ModelTypeOption>()
;(props.pieceTypes ?? []).forEach((type) => { pieceTypes.value.forEach((type) => {
if (type && typeof type.id === 'string') { if (type && typeof type.id === 'string') {
map.set(type.id, type) map.set(type.id, type)
} }
@@ -231,10 +252,18 @@ const pieceTypeMap = computed(() => {
return map return map
}) })
const formatPieceTypeOption = (type: PieceTypeOption | undefined | null) => { const componentTypeMap = computed(() => {
if (!type) return '' const map = new Map<string, ModelTypeOption>()
return type.code ? `${type.name} (${type.code})` : type.name componentTypes.value.forEach((type) => {
} if (type && typeof type.id === 'string') {
map.set(type.id, type)
}
})
return map
})
const formatPieceTypeOption = (type: ModelTypeOption | undefined | null) => formatModelTypeOption(type)
const formatComponentTypeOption = (type: ModelTypeOption | undefined | null) => formatModelTypeOption(type)
const resolvePieceType = (input: string) => { const resolvePieceType = (input: string) => {
const normalized = input.trim().toLowerCase() const normalized = input.trim().toLowerCase()
@@ -242,7 +271,7 @@ const resolvePieceType = (input: string) => {
return null return null
} }
return ( return (
(props.pieceTypes ?? []).find((type) => { pieceTypes.value.find((type) => {
const formatted = formatPieceTypeOption(type).toLowerCase() const formatted = formatPieceTypeOption(type).toLowerCase()
const name = (type?.name ?? '').toLowerCase() const name = (type?.name ?? '').toLowerCase()
const code = (type?.code ?? '').toLowerCase() const code = (type?.code ?? '').toLowerCase()
@@ -255,24 +284,58 @@ const resolvePieceType = (input: string) => {
) )
} }
const resolveComponentType = (input: string) => {
const normalized = input.trim().toLowerCase()
if (!normalized) {
return null
}
return (
componentTypes.value.find((type) => {
const formatted = formatComponentTypeOption(type).toLowerCase()
const name = (type?.name ?? '').toLowerCase()
const code = (type?.code ?? '').toLowerCase()
return (
formatted === normalized
|| name === normalized
|| (!!code && code === normalized)
)
}) ?? null
)
}
const getPieceTypeLabel = (id?: string) => { const getPieceTypeLabel = (id?: string) => {
if (!id) return '' if (!id) return ''
const option = pieceTypeMap.value.get(id) const option = pieceTypeMap.value.get(id)
return formatPieceTypeOption(option) return formatPieceTypeOption(option)
} }
const getComponentTypeLabel = (id?: string) => {
if (!id) return ''
const option = componentTypeMap.value.get(id)
return formatComponentTypeOption(option)
}
const updatePieceTypeLabel = (piece: any) => { const updatePieceTypeLabel = (piece: any) => {
if (!piece) { if (!piece) {
return return
} }
if (piece.typePieceId) { if (piece.typePieceId) {
const option = pieceTypeMap.value.get(piece.typePieceId) const option = pieceTypeMap.value.get(piece.typePieceId)
piece.typePieceLabel = option ? formatPieceTypeOption(option) : piece.typePieceLabel || '' if (option) {
piece.typePieceLabel = formatPieceTypeOption(option)
piece.name = option.name || formatPieceTypeOption(option)
} else if (!piece.typePieceLabel) {
piece.name = ''
}
} else if (piece.typePieceLabel) { } else if (piece.typePieceLabel) {
const match = resolvePieceType(piece.typePieceLabel) const match = resolvePieceType(piece.typePieceLabel)
if (match) { if (match) {
piece.typePieceId = match.id piece.typePieceId = match.id
piece.typePieceLabel = formatPieceTypeOption(match) piece.typePieceLabel = formatPieceTypeOption(match)
piece.name = match.name || formatPieceTypeOption(match)
} else {
piece.typePieceLabel = ''
piece.name = ''
} }
} }
} }
@@ -285,14 +348,18 @@ const handlePieceTypeChange = (piece: any) => {
if (!value) { if (!value) {
piece.typePieceId = '' piece.typePieceId = ''
piece.typePieceLabel = '' piece.typePieceLabel = ''
piece.name = ''
return return
} }
const match = resolvePieceType(value) const match = resolvePieceType(value)
if (match) { if (match) {
piece.typePieceId = match.id piece.typePieceId = match.id
piece.typePieceLabel = formatPieceTypeOption(match) piece.typePieceLabel = formatPieceTypeOption(match)
piece.name = match.name || formatPieceTypeOption(match)
} else { } else {
piece.typePieceId = '' piece.typePieceId = ''
piece.typePieceLabel = ''
piece.name = ''
} }
} }
@@ -310,38 +377,96 @@ const applyPieceLabels = (pieces?: any[]) => {
if (match) { if (match) {
piece.typePieceId = match.id piece.typePieceId = match.id
piece.typePieceLabel = formatPieceTypeOption(match) piece.typePieceLabel = formatPieceTypeOption(match)
piece.name = match.name || formatPieceTypeOption(match)
} else {
piece.typePieceLabel = ''
piece.name = ''
} }
} else if (!piece?.name) {
piece.name = ''
} }
}) })
} }
const applyComponentTypeLabel = (component: any) => {
if (!component) {
return
}
if (component.typeComposantId) {
const option = componentTypeMap.value.get(component.typeComposantId)
if (option) {
component.typeComposantLabel = formatComponentTypeOption(option)
component.name = option.name || formatComponentTypeOption(option)
} else if (!component.typeComposantLabel) {
component.name = ''
}
} else if (component.typeComposantLabel) {
const match = resolveComponentType(component.typeComposantLabel)
if (match) {
component.typeComposantId = match.id
component.typeComposantLabel = formatComponentTypeOption(match)
component.name = match.name || formatComponentTypeOption(match)
} else {
component.typeComposantLabel = ''
component.name = ''
}
}
}
const handleComponentTypeChange = (component: any) => {
if (!component) {
return
}
const value = typeof component.typeComposantLabel === 'string'
? component.typeComposantLabel.trim()
: ''
if (!value) {
component.typeComposantId = ''
component.typeComposantLabel = ''
component.name = ''
return
}
const match = resolveComponentType(value)
if (match) {
component.typeComposantId = match.id
component.typeComposantLabel = formatComponentTypeOption(match)
component.name = match.name || formatComponentTypeOption(match)
} else {
component.typeComposantId = ''
component.typeComposantLabel = ''
component.name = ''
}
}
const traverseSubComponents = (components?: any[]) => { const traverseSubComponents = (components?: any[]) => {
if (!Array.isArray(components)) { if (!Array.isArray(components)) {
return return
} }
components.forEach((component) => { components.forEach((component) => {
applyComponentTypeLabel(component)
applyPieceLabels(component?.pieces) applyPieceLabels(component?.pieces)
traverseSubComponents(component?.subComponents) traverseSubComponents(component?.subComponents)
}) })
} }
const syncPieceTypeLabels = () => { const syncTypeLabels = () => {
applyComponentTypeLabel(props.node)
applyPieceLabels(props.node?.pieces) applyPieceLabels(props.node?.pieces)
traverseSubComponents(props.node?.subComponents) traverseSubComponents(props.node?.subComponents)
} }
watch( watch(pieceTypes, () => {
() => props.pieceTypes, syncTypeLabels()
() => { }, { deep: true, immediate: true })
syncPieceTypeLabels()
}, watch(componentTypes, () => {
{ deep: true, immediate: true } syncTypeLabels()
) }, { deep: true, immediate: true })
watch( watch(
() => props.node, () => props.node,
() => { () => {
syncPieceTypeLabels() syncTypeLabels()
}, },
{ deep: true } { deep: true }
) )
@@ -379,7 +504,6 @@ const addPiece = () => {
ensureArray('pieces') ensureArray('pieces')
props.node.pieces.push({ props.node.pieces.push({
name: '', name: '',
reference: '',
quantity: undefined, quantity: undefined,
typePieceId: '', typePieceId: '',
typePieceLabel: '', typePieceLabel: '',
@@ -397,6 +521,8 @@ const addSubComponent = () => {
name: '', name: '',
description: '', description: '',
quantity: undefined, quantity: undefined,
typeComposantId: '',
typeComposantLabel: '',
customFields: [], customFields: [],
pieces: [], pieces: [],
subComponents: [], subComponents: [],

View File

@@ -38,16 +38,16 @@
aria-hidden="true" aria-hidden="true"
/> />
</div> </div>
<!-- Message --> <!-- Message -->
<div class="flex-1"> <div class="flex-1">
<span class="text-sm font-medium">{{ toast.message }}</span> <span class="text-sm font-medium">{{ toast.message }}</span>
</div> </div>
<!-- Close button --> <!-- Close button -->
<button <button
@click="removeToast(toast.id)"
class="btn btn-ghost btn-xs" class="btn btn-ghost btn-xs"
@click="removeToast(toast.id)"
> >
<IconLucideX class="w-4 h-4" aria-hidden="true" /> <IconLucideX class="w-4 h-4" aria-hidden="true" />
</button> </button>
@@ -102,4 +102,4 @@ const getToastClasses = (type) => {
.toast-move { .toast-move {
transition: transform 0.3s ease; transition: transform 0.3s ease;
} }
</style> </style>

View File

@@ -1,7 +1,7 @@
<template> <template>
<div class="space-y-4"> <div class="space-y-4">
<div class="flex justify-end"> <div class="flex justify-end">
<button <button
type="button" type="button"
class="btn btn-outline btn-sm" class="btn btn-outline btn-sm"
@click="toggleAllComponents" @click="toggleAllComponents"
@@ -10,157 +10,171 @@
class="w-4 h-4 mr-2" class="w-4 h-4 mr-2"
aria-hidden="true" aria-hidden="true"
/> />
</button> </button>
<h5 class="text-sm font-medium">Nouveau composant {{ index + 1 }}</h5> <h5 class="text-sm font-medium">
<span v-if="!isComponentExpanded(index)" class="text-xs text-gray-500 truncate max-w-[160px]"> Nouveau composant {{ index + 1 }}
{{ component.name || 'Sans nom' }} </h5>
</span> <span v-if="!isComponentExpanded(index)" class="text-xs text-gray-500 truncate max-w-[160px]">
</div> {{ component.name || 'Sans nom' }}
<button </span>
type="button"
@click="removeComponent(index)"
class="btn btn-square btn-error btn-sm"
>
<IconLucideChevronRight
class="w-4 h-4"
aria-hidden="true"
/>
</button>
<span class="text-xs font-medium">Champ personnalisé {{ fieldIndex + 1 }}</span>
<span v-if="!isComponentCustomFieldExpanded(index, fieldIndex)" class="text-[10px] text-gray-500 truncate max-w-[120px]">
{{ field.name || 'Sans nom' }}
</span>
</div>
<button
type="button"
@click="removeComponentCustomField(index, fieldIndex)"
class="btn btn-square btn-error btn-xs"
>
<IconLucideChevronRight
class="w-3 h-3"
aria-hidden="true"
/>
</button>
<IconLucideChevronRight
class="w-3 h-3 text-red-500"
aria-hidden="true"
/>
</button>
<span class="text-xs">Champ {{ fieldIndex + 1 }}</span>
<span v-if="!isComponentPieceCustomFieldExpanded(index, pieceIndex, fieldIndex)" class="text-[10px] text-gray-500 truncate max-w-[100px]">
{{ field.name || 'Sans nom' }}
</span>
</div>
<button
type="button"
@click="removeComponentPieceCustomField(index, pieceIndex, fieldIndex)"
class="btn btn-square btn-error btn-xs"
>
<IconLucideChevronRight
class="w-2 h-2"
aria-hidden="true"
/>
</button>
<IconLucideChevronRight
class="w-3 h-3 text-green-500"
aria-hidden="true"
/>
</button>
<span class="text-xs">Champ {{ fieldIndex + 1 }}</span>
<span v-if="!isSubComponentCustomFieldExpanded(index, subIndex, fieldIndex)" class="text-[10px] text-gray-500 truncate max-w-[100px]">
{{ field.name || 'Sans nom' }}
</span>
</div>
<button
type="button"
@click="removeSubComponentCustomField(index, subIndex, fieldIndex)"
class="btn btn-square btn-error btn-xs"
>
<IconLucideChevronRight
class="w-2 h-2"
aria-hidden="true"
/>
</button>
<IconLucideChevronRight
class="w-2 h-2 text-red-500"
aria-hidden="true"
/>
</button>
<span class="text-xs">Champ {{ fieldIndex + 1 }}</span>
<span v-if="!isSubComponentPieceCustomFieldExpanded(index, subIndex, pieceIndex, fieldIndex)" class="text-[10px] text-gray-500 truncate max-w-[100px]">
{{ field.name || 'Sans nom' }}
</span>
</div>
<button
type="button"
@click="removeSubComponentPieceCustomField(index, subIndex, pieceIndex, fieldIndex)"
class="btn btn-square btn-error btn-xs"
>
<IconLucideX
class="w-2 h-2"
aria-hidden="true"
/>
</button>
</div>
<div v-if="isSubComponentPieceCustomFieldExpanded(index, subIndex, pieceIndex, fieldIndex)" class="grid grid-cols-2 gap-1">
<input
v-model="field.name"
type="text"
placeholder="Nom"
class="input input-bordered input-xs"
/>
<select v-model="field.type" class="select select-bordered select-xs">
<option value="">Type</option>
<option value="text">Texte</option>
<option value="number">Nombre</option>
<option value="select">Liste</option>
<option value="boolean">Oui/Non</option>
<option value="date">Date</option>
</select>
</div>
<div v-if="isSubComponentPieceCustomFieldExpanded(index, subIndex, pieceIndex, fieldIndex)" class="mt-1">
<label class="flex items-center gap-1 text-xs">
<input
v-model="field.required"
type="checkbox"
class="checkbox checkbox-xs"
/>
Obligatoire
</label>
</div>
<div v-if="isSubComponentPieceCustomFieldExpanded(index, subIndex, pieceIndex, fieldIndex) && field.type === 'select'" class="mt-1">
<textarea
v-model="field.optionsText"
placeholder="Option 1&#10;Option 2&#10;Option 3"
class="textarea textarea-bordered textarea-xs w-full h-10"
@input="updateSubComponentPieceFieldOptions(index, subIndex, pieceIndex, fieldIndex)"
></textarea>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div> </div>
<button
<button type="button"
type="button" class="btn btn-square btn-error btn-sm"
@click="addComponent" @click="removeComponent(index)"
class="btn btn-outline btn-sm"
> >
<IconLucidePlus <IconLucideChevronRight
class="w-4 h-4 mr-2" class="w-4 h-4"
aria-hidden="true" aria-hidden="true"
/> />
Ajouter un composant
</button> </button>
<span class="text-xs font-medium">Champ personnalisé {{ fieldIndex + 1 }}</span>
<span v-if="!isComponentCustomFieldExpanded(index, fieldIndex)" class="text-[10px] text-gray-500 truncate max-w-[120px]">
{{ field.name || 'Sans nom' }}
</span>
</div>
<button
type="button"
class="btn btn-square btn-error btn-xs"
@click="removeComponentCustomField(index, fieldIndex)"
>
<IconLucideChevronRight
class="w-3 h-3"
aria-hidden="true"
/>
</button>
<IconLucideChevronRight
class="w-3 h-3 text-red-500"
aria-hidden="true"
/>
</button>
<span class="text-xs">Champ {{ fieldIndex + 1 }}</span>
<span v-if="!isComponentPieceCustomFieldExpanded(index, pieceIndex, fieldIndex)" class="text-[10px] text-gray-500 truncate max-w-[100px]">
{{ field.name || 'Sans nom' }}
</span>
</div>
<button
type="button"
class="btn btn-square btn-error btn-xs"
@click="removeComponentPieceCustomField(index, pieceIndex, fieldIndex)"
>
<IconLucideChevronRight
class="w-2 h-2"
aria-hidden="true"
/>
</button>
<IconLucideChevronRight
class="w-3 h-3 text-green-500"
aria-hidden="true"
/>
</button>
<span class="text-xs">Champ {{ fieldIndex + 1 }}</span>
<span v-if="!isSubComponentCustomFieldExpanded(index, subIndex, fieldIndex)" class="text-[10px] text-gray-500 truncate max-w-[100px]">
{{ field.name || 'Sans nom' }}
</span>
</div>
<button
type="button"
class="btn btn-square btn-error btn-xs"
@click="removeSubComponentCustomField(index, subIndex, fieldIndex)"
>
<IconLucideChevronRight
class="w-2 h-2"
aria-hidden="true"
/>
</button>
<IconLucideChevronRight
class="w-2 h-2 text-red-500"
aria-hidden="true"
/>
</button>
<span class="text-xs">Champ {{ fieldIndex + 1 }}</span>
<span v-if="!isSubComponentPieceCustomFieldExpanded(index, subIndex, pieceIndex, fieldIndex)" class="text-[10px] text-gray-500 truncate max-w-[100px]">
{{ field.name || 'Sans nom' }}
</span>
</div>
<button
type="button"
class="btn btn-square btn-error btn-xs"
@click="removeSubComponentPieceCustomField(index, subIndex, pieceIndex, fieldIndex)"
>
<IconLucideX
class="w-2 h-2"
aria-hidden="true"
/>
</button>
</div>
<div v-if="isSubComponentPieceCustomFieldExpanded(index, subIndex, pieceIndex, fieldIndex)" class="grid grid-cols-2 gap-1">
<input
v-model="field.name"
type="text"
placeholder="Nom"
class="input input-bordered input-xs"
>
<select v-model="field.type" class="select select-bordered select-xs">
<option value="">
Type
</option>
<option value="text">
Texte
</option>
<option value="number">
Nombre
</option>
<option value="select">
Liste
</option>
<option value="boolean">
Oui/Non
</option>
<option value="date">
Date
</option>
</select>
</div>
<div v-if="isSubComponentPieceCustomFieldExpanded(index, subIndex, pieceIndex, fieldIndex)" class="mt-1">
<label class="flex items-center gap-1 text-xs">
<input
v-model="field.required"
type="checkbox"
class="checkbox checkbox-xs"
>
Obligatoire
</label>
</div>
<div v-if="isSubComponentPieceCustomFieldExpanded(index, subIndex, pieceIndex, fieldIndex) && field.type === 'select'" class="mt-1">
<textarea
v-model="field.optionsText"
placeholder="Option 1&#10;Option 2&#10;Option 3"
class="textarea textarea-bordered textarea-xs w-full h-10"
@input="updateSubComponentPieceFieldOptions(index, subIndex, pieceIndex, fieldIndex)"
/>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<button
type="button"
class="btn btn-outline btn-sm"
@click="addComponent"
>
<IconLucidePlus
class="w-4 h-4 mr-2"
aria-hidden="true"
/>
Ajouter un composant
</button>
</div> </div>
</template> </template>
@@ -381,7 +395,7 @@ const reorderStore = (store) => {
.forEach((key, position) => { .forEach((key, position) => {
reordered[position] = store[key] reordered[position] = store[key]
}) })
Object.keys(store).forEach((key) => delete store[key]) Object.keys(store).forEach(key => delete store[key])
Object.assign(store, reordered) Object.assign(store, reordered)
} }
@@ -394,19 +408,19 @@ const reorderNestedStore = (store, componentIndex) => {
.forEach((key, position) => { .forEach((key, position) => {
reordered[position] = entries[key] reordered[position] = entries[key]
}) })
Object.keys(entries).forEach((key) => delete entries[key]) Object.keys(entries).forEach(key => delete entries[key])
Object.assign(entries, reordered) Object.assign(entries, reordered)
} }
const clearExpansionState = () => { const clearExpansionState = () => {
expandedComponents.value = [] expandedComponents.value = []
Object.keys(expandedComponentCustomFields).forEach((key) => delete expandedComponentCustomFields[key]) Object.keys(expandedComponentCustomFields).forEach(key => delete expandedComponentCustomFields[key])
Object.keys(expandedComponentPieces).forEach((key) => delete expandedComponentPieces[key]) Object.keys(expandedComponentPieces).forEach(key => delete expandedComponentPieces[key])
Object.keys(expandedComponentPieceCustomFields).forEach((key) => delete expandedComponentPieceCustomFields[key]) Object.keys(expandedComponentPieceCustomFields).forEach(key => delete expandedComponentPieceCustomFields[key])
Object.keys(expandedSubComponents).forEach((key) => delete expandedSubComponents[key]) Object.keys(expandedSubComponents).forEach(key => delete expandedSubComponents[key])
Object.keys(expandedSubComponentCustomFields).forEach((key) => delete expandedSubComponentCustomFields[key]) Object.keys(expandedSubComponentCustomFields).forEach(key => delete expandedSubComponentCustomFields[key])
Object.keys(expandedSubComponentPieces).forEach((key) => delete expandedSubComponentPieces[key]) Object.keys(expandedSubComponentPieces).forEach(key => delete expandedSubComponentPieces[key])
Object.keys(expandedSubComponentPieceCustomFields).forEach((key) => delete expandedSubComponentPieceCustomFields[key]) Object.keys(expandedSubComponentPieceCustomFields).forEach(key => delete expandedSubComponentPieceCustomFields[key])
} }
const setAllExpanded = (value) => { const setAllExpanded = (value) => {
@@ -625,7 +639,7 @@ const removeSubComponent = (componentIndex, subIndex) => {
.forEach((key, position) => { .forEach((key, position) => {
reordered[position] = pieceFieldEntries[key] reordered[position] = pieceFieldEntries[key]
}) })
Object.keys(pieceFieldEntries).forEach((key) => delete pieceFieldEntries[key]) Object.keys(pieceFieldEntries).forEach(key => delete pieceFieldEntries[key])
Object.assign(pieceFieldEntries, reordered) Object.assign(pieceFieldEntries, reordered)
} }
@@ -687,7 +701,7 @@ const removeSubComponentPiece = (componentIndex, subIndex, pieceIndex) => {
.forEach((key, position) => { .forEach((key, position) => {
reordered[position] = store[key] reordered[position] = store[key]
}) })
Object.keys(store || {}).forEach((key) => delete store[key]) Object.keys(store || {}).forEach(key => delete store[key])
Object.assign(store || {}, reordered) Object.assign(store || {}, reordered)
} }
} }

View File

@@ -2,7 +2,7 @@
<div class="card bg-base-100 shadow-lg"> <div class="card bg-base-100 shadow-lg">
<div class="card-body"> <div class="card-body">
<div class="card-actions justify-end"> <div class="card-actions justify-end">
<button type="button" @click="$emit('reset')" class="btn btn-outline"> <button type="button" class="btn btn-outline" @click="$emit('reset')">
Réinitialiser Réinitialiser
</button> </button>
<button type="submit" class="btn btn-primary" :disabled="saving"> <button type="submit" class="btn btn-primary" :disabled="saving">
@@ -26,8 +26,8 @@ import IconLucideCheck from '~icons/lucide/check'
defineProps({ defineProps({
saving: { saving: {
type: Boolean, type: Boolean,
default: false, default: false
}, }
}) })
defineEmits(['reset']) defineEmits(['reset'])

View File

@@ -1,7 +1,9 @@
<template> <template>
<div class="card bg-base-100 shadow-lg"> <div class="card bg-base-100 shadow-lg">
<div class="card-body"> <div class="card-body">
<h3 class="card-title text-lg mb-4">Informations de base</h3> <h3 class="card-title text-lg mb-4">
Informations de base
</h3>
<div class="grid grid-cols-1 md:grid-cols-2 gap-4"> <div class="grid grid-cols-1 md:grid-cols-2 gap-4">
<div class="form-control"> <div class="form-control">
@@ -15,7 +17,7 @@
placeholder="Nom du type de machine" placeholder="Nom du type de machine"
class="input input-bordered" class="input input-bordered"
required required
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
@@ -27,7 +29,7 @@
type="text" type="text"
placeholder="Catégorie du type" placeholder="Catégorie du type"
class="input input-bordered" class="input input-bordered"
/> >
</div> </div>
<div class="form-control md:col-span-2"> <div class="form-control md:col-span-2">
@@ -38,7 +40,7 @@
v-model="descriptionModel" v-model="descriptionModel"
placeholder="Description du type de machine" placeholder="Description du type de machine"
class="textarea textarea-bordered h-24" class="textarea textarea-bordered h-24"
></textarea> />
</div> </div>
<div class="form-control"> <div class="form-control">
@@ -50,7 +52,7 @@
type="text" type="text"
placeholder="ex: Mensuelle, Trimestrielle" placeholder="ex: Mensuelle, Trimestrielle"
class="input input-bordered" class="input input-bordered"
/> >
</div> </div>
</div> </div>
</div> </div>
@@ -63,41 +65,41 @@ import { computed } from 'vue'
const props = defineProps({ const props = defineProps({
name: { name: {
type: String, type: String,
default: '', default: ''
}, },
category: { category: {
type: String, type: String,
default: '', default: ''
}, },
description: { description: {
type: String, type: String,
default: '', default: ''
}, },
maintenanceFrequency: { maintenanceFrequency: {
type: String, type: String,
default: '', default: ''
}, }
}) })
const emit = defineEmits(['update:name', 'update:category', 'update:description', 'update:maintenanceFrequency']) const emit = defineEmits(['update:name', 'update:category', 'update:description', 'update:maintenanceFrequency'])
const nameModel = computed({ const nameModel = computed({
get: () => props.name, get: () => props.name,
set: (value) => emit('update:name', value), set: value => emit('update:name', value)
}) })
const categoryModel = computed({ const categoryModel = computed({
get: () => props.category, get: () => props.category,
set: (value) => emit('update:category', value), set: value => emit('update:category', value)
}) })
const descriptionModel = computed({ const descriptionModel = computed({
get: () => props.description, get: () => props.description,
set: (value) => emit('update:description', value), set: value => emit('update:description', value)
}) })
const maintenanceModel = computed({ const maintenanceModel = computed({
get: () => props.maintenanceFrequency, get: () => props.maintenanceFrequency,
set: (value) => emit('update:maintenanceFrequency', value), set: value => emit('update:maintenanceFrequency', value)
}) })
</script> </script>

View File

@@ -14,7 +14,9 @@
aria-hidden="true" aria-hidden="true"
/> />
</button> </button>
<h3 class="card-title text-lg">Champs personnalisés du type</h3> <h3 class="card-title text-lg">
Champs personnalisés du type
</h3>
<span class="badge badge-primary">{{ fields.length }}</span> <span class="badge badge-primary">{{ fields.length }}</span>
</div> </div>
</div> </div>
@@ -30,8 +32,8 @@
<button <button
type="button" type="button"
class="btn btn-ghost btn-xs p-1" class="btn btn-ghost btn-xs p-1"
@click="toggleField(fieldIndex)"
title="Plier / déplier le champ" title="Plier / déplier le champ"
@click="toggleField(fieldIndex)"
> >
<IconLucideChevronRight <IconLucideChevronRight
class="w-4 h-4 transition-transform duration-200" class="w-4 h-4 transition-transform duration-200"
@@ -40,16 +42,18 @@
/> />
</button> </button>
<IconLucideListChecks class="w-4 h-4 text-blue-500" aria-hidden="true" /> <IconLucideListChecks class="w-4 h-4 text-blue-500" aria-hidden="true" />
<h5 class="text-sm font-medium">Champ personnalisé {{ fieldIndex + 1 }}</h5> <h5 class="text-sm font-medium">
Champ personnalisé {{ fieldIndex + 1 }}
</h5>
<span v-if="!isFieldExpanded(fieldIndex)" class="text-xs text-gray-500 truncate max-w-[160px]"> <span v-if="!isFieldExpanded(fieldIndex)" class="text-xs text-gray-500 truncate max-w-[160px]">
{{ field.name || 'Sans nom' }} {{ field.name || 'Sans nom' }}
</span> </span>
</div> </div>
<button <button
type="button" type="button"
@click="removeField(fieldIndex)"
class="btn btn-square btn-error btn-sm" class="btn btn-square btn-error btn-sm"
title="Supprimer ce champ" title="Supprimer ce champ"
@click="removeField(fieldIndex)"
> >
<IconLucideX class="w-4 h-4" aria-hidden="true" /> <IconLucideX class="w-4 h-4" aria-hidden="true" />
</button> </button>
@@ -68,7 +72,7 @@
class="input input-bordered input-sm" class="input input-bordered input-sm"
required required
@input="updateField(fieldIndex, { name: $event.target.value })" @input="updateField(fieldIndex, { name: $event.target.value })"
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
@@ -82,12 +86,24 @@
:value="field.type" :value="field.type"
@change="updateField(fieldIndex, { type: $event.target.value })" @change="updateField(fieldIndex, { type: $event.target.value })"
> >
<option value="">Sélectionner un type</option> <option value="">
<option value="text">Texte</option> Sélectionner un type
<option value="number">Nombre</option> </option>
<option value="select">Liste déroulante</option> <option value="text">
<option value="boolean">Oui/Non</option> Texte
<option value="date">Date</option> </option>
<option value="number">
Nombre
</option>
<option value="select">
Liste déroulante
</option>
<option value="boolean">
Oui/Non
</option>
<option value="date">
Date
</option>
</select> </select>
</div> </div>
</div> </div>
@@ -99,7 +115,7 @@
class="checkbox checkbox-sm" class="checkbox checkbox-sm"
:checked="field.required" :checked="field.required"
@change="updateField(fieldIndex, { required: $event.target.checked })" @change="updateField(fieldIndex, { required: $event.target.checked })"
/> >
<span class="text-sm">Champ obligatoire</span> <span class="text-sm">Champ obligatoire</span>
</div> </div>
</div> </div>
@@ -117,19 +133,19 @@
placeholder="Option 1&#10;Option 2&#10;Option 3" placeholder="Option 1&#10;Option 2&#10;Option 3"
class="textarea textarea-bordered textarea-sm w-full h-20" class="textarea textarea-bordered textarea-sm w-full h-20"
@input="updateOptions(fieldIndex, $event.target.value)" @input="updateOptions(fieldIndex, $event.target.value)"
></textarea> />
</div> </div>
</div> </div>
<div class="flex justify-end"> <div class="flex justify-end">
<button type="button" @click="addField" class="btn btn-primary btn-sm"> <button type="button" class="btn btn-primary btn-sm" @click="addField">
<IconLucidePlus class="w-4 h-4 mr-2" aria-hidden="true" /> <IconLucidePlus class="w-4 h-4 mr-2" aria-hidden="true" />
Ajouter un champ Ajouter un champ
</button> </button>
</div> </div>
</div> </div>
<div v-else class="flex justify-end"> <div v-else class="flex justify-end">
<button type="button" @click="addField" class="btn btn-primary btn-sm"> <button type="button" class="btn btn-primary btn-sm" @click="addField">
<IconLucidePlus class="w-4 h-4 mr-2" aria-hidden="true" /> <IconLucidePlus class="w-4 h-4 mr-2" aria-hidden="true" />
Ajouter un champ Ajouter un champ
</button> </button>
@@ -148,23 +164,23 @@ import IconLucidePlus from '~icons/lucide/plus'
const props = defineProps({ const props = defineProps({
modelValue: { modelValue: {
type: Array, type: Array,
default: () => [], default: () => []
}, },
allExpanded: { allExpanded: {
type: Boolean, type: Boolean,
default: false, default: false
}, },
expandAllTrigger: { expandAllTrigger: {
type: Number, type: Number,
default: 0, default: 0
}, }
}) })
const emit = defineEmits(['update:modelValue']) const emit = defineEmits(['update:modelValue'])
const fields = computed({ const fields = computed({
get: () => props.modelValue, get: () => props.modelValue,
set: (value) => emit('update:modelValue', value), set: value => emit('update:modelValue', value)
}) })
const expanded = ref(false) const expanded = ref(false)
@@ -213,8 +229,8 @@ const addField = () => {
name: '', name: '',
type: '', type: '',
required: false, required: false,
optionsText: '', optionsText: ''
}, }
] ]
expandedFields.value.push(true) expandedFields.value.push(true)
expanded.value = true expanded.value = true
@@ -231,7 +247,7 @@ const updateField = (index, patch) => {
const updateOptions = (index, value) => { const updateOptions = (index, value) => {
updateField(index, { updateField(index, {
optionsText: value.replace(/\r\n/g, '\n').replace(/\r/g, '\n'), optionsText: value.replace(/\r\n/g, '\n').replace(/\r/g, '\n')
}) })
} }
</script> </script>

View File

@@ -42,17 +42,17 @@ import TypeEditPieceRequirementsSection from '~/components/TypeEditPieceRequirem
const props = defineProps({ const props = defineProps({
modelValue: { modelValue: {
type: Object, type: Object,
required: true, required: true
}, },
saving: { saving: {
type: Boolean, type: Boolean,
default: false, default: false
}, }
}) })
const emit = defineEmits(['update:modelValue', 'submit']) const emit = defineEmits(['update:modelValue', 'submit'])
const deepClone = (value) => JSON.parse(JSON.stringify(value)) const deepClone = value => JSON.parse(JSON.stringify(value))
const createDefaultForm = (source = {}) => ({ const createDefaultForm = (source = {}) => ({
name: source.name || '', name: source.name || '',
@@ -61,7 +61,7 @@ const createDefaultForm = (source = {}) => ({
maintenanceFrequency: source.maintenanceFrequency || '', maintenanceFrequency: source.maintenanceFrequency || '',
customFields: deepClone(source.customFields || []), customFields: deepClone(source.customFields || []),
componentRequirements: deepClone(source.componentRequirements || []), componentRequirements: deepClone(source.componentRequirements || []),
pieceRequirements: deepClone(source.pieceRequirements || []), pieceRequirements: deepClone(source.pieceRequirements || [])
}) })
const formData = reactive(createDefaultForm(props.modelValue)) const formData = reactive(createDefaultForm(props.modelValue))
@@ -69,7 +69,7 @@ const allExpanded = ref(false)
const expandAllTrigger = ref(0) const expandAllTrigger = ref(0)
let syncingFromParent = false let syncingFromParent = false
const toPlainObject = (value) => JSON.parse(JSON.stringify(value)) const toPlainObject = value => JSON.parse(JSON.stringify(value))
const lastSnapshot = ref(toPlainObject(createDefaultForm(props.modelValue))) const lastSnapshot = ref(toPlainObject(createDefaultForm(props.modelValue)))
watch( watch(
@@ -91,7 +91,7 @@ watch(
watch( watch(
formData, formData,
(value) => { (value) => {
if (syncingFromParent) return if (syncingFromParent) { return }
const normalized = createDefaultForm(value) const normalized = createDefaultForm(value)
if (JSON.stringify(normalized) === JSON.stringify(lastSnapshot.value)) { if (JSON.stringify(normalized) === JSON.stringify(lastSnapshot.value)) {
return return

View File

@@ -15,8 +15,8 @@ import IconLucidePlus from '~icons/lucide/plus'
defineProps({ defineProps({
allExpanded: { allExpanded: {
type: Boolean, type: Boolean,
default: false, default: false
}, }
}) })
defineEmits(['toggle']) defineEmits(['toggle'])

View File

@@ -1,13 +1,17 @@
<template> <template>
<div class="alert alert-info mb-6"> <div class="alert alert-info mb-6">
<div> <div>
<h3 class="font-bold">Type existant</h3> <h3 class="font-bold">
Type existant
</h3>
<div class="text-sm"> <div class="text-sm">
<p><strong>Catégorie:</strong> {{ type.category || 'Non définie' }}</p> <p><strong>Catégorie:</strong> {{ type.category || 'Non définie' }}</p>
<p><strong>Maintenance:</strong> {{ type.maintenanceFrequency || 'Non définie' }}</p> <p><strong>Maintenance:</strong> {{ type.maintenanceFrequency || 'Non définie' }}</p>
<p><strong>Familles de composants:</strong> {{ type.componentRequirements?.length || 0 }}</p> <p><strong>Familles de composants:</strong> {{ type.componentRequirements?.length || 0 }}</p>
<p><strong>Groupes de pièces:</strong> {{ type.pieceRequirements?.length || 0 }}</p> <p><strong>Groupes de pièces:</strong> {{ type.pieceRequirements?.length || 0 }}</p>
<p v-if="type.description"><strong>Description:</strong> {{ type.description }}</p> <p v-if="type.description">
<strong>Description:</strong> {{ type.description }}
</p>
</div> </div>
</div> </div>
</div> </div>
@@ -20,4 +24,4 @@ defineProps({
required: true required: true
} }
}) })
</script> </script>

View File

@@ -2,8 +2,12 @@
<div class="card bg-base-100 shadow-lg hover:shadow-xl transition-shadow"> <div class="card bg-base-100 shadow-lg hover:shadow-xl transition-shadow">
<div class="card-body"> <div class="card-body">
<div class="flex items-center justify-between mb-4"> <div class="flex items-center justify-between mb-4">
<h3 class="card-title text-lg">{{ site.name }}</h3> <h3 class="card-title text-lg">
<div class="badge badge-primary badge-sm">{{ machineCount }} machines</div> {{ site.name }}
</h3>
<div class="badge badge-primary badge-sm">
{{ machineCount }} machines
</div>
</div> </div>
<div class="space-y-3 text-sm"> <div class="space-y-3 text-sm">
@@ -20,7 +24,7 @@
<div class="flex items-start gap-2 text-gray-600"> <div class="flex items-start gap-2 text-gray-600">
<IconLucideMapPin class="w-4 h-4 text-accent mt-1" aria-hidden="true" /> <IconLucideMapPin class="w-4 h-4 text-accent mt-1" aria-hidden="true" />
<span> <span>
{{ site.contactAddress }}<br /> {{ site.contactAddress }}<br>
{{ site.contactPostalCode }} {{ site.contactCity }} {{ site.contactPostalCode }} {{ site.contactCity }}
</span> </span>
</div> </div>

View File

@@ -5,7 +5,7 @@
Modifier le site Modifier le site
<span v-if="siteName" class="block text-sm font-normal text-gray-500">{{ siteName }}</span> <span v-if="siteName" class="block text-sm font-normal text-gray-500">{{ siteName }}</span>
</h3> </h3>
<form @submit.prevent="emit('submit')" class="space-y-4"> <form class="space-y-4" @submit.prevent="emit('submit')">
<div class="form-control"> <div class="form-control">
<label class="label"> <label class="label">
<span class="label-text">Nom du site</span> <span class="label-text">Nom du site</span>
@@ -16,7 +16,7 @@
placeholder="Nom du site" placeholder="Nom du site"
class="input input-bordered" class="input input-bordered"
required required
/> >
</div> </div>
<SiteContactFormFields :form="props.form" /> <SiteContactFormFields :form="props.form" />
@@ -24,8 +24,12 @@
<div class="border-t border-base-200 pt-4 space-y-4"> <div class="border-t border-base-200 pt-4 space-y-4">
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<div> <div>
<h4 class="font-semibold text-sm">Documents liés</h4> <h4 class="font-semibold text-sm">
<p class="text-xs text-gray-500">Ajoutez des documents (PDF, images...) relatifs à ce site.</p> Documents liés
</h4>
<p class="text-xs text-gray-500">
Ajoutez des documents (PDF, images...) relatifs à ce site.
</p>
</div> </div>
<span v-if="selectedFilesModel.length" class="badge badge-outline"> <span v-if="selectedFilesModel.length" class="badge badge-outline">
{{ selectedFilesModel.length }} fichier{{ selectedFilesModel.length > 1 ? 's' : '' }} prêt{{ selectedFilesModel.length > 1 ? 's' : '' }} à être ajouté {{ selectedFilesModel.length }} fichier{{ selectedFilesModel.length > 1 ? 's' : '' }} prêt{{ selectedFilesModel.length > 1 ? 's' : '' }} à être ajouté
@@ -39,7 +43,9 @@
/> />
<div v-if="documents.length" class="space-y-3"> <div v-if="documents.length" class="space-y-3">
<h5 class="text-sm font-medium">Documents existants</h5> <h5 class="text-sm font-medium">
Documents existants
</h5>
<div class="space-y-2 max-h-48 overflow-y-auto pr-1"> <div class="space-y-2 max-h-48 overflow-y-auto pr-1">
<div <div
v-for="document in documents" v-for="document in documents"
@@ -51,7 +57,9 @@
<component :is="documentIcon(document).component" class="h-6 w-6" aria-hidden="true" /> <component :is="documentIcon(document).component" class="h-6 w-6" aria-hidden="true" />
</span> </span>
<div> <div>
<div class="font-medium">{{ document.name }}</div> <div class="font-medium">
{{ document.name }}
</div>
<div class="text-xs text-gray-500"> <div class="text-xs text-gray-500">
{{ document.mimeType || 'Inconnu' }} {{ formatSize(document.size) }} {{ document.mimeType || 'Inconnu' }} {{ formatSize(document.size) }}
</div> </div>
@@ -84,7 +92,7 @@
Annuler Annuler
</button> </button>
<button type="submit" class="btn btn-primary" :disabled="uploadingDocuments"> <button type="submit" class="btn btn-primary" :disabled="uploadingDocuments">
<span v-if="uploadingDocuments" class="loading loading-spinner loading-xs mr-2"></span> <span v-if="uploadingDocuments" class="loading loading-spinner loading-xs mr-2" />
Enregistrer Enregistrer
</button> </button>
</div> </div>

View File

@@ -1,6 +1,6 @@
import { useToast } from './useToast' import { useToast } from './useToast'
export function useApi() { export function useApi () {
const { showSuccess, showError, showInfo } = useToast() const { showSuccess, showError, showInfo } = useToast()
const { public: publicConfig } = useRuntimeConfig() const { public: publicConfig } = useRuntimeConfig()
const API_BASE_URL = publicConfig.apiBaseUrl || 'http://localhost:3000' const API_BASE_URL = publicConfig.apiBaseUrl || 'http://localhost:3000'
@@ -11,8 +11,8 @@ export function useApi() {
const url = `${API_BASE_URL}${endpoint}` const url = `${API_BASE_URL}${endpoint}`
const defaultOptions = { const defaultOptions = {
headers: { headers: {
'Content-Type': 'application/json', 'Content-Type': 'application/json'
}, }
} }
// Ajouter un timeout à la requête // Ajouter un timeout à la requête
@@ -20,14 +20,14 @@ export function useApi() {
const timeoutId = setTimeout(() => controller.abort(), API_TIMEOUT) const timeoutId = setTimeout(() => controller.abort(), API_TIMEOUT)
try { try {
const response = await fetch(url, { const response = await fetch(url, {
...defaultOptions, ...defaultOptions,
...options, ...options,
signal: controller.signal signal: controller.signal
}) })
clearTimeout(timeoutId) clearTimeout(timeoutId)
if (response.ok) { if (response.ok) {
const data = await response.json() const data = await response.json()
return { success: true, data } return { success: true, data }

View File

@@ -5,7 +5,7 @@ import { useToast } from './useToast'
const componentModelsBuckets = ref({}) const componentModelsBuckets = ref({})
const loadingComponentModels = ref(false) const loadingComponentModels = ref(false)
export function useComponentModels() { export function useComponentModels () {
const { get, post, patch, delete: del } = useApi() const { get, post, patch, delete: del } = useApi()
const { showSuccess, showError } = useToast() const { showSuccess, showError } = useToast()
@@ -18,7 +18,7 @@ export function useComponentModels() {
const key = typeComposantId || '__all__' const key = typeComposantId || '__all__'
componentModelsBuckets.value = { componentModelsBuckets.value = {
...componentModelsBuckets.value, ...componentModelsBuckets.value,
[key]: result.data, [key]: result.data
} }
} }
return result return result
@@ -39,7 +39,7 @@ export function useComponentModels() {
const bucket = componentModelsBuckets.value[key] || [] const bucket = componentModelsBuckets.value[key] || []
componentModelsBuckets.value = { componentModelsBuckets.value = {
...componentModelsBuckets.value, ...componentModelsBuckets.value,
[key]: [...bucket, result.data], [key]: [...bucket, result.data]
} }
showSuccess(`Modèle de composant "${result.data.name}" créé`) showSuccess(`Modèle de composant "${result.data.name}" créé`)
} }
@@ -59,12 +59,12 @@ export function useComponentModels() {
if (result.success) { if (result.success) {
const key = result.data?.typeComposantId || '__all__' const key = result.data?.typeComposantId || '__all__'
const bucket = componentModelsBuckets.value[key] || [] const bucket = componentModelsBuckets.value[key] || []
const updatedBucket = bucket.map((model) => const updatedBucket = bucket.map(model =>
model.id === id ? result.data : model model.id === id ? result.data : model
) )
componentModelsBuckets.value = { componentModelsBuckets.value = {
...componentModelsBuckets.value, ...componentModelsBuckets.value,
[key]: updatedBucket, [key]: updatedBucket
} }
showSuccess(`Modèle de composant "${result.data.name}" mis à jour`) showSuccess(`Modèle de composant "${result.data.name}" mis à jour`)
} }
@@ -84,7 +84,7 @@ export function useComponentModels() {
if (result.success) { if (result.success) {
const updatedBuckets = {} const updatedBuckets = {}
for (const [key, bucket] of Object.entries(componentModelsBuckets.value)) { for (const [key, bucket] of Object.entries(componentModelsBuckets.value)) {
updatedBuckets[key] = bucket.filter((model) => model.id !== id) updatedBuckets[key] = bucket.filter(model => model.id !== id)
} }
componentModelsBuckets.value = updatedBuckets componentModelsBuckets.value = updatedBuckets
showSuccess('Modèle de composant supprimé') showSuccess('Modèle de composant supprimé')
@@ -101,7 +101,7 @@ export function useComponentModels() {
const allComponentModels = computed(() => { const allComponentModels = computed(() => {
return Object.values(componentModelsBuckets.value).reduce((acc, bucket) => { return Object.values(componentModelsBuckets.value).reduce((acc, bucket) => {
bucket.forEach((model) => { bucket.forEach((model) => {
if (!acc.find((existing) => existing.id === model.id)) { if (!acc.find(existing => existing.id === model.id)) {
acc.push(model) acc.push(model)
} }
}) })
@@ -126,6 +126,6 @@ export function useComponentModels() {
deleteComponentModel, deleteComponentModel,
getComponentModels, getComponentModels,
getComponentModelsForType, getComponentModelsForType,
isComponentModelLoading, isComponentModelLoading
} }
} }

View File

@@ -1,17 +1,17 @@
import { ref } from 'vue' import { ref } from 'vue'
import { listModelTypes, createModelType, updateModelType, deleteModelType } from '~/services/modelTypes'
import { useToast } from './useToast' import { useToast } from './useToast'
import { listModelTypes, createModelType, updateModelType, deleteModelType } from '~/services/modelTypes'
const componentTypes = ref([]) const componentTypes = ref([])
const loadingComponentTypes = ref(false) const loadingComponentTypes = ref(false)
export function useComponentTypes() { export function useComponentTypes () {
const { showSuccess, showError } = useToast() const { showSuccess, showError } = useToast()
const generateCodeFromName = (name) => { const generateCodeFromName = (name) => {
return (name || '') return (name || '')
.normalize('NFD') .normalize('NFD')
.replace(/[\u0300-\u036f]/g, '') .replace(/[\u0300-\u036F]/g, '')
.toLowerCase() .toLowerCase()
.replace(/[^a-z0-9]+/g, '-') .replace(/[^a-z0-9]+/g, '-')
.replace(/^-+|-+$/g, '') .replace(/^-+|-+$/g, '')
@@ -25,12 +25,12 @@ export function useComponentTypes() {
category: 'COMPONENT', category: 'COMPONENT',
sort: 'name', sort: 'name',
dir: 'asc', dir: 'asc',
limit: 200, limit: 200
}) })
componentTypes.value = data.items.map((item) => ({ componentTypes.value = data.items.map(item => ({
...item, ...item,
description: item.description ?? item.notes ?? null, description: item.description ?? item.notes ?? null
})) }))
return { success: true, data: componentTypes.value } return { success: true, data: componentTypes.value }
@@ -51,12 +51,12 @@ export function useComponentTypes() {
code: payload.code || generateCodeFromName(payload.name), code: payload.code || generateCodeFromName(payload.name),
category: 'COMPONENT', category: 'COMPONENT',
notes: payload.description ?? payload.notes, notes: payload.description ?? payload.notes,
description: payload.description ?? null, description: payload.description ?? null
}) })
const normalized = { const normalized = {
...data, ...data,
description: data.description ?? data.notes ?? null, description: data.description ?? data.notes ?? null
} }
componentTypes.value.push(normalized) componentTypes.value.push(normalized)
@@ -79,15 +79,15 @@ export function useComponentTypes() {
name: payload.name, name: payload.name,
description: payload.description, description: payload.description,
notes: payload.notes, notes: payload.notes,
code: payload.code, code: payload.code
}) })
const normalized = { const normalized = {
...data, ...data,
description: data.description ?? data.notes ?? null, description: data.description ?? data.notes ?? null
} }
const index = componentTypes.value.findIndex((type) => type.id === id) const index = componentTypes.value.findIndex(type => type.id === id)
if (index !== -1) { if (index !== -1) {
componentTypes.value[index] = normalized componentTypes.value[index] = normalized
} }
@@ -95,7 +95,7 @@ export function useComponentTypes() {
return { return {
success: true, success: true,
data: normalized, data: normalized
} }
} catch (error) { } catch (error) {
const message = error?.data?.message || error?.message || 'Erreur inconnue' const message = error?.data?.message || error?.message || 'Erreur inconnue'
@@ -110,7 +110,7 @@ export function useComponentTypes() {
loadingComponentTypes.value = true loadingComponentTypes.value = true
try { try {
await deleteModelType(id) await deleteModelType(id)
componentTypes.value = componentTypes.value.filter((type) => type.id !== id) componentTypes.value = componentTypes.value.filter(type => type.id !== id)
showSuccess('Type de composant supprimé') showSuccess('Type de composant supprimé')
return { success: true } return { success: true }
} catch (error) { } catch (error) {
@@ -133,6 +133,6 @@ export function useComponentTypes() {
updateComponentType, updateComponentType,
deleteComponentType, deleteComponentType,
getComponentTypes, getComponentTypes,
isComponentTypeLoading, isComponentTypeLoading
} }
} }

View File

@@ -5,7 +5,7 @@ import { useApi } from './useApi'
const composants = ref([]) const composants = ref([])
const loading = ref(false) const loading = ref(false)
export function useComposants() { export function useComposants () {
const { showSuccess, showError, showInfo } = useToast() const { showSuccess, showError, showInfo } = useToast()
const { get, post, patch, delete: del } = useApi() const { get, post, patch, delete: del } = useApi()
@@ -132,4 +132,4 @@ export function useComposants() {
getComposants, getComposants,
isLoading isLoading
} }
} }

View File

@@ -5,7 +5,7 @@ import { useToast } from './useToast'
const constructeurs = ref([]) const constructeurs = ref([])
const loading = ref(false) const loading = ref(false)
export function useConstructeurs() { export function useConstructeurs () {
const { get, post, patch, delete: del } = useApi() const { get, post, patch, delete: del } = useApi()
const { showSuccess, showError } = useToast() const { showSuccess, showError } = useToast()
@@ -43,7 +43,7 @@ export function useConstructeurs() {
return result return result
} catch (error) { } catch (error) {
console.error('Erreur lors de la création du constructeur:', error) console.error('Erreur lors de la création du constructeur:', error)
showError("Impossible de créer le constructeur") showError('Impossible de créer le constructeur')
return { success: false, error: error.message } return { success: false, error: error.message }
} finally { } finally {
loading.value = false loading.value = false
@@ -66,7 +66,7 @@ export function useConstructeurs() {
return result return result
} catch (error) { } catch (error) {
console.error('Erreur lors de la mise à jour du constructeur:', error) console.error('Erreur lors de la mise à jour du constructeur:', error)
showError("Impossible de mettre à jour le constructeur") showError('Impossible de mettre à jour le constructeur')
return { success: false, error: error.message } return { success: false, error: error.message }
} finally { } finally {
loading.value = false loading.value = false
@@ -86,14 +86,14 @@ export function useConstructeurs() {
return result return result
} catch (error) { } catch (error) {
console.error('Erreur lors de la suppression du constructeur:', error) console.error('Erreur lors de la suppression du constructeur:', error)
showError("Impossible de supprimer le constructeur") showError('Impossible de supprimer le constructeur')
return { success: false, error: error.message } return { success: false, error: error.message }
} finally { } finally {
loading.value = false loading.value = false
} }
} }
const getConstructeurById = (id) => constructeurs.value.find(item => item.id === id) const getConstructeurById = id => constructeurs.value.find(item => item.id === id)
return { return {
constructeurs, constructeurs,
@@ -103,6 +103,6 @@ export function useConstructeurs() {
createConstructeur, createConstructeur,
updateConstructeur, updateConstructeur,
deleteConstructeur, deleteConstructeur,
getConstructeurById, getConstructeurById
} }
} }

View File

@@ -1,7 +1,7 @@
import { ref } from 'vue' import { ref } from 'vue'
import { useApi } from './useApi' import { useApi } from './useApi'
export function useCustomFields() { export function useCustomFields () {
const { apiCall } = useApi() const { apiCall } = useApi()
const customFieldValues = ref([]) const customFieldValues = ref([])
const loading = ref(false) const loading = ref(false)
@@ -94,4 +94,4 @@ export function useCustomFields() {
upsertCustomFieldValue, upsertCustomFieldValue,
deleteCustomFieldValue deleteCustomFieldValue
} }
} }

View File

@@ -5,7 +5,7 @@ import { useToast } from './useToast'
const documents = ref([]) const documents = ref([])
const loading = ref(false) const loading = ref(false)
const fileToBase64 = (file) => const fileToBase64 = file =>
new Promise((resolve, reject) => { new Promise((resolve, reject) => {
const reader = new FileReader() const reader = new FileReader()
reader.onload = () => resolve(reader.result) reader.onload = () => resolve(reader.result)
@@ -13,7 +13,7 @@ const fileToBase64 = (file) =>
reader.readAsDataURL(file) reader.readAsDataURL(file)
}) })
export function useDocuments() { export function useDocuments () {
const { get, post, delete: del } = useApi() const { get, post, delete: del } = useApi()
const { showError, showSuccess } = useToast() const { showError, showSuccess } = useToast()
@@ -34,7 +34,7 @@ export function useDocuments() {
return result return result
} catch (error) { } catch (error) {
console.error(`Erreur lors du chargement des documents (${endpoint}):`, error) console.error(`Erreur lors du chargement des documents (${endpoint}):`, error)
showError("Impossible de charger les documents") showError('Impossible de charger les documents')
return { success: false, error: error.message } return { success: false, error: error.message }
} finally { } finally {
loading.value = false loading.value = false
@@ -46,27 +46,27 @@ export function useDocuments() {
} }
const loadDocumentsBySite = async (siteId, options = {}) => { const loadDocumentsBySite = async (siteId, options = {}) => {
if (!siteId) return { success: false, error: 'Aucun site sélectionné' } if (!siteId) { return { success: false, error: 'Aucun site sélectionné' } }
return loadFromEndpoint(`/documents/site/${siteId}`, { updateStore: options.updateStore ?? false }) return loadFromEndpoint(`/documents/site/${siteId}`, { updateStore: options.updateStore ?? false })
} }
const loadDocumentsByMachine = async (machineId, options = {}) => { const loadDocumentsByMachine = async (machineId, options = {}) => {
if (!machineId) return { success: false, error: 'Aucune machine sélectionnée' } if (!machineId) { return { success: false, error: 'Aucune machine sélectionnée' } }
return loadFromEndpoint(`/documents/machine/${machineId}`, { updateStore: options.updateStore ?? false }) return loadFromEndpoint(`/documents/machine/${machineId}`, { updateStore: options.updateStore ?? false })
} }
const loadDocumentsByComponent = async (componentId, options = {}) => { const loadDocumentsByComponent = async (componentId, options = {}) => {
if (!componentId) return { success: false, error: 'Aucun composant sélectionné' } if (!componentId) { return { success: false, error: 'Aucun composant sélectionné' } }
return loadFromEndpoint(`/documents/composant/${componentId}`, { updateStore: options.updateStore ?? false }) return loadFromEndpoint(`/documents/composant/${componentId}`, { updateStore: options.updateStore ?? false })
} }
const loadDocumentsByPiece = async (pieceId, options = {}) => { const loadDocumentsByPiece = async (pieceId, options = {}) => {
if (!pieceId) return { success: false, error: 'Aucune pièce sélectionnée' } if (!pieceId) { return { success: false, error: 'Aucune pièce sélectionnée' } }
return loadFromEndpoint(`/documents/piece/${pieceId}`, { updateStore: options.updateStore ?? false }) return loadFromEndpoint(`/documents/piece/${pieceId}`, { updateStore: options.updateStore ?? false })
} }
const uploadDocuments = async ({ files = [], context = {} }, { updateStore = false } = {}) => { const uploadDocuments = async ({ files = [], context = {} }, { updateStore = false } = {}) => {
if (!files.length) return { success: false, error: 'Aucun fichier sélectionné' } if (!files.length) { return { success: false, error: 'Aucun fichier sélectionné' } }
loading.value = true loading.value = true
const created = [] const created = []
@@ -81,7 +81,7 @@ export function useDocuments() {
mimeType: file.type || 'application/octet-stream', mimeType: file.type || 'application/octet-stream',
size: file.size, size: file.size,
path: dataUrl, path: dataUrl,
...context, ...context
} }
const result = await post('/documents', payload) const result = await post('/documents', payload)
@@ -111,7 +111,7 @@ export function useDocuments() {
} }
const deleteDocument = async (id, { updateStore = false } = {}) => { const deleteDocument = async (id, { updateStore = false } = {}) => {
if (!id) return { success: false, error: 'Identifiant manquant' } if (!id) { return { success: false, error: 'Identifiant manquant' } }
loading.value = true loading.value = true
try { try {
@@ -125,7 +125,7 @@ export function useDocuments() {
return result return result
} catch (error) { } catch (error) {
console.error('Erreur lors de la suppression du document:', error) console.error('Erreur lors de la suppression du document:', error)
showError("Impossible de supprimer le document") showError('Impossible de supprimer le document')
return { success: false, error: error.message } return { success: false, error: error.message }
} finally { } finally {
loading.value = false loading.value = false
@@ -141,6 +141,6 @@ export function useDocuments() {
loadDocumentsByComponent, loadDocumentsByComponent,
loadDocumentsByPiece, loadDocumentsByPiece,
uploadDocuments, uploadDocuments,
deleteDocument, deleteDocument
} }
} }

View File

@@ -5,7 +5,7 @@ import { useApi } from './useApi'
const machineTypes = ref([]) const machineTypes = ref([])
const loading = ref(false) const loading = ref(false)
export function useMachineTypesApi() { export function useMachineTypesApi () {
const { showSuccess, showError, showInfo } = useToast() const { showSuccess, showError, showInfo } = useToast()
const { get, post, patch, delete: del } = useApi() const { get, post, patch, delete: del } = useApi()
@@ -85,7 +85,7 @@ export function useMachineTypesApi() {
if (localType) { if (localType) {
return { success: true, data: localType } return { success: true, data: localType }
} }
// Si pas trouvé localement, récupérer depuis l'API // Si pas trouvé localement, récupérer depuis l'API
try { try {
const result = await get(`/types/machines/${id}`) const result = await get(`/types/machines/${id}`)
@@ -114,4 +114,4 @@ export function useMachineTypesApi() {
getMachineTypes, getMachineTypes,
isLoading isLoading
} }
} }

View File

@@ -5,7 +5,7 @@ import { useApi } from './useApi'
const machines = ref([]) const machines = ref([])
const loading = ref(false) const loading = ref(false)
export function useMachines() { export function useMachines () {
const { showSuccess, showError, showInfo } = useToast() const { showSuccess, showError, showInfo } = useToast()
const { get, post, patch, delete: del } = useApi() const { get, post, patch, delete: del } = useApi()
@@ -45,7 +45,7 @@ export function useMachines() {
// Créer la machine avec la structure héritée du type // Créer la machine avec la structure héritée du type
const machineWithStructure = { const machineWithStructure = {
...machineData, ...machineData,
typeMachineId: typeMachine.id, typeMachineId: typeMachine.id
// La structure sera automatiquement héritée du type // La structure sera automatiquement héritée du type
// Les composants et pièces seront créés automatiquement // Les composants et pièces seront créés automatiquement
} }

View File

@@ -5,7 +5,7 @@ import { useToast } from './useToast'
const pieceModelsBuckets = ref({}) const pieceModelsBuckets = ref({})
const loadingPieceModels = ref(false) const loadingPieceModels = ref(false)
export function usePieceModels() { export function usePieceModels () {
const { get, post, patch, delete: del } = useApi() const { get, post, patch, delete: del } = useApi()
const { showSuccess, showError } = useToast() const { showSuccess, showError } = useToast()
@@ -18,7 +18,7 @@ export function usePieceModels() {
const key = typePieceId || '__all__' const key = typePieceId || '__all__'
pieceModelsBuckets.value = { pieceModelsBuckets.value = {
...pieceModelsBuckets.value, ...pieceModelsBuckets.value,
[key]: result.data, [key]: result.data
} }
} }
return result return result
@@ -39,7 +39,7 @@ export function usePieceModels() {
const bucket = pieceModelsBuckets.value[key] || [] const bucket = pieceModelsBuckets.value[key] || []
pieceModelsBuckets.value = { pieceModelsBuckets.value = {
...pieceModelsBuckets.value, ...pieceModelsBuckets.value,
[key]: [...bucket, result.data], [key]: [...bucket, result.data]
} }
showSuccess(`Modèle de pièce "${result.data.name}" créé`) showSuccess(`Modèle de pièce "${result.data.name}" créé`)
} }
@@ -59,12 +59,12 @@ export function usePieceModels() {
if (result.success) { if (result.success) {
const key = result.data?.typePieceId || '__all__' const key = result.data?.typePieceId || '__all__'
const bucket = pieceModelsBuckets.value[key] || [] const bucket = pieceModelsBuckets.value[key] || []
const updatedBucket = bucket.map((model) => const updatedBucket = bucket.map(model =>
model.id === id ? result.data : model model.id === id ? result.data : model
) )
pieceModelsBuckets.value = { pieceModelsBuckets.value = {
...pieceModelsBuckets.value, ...pieceModelsBuckets.value,
[key]: updatedBucket, [key]: updatedBucket
} }
showSuccess(`Modèle de pièce "${result.data.name}" mis à jour`) showSuccess(`Modèle de pièce "${result.data.name}" mis à jour`)
} }
@@ -84,7 +84,7 @@ export function usePieceModels() {
if (result.success) { if (result.success) {
const updatedBuckets = {} const updatedBuckets = {}
for (const [key, bucket] of Object.entries(pieceModelsBuckets.value)) { for (const [key, bucket] of Object.entries(pieceModelsBuckets.value)) {
updatedBuckets[key] = bucket.filter((model) => model.id !== id) updatedBuckets[key] = bucket.filter(model => model.id !== id)
} }
pieceModelsBuckets.value = updatedBuckets pieceModelsBuckets.value = updatedBuckets
showSuccess('Modèle de pièce supprimé') showSuccess('Modèle de pièce supprimé')
@@ -101,7 +101,7 @@ export function usePieceModels() {
const allPieceModels = computed(() => { const allPieceModels = computed(() => {
return Object.values(pieceModelsBuckets.value).reduce((acc, bucket) => { return Object.values(pieceModelsBuckets.value).reduce((acc, bucket) => {
bucket.forEach((model) => { bucket.forEach((model) => {
if (!acc.find((existing) => existing.id === model.id)) { if (!acc.find(existing => existing.id === model.id)) {
acc.push(model) acc.push(model)
} }
}) })
@@ -126,6 +126,6 @@ export function usePieceModels() {
deletePieceModel, deletePieceModel,
getPieceModels, getPieceModels,
getPieceModelsForType, getPieceModelsForType,
isPieceModelLoading, isPieceModelLoading
} }
} }

View File

@@ -1,17 +1,17 @@
import { ref } from 'vue' import { ref } from 'vue'
import { listModelTypes, createModelType, updateModelType, deleteModelType } from '~/services/modelTypes'
import { useToast } from './useToast' import { useToast } from './useToast'
import { listModelTypes, createModelType, updateModelType, deleteModelType } from '~/services/modelTypes'
const pieceTypes = ref([]) const pieceTypes = ref([])
const loadingPieceTypes = ref(false) const loadingPieceTypes = ref(false)
export function usePieceTypes() { export function usePieceTypes () {
const { showSuccess, showError } = useToast() const { showSuccess, showError } = useToast()
const generateCodeFromName = (name) => { const generateCodeFromName = (name) => {
return (name || '') return (name || '')
.normalize('NFD') .normalize('NFD')
.replace(/[\u0300-\u036f]/g, '') .replace(/[\u0300-\u036F]/g, '')
.toLowerCase() .toLowerCase()
.replace(/[^a-z0-9]+/g, '-') .replace(/[^a-z0-9]+/g, '-')
.replace(/^-+|-+$/g, '') .replace(/^-+|-+$/g, '')
@@ -25,12 +25,12 @@ export function usePieceTypes() {
category: 'PIECE', category: 'PIECE',
sort: 'name', sort: 'name',
dir: 'asc', dir: 'asc',
limit: 200, limit: 200
}) })
pieceTypes.value = data.items.map((item) => ({ pieceTypes.value = data.items.map(item => ({
...item, ...item,
description: item.description ?? item.notes ?? null, description: item.description ?? item.notes ?? null
})) }))
return { success: true, data: pieceTypes.value } return { success: true, data: pieceTypes.value }
@@ -51,12 +51,12 @@ export function usePieceTypes() {
code: payload.code || generateCodeFromName(payload.name), code: payload.code || generateCodeFromName(payload.name),
category: 'PIECE', category: 'PIECE',
notes: payload.description ?? payload.notes, notes: payload.description ?? payload.notes,
description: payload.description ?? null, description: payload.description ?? null
}) })
const normalized = { const normalized = {
...data, ...data,
description: data.description ?? data.notes ?? null, description: data.description ?? data.notes ?? null
} }
pieceTypes.value.push(normalized) pieceTypes.value.push(normalized)
@@ -79,15 +79,15 @@ export function usePieceTypes() {
name: payload.name, name: payload.name,
description: payload.description, description: payload.description,
notes: payload.notes, notes: payload.notes,
code: payload.code, code: payload.code
}) })
const normalized = { const normalized = {
...data, ...data,
description: data.description ?? data.notes ?? null, description: data.description ?? data.notes ?? null
} }
const index = pieceTypes.value.findIndex((type) => type.id === id) const index = pieceTypes.value.findIndex(type => type.id === id)
if (index !== -1) { if (index !== -1) {
pieceTypes.value[index] = normalized pieceTypes.value[index] = normalized
} }
@@ -95,7 +95,7 @@ export function usePieceTypes() {
return { return {
success: true, success: true,
data: normalized, data: normalized
} }
} catch (error) { } catch (error) {
const message = error?.data?.message || error?.message || 'Erreur inconnue' const message = error?.data?.message || error?.message || 'Erreur inconnue'
@@ -110,7 +110,7 @@ export function usePieceTypes() {
loadingPieceTypes.value = true loadingPieceTypes.value = true
try { try {
await deleteModelType(id) await deleteModelType(id)
pieceTypes.value = pieceTypes.value.filter((type) => type.id !== id) pieceTypes.value = pieceTypes.value.filter(type => type.id !== id)
showSuccess('Type de pièce supprimé') showSuccess('Type de pièce supprimé')
return { success: true } return { success: true }
} catch (error) { } catch (error) {
@@ -133,6 +133,6 @@ export function usePieceTypes() {
updatePieceType, updatePieceType,
deletePieceType, deletePieceType,
getPieceTypes, getPieceTypes,
isPieceTypeLoading, isPieceTypeLoading
} }
} }

View File

@@ -5,7 +5,7 @@ import { useApi } from './useApi'
const pieces = ref([]) const pieces = ref([])
const loading = ref(false) const loading = ref(false)
export function usePieces() { export function usePieces () {
const { showSuccess, showError, showInfo } = useToast() const { showSuccess, showError, showInfo } = useToast()
const { get, post, patch, delete: del } = useApi() const { get, post, patch, delete: del } = useApi()
@@ -132,4 +132,4 @@ export function usePieces() {
getPieces, getPieces,
isLoading isLoading
} }
} }

View File

@@ -6,13 +6,13 @@ const buildUrl = (path) => {
return `${base}${path}` return `${base}${path}`
} }
export function useProfileSession() { export function useProfileSession () {
const activeProfile = useState('profileSession:active', () => null) const activeProfile = useState('profileSession:active', () => null)
const sessionLoaded = useState('profileSession:loaded', () => false) const sessionLoaded = useState('profileSession:loaded', () => false)
const loading = useState('profileSession:loading', () => false) const loading = useState('profileSession:loading', () => false)
const getSessionHeaders = () => { const getSessionHeaders = () => {
if (!process.server) return undefined if (!process.server) { return undefined }
const headers = useRequestHeaders(['cookie']) const headers = useRequestHeaders(['cookie'])
return headers?.cookie ? { cookie: headers.cookie } : undefined return headers?.cookie ? { cookie: headers.cookie } : undefined
} }
@@ -23,7 +23,7 @@ export function useProfileSession() {
activeProfile.value = await $fetch(buildUrl('/session/profile'), { activeProfile.value = await $fetch(buildUrl('/session/profile'), {
method: 'GET', method: 'GET',
credentials: 'include', credentials: 'include',
headers: getSessionHeaders(), headers: getSessionHeaders()
}) })
} catch (error) { } catch (error) {
if (error?.status === 401) { if (error?.status === 401) {
@@ -51,7 +51,7 @@ export function useProfileSession() {
method: 'POST', method: 'POST',
credentials: 'include', credentials: 'include',
body: { profileId }, body: { profileId },
headers: getSessionHeaders(), headers: getSessionHeaders()
}) })
await fetchCurrentProfile() await fetchCurrentProfile()
} }
@@ -61,7 +61,7 @@ export function useProfileSession() {
await $fetch(buildUrl('/session/profile'), { await $fetch(buildUrl('/session/profile'), {
method: 'DELETE', method: 'DELETE',
credentials: 'include', credentials: 'include',
headers: getSessionHeaders(), headers: getSessionHeaders()
}) })
} finally { } finally {
activeProfile.value = null activeProfile.value = null
@@ -76,6 +76,6 @@ export function useProfileSession() {
ensureSession, ensureSession,
fetchCurrentProfile, fetchCurrentProfile,
activateProfile, activateProfile,
logout, logout
} }
} }

View File

@@ -6,13 +6,13 @@ const buildUrl = (path) => {
return `${base}${path}` return `${base}${path}`
} }
export function useProfiles() { export function useProfiles () {
const profiles = useState('profiles:list', () => []) const profiles = useState('profiles:list', () => [])
const loadingProfiles = useState('profiles:loading', () => false) const loadingProfiles = useState('profiles:loading', () => false)
const profilesLoaded = useState('profiles:loaded', () => false) const profilesLoaded = useState('profiles:loaded', () => false)
const getSessionHeaders = () => { const getSessionHeaders = () => {
if (!process.server) return undefined if (!process.server) { return undefined }
const headers = useRequestHeaders(['cookie']) const headers = useRequestHeaders(['cookie'])
return headers?.cookie ? { cookie: headers.cookie } : undefined return headers?.cookie ? { cookie: headers.cookie } : undefined
} }
@@ -23,7 +23,7 @@ export function useProfiles() {
profiles.value = await $fetch(buildUrl('/profiles'), { profiles.value = await $fetch(buildUrl('/profiles'), {
method: 'GET', method: 'GET',
credentials: 'include', credentials: 'include',
headers: getSessionHeaders(), headers: getSessionHeaders()
}) })
profilesLoaded.value = true profilesLoaded.value = true
} catch (error) { } catch (error) {
@@ -41,7 +41,7 @@ export function useProfiles() {
method: 'POST', method: 'POST',
credentials: 'include', credentials: 'include',
body: { firstName, lastName }, body: { firstName, lastName },
headers: getSessionHeaders(), headers: getSessionHeaders()
}) })
await fetchProfiles() await fetchProfiles()
return profile return profile
@@ -51,7 +51,7 @@ export function useProfiles() {
await $fetch(buildUrl(`/profiles/${profileId}`), { await $fetch(buildUrl(`/profiles/${profileId}`), {
method: 'DELETE', method: 'DELETE',
credentials: 'include', credentials: 'include',
headers: getSessionHeaders(), headers: getSessionHeaders()
}) })
await fetchProfiles() await fetchProfiles()
} }
@@ -62,6 +62,6 @@ export function useProfiles() {
profilesLoaded, profilesLoaded,
fetchProfiles, fetchProfiles,
createProfile, createProfile,
deleteProfile, deleteProfile
} }
} }

View File

@@ -5,7 +5,7 @@ import { useApi } from './useApi'
const sites = ref([]) const sites = ref([])
const loading = ref(false) const loading = ref(false)
export function useSites() { export function useSites () {
const { showSuccess, showInfo } = useToast() const { showSuccess, showInfo } = useToast()
const { get, post, patch, delete: del } = useApi() const { get, post, patch, delete: del } = useApi()
@@ -97,4 +97,4 @@ export function useSites() {
getSites, getSites,
isLoading isLoading
} }
} }

View File

@@ -3,7 +3,7 @@ import { ref } from 'vue'
const toasts = ref([]) const toasts = ref([])
let nextId = 1 let nextId = 1
export function useToast() { export function useToast () {
const showToast = (message, type = 'info', duration = 5000) => { const showToast = (message, type = 'info', duration = 5000) => {
const id = nextId++ const id = nextId++
const toast = { const toast = {
@@ -12,14 +12,14 @@ export function useToast() {
type, type,
visible: true visible: true
} }
toasts.value.push(toast) toasts.value.push(toast)
// Auto-remove after duration // Auto-remove after duration
setTimeout(() => { setTimeout(() => {
removeToast(id) removeToast(id)
}, duration) }, duration)
return id return id
} }
@@ -63,4 +63,4 @@ export function useToast() {
removeToast, removeToast,
clearAll clearAll
} }
} }

View File

@@ -2,12 +2,20 @@
<main class="container mx-auto px-6 py-8 space-y-8"> <main class="container mx-auto px-6 py-8 space-y-8">
<header class="flex flex-col gap-4 md:flex-row md:items-center md:justify-between"> <header class="flex flex-col gap-4 md:flex-row md:items-center md:justify-between">
<div class="space-y-1"> <div class="space-y-1">
<h1 class="text-3xl font-bold text-gray-800">Catalogue de composant</h1> <h1 class="text-3xl font-bold text-gray-800">
<p class="text-sm text-gray-500">Gérez les modèles disponibles pour chaque famille de composant.</p> Catalogue de composant
</h1>
<p class="text-sm text-gray-500">
Gérez les modèles disponibles pour chaque famille de composant.
</p>
</div> </div>
<div class="tabs tabs-boxed"> <div class="tabs tabs-boxed">
<NuxtLink to="/component-catalog" class="tab tab-active">Composants</NuxtLink> <NuxtLink to="/component-catalog" class="tab tab-active">
<NuxtLink to="/pieces-catalog" class="tab">Pièces</NuxtLink> Composants
</NuxtLink>
<NuxtLink to="/pieces-catalog" class="tab">
Pièces
</NuxtLink>
</div> </div>
</header> </header>
@@ -36,7 +44,7 @@
type="search" type="search"
placeholder="Rechercher un modèle..." placeholder="Rechercher un modèle..."
class="input input-bordered input-sm" class="input input-bordered input-sm"
/> >
</label> </label>
<span class="text-xs text-gray-500">{{ filteredModels.length }} modèle(s)</span> <span class="text-xs text-gray-500">{{ filteredModels.length }} modèle(s)</span>
</div> </div>
@@ -47,7 +55,7 @@
</div> </div>
<div v-if="loadingComponentModels" class="flex justify-center py-16"> <div v-if="loadingComponentModels" class="flex justify-center py-16">
<span class="loading loading-spinner loading-lg"></span> <span class="loading loading-spinner loading-lg" />
</div> </div>
<div v-else-if="filteredModels.length === 0" class="py-16 text-center text-sm text-gray-500"> <div v-else-if="filteredModels.length === 0" class="py-16 text-center text-sm text-gray-500">
@@ -59,11 +67,19 @@
<thead> <thead>
<tr class="text-sm text-gray-500"> <tr class="text-sm text-gray-500">
<th>Nom</th> <th>Nom</th>
<th class="hidden md:table-cell">Description</th> <th class="hidden md:table-cell">
Description
</th>
<th>Type</th> <th>Type</th>
<th class="hidden xl:table-cell">Structure</th> <th class="hidden xl:table-cell">
<th class="hidden lg:table-cell">Modifié</th> Structure
<th class="text-right">Actions</th> </th>
<th class="hidden lg:table-cell">
Modifié
</th>
<th class="text-right">
Actions
</th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
@@ -80,10 +96,16 @@
<span class="font-medium">{{ model.name }}</span> <span class="font-medium">{{ model.name }}</span>
</div> </div>
</td> </td>
<td class="hidden md:table-cell">{{ model.description || '—' }}</td> <td class="hidden md:table-cell">
{{ model.description || '—' }}
</td>
<td>{{ model.typeComposant?.name || 'Non défini' }}</td> <td>{{ model.typeComposant?.name || 'Non défini' }}</td>
<td class="hidden xl:table-cell text-xs text-gray-500">{{ formatStructurePreview(model.structure) }}</td> <td class="hidden xl:table-cell text-xs text-gray-500">
<td class="hidden lg:table-cell text-xs text-gray-500">{{ formatFrenchDate(model.updatedAt || model.createdAt) }}</td> {{ formatStructurePreview(model.structure) }}
</td>
<td class="hidden lg:table-cell text-xs text-gray-500">
{{ formatFrenchDate(model.updatedAt || model.createdAt) }}
</td>
<td class="text-right space-x-2"> <td class="text-right space-x-2">
<button type="button" class="btn btn-sm btn-outline" @click="startEdit(model)"> <button type="button" class="btn btn-sm btn-outline" @click="startEdit(model)">
Éditer Éditer
@@ -124,7 +146,7 @@
class="input input-bordered input-sm" class="input input-bordered input-sm"
placeholder="Nom du modèle" placeholder="Nom du modèle"
required required
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
<label class="label"><span class="label-text">Description</span></label> <label class="label"><span class="label-text">Description</span></label>
@@ -133,7 +155,7 @@
class="textarea textarea-bordered textarea-sm" class="textarea textarea-bordered textarea-sm"
rows="3" rows="3"
placeholder="Notes optionnelles" placeholder="Notes optionnelles"
></textarea> />
</div> </div>
<div class="form-control"> <div class="form-control">
<label class="label"><span class="label-text">Type de composant</span></label> <label class="label"><span class="label-text">Type de composant</span></label>
@@ -142,7 +164,9 @@
class="select select-bordered select-sm" class="select select-bordered select-sm"
required required
> >
<option value="" disabled>Sélectionner un type</option> <option value="" disabled>
Sélectionner un type
</option>
<option <option
v-for="type in componentTypes" v-for="type in componentTypes"
:key="type.id" :key="type.id"
@@ -153,7 +177,9 @@
</select> </select>
</div> </div>
<div class="divider my-0">Structure</div> <div class="divider my-0">
Structure
</div>
<ComponentModelStructureEditor v-model="form.data.structure" /> <ComponentModelStructureEditor v-model="form.data.structure" />
<div class="rounded-lg border border-base-200 bg-base-200/60 p-3 text-xs text-gray-500"> <div class="rounded-lg border border-base-200 bg-base-200/60 p-3 text-xs text-gray-500">
Aperçu : {{ formatStructurePreview(form.data.structure) }} Aperçu : {{ formatStructurePreview(form.data.structure) }}
@@ -190,7 +216,7 @@ const {
updateComponentModel, updateComponentModel,
deleteComponentModel, deleteComponentModel,
loadingComponentModels, loadingComponentModels,
getComponentModelsForType, getComponentModelsForType
} = useComponentModels() } = useComponentModels()
const { showError, showSuccess } = useToast() const { showError, showSuccess } = useToast()
@@ -205,15 +231,15 @@ const form = reactive({
name: '', name: '',
description: '', description: '',
typeComposantId: '', typeComposantId: '',
structure: {}, structure: {}
}, }
}) })
const ensureTypeSelected = () => { const ensureTypeSelected = () => {
if (form.data.typeComposantId && componentTypes.value.some((type) => type.id === form.data.typeComposantId)) { if (form.data.typeComposantId && componentTypes.value.some(type => type.id === form.data.typeComposantId)) {
return return
} }
if (selectedType.value !== 'all' && componentTypes.value.some((type) => type.id === selectedType.value)) { if (selectedType.value !== 'all' && componentTypes.value.some(type => type.id === selectedType.value)) {
form.data.typeComposantId = selectedType.value form.data.typeComposantId = selectedType.value
return return
} }
@@ -227,7 +253,7 @@ const startCreate = () => {
name: '', name: '',
description: '', description: '',
typeComposantId: selectedType.value !== 'all' ? selectedType.value : '', typeComposantId: selectedType.value !== 'all' ? selectedType.value : '',
structure: {}, structure: {}
} }
ensureTypeSelected() ensureTypeSelected()
} }
@@ -239,7 +265,7 @@ const startEdit = (model) => {
name: model.name, name: model.name,
description: model.description || '', description: model.description || '',
typeComposantId: model.typeComposantId || model.typeComposant?.id || '', typeComposantId: model.typeComposantId || model.typeComposant?.id || '',
structure: model.structure || {}, structure: model.structure || {}
} }
} }
@@ -254,9 +280,9 @@ const filteredModels = computed(() => {
const term = searchQuery.value.toLowerCase() const term = searchQuery.value.toLowerCase()
return list.filter((model) => { return list.filter((model) => {
return ( return (
model.name?.toLowerCase().includes(term) model.name?.toLowerCase().includes(term) ||
|| model.description?.toLowerCase().includes(term) model.description?.toLowerCase().includes(term) ||
|| model.typeComposant?.name?.toLowerCase().includes(term) model.typeComposant?.name?.toLowerCase().includes(term)
) )
}) })
}) })
@@ -282,7 +308,7 @@ const handleSubmit = async () => {
name: form.data.name.trim(), name: form.data.name.trim(),
description: form.data.description.trim() || undefined, description: form.data.description.trim() || undefined,
typeComposantId: form.data.typeComposantId, typeComposantId: form.data.typeComposantId,
structure: form.data.structure || {}, structure: form.data.structure || {}
}) })
if (!result.success) { if (!result.success) {
showError(result.error || 'Impossible de créer le modèle') showError(result.error || 'Impossible de créer le modèle')
@@ -294,7 +320,7 @@ const handleSubmit = async () => {
name: form.data.name.trim(), name: form.data.name.trim(),
description: form.data.description.trim() || undefined, description: form.data.description.trim() || undefined,
typeComposantId: form.data.typeComposantId, typeComposantId: form.data.typeComposantId,
structure: form.data.structure || {}, structure: form.data.structure || {}
}) })
if (!result.success) { if (!result.success) {
showError(result.error || 'Impossible de mettre à jour le modèle') showError(result.error || 'Impossible de mettre à jour le modèle')
@@ -312,7 +338,7 @@ const handleSubmit = async () => {
const confirmDelete = async (model) => { const confirmDelete = async (model) => {
const ok = confirm(`Supprimer le modèle "${model.name}" ?`) const ok = confirm(`Supprimer le modèle "${model.name}" ?`)
if (!ok) return if (!ok) { return }
const result = await deleteComponentModel(model.id) const result = await deleteComponentModel(model.id)
if (!result.success) { if (!result.success) {
showError(result.error || 'Impossible de supprimer ce modèle') showError(result.error || 'Impossible de supprimer ce modèle')

View File

@@ -2,8 +2,12 @@
<main class="container mx-auto px-6 py-8 space-y-6"> <main class="container mx-auto px-6 py-8 space-y-6">
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<div> <div>
<h1 class="text-3xl font-bold">Constructeurs</h1> <h1 class="text-3xl font-bold">
<p class="text-sm text-gray-500">Gérez les constructeurs et leurs coordonnées.</p> Constructeurs
</h1>
<p class="text-sm text-gray-500">
Gérez les constructeurs et leurs coordonnées.
</p>
</div> </div>
<button class="btn btn-primary" @click="openCreateModal"> <button class="btn btn-primary" @click="openCreateModal">
<IconLucidePlus class="w-4 h-4 mr-2" aria-hidden="true" /> <IconLucidePlus class="w-4 h-4 mr-2" aria-hidden="true" />
@@ -22,20 +26,26 @@
class="input input-bordered w-full" class="input input-bordered w-full"
placeholder="Nom, email ou téléphone" placeholder="Nom, email ou téléphone"
@input="debouncedSearch" @input="debouncedSearch"
/> >
</div> </div>
<div class="md:w-1/3"> <div class="md:w-1/3">
<label class="label"><span class="label-text">Tri</span></label> <label class="label"><span class="label-text">Tri</span></label>
<select v-model="sortKey" class="select select-bordered w-full"> <select v-model="sortKey" class="select select-bordered w-full">
<option value="name">Nom</option> <option value="name">
<option value="email">Email</option> Nom
<option value="phone">Téléphone</option> </option>
<option value="email">
Email
</option>
<option value="phone">
Téléphone
</option>
</select> </select>
</div> </div>
</div> </div>
<div v-if="loading" class="py-16 text-center text-sm text-gray-500"> <div v-if="loading" class="py-16 text-center text-sm text-gray-500">
<span class="loading loading-spinner loading-lg mb-2"></span> <span class="loading loading-spinner loading-lg mb-2" />
Chargement des constructeurs... Chargement des constructeurs...
</div> </div>
@@ -50,7 +60,9 @@
<th>Nom</th> <th>Nom</th>
<th>Email</th> <th>Email</th>
<th>Téléphone</th> <th>Téléphone</th>
<th class="text-right">Actions</th> <th class="text-right">
Actions
</th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
@@ -60,8 +72,12 @@
<td>{{ constructeur.phone || '—' }}</td> <td>{{ constructeur.phone || '—' }}</td>
<td class="text-right"> <td class="text-right">
<div class="flex justify-end gap-2"> <div class="flex justify-end gap-2">
<button class="btn btn-ghost btn-xs" @click="openEditModal(constructeur)">Modifier</button> <button class="btn btn-ghost btn-xs" @click="openEditModal(constructeur)">
<button class="btn btn-error btn-xs" @click="confirmDelete(constructeur)">Supprimer</button> Modifier
</button>
<button class="btn btn-error btn-xs" @click="confirmDelete(constructeur)">
Supprimer
</button>
</div> </div>
</td> </td>
</tr> </tr>
@@ -73,20 +89,24 @@
<dialog class="modal" :class="{ 'modal-open': modalOpen }"> <dialog class="modal" :class="{ 'modal-open': modalOpen }">
<div class="modal-box"> <div class="modal-box">
<h3 class="font-bold text-lg mb-4">{{ editingConstructeur ? 'Modifier' : 'Nouveau' }} constructeur</h3> <h3 class="font-bold text-lg mb-4">
<form @submit.prevent="saveConstructeur" class="space-y-4"> {{ editingConstructeur ? 'Modifier' : 'Nouveau' }} constructeur
</h3>
<form class="space-y-4" @submit.prevent="saveConstructeur">
<div class="form-control"> <div class="form-control">
<label class="label"><span class="label-text">Nom</span></label> <label class="label"><span class="label-text">Nom</span></label>
<input v-model="form.name" type="text" class="input input-bordered" required /> <input v-model="form.name" type="text" class="input input-bordered" required>
</div> </div>
<div class="grid grid-cols-1 md:grid-cols-2 gap-4"> <div class="grid grid-cols-1 md:grid-cols-2 gap-4">
<FieldEmail v-model="form.email" label="Email" /> <FieldEmail v-model="form.email" label="Email" />
<FieldPhone v-model="form.phone" label="Téléphone" /> <FieldPhone v-model="form.phone" label="Téléphone" />
</div> </div>
<div class="modal-action"> <div class="modal-action">
<button type="button" class="btn" @click="closeModal">Annuler</button> <button type="button" class="btn" @click="closeModal">
Annuler
</button>
<button type="submit" class="btn btn-primary" :disabled="saving"> <button type="submit" class="btn btn-primary" :disabled="saving">
<span v-if="saving" class="loading loading-spinner loading-xs mr-2"></span> <span v-if="saving" class="loading loading-spinner loading-xs mr-2" />
{{ editingConstructeur ? 'Enregistrer' : 'Créer' }} {{ editingConstructeur ? 'Enregistrer' : 'Créer' }}
</button> </button>
</div> </div>
@@ -119,7 +139,7 @@ const filteredConstructeurs = computed(() => {
const key = sortKey.value const key = sortKey.value
return (a[key] || '').localeCompare(b[key] || '') return (a[key] || '').localeCompare(b[key] || '')
}) })
if (!searchTerm.value) return sorted if (!searchTerm.value) { return sorted }
const term = searchTerm.value.toLowerCase() const term = searchTerm.value.toLowerCase()
return sorted.filter(item => return sorted.filter(item =>
[item.name, item.email, item.phone].some(value => value && value.toLowerCase().includes(term)) [item.name, item.email, item.phone].some(value => value && value.toLowerCase().includes(term))
@@ -130,7 +150,7 @@ const debouncedSearch = debounce(async () => {
await searchConstructeurs(searchTerm.value) await searchConstructeurs(searchTerm.value)
}, 300) }, 300)
function debounce(fn, delay) { function debounce (fn, delay) {
let timeout let timeout
return (...args) => { return (...args) => {
clearTimeout(timeout) clearTimeout(timeout)
@@ -153,7 +173,7 @@ const openEditModal = (constructeur) => {
form.value = { form.value = {
name: constructeur.name, name: constructeur.name,
email: constructeur.email || '', email: constructeur.email || '',
phone: constructeur.phone || '', phone: constructeur.phone || ''
} }
modalOpen.value = true modalOpen.value = true
} }
@@ -166,8 +186,8 @@ const closeModal = () => {
const saveConstructeur = async () => { const saveConstructeur = async () => {
saving.value = true saving.value = true
const payload = { ...form.value } const payload = { ...form.value }
if (!payload.email) delete payload.email if (!payload.email) { delete payload.email }
if (!payload.phone) delete payload.phone if (!payload.phone) { delete payload.phone }
let result let result
if (editingConstructeur.value) { if (editingConstructeur.value) {
result = await updateConstructeur(editingConstructeur.value.id, payload) result = await updateConstructeur(editingConstructeur.value.id, payload)
@@ -182,7 +202,7 @@ const saveConstructeur = async () => {
} }
const confirmDelete = async (constructeur) => { const confirmDelete = async (constructeur) => {
if (!confirm(`Supprimer le constructeur "${constructeur.name}" ?`)) return if (!confirm(`Supprimer le constructeur "${constructeur.name}" ?`)) { return }
const result = await deleteConstructeur(constructeur.id) const result = await deleteConstructeur(constructeur.id)
if (!result.success && result.error) { if (!result.success && result.error) {
showError(result.error) showError(result.error)

View File

@@ -1,7 +1,5 @@
<template> <template>
<main class="container mx-auto px-6 py-8 space-y-8"> <main class="container mx-auto px-6 py-8 space-y-8">
<DocumentPreviewModal <DocumentPreviewModal
:document="previewDocument" :document="previewDocument"
:visible="previewVisible" :visible="previewVisible"
@@ -20,7 +18,7 @@
type="search" type="search"
placeholder="Nom du document, type, site, machine..." placeholder="Nom du document, type, site, machine..."
class="input input-bordered w-full" class="input input-bordered w-full"
/> >
</div> </div>
<div class="w-full md:w-1/3"> <div class="w-full md:w-1/3">
@@ -28,19 +26,29 @@
<span class="label-text">Filtrer par rattachement</span> <span class="label-text">Filtrer par rattachement</span>
</label> </label>
<select v-model="attachmentFilter" class="select select-bordered w-full"> <select v-model="attachmentFilter" class="select select-bordered w-full">
<option value="all">Tous</option> <option value="all">
<option value="site">Sites</option> Tous
<option value="machine">Machines</option> </option>
<option value="composant">Composants</option> <option value="site">
<option value="piece">Pièces</option> Sites
</option>
<option value="machine">
Machines
</option>
<option value="composant">
Composants
</option>
<option value="piece">
Pièces
</option>
</select> </select>
</div> </div>
</div> </div>
<div class="divider my-0"></div> <div class="divider my-0" />
<div v-if="loading" class="flex flex-col items-center justify-center py-16 text-sm text-gray-500"> <div v-if="loading" class="flex flex-col items-center justify-center py-16 text-sm text-gray-500">
<span class="loading loading-spinner loading-lg mb-3"></span> <span class="loading loading-spinner loading-lg mb-3" />
Chargement des documents... Chargement des documents...
</div> </div>
@@ -58,7 +66,9 @@
<th>Taille</th> <th>Taille</th>
<th>Rattaché à</th> <th>Rattaché à</th>
<th>Date</th> <th>Date</th>
<th class="text-right">Actions</th> <th class="text-right">
Actions
</th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
@@ -73,8 +83,12 @@
/> />
</span> </span>
<div> <div>
<div class="font-semibold">{{ document.name }}</div> <div class="font-semibold">
<div class="text-xs text-gray-500">{{ document.filename }}</div> {{ document.name }}
</div>
<div class="text-xs text-gray-500">
{{ document.filename }}
</div>
</div> </div>
</div> </div>
</td> </td>
@@ -124,7 +138,6 @@ import { formatFrenchDate } from '~/utils/date'
import DocumentPreviewModal from '~/components/DocumentPreviewModal.vue' import DocumentPreviewModal from '~/components/DocumentPreviewModal.vue'
import IconLucideFileSearch from '~icons/lucide/file-search' import IconLucideFileSearch from '~icons/lucide/file-search'
const { documents, loading, loadDocuments } = useDocuments() const { documents, loading, loadDocuments } = useDocuments()
const searchTerm = ref('') const searchTerm = ref('')
@@ -148,9 +161,9 @@ const filteredDocuments = computed(() => {
(filter === 'composant' && document.composantId) || (filter === 'composant' && document.composantId) ||
(filter === 'piece' && document.pieceId) (filter === 'piece' && document.pieceId)
if (!matchesFilter) return false if (!matchesFilter) { return false }
if (!term) return true if (!term) { return true }
const searchable = [ const searchable = [
document.name, document.name,
@@ -159,7 +172,7 @@ const filteredDocuments = computed(() => {
document.site?.name, document.site?.name,
document.machine?.name, document.machine?.name,
document.composant?.name, document.composant?.name,
document.piece?.name, document.piece?.name
] ]
.filter(Boolean) .filter(Boolean)
.map(value => value.toLowerCase()) .map(value => value.toLowerCase())
@@ -169,18 +182,18 @@ const filteredDocuments = computed(() => {
}) })
const formatSize = (size) => { const formatSize = (size) => {
if (size === undefined || size === null) return '—' if (size === undefined || size === null) { return '—' }
if (size === 0) return '0 B' if (size === 0) { return '0 B' }
const units = ['B', 'KB', 'MB', 'GB'] const units = ['B', 'KB', 'MB', 'GB']
const index = Math.min(units.length - 1, Math.floor(Math.log(size) / Math.log(1024))) const index = Math.min(units.length - 1, Math.floor(Math.log(size) / Math.log(1024)))
const formatted = size / Math.pow(1024, index) const formatted = size / Math.pow(1024, index)
return `${formatted.toFixed(1)} ${units[index]}` return `${formatted.toFixed(1)} ${units[index]}`
} }
const documentIcon = (doc) => getFileIcon({ name: doc.filename || doc.name, mime: doc.mimeType }) const documentIcon = doc => getFileIcon({ name: doc.filename || doc.name, mime: doc.mimeType })
const downloadDocument = (doc) => { const downloadDocument = (doc) => {
if (!doc?.path) return if (!doc?.path) { return }
if (doc.path.startsWith('data:')) { if (doc.path.startsWith('data:')) {
const link = document.createElement('a') const link = document.createElement('a')
@@ -194,7 +207,7 @@ const downloadDocument = (doc) => {
} }
const openPreview = (doc) => { const openPreview = (doc) => {
if (!canPreviewDocument(doc)) return if (!canPreviewDocument(doc)) { return }
previewDocument.value = doc previewDocument.value = doc
previewVisible.value = true previewVisible.value = true
} }

View File

@@ -1,7 +1,5 @@
<template> <template>
<main class="container mx-auto px-6 py-8 space-y-8"> <main class="container mx-auto px-6 py-8 space-y-8">
<div class="card bg-base-100 shadow-xl"> <div class="card bg-base-100 shadow-xl">
<div class="card-body space-y-6"> <div class="card-body space-y-6">
<div class="flex items-center gap-3"> <div class="flex items-center gap-3">
@@ -12,7 +10,9 @@
/> />
</span> </span>
<div> <div>
<h2 class="card-title text-2xl">Nouveau type de machine</h2> <h2 class="card-title text-2xl">
Nouveau type de machine
</h2>
<p class="text-sm text-gray-500"> <p class="text-sm text-gray-500">
Complétez les informations puis enregistrez pour générer le nouveau type. Complétez les informations puis enregistrez pour générer le nouveau type.
</p> </p>
@@ -37,7 +37,9 @@
</div> </div>
<template v-else> <template v-else>
<div v-if="recentTypes.length" class="space-y-4"> <div v-if="recentTypes.length" class="space-y-4">
<h3 class="text-xl font-semibold">Types générés récemment</h3> <h3 class="text-xl font-semibold">
Types générés récemment
</h3>
<div class="grid grid-cols-1 gap-4 md:grid-cols-2 lg:grid-cols-3"> <div class="grid grid-cols-1 gap-4 md:grid-cols-2 lg:grid-cols-3">
<article <article
v-for="type in recentTypes" v-for="type in recentTypes"
@@ -46,10 +48,14 @@
> >
<div class="card-body space-y-2"> <div class="card-body space-y-2">
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<h4 class="card-title text-base">{{ type.name }}</h4> <h4 class="card-title text-base">
{{ type.name }}
</h4>
<span v-if="type.category" class="badge badge-outline badge-sm">{{ type.category }}</span> <span v-if="type.category" class="badge badge-outline badge-sm">{{ type.category }}</span>
</div> </div>
<p class="text-sm text-gray-600 line-clamp-3">{{ type.description || 'Aucune description' }}</p> <p class="text-sm text-gray-600 line-clamp-3">
{{ type.description || 'Aucune description' }}
</p>
<div class="text-xs text-gray-500 flex items-center gap-2"> <div class="text-xs text-gray-500 flex items-center gap-2">
<span class="inline-flex items-center gap-1"> <span class="inline-flex items-center gap-1">
<IconLucideClipboardList class="h-4 w-4" aria-hidden="true" /> <IconLucideClipboardList class="h-4 w-4" aria-hidden="true" />
@@ -94,7 +100,7 @@ const createEmptyType = () => ({
maintenanceFrequency: '', maintenanceFrequency: '',
customFields: [], customFields: [],
componentRequirements: [], componentRequirements: [],
pieceRequirements: [], pieceRequirements: []
}) })
const draftType = ref(createEmptyType()) const draftType = ref(createEmptyType())
@@ -115,7 +121,7 @@ onMounted(async () => {
}) })
const parseOptions = (field = {}) => { const parseOptions = (field = {}) => {
if (field.type !== 'select') return [] if (field.type !== 'select') { return [] }
if (field.optionsText && typeof field.optionsText === 'string') { if (field.optionsText && typeof field.optionsText === 'string') {
return field.optionsText return field.optionsText
.split('\n') .split('\n')
@@ -157,7 +163,7 @@ const normalizeComponentRequirements = (requirements = []) =>
minCount: toIntegerOrNull(req.minCount, 1), minCount: toIntegerOrNull(req.minCount, 1),
maxCount: toIntegerOrNull(req.maxCount, null), maxCount: toIntegerOrNull(req.maxCount, null),
required: req.required ?? true, required: req.required ?? true,
allowNewModels: req.allowNewModels ?? true, allowNewModels: req.allowNewModels ?? true
})) }))
const normalizePieceRequirements = (requirements = []) => const normalizePieceRequirements = (requirements = []) =>
@@ -169,10 +175,10 @@ const normalizePieceRequirements = (requirements = []) =>
minCount: toIntegerOrNull(req.minCount, 0), minCount: toIntegerOrNull(req.minCount, 0),
maxCount: toIntegerOrNull(req.maxCount, null), maxCount: toIntegerOrNull(req.maxCount, null),
required: req.required ?? false, required: req.required ?? false,
allowNewModels: req.allowNewModels ?? true, allowNewModels: req.allowNewModels ?? true
})) }))
const buildPayload = (typeData) => ({ const buildPayload = typeData => ({
name: typeData.name, name: typeData.name,
description: typeData.description, description: typeData.description,
category: typeData.category, category: typeData.category,
@@ -204,7 +210,7 @@ const handleSubmit = async () => {
} else if (result?.error) { } else if (result?.error) {
showError(result.error) showError(result.error)
} else { } else {
showError("Impossible de créer le type.") showError('Impossible de créer le type.')
} }
} }
</script> </script>

View File

@@ -5,17 +5,29 @@
<!-- Header with Stats --> <!-- Header with Stats -->
<div class="flex justify-between items-center mb-6"> <div class="flex justify-between items-center mb-6">
<div> <div>
<h2 class="text-2xl font-bold text-gray-800">Vue d'ensemble</h2> <h2 class="text-2xl font-bold text-gray-800">
<p class="text-gray-600">Machines organisées par site</p> Vue d'ensemble
</h2>
<p class="text-gray-600">
Machines organisées par site
</p>
</div> </div>
<div class="stats shadow"> <div class="stats shadow">
<div class="stat"> <div class="stat">
<div class="stat-title">Sites</div> <div class="stat-title">
<div class="stat-value text-primary">{{ sites.length }}</div> Sites
</div>
<div class="stat-value text-primary">
{{ sites.length }}
</div>
</div> </div>
<div class="stat"> <div class="stat">
<div class="stat-title">Machines</div> <div class="stat-title">
<div class="stat-value text-secondary">{{ totalMachines }}</div> Machines
</div>
<div class="stat-value text-secondary">
{{ totalMachines }}
</div>
</div> </div>
</div> </div>
</div> </div>
@@ -33,14 +45,16 @@
type="text" type="text"
placeholder="Nom de machine ou site..." placeholder="Nom de machine ou site..."
class="input input-bordered" class="input input-bordered"
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
<label class="label"> <label class="label">
<span class="label-text">Type de machine</span> <span class="label-text">Type de machine</span>
</label> </label>
<select v-model="selectedType" class="select select-bordered"> <select v-model="selectedType" class="select select-bordered">
<option value="">Tous les types</option> <option value="">
Tous les types
</option>
<option <option
v-for="type in machineTypes" v-for="type in machineTypes"
:key="type.id" :key="type.id"
@@ -55,7 +69,9 @@
<span class="label-text">Catégorie</span> <span class="label-text">Catégorie</span>
</label> </label>
<select v-model="selectedCategory" class="select select-bordered"> <select v-model="selectedCategory" class="select select-bordered">
<option value="">Toutes les catégories</option> <option value="">
Toutes les catégories
</option>
<option <option
v-for="category in categories" v-for="category in categories"
:key="category" :key="category"
@@ -71,7 +87,7 @@
<!-- Loading State --> <!-- Loading State -->
<div v-if="loading" class="flex justify-center items-center py-12"> <div v-if="loading" class="flex justify-center items-center py-12">
<span class="loading loading-spinner loading-lg"></span> <span class="loading loading-spinner loading-lg" />
</div> </div>
<!-- Hierarchical Machines View --> <!-- Hierarchical Machines View -->
@@ -88,12 +104,12 @@
Commencez par ajouter des sites et des machines. Commencez par ajouter des sites et des machines.
</p> </p>
<div class="flex gap-2 justify-center"> <div class="flex gap-2 justify-center">
<button @click="showAddSiteModal = true" class="btn btn-primary"> <button class="btn btn-primary" @click="showAddSiteModal = true">
Ajouter un site Ajouter un site
</button> </button>
<button <button
@click="showAddMachineModal = true"
class="btn btn-secondary" class="btn btn-secondary"
@click="showAddMachineModal = true"
> >
Ajouter une machine Ajouter une machine
</button> </button>
@@ -119,7 +135,9 @@
</div> </div>
</div> </div>
<div> <div>
<h3 class="text-xl font-bold">{{ site.name }}</h3> <h3 class="text-xl font-bold">
{{ site.name }}
</h3>
<div class="text-sm text-gray-600 space-y-1"> <div class="text-sm text-gray-600 space-y-1">
<div class="flex items-center gap-2"> <div class="flex items-center gap-2">
<IconLucideUser <IconLucideUser
@@ -141,7 +159,7 @@
aria-hidden="true" aria-hidden="true"
/> />
<span> <span>
{{ site.contactAddress }}<br /> {{ site.contactAddress }}<br>
{{ site.contactPostalCode }} {{ site.contactCity }} {{ site.contactPostalCode }} {{ site.contactCity }}
</span> </span>
</div> </div>
@@ -153,8 +171,8 @@
{{ site.machines?.length || 0 }} machines {{ site.machines?.length || 0 }} machines
</div> </div>
<button <button
@click="toggleSiteCollapse(site.id)"
class="btn btn-ghost btn-sm" class="btn btn-ghost btn-sm"
@click="toggleSiteCollapse(site.id)"
> >
<IconLucideChevronDown <IconLucideChevronDown
class="w-5 h-5 transition-transform" class="w-5 h-5 transition-transform"
@@ -171,12 +189,12 @@
<div <div
v-if=" v-if="
!collapsedSites.includes(site.id) && !collapsedSites.includes(site.id) &&
site.machines && site.machines &&
site.machines.length > 0 site.machines.length > 0
" "
class="space-y-3" class="space-y-3"
> >
<div class="divider"></div> <div class="divider" />
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4"> <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
<div <div
v-for="machine in site.machines" v-for="machine in site.machines"
@@ -186,7 +204,9 @@
> >
<div class="card-body p-4"> <div class="card-body p-4">
<div class="flex items-center justify-between mb-2"> <div class="flex items-center justify-between mb-2">
<h4 class="font-semibold text-sm">{{ machine.name }}</h4> <h4 class="font-semibold text-sm">
{{ machine.name }}
</h4>
<div <div
class="badge badge-sm" class="badge badge-sm"
:class=" :class="
@@ -246,7 +266,7 @@
<div <div
v-else-if=" v-else-if="
!collapsedSites.includes(site.id) && !collapsedSites.includes(site.id) &&
(!site.machines || site.machines.length === 0) (!site.machines || site.machines.length === 0)
" "
class="text-center py-6" class="text-center py-6"
> >
@@ -257,8 +277,8 @@
Aucune machine dans ce site Aucune machine dans ce site
</p> </p>
<button <button
@click="addMachineToSite(site)"
class="btn btn-sm btn-primary" class="btn btn-sm btn-primary"
@click="addMachineToSite(site)"
> >
Ajouter une machine Ajouter une machine
</button> </button>
@@ -271,8 +291,10 @@
<!-- Add Site Modal --> <!-- Add Site Modal -->
<div v-if="showAddSiteModal" class="modal modal-open"> <div v-if="showAddSiteModal" class="modal modal-open">
<div class="modal-box"> <div class="modal-box">
<h3 class="font-bold text-lg mb-4">Ajouter un nouveau site</h3> <h3 class="font-bold text-lg mb-4">
<form @submit.prevent="handleCreateSite" class="space-y-4"> Ajouter un nouveau site
</h3>
<form class="space-y-4" @submit.prevent="handleCreateSite">
<div class="form-control"> <div class="form-control">
<label class="label"> <label class="label">
<span class="label-text">Nom du site</span> <span class="label-text">Nom du site</span>
@@ -283,7 +305,7 @@
placeholder="Ex: Usine de production" placeholder="Ex: Usine de production"
class="input input-bordered" class="input input-bordered"
required required
/> >
</div> </div>
<SiteContactFormFields :form="newSite" /> <SiteContactFormFields :form="newSite" />
@@ -291,12 +313,14 @@
<div class="modal-action"> <div class="modal-action">
<button <button
type="button" type="button"
@click="showAddSiteModal = false"
class="btn btn-outline" class="btn btn-outline"
@click="showAddSiteModal = false"
> >
Annuler Annuler
</button> </button>
<button type="submit" class="btn btn-primary">Créer le site</button> <button type="submit" class="btn btn-primary">
Créer le site
</button>
</div> </div>
</form> </form>
</div> </div>
@@ -305,7 +329,9 @@
<!-- Add Machine Modal --> <!-- Add Machine Modal -->
<div v-if="showAddMachineModal" class="modal modal-open"> <div v-if="showAddMachineModal" class="modal modal-open">
<div class="modal-box max-w-2xl"> <div class="modal-box max-w-2xl">
<h3 class="font-bold text-lg mb-4">Ajouter une nouvelle machine</h3> <h3 class="font-bold text-lg mb-4">
Ajouter une nouvelle machine
</h3>
<form @submit.prevent="handleCreateMachine"> <form @submit.prevent="handleCreateMachine">
<div class="grid grid-cols-1 md:grid-cols-2 gap-4 mb-4"> <div class="grid grid-cols-1 md:grid-cols-2 gap-4 mb-4">
<div class="form-control"> <div class="form-control">
@@ -318,7 +344,7 @@
placeholder="Ex: Presse hydraulique #1" placeholder="Ex: Presse hydraulique #1"
class="input input-bordered" class="input input-bordered"
required required
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
@@ -330,7 +356,9 @@
class="select select-bordered" class="select select-bordered"
required required
> >
<option value="">Sélectionner un site</option> <option value="">
Sélectionner un site
</option>
<option v-for="site in sites" :key="site.id" :value="site.id"> <option v-for="site in sites" :key="site.id" :value="site.id">
{{ site.name }} {{ site.name }}
</option> </option>
@@ -348,7 +376,9 @@
class="select select-bordered" class="select select-bordered"
required required
> >
<option value="">Sélectionner un type</option> <option value="">
Sélectionner un type
</option>
<option <option
v-for="type in machineTypes" v-for="type in machineTypes"
:key="type.id" :key="type.id"
@@ -368,7 +398,7 @@
type="text" type="text"
placeholder="Ex: PRESS-001" placeholder="Ex: PRESS-001"
class="input input-bordered" class="input input-bordered"
/> >
</div> </div>
</div> </div>
@@ -405,8 +435,8 @@
<div class="modal-action"> <div class="modal-action">
<button <button
type="button" type="button"
@click="showAddMachineModal = false"
class="btn btn-outline" class="btn btn-outline"
@click="showAddMachineModal = false"
> >
Annuler Annuler
</button> </button>
@@ -421,129 +451,129 @@
</template> </template>
<script setup> <script setup>
import { ref, reactive, onMounted, computed } from "vue"; import { ref, reactive, onMounted, computed } from 'vue'
import SiteContactFormFields from "~/components/sites/SiteContactFormFields.vue"; import SiteContactFormFields from '~/components/sites/SiteContactFormFields.vue'
import { useSites } from "~/composables/useSites"; import { useSites } from '~/composables/useSites'
import { useMachineTypesApi } from "~/composables/useMachineTypesApi"; import { useMachineTypesApi } from '~/composables/useMachineTypesApi'
import { useMachines } from "~/composables/useMachines"; import { useMachines } from '~/composables/useMachines'
import { useToast } from "~/composables/useToast"; import { useToast } from '~/composables/useToast'
import IconLucideFactory from "~icons/lucide/factory"; import IconLucideFactory from '~icons/lucide/factory'
import IconLucideMapPin from "~icons/lucide/map-pin"; import IconLucideMapPin from '~icons/lucide/map-pin'
import IconLucideUser from "~icons/lucide/user"; import IconLucideUser from '~icons/lucide/user'
import IconLucidePhone from "~icons/lucide/phone"; import IconLucidePhone from '~icons/lucide/phone'
import IconLucideMapPinned from "~icons/lucide/map-pinned"; import IconLucideMapPinned from '~icons/lucide/map-pinned'
import IconLucideChevronDown from "~icons/lucide/chevron-down"; import IconLucideChevronDown from '~icons/lucide/chevron-down'
import IconLucideSettings2 from "~icons/lucide/settings-2"; import IconLucideSettings2 from '~icons/lucide/settings-2'
import IconLucideTag from "~icons/lucide/tag"; import IconLucideTag from '~icons/lucide/tag'
const { sites, loading, loadSites, createSite } = useSites(); const { sites, loading, loadSites, createSite } = useSites()
const { machineTypes, loadMachineTypes } = useMachineTypesApi(); const { machineTypes, loadMachineTypes } = useMachineTypesApi()
const { createMachineFromType, deleteMachine } = useMachines(); const { createMachineFromType, deleteMachine } = useMachines()
// Data // Data
const showAddSiteModal = ref(false); const showAddSiteModal = ref(false)
const showAddMachineModal = ref(false); const showAddMachineModal = ref(false)
const searchTerm = ref(""); const searchTerm = ref('')
const selectedType = ref(""); const selectedType = ref('')
const selectedCategory = ref(""); const selectedCategory = ref('')
const collapsedSites = ref([]); const collapsedSites = ref([])
const newSite = reactive({ const newSite = reactive({
name: "", name: '',
contactName: "", contactName: '',
contactPhone: "", contactPhone: '',
contactAddress: "", contactAddress: '',
contactPostalCode: "", contactPostalCode: '',
contactCity: "", contactCity: ''
}); })
const newMachine = reactive({ const newMachine = reactive({
name: "", name: '',
siteId: "", siteId: '',
typeMachineId: "", typeMachineId: '',
reference: "", reference: ''
}); })
// Computed // Computed
const selectedMachineType = computed(() => { const selectedMachineType = computed(() => {
if (!newMachine.typeMachineId) return null; if (!newMachine.typeMachineId) { return null }
return machineTypes.value.find( return machineTypes.value.find(
(type) => type.id === newMachine.typeMachineId type => type.id === newMachine.typeMachineId
); )
}); })
const categories = computed(() => { const categories = computed(() => {
const cats = new Set(); const cats = new Set()
machineTypes.value.forEach((type) => { machineTypes.value.forEach((type) => {
if (type.category) cats.add(type.category); if (type.category) { cats.add(type.category) }
}); })
return Array.from(cats); return Array.from(cats)
}); })
const totalMachines = computed(() => { const totalMachines = computed(() => {
return sites.value.reduce((total, site) => { return sites.value.reduce((total, site) => {
return total + (site.machines?.length || 0); return total + (site.machines?.length || 0)
}, 0); }, 0)
}); })
const filteredSites = computed(() => { const filteredSites = computed(() => {
let filtered = sites.value; let filtered = sites.value
// Filtrer par terme de recherche // Filtrer par terme de recherche
if (searchTerm.value) { if (searchTerm.value) {
filtered = filtered.filter((site) => { filtered = filtered.filter((site) => {
const lowerTerm = searchTerm.value.toLowerCase(); const lowerTerm = searchTerm.value.toLowerCase()
const siteMatches = [ const siteMatches = [
site.name, site.name,
site.contactName, site.contactName,
site.contactPhone, site.contactPhone,
site.contactAddress, site.contactAddress,
site.contactPostalCode, site.contactPostalCode,
site.contactCity, site.contactCity
].some((field) => { ].some((field) => {
if (!field) return false; if (!field) { return false }
return field.toLowerCase().includes(lowerTerm); return field.toLowerCase().includes(lowerTerm)
}); })
const machineMatches = site.machines?.some( const machineMatches = site.machines?.some(
(machine) => machine =>
machine.name.toLowerCase().includes(lowerTerm) || machine.name.toLowerCase().includes(lowerTerm) ||
machine.reference?.toLowerCase().includes(lowerTerm) machine.reference?.toLowerCase().includes(lowerTerm)
); )
return siteMatches || machineMatches; return siteMatches || machineMatches
}); })
} }
// Filtrer par type de machine // Filtrer par type de machine
if (selectedType.value) { if (selectedType.value) {
filtered = filtered filtered = filtered
.map((site) => ({ .map(site => ({
...site, ...site,
machines: machines:
site.machines?.filter( site.machines?.filter(
(machine) => machine.typeMachineId === selectedType.value machine => machine.typeMachineId === selectedType.value
) || [], ) || []
})) }))
.filter((site) => site.machines.length > 0); .filter(site => site.machines.length > 0)
} }
// Filtrer par catégorie // Filtrer par catégorie
if (selectedCategory.value) { if (selectedCategory.value) {
filtered = filtered filtered = filtered
.map((site) => ({ .map(site => ({
...site, ...site,
machines: machines:
site.machines?.filter( site.machines?.filter(
(machine) => machine =>
machine.typeMachine?.category === selectedCategory.value machine.typeMachine?.category === selectedCategory.value
) || [], ) || []
})) }))
.filter((site) => site.machines.length > 0); .filter(site => site.machines.length > 0)
} }
return filtered; return filtered
}); })
// Methods // Methods
const handleCreateSite = async () => { const handleCreateSite = async () => {
@@ -553,69 +583,69 @@ const handleCreateSite = async () => {
contactPhone: newSite.contactPhone, contactPhone: newSite.contactPhone,
contactAddress: newSite.contactAddress, contactAddress: newSite.contactAddress,
contactPostalCode: newSite.contactPostalCode, contactPostalCode: newSite.contactPostalCode,
contactCity: newSite.contactCity, contactCity: newSite.contactCity
}); })
if (result.success) { if (result.success) {
showAddSiteModal.value = false; showAddSiteModal.value = false
// Reset form // Reset form
newSite.name = ""; newSite.name = ''
newSite.contactName = ""; newSite.contactName = ''
newSite.contactPhone = ""; newSite.contactPhone = ''
newSite.contactAddress = ""; newSite.contactAddress = ''
newSite.contactPostalCode = ""; newSite.contactPostalCode = ''
newSite.contactCity = ""; newSite.contactCity = ''
} }
}; }
const handleCreateMachine = async () => { const handleCreateMachine = async () => {
if (!selectedMachineType.value) { if (!selectedMachineType.value) {
console.error("Aucun type de machine sélectionné"); console.error('Aucun type de machine sélectionné')
return; return
} }
const machineData = { const machineData = {
name: newMachine.name, name: newMachine.name,
siteId: newMachine.siteId, siteId: newMachine.siteId,
reference: newMachine.reference, reference: newMachine.reference
}; }
const result = await createMachineFromType( const result = await createMachineFromType(
machineData, machineData,
selectedMachineType.value selectedMachineType.value
); )
if (result.success) { if (result.success) {
// Reset form // Reset form
newMachine.name = ""; newMachine.name = ''
newMachine.siteId = ""; newMachine.siteId = ''
newMachine.typeMachineId = ""; newMachine.typeMachineId = ''
newMachine.reference = ""; newMachine.reference = ''
showAddMachineModal.value = false; showAddMachineModal.value = false
} }
}; }
const toggleSiteCollapse = (siteId) => { const toggleSiteCollapse = (siteId) => {
const index = collapsedSites.value.indexOf(siteId); const index = collapsedSites.value.indexOf(siteId)
if (index > -1) { if (index > -1) {
collapsedSites.value.splice(index, 1); collapsedSites.value.splice(index, 1)
} else { } else {
collapsedSites.value.push(siteId); collapsedSites.value.push(siteId)
} }
}; }
const viewMachineDetails = (machine) => { const viewMachineDetails = (machine) => {
// Navigation vers la page de détails de la machine // Navigation vers la page de détails de la machine
navigateTo(`/machine/${machine.id}`); navigateTo(`/machine/${machine.id}`)
}; }
const editMachine = (machine) => { const editMachine = (machine) => {
// Rediriger vers la page d'édition de la machine // Rediriger vers la page d'édition de la machine
navigateTo(`/machine/${machine.id}?edit=true`); navigateTo(`/machine/${machine.id}?edit=true`)
}; }
const confirmDeleteMachine = async (machine) => { const confirmDeleteMachine = async (machine) => {
const { showError, showSuccess } = useToast(); const { showError, showSuccess } = useToast()
if ( if (
confirm( confirm(
@@ -623,36 +653,36 @@ const confirmDeleteMachine = async (machine) => {
) )
) { ) {
try { try {
const result = await deleteMachine(machine.id); const result = await deleteMachine(machine.id)
if (result.success) { if (result.success) {
showSuccess(`Machine "${machine.name}" supprimée avec succès`); showSuccess(`Machine "${machine.name}" supprimée avec succès`)
} else { } else {
showError(`Erreur lors de la suppression: ${result.error}`); showError(`Erreur lors de la suppression: ${result.error}`)
} }
} catch (error) { } catch (error) {
showError(`Erreur lors de la suppression: ${error.message}`); showError(`Erreur lors de la suppression: ${error.message}`)
} }
} }
}; }
const addMachineToSite = (site) => { const addMachineToSite = (site) => {
newMachine.siteId = site.id; newMachine.siteId = site.id
showAddMachineModal.value = true; showAddMachineModal.value = true
}; }
const getCategoryBadgeClass = (category) => { const getCategoryBadgeClass = (category) => {
const classes = { const classes = {
Production: "badge-primary", Production: 'badge-primary',
Transformation: "badge-secondary", Transformation: 'badge-secondary',
Manutention: "badge-accent", Manutention: 'badge-accent',
Traitement: "badge-info", Traitement: 'badge-info',
Contrôle: "badge-warning", Contrôle: 'badge-warning'
}; }
return classes[category] || "badge-neutral"; return classes[category] || 'badge-neutral'
}; }
// Lifecycle // Lifecycle
onMounted(async () => { onMounted(async () => {
await Promise.all([loadSites(), loadMachineTypes()]); await Promise.all([loadSites(), loadMachineTypes()])
}); })
</script> </script>

View File

@@ -1,12 +1,12 @@
<template> <template>
<main class="container mx-auto px-6 py-8"> <main class="container mx-auto px-6 py-8">
<!-- Machine Types List --> <!-- Machine Types List -->
<div class="my-8"> <div class="my-8">
<!-- Header with Add Button --> <!-- Header with Add Button -->
<div class="flex justify-between items-center mb-6"> <div class="flex justify-between items-center mb-6">
<h2 class="text-2xl font-bold text-gray-800">Squelettes de machine</h2> <h2 class="text-2xl font-bold text-gray-800">
Squelettes de machine
</h2>
<NuxtLink to="/generator" class="btn btn-primary"> <NuxtLink to="/generator" class="btn btn-primary">
<IconLucidePlus <IconLucidePlus
class="w-5 h-5 mr-2" class="w-5 h-5 mr-2"
@@ -18,8 +18,8 @@
<!-- Categories Tabs --> <!-- Categories Tabs -->
<div class="tabs tabs-boxed mb-6"> <div class="tabs tabs-boxed mb-6">
<a <a
v-for="category in categories" v-for="category in categories"
:key="category" :key="category"
class="tab" class="tab"
:class="{ 'tab-active': selectedCategory === category }" :class="{ 'tab-active': selectedCategory === category }"
@@ -31,17 +31,23 @@
<!-- Machine Types Grid --> <!-- Machine Types Grid -->
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6"> <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
<div <div
v-for="type in filteredTypes" v-for="type in filteredTypes"
:key="type.id" :key="type.id"
class="card bg-base-100 shadow-lg hover:shadow-xl transition-all duration-300 cursor-pointer" class="card bg-base-100 shadow-lg hover:shadow-xl transition-all duration-300 cursor-pointer"
> >
<div class="card-body"> <div class="card-body">
<div class="flex items-center justify-between mb-4"> <div class="flex items-center justify-between mb-4">
<h3 class="card-title text-lg">{{ type.name }}</h3> <h3 class="card-title text-lg">
<div class="badge badge-primary">{{ type.category }}</div> {{ type.name }}
</h3>
<div class="badge badge-primary">
{{ type.category }}
</div>
</div> </div>
<p class="text-gray-600 mb-4">{{ type.description }}</p> <p class="text-gray-600 mb-4">
{{ type.description }}
</p>
<div class="space-y-2 text-sm text-gray-500"> <div class="space-y-2 text-sm text-gray-500">
<div class="flex items-center gap-2"> <div class="flex items-center gap-2">
<IconLucidePackage class="w-4 h-4" aria-hidden="true" /> <IconLucidePackage class="w-4 h-4" aria-hidden="true" />
@@ -56,8 +62,12 @@
<button class="btn btn-sm btn-error" @click.stop="confirmDeleteType(type)"> <button class="btn btn-sm btn-error" @click.stop="confirmDeleteType(type)">
Supprimer Supprimer
</button> </button>
<NuxtLink :to="`/type/${type.id}`" class="btn btn-sm btn-outline">Voir détails</NuxtLink> <NuxtLink :to="`/type/${type.id}`" class="btn btn-sm btn-outline">
<button class="btn btn-sm btn-primary">Utiliser</button> Voir détails
</NuxtLink>
<button class="btn btn-sm btn-primary">
Utiliser
</button>
</div> </div>
</div> </div>
</div> </div>
@@ -65,13 +75,17 @@
<!-- Empty State --> <!-- Empty State -->
<div v-if="filteredTypes.length === 0" class="text-center py-12"> <div v-if="filteredTypes.length === 0" class="text-center py-12">
<div class="avatar placeholder"> <div class="avatar placeholder">
<div class="bg-neutral text-neutral-content rounded-full w-16"> <div class="bg-neutral text-neutral-content rounded-full w-16">
<IconLucideLayoutGrid class="w-8 h-8" aria-hidden="true" /> <IconLucideLayoutGrid class="w-8 h-8" aria-hidden="true" />
</div> </div>
</div> </div>
<h3 class="text-lg font-semibold text-gray-600 mt-4">Aucun type trouvé</h3> <h3 class="text-lg font-semibold text-gray-600 mt-4">
<p class="text-gray-500">Aucun type de machine ne correspond à cette catégorie.</p> Aucun type trouvé
</h3>
<p class="text-gray-500">
Aucun type de machine ne correspond à cette catégorie.
</p>
</div> </div>
</div> </div>
</main> </main>
@@ -90,7 +104,7 @@ const { machineTypes, loading, loadMachineTypes, deleteMachineType } = useMachin
const categories = ref([ const categories = ref([
'Toutes', 'Toutes',
'Production', 'Production',
'Transformation', 'Transformation',
'Manutention', 'Manutention',
'Traitement', 'Traitement',
'Contrôle' 'Contrôle'
@@ -107,7 +121,7 @@ const filteredTypes = computed(() => {
const confirmDeleteType = async (type) => { const confirmDeleteType = async (type) => {
const { showError, showSuccess } = useToast() const { showError, showSuccess } = useToast()
if (confirm(`Êtes-vous sûr de vouloir supprimer le type "${type.name}" ? Cette action est irréversible.`)) { if (confirm(`Êtes-vous sûr de vouloir supprimer le type "${type.name}" ? Cette action est irréversible.`)) {
try { try {
const result = await deleteMachineType(type.id) const result = await deleteMachineType(type.id)
@@ -126,4 +140,4 @@ const confirmDeleteType = async (type) => {
onMounted(async () => { onMounted(async () => {
await loadMachineTypes() await loadMachineTypes()
}) })
</script> </script>

View File

@@ -2,7 +2,9 @@
<main class="container mx-auto px-6 py-8"> <main class="container mx-auto px-6 py-8">
<div class="my-8"> <div class="my-8">
<div class="flex justify-between items-center mb-6"> <div class="flex justify-between items-center mb-6">
<h2 class="text-2xl font-bold">Parc Machines</h2> <h2 class="text-2xl font-bold">
Parc Machines
</h2>
<NuxtLink to="/machines/new" class="btn btn-primary"> <NuxtLink to="/machines/new" class="btn btn-primary">
<IconLucidePlus class="w-5 h-5 mr-2" aria-hidden="true" /> <IconLucidePlus class="w-5 h-5 mr-2" aria-hidden="true" />
Ajouter une machine Ajouter une machine
@@ -17,7 +19,9 @@
<span class="label-text">Site</span> <span class="label-text">Site</span>
</label> </label>
<select v-model="selectedSite" class="select select-bordered"> <select v-model="selectedSite" class="select select-bordered">
<option value="">Tous les sites</option> <option value="">
Tous les sites
</option>
<option v-for="site in sites" :key="site.id" :value="site.id"> <option v-for="site in sites" :key="site.id" :value="site.id">
{{ site.name }} {{ site.name }}
</option> </option>
@@ -28,7 +32,9 @@
<span class="label-text">Type de machine</span> <span class="label-text">Type de machine</span>
</label> </label>
<select v-model="selectedType" class="select select-bordered"> <select v-model="selectedType" class="select select-bordered">
<option value="">Tous les types</option> <option value="">
Tous les types
</option>
<option v-for="type in machineTypes" :key="type.id" :value="type.id"> <option v-for="type in machineTypes" :key="type.id" :value="type.id">
{{ type.name }} {{ type.name }}
</option> </option>
@@ -39,7 +45,9 @@
<span class="label-text">Catégorie</span> <span class="label-text">Catégorie</span>
</label> </label>
<select v-model="selectedCategory" class="select select-bordered"> <select v-model="selectedCategory" class="select select-bordered">
<option value="">Toutes les catégories</option> <option value="">
Toutes les catégories
</option>
<option v-for="category in categories" :key="category" :value="category"> <option v-for="category in categories" :key="category" :value="category">
{{ category }} {{ category }}
</option> </option>
@@ -50,14 +58,18 @@
</div> </div>
<div v-if="loading" class="flex justify-center items-center py-12"> <div v-if="loading" class="flex justify-center items-center py-12">
<span class="loading loading-spinner loading-lg"></span> <span class="loading loading-spinner loading-lg" />
</div> </div>
<div v-else-if="filteredMachines.length === 0" class="text-center py-12"> <div v-else-if="filteredMachines.length === 0" class="text-center py-12">
<div class="max-w-md mx-auto"> <div class="max-w-md mx-auto">
<IconLucideFactory class="w-16 h-16 mx-auto text-gray-400 mb-4" aria-hidden="true" /> <IconLucideFactory class="w-16 h-16 mx-auto text-gray-400 mb-4" aria-hidden="true" />
<h3 class="text-lg font-medium text-gray-900 mb-2">Aucune machine trouvée</h3> <h3 class="text-lg font-medium text-gray-900 mb-2">
<p class="text-gray-500 mb-4">Commencez par ajouter votre première machine.</p> Aucune machine trouvée
</h3>
<p class="text-gray-500 mb-4">
Commencez par ajouter votre première machine.
</p>
<NuxtLink to="/machines/new" class="btn btn-primary"> <NuxtLink to="/machines/new" class="btn btn-primary">
Ajouter une machine Ajouter une machine
</NuxtLink> </NuxtLink>
@@ -73,11 +85,17 @@
> >
<div class="card-body"> <div class="card-body">
<div class="flex items-center justify-between mb-2"> <div class="flex items-center justify-between mb-2">
<h3 class="card-title text-lg">{{ machine.name }}</h3> <h3 class="card-title text-lg">
<div class="badge badge-primary badge-sm">{{ machine.typeMachine?.category || 'N/A' }}</div> {{ machine.name }}
</h3>
<div class="badge badge-primary badge-sm">
{{ machine.typeMachine?.category || 'N/A' }}
</div>
</div> </div>
<p class="text-sm text-gray-600 mb-3">{{ machine.description || machine.typeMachine?.description }}</p> <p class="text-sm text-gray-600 mb-3">
{{ machine.description || machine.typeMachine?.description }}
</p>
<div class="space-y-2 text-sm"> <div class="space-y-2 text-sm">
<div class="flex items-center gap-2"> <div class="flex items-center gap-2">
@@ -94,7 +112,6 @@
<IconLucideTag class="w-4 h-4 text-orange-500" aria-hidden="true" /> <IconLucideTag class="w-4 h-4 text-orange-500" aria-hidden="true" />
<span class="text-gray-600">{{ machine.reference }}</span> <span class="text-gray-600">{{ machine.reference }}</span>
</div> </div>
</div> </div>
<div class="card-actions justify-end mt-4"> <div class="card-actions justify-end mt-4">
@@ -150,15 +167,15 @@ const filteredMachines = computed(() => {
let filtered = machines.value let filtered = machines.value
if (selectedSite.value) { if (selectedSite.value) {
filtered = filtered.filter((machine) => machine.siteId === selectedSite.value) filtered = filtered.filter(machine => machine.siteId === selectedSite.value)
} }
if (selectedType.value) { if (selectedType.value) {
filtered = filtered.filter((machine) => machine.typeMachineId === selectedType.value) filtered = filtered.filter(machine => machine.typeMachineId === selectedType.value)
} }
if (selectedCategory.value) { if (selectedCategory.value) {
filtered = filtered.filter((machine) => machine.typeMachine?.category === selectedCategory.value) filtered = filtered.filter(machine => machine.typeMachine?.category === selectedCategory.value)
} }
return filtered return filtered
@@ -193,7 +210,7 @@ onMounted(async () => {
await Promise.all([ await Promise.all([
loadMachines(), loadMachines(),
loadSites(), loadSites(),
loadMachineTypes(), loadMachineTypes()
]) ])
}) })
</script> </script>

View File

@@ -3,10 +3,16 @@
<div class="max-w-5xl mx-auto"> <div class="max-w-5xl mx-auto">
<div class="flex flex-wrap items-center justify-between gap-3 mb-6"> <div class="flex flex-wrap items-center justify-between gap-3 mb-6">
<div> <div>
<h1 class="text-2xl font-bold">Nouvelle machine</h1> <h1 class="text-2xl font-bold">
<p class="text-sm text-gray-500">Renseignez les informations et la configuration avant de créer la machine.</p> Nouvelle machine
</h1>
<p class="text-sm text-gray-500">
Renseignez les informations et la configuration avant de créer la machine.
</p>
</div> </div>
<NuxtLink to="/machines" class="btn btn-ghost">Annuler</NuxtLink> <NuxtLink to="/machines" class="btn btn-ghost">
Annuler
</NuxtLink>
</div> </div>
<form class="space-y-6" @submit.prevent="finalizeMachineCreation"> <form class="space-y-6" @submit.prevent="finalizeMachineCreation">
@@ -24,7 +30,7 @@
placeholder="Ex: Presse hydraulique #1" placeholder="Ex: Presse hydraulique #1"
class="input input-bordered" class="input input-bordered"
required required
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
@@ -32,7 +38,9 @@
<span class="label-text">Site</span> <span class="label-text">Site</span>
</label> </label>
<select id="machine-field-site" v-model="newMachine.siteId" class="select select-bordered" required> <select id="machine-field-site" v-model="newMachine.siteId" class="select select-bordered" required>
<option value="">Sélectionner un site</option> <option value="">
Sélectionner un site
</option>
<option v-for="site in sites" :key="site.id" :value="site.id"> <option v-for="site in sites" :key="site.id" :value="site.id">
{{ site.name }} {{ site.name }}
</option> </option>
@@ -46,7 +54,9 @@
<span class="label-text">Type de machine</span> <span class="label-text">Type de machine</span>
</label> </label>
<select v-model="newMachine.typeMachineId" class="select select-bordered" required> <select v-model="newMachine.typeMachineId" class="select select-bordered" required>
<option value="">Sélectionner un type</option> <option value="">
Sélectionner un type
</option>
<option v-for="type in machineTypes" :key="type.id" :value="type.id"> <option v-for="type in machineTypes" :key="type.id" :value="type.id">
{{ type.name }} ({{ type.category }}) {{ type.name }} ({{ type.category }})
</option> </option>
@@ -62,12 +72,14 @@
type="text" type="text"
placeholder="Ex: PRESS-001" placeholder="Ex: PRESS-001"
class="input input-bordered" class="input input-bordered"
/> >
</div> </div>
</div> </div>
<div v-if="selectedMachineType" class="p-4 bg-gray-50 rounded-lg space-y-2 text-sm"> <div v-if="selectedMachineType" class="p-4 bg-gray-50 rounded-lg space-y-2 text-sm">
<h4 class="font-semibold text-sm">Structure du type sélectionné :</h4> <h4 class="font-semibold text-sm">
Structure du type sélectionné :
</h4>
<div class="flex flex-wrap gap-3"> <div class="flex flex-wrap gap-3">
<span class="inline-flex items-center gap-2"> <span class="inline-flex items-center gap-2">
<span class="font-medium">Familles de composants :</span> <span class="font-medium">Familles de composants :</span>
@@ -91,13 +103,15 @@
</div> </div>
<div v-if="selectedMachineType?.componentRequirements?.length" class="space-y-4"> <div v-if="selectedMachineType?.componentRequirements?.length" class="space-y-4">
<h4 class="text-sm font-semibold">Sélection des composants</h4> <h4 class="text-sm font-semibold">
Sélection des composants
</h4>
<div <div
v-for="requirement in selectedMachineType.componentRequirements" v-for="requirement in selectedMachineType.componentRequirements"
:id="`component-group-${requirement.id}`"
:key="requirement.id" :key="requirement.id"
class="border border-base-200 rounded-lg p-4 space-y-3" class="border border-base-200 rounded-lg p-4 space-y-3"
:id="`component-group-${requirement.id}`"
> >
<div class="flex flex-wrap items-start justify-between gap-3"> <div class="flex flex-wrap items-start justify-between gap-3">
<div> <div>
@@ -115,8 +129,8 @@
<button <button
type="button" type="button"
class="btn btn-sm btn-outline" class="btn btn-sm btn-outline"
@click="addComponentSelectionEntry(requirement)"
:disabled="requirement.maxCount !== null && getComponentRequirementEntries(requirement.id).length >= requirement.maxCount" :disabled="requirement.maxCount !== null && getComponentRequirementEntries(requirement.id).length >= requirement.maxCount"
@click="addComponentSelectionEntry(requirement)"
> >
<IconLucidePlus class="w-4 h-4 mr-2" aria-hidden="true" /> <IconLucidePlus class="w-4 h-4 mr-2" aria-hidden="true" />
Ajouter Ajouter
@@ -148,7 +162,7 @@
class="radio radio-xs" class="radio radio-xs"
:checked="entry.mode === 'model'" :checked="entry.mode === 'model'"
@change="setComponentSelectionMode(requirement.id, entryIndex, 'model')" @change="setComponentSelectionMode(requirement.id, entryIndex, 'model')"
/> >
Modèle existant Modèle existant
</label> </label>
<label class="inline-flex items-center gap-1"> <label class="inline-flex items-center gap-1">
@@ -156,9 +170,9 @@
type="radio" type="radio"
class="radio radio-xs" class="radio radio-xs"
:checked="entry.mode === 'manual'" :checked="entry.mode === 'manual'"
@change="setComponentSelectionMode(requirement.id, entryIndex, 'manual')"
:disabled="!requirement.allowNewModels" :disabled="!requirement.allowNewModels"
/> @change="setComponentSelectionMode(requirement.id, entryIndex, 'manual')"
>
Définir manuellement Définir manuellement
</label> </label>
</div> </div>
@@ -173,7 +187,9 @@
:value="entry.componentModelId || ''" :value="entry.componentModelId || ''"
@change="updateComponentSelectionEntry(requirement.id, entryIndex, { componentModelId: $event.target.value || '' })" @change="updateComponentSelectionEntry(requirement.id, entryIndex, { componentModelId: $event.target.value || '' })"
> >
<option value="">Sélectionner un modèle</option> <option value="">
Sélectionner un modèle
</option>
<option <option
v-for="model in getComponentModelsForType(requirement.typeComposantId)" v-for="model in getComponentModelsForType(requirement.typeComposantId)"
:key="model.id" :key="model.id"
@@ -182,7 +198,9 @@
{{ model.name }} {{ model.name }}
</option> </option>
</select> </select>
<p v-if="loadingComponentModels" class="text-[10px] text-gray-500 mt-1">Chargement des modèles...</p> <p v-if="loadingComponentModels" class="text-[10px] text-gray-500 mt-1">
Chargement des modèles...
</p>
<p <p
v-else-if="getComponentModelsForType(requirement.typeComposantId).length === 0" v-else-if="getComponentModelsForType(requirement.typeComposantId).length === 0"
class="text-[10px] text-gray-500 mt-1" class="text-[10px] text-gray-500 mt-1"
@@ -203,7 +221,7 @@
:value="entry.name" :value="entry.name"
placeholder="Nom du composant" placeholder="Nom du composant"
@input="updateComponentSelectionEntry(requirement.id, entryIndex, { name: $event.target.value })" @input="updateComponentSelectionEntry(requirement.id, entryIndex, { name: $event.target.value })"
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
<label class="label"> <label class="label">
@@ -214,7 +232,7 @@
class="input input-bordered input-sm" class="input input-bordered input-sm"
placeholder="(Non géré pour l'instant)" placeholder="(Non géré pour l'instant)"
disabled disabled
/> >
</div> </div>
</div> </div>
@@ -231,13 +249,15 @@
</div> </div>
</div> </div>
<div v-if="selectedMachineType?.pieceRequirements?.length" class="space-y-4"> <div v-if="selectedMachineType?.pieceRequirements?.length" class="space-y-4">
<h4 class="text-sm font-semibold">Sélection des pièces principales</h4> <h4 class="text-sm font-semibold">
Sélection des pièces principales
</h4>
<div <div
v-for="requirement in selectedMachineType.pieceRequirements" v-for="requirement in selectedMachineType.pieceRequirements"
:id="`piece-group-${requirement.id}`"
:key="requirement.id" :key="requirement.id"
class="border border-base-200 rounded-lg p-4 space-y-3" class="border border-base-200 rounded-lg p-4 space-y-3"
:id="`piece-group-${requirement.id}`"
> >
<div class="flex flex-wrap items-start justify-between gap-3"> <div class="flex flex-wrap items-start justify-between gap-3">
<div> <div>
@@ -255,8 +275,8 @@
<button <button
type="button" type="button"
class="btn btn-sm btn-outline" class="btn btn-sm btn-outline"
@click="addPieceSelectionEntry(requirement)"
:disabled="requirement.maxCount !== null && getPieceRequirementEntries(requirement.id).length >= requirement.maxCount" :disabled="requirement.maxCount !== null && getPieceRequirementEntries(requirement.id).length >= requirement.maxCount"
@click="addPieceSelectionEntry(requirement)"
> >
<IconLucidePlus class="w-4 h-4 mr-2" aria-hidden="true" /> <IconLucidePlus class="w-4 h-4 mr-2" aria-hidden="true" />
Ajouter Ajouter
@@ -291,7 +311,9 @@
:value="entry.pieceModelId || ''" :value="entry.pieceModelId || ''"
@change="updatePieceSelectionEntry(requirement.id, entryIndex, { pieceModelId: $event.target.value || '' })" @change="updatePieceSelectionEntry(requirement.id, entryIndex, { pieceModelId: $event.target.value || '' })"
> >
<option value="">Sélectionner un modèle</option> <option value="">
Sélectionner un modèle
</option>
<option <option
v-for="model in getPieceModelsForType(requirement.typePieceId)" v-for="model in getPieceModelsForType(requirement.typePieceId)"
:key="model.id" :key="model.id"
@@ -300,7 +322,9 @@
{{ model.name }} {{ model.name }}
</option> </option>
</select> </select>
<p v-if="loadingPieceModels" class="text-[10px] text-gray-500 mt-1">Chargement des modèles...</p> <p v-if="loadingPieceModels" class="text-[10px] text-gray-500 mt-1">
Chargement des modèles...
</p>
<p <p
v-else-if="getPieceModelsForType(requirement.typePieceId).length === 0" v-else-if="getPieceModelsForType(requirement.typePieceId).length === 0"
class="text-[10px] text-gray-500 mt-1" class="text-[10px] text-gray-500 mt-1"
@@ -363,7 +387,9 @@
</div> </div>
<div v-if="machinePreview.base.issues.length" class="rounded-md bg-warning/10 border border-warning/30 p-3 text-xs text-warning"> <div v-if="machinePreview.base.issues.length" class="rounded-md bg-warning/10 border border-warning/30 p-3 text-xs text-warning">
<p class="font-medium mb-1">Informations générales incomplètes :</p> <p class="font-medium mb-1">
Informations générales incomplètes :
</p>
<ul class="space-y-1"> <ul class="space-y-1">
<li v-for="issue in machinePreview.base.issues" :key="issue.message"> <li v-for="issue in machinePreview.base.issues" :key="issue.message">
<button <button
@@ -379,7 +405,9 @@
</div> </div>
<div v-if="machinePreview.componentGroups.length" class="space-y-3"> <div v-if="machinePreview.componentGroups.length" class="space-y-3">
<h5 class="text-xs font-semibold uppercase tracking-wide text-gray-500">Composants hérités</h5> <h5 class="text-xs font-semibold uppercase tracking-wide text-gray-500">
Composants hérités
</h5>
<div <div
v-for="group in machinePreview.componentGroups" v-for="group in machinePreview.componentGroups"
:key="group.id" :key="group.id"
@@ -387,7 +415,9 @@
> >
<div class="flex flex-wrap items-start justify-between gap-2"> <div class="flex flex-wrap items-start justify-between gap-2">
<div> <div>
<p class="text-sm font-semibold">{{ group.label }}</p> <p class="text-sm font-semibold">
{{ group.label }}
</p>
<p class="text-xs text-gray-500"> <p class="text-xs text-gray-500">
Type : {{ group.typeName }} · Min {{ group.min }} · Type : {{ group.typeName }} · Min {{ group.min }} ·
{{ group.max !== null ? `Max ${group.max}` : 'Max ∞' }} {{ group.max !== null ? `Max ${group.max}` : 'Max ∞' }}
@@ -400,7 +430,9 @@
<div v-if="group.issues.length" class="rounded bg-warning/10 border border-warning/30 p-2 text-[11px] text-warning"> <div v-if="group.issues.length" class="rounded bg-warning/10 border border-warning/30 p-2 text-[11px] text-warning">
<ul class="list-disc pl-4 space-y-1"> <ul class="list-disc pl-4 space-y-1">
<li v-for="issue in group.issues" :key="issue.message">{{ issue.message }}</li> <li v-for="issue in group.issues" :key="issue.message">
{{ issue.message }}
</li>
</ul> </ul>
</div> </div>
@@ -420,7 +452,9 @@
<p class="text-sm font-medium" :class="entry.status === 'complete' ? 'text-gray-900' : 'text-gray-500'"> <p class="text-sm font-medium" :class="entry.status === 'complete' ? 'text-gray-900' : 'text-gray-500'">
{{ entry.title }} {{ entry.title }}
</p> </p>
<p v-if="entry.subtitle" class="text-xs text-gray-500">{{ entry.subtitle }}</p> <p v-if="entry.subtitle" class="text-xs text-gray-500">
{{ entry.subtitle }}
</p>
</div> </div>
</li> </li>
</ul> </ul>
@@ -432,7 +466,9 @@
</div> </div>
<div v-if="machinePreview.pieceGroups.length" class="space-y-3"> <div v-if="machinePreview.pieceGroups.length" class="space-y-3">
<h5 class="text-xs font-semibold uppercase tracking-wide text-gray-500">Pièces associées</h5> <h5 class="text-xs font-semibold uppercase tracking-wide text-gray-500">
Pièces associées
</h5>
<div <div
v-for="group in machinePreview.pieceGroups" v-for="group in machinePreview.pieceGroups"
:key="group.id" :key="group.id"
@@ -440,7 +476,9 @@
> >
<div class="flex flex-wrap items-start justify-between gap-2"> <div class="flex flex-wrap items-start justify-between gap-2">
<div> <div>
<p class="text-sm font-semibold">{{ group.label }}</p> <p class="text-sm font-semibold">
{{ group.label }}
</p>
<p class="text-xs text-gray-500"> <p class="text-xs text-gray-500">
Type : {{ group.typeName }} · Min {{ group.min }} · Type : {{ group.typeName }} · Min {{ group.min }} ·
{{ group.max !== null ? `Max ${group.max}` : 'Max ∞' }} {{ group.max !== null ? `Max ${group.max}` : 'Max ∞' }}
@@ -453,7 +491,9 @@
<div v-if="group.issues.length" class="rounded bg-warning/10 border border-warning/30 p-2 text-[11px] text-warning"> <div v-if="group.issues.length" class="rounded bg-warning/10 border border-warning/30 p-2 text-[11px] text-warning">
<ul class="list-disc pl-4 space-y-1"> <ul class="list-disc pl-4 space-y-1">
<li v-for="issue in group.issues" :key="issue.message">{{ issue.message }}</li> <li v-for="issue in group.issues" :key="issue.message">
{{ issue.message }}
</li>
</ul> </ul>
</div> </div>
@@ -473,7 +513,9 @@
<p class="text-sm font-medium" :class="entry.status === 'complete' ? 'text-gray-900' : 'text-gray-500'"> <p class="text-sm font-medium" :class="entry.status === 'complete' ? 'text-gray-900' : 'text-gray-500'">
{{ entry.title }} {{ entry.title }}
</p> </p>
<p v-if="entry.subtitle" class="text-xs text-gray-500">{{ entry.subtitle }}</p> <p v-if="entry.subtitle" class="text-xs text-gray-500">
{{ entry.subtitle }}
</p>
</div> </div>
<span v-if="entry.mode === 'manual'" class="badge badge-ghost badge-xs">manuel</span> <span v-if="entry.mode === 'manual'" class="badge badge-ghost badge-xs">manuel</span>
</li> </li>
@@ -492,7 +534,9 @@
<div class="flex items-start gap-2"> <div class="flex items-start gap-2">
<IconLucideAlertTriangle class="w-4 h-4 mt-0.5" aria-hidden="true" /> <IconLucideAlertTriangle class="w-4 h-4 mt-0.5" aria-hidden="true" />
<div class="space-y-1"> <div class="space-y-1">
<p class="font-medium">Points à vérifier avant la création :</p> <p class="font-medium">
Points à vérifier avant la création :
</p>
<ul class="space-y-1"> <ul class="space-y-1">
<li v-for="issue in machinePreview.issues" :key="`${issue.scope}-${issue.message}`"> <li v-for="issue in machinePreview.issues" :key="`${issue.scope}-${issue.message}`">
<button <button
@@ -522,7 +566,9 @@
</div> </div>
<div class="flex justify-end gap-3 pt-4 border-t border-base-200"> <div class="flex justify-end gap-3 pt-4 border-t border-base-200">
<NuxtLink to="/machines" class="btn btn-outline">Annuler</NuxtLink> <NuxtLink to="/machines" class="btn btn-outline">
Annuler
</NuxtLink>
<button <button
type="submit" type="submit"
class="btn btn-primary" class="btn btn-primary"
@@ -539,7 +585,9 @@
<div v-if="createComponentModelModal.open" class="modal modal-open"> <div v-if="createComponentModelModal.open" class="modal modal-open">
<div class="modal-box max-w-md"> <div class="modal-box max-w-md">
<h3 class="font-bold text-lg mb-1">Nouveau modèle de composant</h3> <h3 class="font-bold text-lg mb-1">
Nouveau modèle de composant
</h3>
<p class="text-sm text-gray-500 mb-4"> <p class="text-sm text-gray-500 mb-4">
{{ createComponentModelModal.requirement?.label || createComponentModelModal.requirement?.typeComposant?.name || 'Famille de composants' }} {{ createComponentModelModal.requirement?.label || createComponentModelModal.requirement?.typeComposant?.name || 'Famille de composants' }}
</p> </p>
@@ -556,7 +604,7 @@
class="input input-bordered input-sm" class="input input-bordered input-sm"
placeholder="Ex: Sangle 20 m" placeholder="Ex: Sangle 20 m"
required required
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
@@ -569,11 +617,13 @@
class="textarea textarea-bordered textarea-sm" class="textarea textarea-bordered textarea-sm"
rows="3" rows="3"
placeholder="Notes sur ce modèle" placeholder="Notes sur ce modèle"
></textarea> />
</div> </div>
<div class="modal-action"> <div class="modal-action">
<button type="button" class="btn btn-outline" @click="closeCreateComponentModelModal">Annuler</button> <button type="button" class="btn btn-outline" @click="closeCreateComponentModelModal">
Annuler
</button>
<button type="submit" class="btn btn-primary" :class="{ loading: createComponentModelModal.submitting }"> <button type="submit" class="btn btn-primary" :class="{ loading: createComponentModelModal.submitting }">
Créer Créer
</button> </button>
@@ -584,7 +634,9 @@
<div v-if="createPieceModelModal.open" class="modal modal-open"> <div v-if="createPieceModelModal.open" class="modal modal-open">
<div class="modal-box max-w-md"> <div class="modal-box max-w-md">
<h3 class="font-bold text-lg mb-1">Nouveau modèle de pièce</h3> <h3 class="font-bold text-lg mb-1">
Nouveau modèle de pièce
</h3>
<p class="text-sm text-gray-500 mb-4"> <p class="text-sm text-gray-500 mb-4">
{{ createPieceModelModal.requirement?.label || createPieceModelModal.requirement?.typePiece?.name || 'Groupe de pièces' }} {{ createPieceModelModal.requirement?.label || createPieceModelModal.requirement?.typePiece?.name || 'Groupe de pièces' }}
</p> </p>
@@ -601,7 +653,7 @@
class="input input-bordered input-sm" class="input input-bordered input-sm"
placeholder="Ex: Vis standard" placeholder="Ex: Vis standard"
required required
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
@@ -614,11 +666,13 @@
class="textarea textarea-bordered textarea-sm" class="textarea textarea-bordered textarea-sm"
rows="3" rows="3"
placeholder="Notes sur ce modèle" placeholder="Notes sur ce modèle"
></textarea> />
</div> </div>
<div class="modal-action"> <div class="modal-action">
<button type="button" class="btn btn-outline" @click="closeCreatePieceModelModal">Annuler</button> <button type="button" class="btn btn-outline" @click="closeCreatePieceModelModal">
Annuler
</button>
<button type="submit" class="btn btn-primary" :class="{ loading: createPieceModelModal.submitting }"> <button type="submit" class="btn btn-primary" :class="{ loading: createPieceModelModal.submitting }">
Créer Créer
</button> </button>
@@ -657,7 +711,7 @@ const newMachine = reactive({
name: '', name: '',
siteId: '', siteId: '',
typeMachineId: '', typeMachineId: '',
reference: '', reference: ''
}) })
const componentRequirementSelections = reactive({}) const componentRequirementSelections = reactive({})
@@ -670,7 +724,7 @@ const createComponentModelModal = reactive({
entryIndex: null, entryIndex: null,
name: '', name: '',
description: '', description: '',
submitting: false, submitting: false
}) })
const createPieceModelModal = reactive({ const createPieceModelModal = reactive({
@@ -680,14 +734,14 @@ const createPieceModelModal = reactive({
entryIndex: null, entryIndex: null,
name: '', name: '',
description: '', description: '',
submitting: false, submitting: false
}) })
const selectedMachineType = computed(() => { const selectedMachineType = computed(() => {
if (!newMachine.typeMachineId) { if (!newMachine.typeMachineId) {
return null return null
} }
return machineTypes.value.find((type) => type.id === newMachine.typeMachineId) || null return machineTypes.value.find(type => type.id === newMachine.typeMachineId) || null
}) })
const getStatusBadgeClass = (status) => { const getStatusBadgeClass = (status) => {
@@ -700,8 +754,8 @@ const getStatusBadgeClass = (status) => {
return 'badge-error' return 'badge-error'
} }
const getComponentRequirementEntries = (requirementId) => componentRequirementSelections[requirementId] || [] const getComponentRequirementEntries = requirementId => componentRequirementSelections[requirementId] || []
const getPieceRequirementEntries = (requirementId) => pieceRequirementSelections[requirementId] || [] const getPieceRequirementEntries = requirementId => pieceRequirementSelections[requirementId] || []
const machinePreview = computed(() => { const machinePreview = computed(() => {
const type = selectedMachineType.value const type = selectedMachineType.value
@@ -711,7 +765,7 @@ const machinePreview = computed(() => {
const trimmedName = (newMachine.name || '').trim() const trimmedName = (newMachine.name || '').trim()
const currentSite = newMachine.siteId const currentSite = newMachine.siteId
? sites.value.find((site) => site.id === newMachine.siteId) || null ? sites.value.find(site => site.id === newMachine.siteId) || null
: null : null
const trimmedReference = (newMachine.reference || '').trim() const trimmedReference = (newMachine.reference || '').trim()
@@ -720,26 +774,26 @@ const machinePreview = computed(() => {
key: 'name', key: 'name',
label: 'Nom', label: 'Nom',
display: trimmedName || 'À renseigner', display: trimmedName || 'À renseigner',
status: trimmedName ? 'complete' : 'missing', status: trimmedName ? 'complete' : 'missing'
}, },
{ {
key: 'site', key: 'site',
label: 'Site', label: 'Site',
display: currentSite?.name || 'Sélectionner un site', display: currentSite?.name || 'Sélectionner un site',
status: currentSite ? 'complete' : 'missing', status: currentSite ? 'complete' : 'missing'
}, },
{ {
key: 'type', key: 'type',
label: 'Type sélectionné', label: 'Type sélectionné',
display: type.name, display: type.name,
status: 'complete', status: 'complete'
}, },
{ {
key: 'reference', key: 'reference',
label: 'Référence', label: 'Référence',
display: trimmedReference || 'Non renseignée', display: trimmedReference || 'Non renseignée',
status: trimmedReference ? 'complete' : 'optional', status: trimmedReference ? 'complete' : 'optional'
}, }
] ]
const baseIssues = [] const baseIssues = []
@@ -749,20 +803,20 @@ const machinePreview = computed(() => {
if (!currentSite) { if (!currentSite) {
baseIssues.push({ message: "Sélectionner un site d'affectation.", kind: 'error', anchor: 'machine-field-site' }) baseIssues.push({ message: "Sélectionner un site d'affectation.", kind: 'error', anchor: 'machine-field-site' })
} }
const baseStatus = baseIssues.some((issue) => issue.kind === 'error') ? 'error' : 'ready' const baseStatus = baseIssues.some(issue => issue.kind === 'error') ? 'error' : 'ready'
const resolveComponentModel = (requirement, modelId) => { const resolveComponentModel = (requirement, modelId) => {
if (!modelId) { if (!modelId) {
return null return null
} }
return (getComponentModelsForType(requirement.typeComposantId) || []).find((model) => model.id === modelId) || null return (getComponentModelsForType(requirement.typeComposantId) || []).find(model => model.id === modelId) || null
} }
const resolvePieceModel = (requirement, modelId) => { const resolvePieceModel = (requirement, modelId) => {
if (!modelId) { if (!modelId) {
return null return null
} }
return (getPieceModelsForType(requirement.typePieceId) || []).find((model) => model.id === modelId) || null return (getPieceModelsForType(requirement.typePieceId) || []).find(model => model.id === modelId) || null
} }
const componentGroups = (type.componentRequirements || []).map((requirement) => { const componentGroups = (type.componentRequirements || []).map((requirement) => {
@@ -776,7 +830,7 @@ const machinePreview = computed(() => {
mode: 'model', mode: 'model',
status: model ? 'complete' : 'pending', status: model ? 'complete' : 'pending',
title: model ? model.name : 'Sélectionner un modèle', title: model ? model.name : 'Sélectionner un modèle',
subtitle: model?.description || null, subtitle: model?.description || null
} }
} }
const manualName = (entry.name || '').trim() const manualName = (entry.name || '').trim()
@@ -785,13 +839,13 @@ const machinePreview = computed(() => {
mode: 'manual', mode: 'manual',
status: manualName ? 'complete' : 'pending', status: manualName ? 'complete' : 'pending',
title: manualName || 'Nom à renseigner', title: manualName || 'Nom à renseigner',
subtitle: manualName ? null : null, subtitle: manualName ? null : null
} }
}) })
const min = requirement.minCount ?? (requirement.required ? 1 : 0) const min = requirement.minCount ?? (requirement.required ? 1 : 0)
const max = requirement.maxCount ?? null const max = requirement.maxCount ?? null
const completed = normalizedEntries.filter((entry) => entry.status === 'complete').length const completed = normalizedEntries.filter(entry => entry.status === 'complete').length
const issues = [] const issues = []
if (completed < min) { if (completed < min) {
@@ -802,15 +856,15 @@ const machinePreview = computed(() => {
issues.push({ message: `Maximum ${max} dépassé`, kind: 'error', anchor: `component-group-${requirement.id}` }) issues.push({ message: `Maximum ${max} dépassé`, kind: 'error', anchor: `component-group-${requirement.id}` })
} }
if (!requirement.allowNewModels && normalizedEntries.some((entry) => entry.mode === 'manual' && entry.status === 'complete')) { if (!requirement.allowNewModels && normalizedEntries.some(entry => entry.mode === 'manual' && entry.status === 'complete')) {
issues.push({ message: "Ce groupe n'autorise que les modèles existants.", kind: 'error', anchor: `component-group-${requirement.id}` }) issues.push({ message: "Ce groupe n'autorise que les modèles existants.", kind: 'error', anchor: `component-group-${requirement.id}` })
} }
if (normalizedEntries.some((entry) => entry.status !== 'complete')) { if (normalizedEntries.some(entry => entry.status !== 'complete')) {
issues.push({ message: 'Compléter les sélections restantes.', kind: 'warning', anchor: `component-group-${requirement.id}` }) issues.push({ message: 'Compléter les sélections restantes.', kind: 'warning', anchor: `component-group-${requirement.id}` })
} }
const status = issues.some((issue) => issue.kind === 'error') const status = issues.some(issue => issue.kind === 'error')
? 'error' ? 'error'
: issues.length > 0 : issues.length > 0
? 'warning' ? 'warning'
@@ -826,7 +880,7 @@ const machinePreview = computed(() => {
issues, issues,
completed, completed,
total: normalizedEntries.length, total: normalizedEntries.length,
status, status
} }
}) })
@@ -839,13 +893,13 @@ const machinePreview = computed(() => {
key: `${requirement.id}-${index}`, key: `${requirement.id}-${index}`,
status: model ? 'complete' : 'pending', status: model ? 'complete' : 'pending',
title: model ? model.name : 'Sélectionner un modèle', title: model ? model.name : 'Sélectionner un modèle',
subtitle: model?.description || null, subtitle: model?.description || null
} }
}) })
const min = requirement.minCount ?? (requirement.required ? 1 : 0) const min = requirement.minCount ?? (requirement.required ? 1 : 0)
const max = requirement.maxCount ?? null const max = requirement.maxCount ?? null
const completed = normalizedEntries.filter((entry) => entry.status === 'complete').length const completed = normalizedEntries.filter(entry => entry.status === 'complete').length
const issues = [] const issues = []
if (completed < min) { if (completed < min) {
@@ -856,11 +910,11 @@ const machinePreview = computed(() => {
issues.push({ message: `Maximum ${max} dépassé`, kind: 'error', anchor: `piece-group-${requirement.id}` }) issues.push({ message: `Maximum ${max} dépassé`, kind: 'error', anchor: `piece-group-${requirement.id}` })
} }
if (normalizedEntries.some((entry) => entry.status !== 'complete')) { if (normalizedEntries.some(entry => entry.status !== 'complete')) {
issues.push({ message: 'Compléter les sélections restantes.', kind: 'warning', anchor: `piece-group-${requirement.id}` }) issues.push({ message: 'Compléter les sélections restantes.', kind: 'warning', anchor: `piece-group-${requirement.id}` })
} }
const status = issues.some((issue) => issue.kind === 'error') const status = issues.some(issue => issue.kind === 'error')
? 'error' ? 'error'
: issues.length > 0 : issues.length > 0
? 'warning' ? 'warning'
@@ -876,20 +930,20 @@ const machinePreview = computed(() => {
issues, issues,
completed, completed,
total: normalizedEntries.length, total: normalizedEntries.length,
status, status
} }
}) })
const aggregatedIssues = [ const aggregatedIssues = [
...baseIssues.map((issue) => ({ ...issue, scope: 'Informations générales' })), ...baseIssues.map(issue => ({ ...issue, scope: 'Informations générales' })),
...componentGroups.flatMap((group) => group.issues.map((issue) => ({ ...issue, scope: group.label }))), ...componentGroups.flatMap(group => group.issues.map(issue => ({ ...issue, scope: group.label }))),
...pieceGroups.flatMap((group) => group.issues.map((issue) => ({ ...issue, scope: group.label }))), ...pieceGroups.flatMap(group => group.issues.map(issue => ({ ...issue, scope: group.label })))
] ]
const statuses = [ const statuses = [
baseStatus, baseStatus,
...componentGroups.map((group) => group.status), ...componentGroups.map(group => group.status),
...pieceGroups.map((group) => group.status), ...pieceGroups.map(group => group.status)
] ]
const overallStatus = statuses.includes('error') const overallStatus = statuses.includes('error')
@@ -902,7 +956,7 @@ const machinePreview = computed(() => {
base: { base: {
fields: baseFields, fields: baseFields,
issues: baseIssues, issues: baseIssues,
status: baseStatus, status: baseStatus
}, },
componentGroups, componentGroups,
pieceGroups, pieceGroups,
@@ -910,11 +964,11 @@ const machinePreview = computed(() => {
name: type.name, name: type.name,
category: type.category || null, category: type.category || null,
hasStructuredDefinition: hasStructuredDefinition:
(type.componentRequirements?.length || 0) > 0 || (type.pieceRequirements?.length || 0) > 0, (type.componentRequirements?.length || 0) > 0 || (type.pieceRequirements?.length || 0) > 0
}, },
status: overallStatus, status: overallStatus,
ready: overallStatus === 'ready', ready: overallStatus === 'ready',
issues: aggregatedIssues, issues: aggregatedIssues
} }
}) })
@@ -922,7 +976,7 @@ const blockingPreviewIssues = computed(() => {
if (!machinePreview.value) { if (!machinePreview.value) {
return [] return []
} }
return machinePreview.value.issues.filter((issue) => issue.kind === 'error') return machinePreview.value.issues.filter(issue => issue.kind === 'error')
}) })
const canCreateMachine = computed(() => { const canCreateMachine = computed(() => {
@@ -943,9 +997,9 @@ const scrollToAnchor = (anchor) => {
return return
} }
target.scrollIntoView({ behavior: 'smooth', block: 'center' }) target.scrollIntoView({ behavior: 'smooth', block: 'center' })
highlightClasses.forEach((cls) => target.classList.add(cls)) highlightClasses.forEach(cls => target.classList.add(cls))
window.setTimeout(() => { window.setTimeout(() => {
highlightClasses.forEach((cls) => target.classList.remove(cls)) highlightClasses.forEach(cls => target.classList.remove(cls))
}, 1500) }, 1500)
} }
@@ -968,11 +1022,11 @@ const clearRequirementSelections = () => {
const createComponentSelectionEntry = () => ({ const createComponentSelectionEntry = () => ({
mode: 'model', mode: 'model',
componentModelId: '', componentModelId: '',
name: '', name: ''
}) })
const createPieceSelectionEntry = () => ({ const createPieceSelectionEntry = () => ({
pieceModelId: '', pieceModelId: ''
}) })
const addComponentSelectionEntry = (requirement) => { const addComponentSelectionEntry = (requirement) => {
@@ -993,7 +1047,7 @@ const removeComponentSelectionEntry = (requirementId, index) => {
const setComponentSelectionMode = (requirementId, index, mode) => { const setComponentSelectionMode = (requirementId, index, mode) => {
const entries = getComponentRequirementEntries(requirementId) const entries = getComponentRequirementEntries(requirementId)
componentRequirementSelections[requirementId] = entries.map((entry, i) => { componentRequirementSelections[requirementId] = entries.map((entry, i) => {
if (i !== index) return entry if (i !== index) { return entry }
if (mode === 'model') { if (mode === 'model') {
return { ...entry, mode: 'model', componentModelId: entry.componentModelId || '', name: '' } return { ...entry, mode: 'model', componentModelId: entry.componentModelId || '', name: '' }
} }
@@ -1055,7 +1109,7 @@ const validateRequirementSelections = (type) => {
errors.push(`Le groupe "${requirement.label || requirement.typeComposant?.name || 'Composants'}" ne peut dépasser ${max} élément(s).`) errors.push(`Le groupe "${requirement.label || requirement.typeComposant?.name || 'Composants'}" ne peut dépasser ${max} élément(s).`)
} }
if (!requirement.allowNewModels && usableEntries.some((entry) => entry.mode === 'manual')) { if (!requirement.allowNewModels && usableEntries.some(entry => entry.mode === 'manual')) {
errors.push(`Le groupe "${requirement.label || requirement.typeComposant?.name || 'Composants'}" n'autorise que les modèles existants.`) errors.push(`Le groupe "${requirement.label || requirement.typeComposant?.name || 'Composants'}" n'autorise que les modèles existants.`)
} }
@@ -1063,14 +1117,14 @@ const validateRequirementSelections = (type) => {
if (entry.mode === 'model') { if (entry.mode === 'model') {
componentSelectionsPayload.push({ componentSelectionsPayload.push({
requirementId: requirement.id, requirementId: requirement.id,
componentModelId: entry.componentModelId, componentModelId: entry.componentModelId
}) })
} else { } else {
componentSelectionsPayload.push({ componentSelectionsPayload.push({
requirementId: requirement.id, requirementId: requirement.id,
definition: { definition: {
name: entry.name.trim(), name: entry.name.trim()
}, }
}) })
} }
}) })
@@ -1078,7 +1132,7 @@ const validateRequirementSelections = (type) => {
for (const requirement of type.pieceRequirements || []) { for (const requirement of type.pieceRequirements || []) {
const entries = getPieceRequirementEntries(requirement.id) const entries = getPieceRequirementEntries(requirement.id)
const usableEntries = entries.filter((entry) => !!entry.pieceModelId) const usableEntries = entries.filter(entry => !!entry.pieceModelId)
const min = requirement.minCount ?? (requirement.required ? 1 : 0) const min = requirement.minCount ?? (requirement.required ? 1 : 0)
const max = requirement.maxCount ?? null const max = requirement.maxCount ?? null
@@ -1094,7 +1148,7 @@ const validateRequirementSelections = (type) => {
usableEntries.forEach((entry) => { usableEntries.forEach((entry) => {
pieceSelectionsPayload.push({ pieceSelectionsPayload.push({
requirementId: requirement.id, requirementId: requirement.id,
pieceModelId: entry.pieceModelId, pieceModelId: entry.pieceModelId
}) })
}) })
} }
@@ -1106,7 +1160,7 @@ const validateRequirementSelections = (type) => {
return { return {
valid: true, valid: true,
componentSelections: componentSelectionsPayload, componentSelections: componentSelectionsPayload,
pieceSelections: pieceSelectionsPayload, pieceSelections: pieceSelectionsPayload
} }
} }
@@ -1139,17 +1193,17 @@ const submitCreateComponentModel = async () => {
name: createComponentModelModal.name.trim(), name: createComponentModelModal.name.trim(),
description: createComponentModelModal.description.trim() || undefined, description: createComponentModelModal.description.trim() || undefined,
typeComposantId: createComponentModelModal.requirement.typeComposantId, typeComposantId: createComponentModelModal.requirement.typeComposantId,
structure: {}, structure: {}
} }
const result = await createComponentModel(payload) const result = await createComponentModel(payload)
if (result.success) { if (result.success) {
await loadComponentModels(createComponentModelModal.requirement.typeComposantId) await loadComponentModels(createComponentModelModal.requirement.typeComposantId)
const entries = getComponentRequirementEntries(createComponentModelModal.requirement.id) const entries = getComponentRequirementEntries(createComponentModelModal.requirement.id)
const targetIndex = entries.findIndex((entry) => entry.mode === 'model' && !entry.componentModelId) const targetIndex = entries.findIndex(entry => entry.mode === 'model' && !entry.componentModelId)
if (targetIndex !== -1) { if (targetIndex !== -1) {
updateComponentSelectionEntry(createComponentModelModal.requirement.id, targetIndex, { updateComponentSelectionEntry(createComponentModelModal.requirement.id, targetIndex, {
mode: 'model', mode: 'model',
componentModelId: result.data.id, componentModelId: result.data.id
}) })
} else { } else {
addComponentSelectionEntry(createComponentModelModal.requirement) addComponentSelectionEntry(createComponentModelModal.requirement)
@@ -1158,8 +1212,8 @@ const submitCreateComponentModel = async () => {
getComponentRequirementEntries(createComponentModelModal.requirement.id).length - 1, getComponentRequirementEntries(createComponentModelModal.requirement.id).length - 1,
{ {
mode: 'model', mode: 'model',
componentModelId: result.data.id, componentModelId: result.data.id
}, }
) )
} }
toast.showSuccess(`Modèle "${result.data.name}" créé`) toast.showSuccess(`Modèle "${result.data.name}" créé`)
@@ -1201,16 +1255,16 @@ const submitCreatePieceModel = async () => {
name: createPieceModelModal.name.trim(), name: createPieceModelModal.name.trim(),
description: createPieceModelModal.description.trim() || undefined, description: createPieceModelModal.description.trim() || undefined,
typePieceId: createPieceModelModal.requirement.typePieceId, typePieceId: createPieceModelModal.requirement.typePieceId,
structure: {}, structure: {}
} }
const result = await createPieceModel(payload) const result = await createPieceModel(payload)
if (result.success) { if (result.success) {
await loadPieceModels(createPieceModelModal.requirement.typePieceId) await loadPieceModels(createPieceModelModal.requirement.typePieceId)
const entries = getPieceRequirementEntries(createPieceModelModal.requirement.id) const entries = getPieceRequirementEntries(createPieceModelModal.requirement.id)
const targetIndex = entries.findIndex((entry) => !entry.pieceModelId) const targetIndex = entries.findIndex(entry => !entry.pieceModelId)
if (targetIndex !== -1) { if (targetIndex !== -1) {
updatePieceSelectionEntry(createPieceModelModal.requirement.id, targetIndex, { updatePieceSelectionEntry(createPieceModelModal.requirement.id, targetIndex, {
pieceModelId: result.data.id, pieceModelId: result.data.id
}) })
} else { } else {
addPieceSelectionEntry(createPieceModelModal.requirement) addPieceSelectionEntry(createPieceModelModal.requirement)
@@ -1218,8 +1272,8 @@ const submitCreatePieceModel = async () => {
createPieceModelModal.requirement.id, createPieceModelModal.requirement.id,
getPieceRequirementEntries(createPieceModelModal.requirement.id).length - 1, getPieceRequirementEntries(createPieceModelModal.requirement.id).length - 1,
{ {
pieceModelId: result.data.id, pieceModelId: result.data.id
}, }
) )
} }
toast.showSuccess(`Modèle "${result.data.name}" créé`) toast.showSuccess(`Modèle "${result.data.name}" créé`)
@@ -1262,8 +1316,8 @@ const initializeRequirementSelections = async (type) => {
}) })
await Promise.all([ await Promise.all([
...Array.from(componentTypeIds).map((id) => loadComponentModels(id)), ...Array.from(componentTypeIds).map(id => loadComponentModels(id)),
...Array.from(pieceTypeIds).map((id) => loadPieceModels(id)), ...Array.from(pieceTypeIds).map(id => loadPieceModels(id))
]) ])
} }
@@ -1287,7 +1341,7 @@ const finalizeMachineCreation = async () => {
name: newMachine.name, name: newMachine.name,
siteId: newMachine.siteId, siteId: newMachine.siteId,
reference: newMachine.reference, reference: newMachine.reference,
typeMachineId: type.id, typeMachineId: type.id
} }
const hasRequirements = (type.componentRequirements?.length || 0) > 0 || (type.pieceRequirements?.length || 0) > 0 const hasRequirements = (type.componentRequirements?.length || 0) > 0 || (type.pieceRequirements?.length || 0) > 0
@@ -1310,9 +1364,9 @@ const finalizeMachineCreation = async () => {
...(hasRequirements ...(hasRequirements
? { ? {
componentSelections, componentSelections,
pieceSelections, pieceSelections
} }
: {}), : {})
} }
const result = hasRequirements const result = hasRequirements
@@ -1344,7 +1398,7 @@ watch(
return return
} }
const type = machineTypes.value.find((item) => item.id === typeId) const type = machineTypes.value.find(item => item.id === typeId)
if (!type) { if (!type) {
return return
} }
@@ -1356,7 +1410,7 @@ watch(
onMounted(async () => { onMounted(async () => {
await Promise.all([ await Promise.all([
loadSites(), loadSites(),
loadMachineTypes(), loadMachineTypes()
]) ])
}) })
</script> </script>

View File

@@ -1,7 +1,9 @@
<template> <template>
<div class="min-h-[60vh] flex flex-col items-center justify-center gap-6 text-center"> <div class="min-h-[60vh] flex flex-col items-center justify-center gap-6 text-center">
<div class="space-y-2"> <div class="space-y-2">
<h1 class="text-3xl font-semibold">Gestion des catalogues</h1> <h1 class="text-3xl font-semibold">
Gestion des catalogues
</h1>
<p class="text-sm text-gray-500"> <p class="text-sm text-gray-500">
Administrez les modèles de composants et de pièces utilisés lors de la configuration des machines. Administrez les modèles de composants et de pièces utilisés lors de la configuration des machines.
</p> </p>

View File

@@ -2,12 +2,20 @@
<main class="container mx-auto px-6 py-8 space-y-8"> <main class="container mx-auto px-6 py-8 space-y-8">
<header class="flex flex-col gap-4 md:flex-row md:items-center md:justify-between"> <header class="flex flex-col gap-4 md:flex-row md:items-center md:justify-between">
<div class="space-y-1"> <div class="space-y-1">
<h1 class="text-3xl font-bold text-gray-800">Catalogue de pièce</h1> <h1 class="text-3xl font-bold text-gray-800">
<p class="text-sm text-gray-500">Gérez les modèles disponibles pour chaque groupe de pièces.</p> Catalogue de pièce
</h1>
<p class="text-sm text-gray-500">
Gérez les modèles disponibles pour chaque groupe de pièces.
</p>
</div> </div>
<div class="tabs tabs-boxed"> <div class="tabs tabs-boxed">
<NuxtLink to="/component-catalog" class="tab">Composants</NuxtLink> <NuxtLink to="/component-catalog" class="tab">
<NuxtLink to="/pieces-catalog" class="tab tab-active">Pièces</NuxtLink> Composants
</NuxtLink>
<NuxtLink to="/pieces-catalog" class="tab tab-active">
Pièces
</NuxtLink>
</div> </div>
</header> </header>
@@ -32,7 +40,7 @@
type="search" type="search"
placeholder="Rechercher un modèle..." placeholder="Rechercher un modèle..."
class="input input-bordered input-sm" class="input input-bordered input-sm"
/> >
</label> </label>
<span class="text-xs text-gray-500">{{ filteredModels.length }} modèle(s)</span> <span class="text-xs text-gray-500">{{ filteredModels.length }} modèle(s)</span>
</div> </div>
@@ -43,7 +51,7 @@
</div> </div>
<div v-if="loadingPieceModels" class="flex justify-center py-16"> <div v-if="loadingPieceModels" class="flex justify-center py-16">
<span class="loading loading-spinner loading-lg"></span> <span class="loading loading-spinner loading-lg" />
</div> </div>
<div v-else-if="filteredModels.length === 0" class="py-16 text-center text-sm text-gray-500"> <div v-else-if="filteredModels.length === 0" class="py-16 text-center text-sm text-gray-500">
@@ -55,10 +63,16 @@
<thead> <thead>
<tr class="text-sm text-gray-500"> <tr class="text-sm text-gray-500">
<th>Nom</th> <th>Nom</th>
<th class="hidden md:table-cell">Description</th> <th class="hidden md:table-cell">
Description
</th>
<th>Type</th> <th>Type</th>
<th class="hidden lg:table-cell">Modifié</th> <th class="hidden lg:table-cell">
<th class="text-right">Actions</th> Modifié
</th>
<th class="text-right">
Actions
</th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
@@ -75,9 +89,13 @@
<span class="font-medium">{{ model.name }}</span> <span class="font-medium">{{ model.name }}</span>
</div> </div>
</td> </td>
<td class="hidden md:table-cell">{{ model.description || '—' }}</td> <td class="hidden md:table-cell">
{{ model.description || '—' }}
</td>
<td>{{ model.typePiece?.name || 'Non défini' }}</td> <td>{{ model.typePiece?.name || 'Non défini' }}</td>
<td class="hidden lg:table-cell text-xs text-gray-500">{{ formatFrenchDate(model.updatedAt || model.createdAt) }}</td> <td class="hidden lg:table-cell text-xs text-gray-500">
{{ formatFrenchDate(model.updatedAt || model.createdAt) }}
</td>
<td class="text-right space-x-2"> <td class="text-right space-x-2">
<button type="button" class="btn btn-sm btn-outline" @click="startEdit(model)"> <button type="button" class="btn btn-sm btn-outline" @click="startEdit(model)">
Éditer Éditer
@@ -118,7 +136,7 @@
class="input input-bordered input-sm" class="input input-bordered input-sm"
placeholder="Nom du modèle" placeholder="Nom du modèle"
required required
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
<label class="label"><span class="label-text">Description</span></label> <label class="label"><span class="label-text">Description</span></label>
@@ -127,7 +145,7 @@
class="textarea textarea-bordered textarea-sm" class="textarea textarea-bordered textarea-sm"
rows="3" rows="3"
placeholder="Notes optionnelles" placeholder="Notes optionnelles"
></textarea> />
</div> </div>
<div class="form-control"> <div class="form-control">
<label class="label"><span class="label-text">Type de pièce</span></label> <label class="label"><span class="label-text">Type de pièce</span></label>
@@ -136,14 +154,18 @@
class="select select-bordered select-sm" class="select select-bordered select-sm"
required required
> >
<option value="" disabled>Sélectionner un type</option> <option value="" disabled>
Sélectionner un type
</option>
<option v-for="type in pieceTypes" :key="type.id" :value="type.id"> <option v-for="type in pieceTypes" :key="type.id" :value="type.id">
{{ type.name }} {{ type.name }}
</option> </option>
</select> </select>
</div> </div>
<div class="divider my-0">Structure</div> <div class="divider my-0">
Structure
</div>
<PieceModelStructureEditor v-model="form.data.structure" /> <PieceModelStructureEditor v-model="form.data.structure" />
<div class="rounded-lg border border-base-200 bg-base-200/60 p-3 text-xs text-gray-500"> <div class="rounded-lg border border-base-200 bg-base-200/60 p-3 text-xs text-gray-500">
@@ -181,7 +203,7 @@ const {
updatePieceModel, updatePieceModel,
deletePieceModel, deletePieceModel,
loadingPieceModels, loadingPieceModels,
getPieceModelsForType, getPieceModelsForType
} = usePieceModels() } = usePieceModels()
const { showError, showSuccess } = useToast() const { showError, showSuccess } = useToast()
@@ -206,15 +228,15 @@ const form = reactive({
name: '', name: '',
description: '', description: '',
typePieceId: '', typePieceId: '',
structure: defaultStructure(), structure: defaultStructure()
}, }
}) })
const ensureTypeSelected = () => { const ensureTypeSelected = () => {
if (form.data.typePieceId && pieceTypes.value.some((type) => type.id === form.data.typePieceId)) { if (form.data.typePieceId && pieceTypes.value.some(type => type.id === form.data.typePieceId)) {
return return
} }
if (selectedType.value !== 'all' && pieceTypes.value.some((type) => type.id === selectedType.value)) { if (selectedType.value !== 'all' && pieceTypes.value.some(type => type.id === selectedType.value)) {
form.data.typePieceId = selectedType.value form.data.typePieceId = selectedType.value
return return
} }
@@ -228,7 +250,7 @@ const startCreate = () => {
name: '', name: '',
description: '', description: '',
typePieceId: selectedType.value !== 'all' ? selectedType.value : '', typePieceId: selectedType.value !== 'all' ? selectedType.value : '',
structure: defaultStructure(), structure: defaultStructure()
} }
ensureTypeSelected() ensureTypeSelected()
} }
@@ -240,7 +262,7 @@ const startEdit = (model) => {
name: model.name, name: model.name,
description: model.description || '', description: model.description || '',
typePieceId: model.typePieceId || model.typePiece?.id || '', typePieceId: model.typePieceId || model.typePiece?.id || '',
structure: cloneStructure(model.structure || defaultStructure()), structure: cloneStructure(model.structure || defaultStructure())
} }
} }
@@ -255,9 +277,9 @@ const filteredModels = computed(() => {
const term = searchQuery.value.toLowerCase() const term = searchQuery.value.toLowerCase()
return list.filter((model) => { return list.filter((model) => {
return ( return (
model.name?.toLowerCase().includes(term) model.name?.toLowerCase().includes(term) ||
|| model.description?.toLowerCase().includes(term) model.description?.toLowerCase().includes(term) ||
|| model.typePiece?.name?.toLowerCase().includes(term) model.typePiece?.name?.toLowerCase().includes(term)
) )
}) })
}) })
@@ -284,7 +306,7 @@ const handleSubmit = async () => {
name: form.data.name.trim(), name: form.data.name.trim(),
description: form.data.description.trim() || undefined, description: form.data.description.trim() || undefined,
typePieceId: form.data.typePieceId, typePieceId: form.data.typePieceId,
structure, structure
}) })
if (!result.success) { if (!result.success) {
showError(result.error || 'Impossible de créer le modèle') showError(result.error || 'Impossible de créer le modèle')
@@ -296,7 +318,7 @@ const handleSubmit = async () => {
name: form.data.name.trim(), name: form.data.name.trim(),
description: form.data.description.trim() || undefined, description: form.data.description.trim() || undefined,
typePieceId: form.data.typePieceId, typePieceId: form.data.typePieceId,
structure, structure
}) })
if (!result.success) { if (!result.success) {
showError(result.error || 'Impossible de mettre à jour le modèle') showError(result.error || 'Impossible de mettre à jour le modèle')
@@ -314,7 +336,7 @@ const handleSubmit = async () => {
const confirmDelete = async (model) => { const confirmDelete = async (model) => {
const ok = confirm(`Supprimer le modèle "${model.name}" ?`) const ok = confirm(`Supprimer le modèle "${model.name}" ?`)
if (!ok) return if (!ok) { return }
const result = await deletePieceModel(model.id) const result = await deletePieceModel(model.id)
if (!result.success) { if (!result.success) {
showError(result.error || 'Impossible de supprimer ce modèle') showError(result.error || 'Impossible de supprimer ce modèle')

View File

@@ -3,26 +3,30 @@
<div class="w-full max-w-2xl"> <div class="w-full max-w-2xl">
<div class="card bg-base-100 shadow-2xl"> <div class="card bg-base-100 shadow-2xl">
<div class="card-body"> <div class="card-body">
<h1 class="text-2xl font-bold mb-2">Choisir un profil</h1> <h1 class="text-2xl font-bold mb-2">
Choisir un profil
</h1>
<p class="text-sm text-base-content/70 mb-6"> <p class="text-sm text-base-content/70 mb-6">
Sélectionnez votre profil pour accéder à l'application. La création et la gestion se font via le menu utilisateur. Sélectionnez votre profil pour accéder à l'application. La création et la gestion se font via le menu utilisateur.
</p> </p>
<section class="space-y-4"> <section class="space-y-4">
<header class="flex items-center justify-between"> <header class="flex items-center justify-between">
<h2 class="font-semibold">Profils disponibles</h2> <h2 class="font-semibold">
Profils disponibles
</h2>
<button <button
type="button" type="button"
class="btn btn-ghost btn-xs" class="btn btn-ghost btn-xs"
@click="refreshProfiles"
:disabled="loadingProfiles" :disabled="loadingProfiles"
@click="refreshProfiles"
> >
<span v-if="loadingProfiles" class="loading loading-spinner loading-xs"></span> <span v-if="loadingProfiles" class="loading loading-spinner loading-xs" />
<span v-else>Rafraîchir</span> <span v-else>Rafraîchir</span>
</button> </button>
</header> </header>
<div class="space-y-2 max-h-64 overflow-y-auto" v-if="profiles.length"> <div v-if="profiles.length" class="space-y-2 max-h-64 overflow-y-auto">
<button <button
v-for="profile in profiles" v-for="profile in profiles"
:key="profile.id" :key="profile.id"
@@ -34,10 +38,12 @@
<IconLucideChevronRight class="w-4 h-4" aria-hidden="true" /> <IconLucideChevronRight class="w-4 h-4" aria-hidden="true" />
</button> </button>
</div> </div>
<p v-else class="text-sm text-base-content/60">Aucun profil enregistré.</p> <p v-else class="text-sm text-base-content/60">
Aucun profil enregistré.
</p>
</section> </section>
<footer class="mt-6 flex justify-between items-center" v-if="activeProfile"> <footer v-if="activeProfile" class="mt-6 flex justify-between items-center">
<div class="text-sm text-base-content/70"> <div class="text-sm text-base-content/70">
Profil actuel : Profil actuel :
<span class="font-semibold">{{ activeProfile.firstName }} {{ activeProfile.lastName }}</span> <span class="font-semibold">{{ activeProfile.firstName }} {{ activeProfile.lastName }}</span>
@@ -55,8 +61,7 @@
<script setup> <script setup>
import { onMounted } from 'vue' import { onMounted } from 'vue'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { useProfiles } from '#imports' import { useProfiles, useProfileSession } from '#imports'
import { useProfileSession } from '#imports'
import IconLucideChevronRight from '~icons/lucide/chevron-right' import IconLucideChevronRight from '~icons/lucide/chevron-right'
const router = useRouter() const router = useRouter()

View File

@@ -3,32 +3,44 @@
<div class="max-w-4xl mx-auto"> <div class="max-w-4xl mx-auto">
<div class="flex items-center justify-between mb-6"> <div class="flex items-center justify-between mb-6">
<div> <div>
<h1 class="text-2xl font-bold">Gestion des profils</h1> <h1 class="text-2xl font-bold">
<p class="text-sm text-base-content/70">Sélectionnez, créez ou supprimez des profils.</p> Gestion des profils
</h1>
<p class="text-sm text-base-content/70">
Sélectionnez, créez ou supprimez des profils.
</p>
</div> </div>
<NuxtLink to="/" class="btn btn-ghost btn-sm">Retour</NuxtLink> <NuxtLink to="/" class="btn btn-ghost btn-sm">
Retour
</NuxtLink>
</div> </div>
<div class="grid grid-cols-1 md:grid-cols-2 gap-6"> <div class="grid grid-cols-1 md:grid-cols-2 gap-6">
<section class="card bg-base-100 shadow-lg"> <section class="card bg-base-100 shadow-lg">
<div class="card-body space-y-4"> <div class="card-body space-y-4">
<header class="flex items-center justify-between"> <header class="flex items-center justify-between">
<h2 class="card-title text-lg">Profils existants</h2> <h2 class="card-title text-lg">
<button type="button" class="btn btn-ghost btn-xs" @click="refresh" :disabled="loadingProfiles"> Profils existants
<span v-if="loadingProfiles" class="loading loading-spinner loading-xs"></span> </h2>
<button type="button" class="btn btn-ghost btn-xs" :disabled="loadingProfiles" @click="refresh">
<span v-if="loadingProfiles" class="loading loading-spinner loading-xs" />
<span v-else>Rafraîchir</span> <span v-else>Rafraîchir</span>
</button> </button>
</header> </header>
<div class="space-y-2 max-h-80 overflow-y-auto" v-if="profiles.length"> <div v-if="profiles.length" class="space-y-2 max-h-80 overflow-y-auto">
<div <div
v-for="profile in profiles" v-for="profile in profiles"
:key="profile.id" :key="profile.id"
class="flex items-center justify-between rounded-lg border border-base-200 bg-base-100 px-3 py-2" class="flex items-center justify-between rounded-lg border border-base-200 bg-base-100 px-3 py-2"
> >
<div> <div>
<p class="font-medium">{{ profile.firstName }} {{ profile.lastName }}</p> <p class="font-medium">
<p class="text-xs text-base-content/60">ID : {{ profile.id }}</p> {{ profile.firstName }} {{ profile.lastName }}
</p>
<p class="text-xs text-base-content/60">
ID : {{ profile.id }}
</p>
</div> </div>
<div class="flex items-center gap-2"> <div class="flex items-center gap-2">
<button <button
@@ -44,19 +56,23 @@
class="btn btn-error btn-sm" class="btn btn-error btn-sm"
@click="remove(profile.id)" @click="remove(profile.id)"
> >
<span v-if="deleting === profile.id" class="loading loading-spinner loading-xs"></span> <span v-if="deleting === profile.id" class="loading loading-spinner loading-xs" />
<span v-else>Supprimer</span> <span v-else>Supprimer</span>
</button> </button>
</div> </div>
</div> </div>
</div> </div>
<p v-else class="text-sm text-base-content/60">Aucun profil enregistré.</p> <p v-else class="text-sm text-base-content/60">
Aucun profil enregistré.
</p>
</div> </div>
</section> </section>
<section class="card bg-base-100 shadow-lg"> <section class="card bg-base-100 shadow-lg">
<div class="card-body space-y-4"> <div class="card-body space-y-4">
<h2 class="card-title text-lg">Créer un profil</h2> <h2 class="card-title text-lg">
Créer un profil
</h2>
<form class="space-y-3" @submit.prevent="create"> <form class="space-y-3" @submit.prevent="create">
<div class="form-control"> <div class="form-control">
<label class="label"><span class="label-text">Prénom</span></label> <label class="label"><span class="label-text">Prénom</span></label>
@@ -66,7 +82,7 @@
class="input input-bordered" class="input input-bordered"
placeholder="Prénom" placeholder="Prénom"
required required
/> >
</div> </div>
<div class="form-control"> <div class="form-control">
<label class="label"><span class="label-text">Nom</span></label> <label class="label"><span class="label-text">Nom</span></label>
@@ -76,10 +92,10 @@
class="input input-bordered" class="input input-bordered"
placeholder="Nom" placeholder="Nom"
required required
/> >
</div> </div>
<button type="submit" class="btn btn-primary w-full" :disabled="creating"> <button type="submit" class="btn btn-primary w-full" :disabled="creating">
<span v-if="creating" class="loading loading-spinner loading-sm"></span> <span v-if="creating" class="loading loading-spinner loading-sm" />
<span v-else>Créer et activer</span> <span v-else>Créer et activer</span>
</button> </button>
</form> </form>
@@ -87,10 +103,14 @@
</section> </section>
</div> </div>
<div class="mt-8 flex items-center justify-between bg-base-100 shadow-lg rounded-lg p-4" v-if="activeProfile"> <div v-if="activeProfile" class="mt-8 flex items-center justify-between bg-base-100 shadow-lg rounded-lg p-4">
<div> <div>
<p class="text-sm text-base-content/70">Profil actif :</p> <p class="text-sm text-base-content/70">
<p class="font-semibold text-base-content">{{ activeProfile.firstName }} {{ activeProfile.lastName }}</p> Profil actif :
</p>
<p class="font-semibold text-base-content">
{{ activeProfile.firstName }} {{ activeProfile.lastName }}
</p>
</div> </div>
<button type="button" class="btn btn-outline" @click="handleLogout"> <button type="button" class="btn btn-outline" @click="handleLogout">
Déconnexion Déconnexion
@@ -111,7 +131,7 @@ const { activeProfile, activateProfile, fetchCurrentProfile, logout } = useProfi
const createForm = reactive({ const createForm = reactive({
firstName: '', firstName: '',
lastName: '', lastName: ''
}) })
const creating = ref(false) const creating = ref(false)
@@ -136,7 +156,7 @@ const create = async () => {
try { try {
const profile = await createProfile({ const profile = await createProfile({
firstName: createForm.firstName, firstName: createForm.firstName,
lastName: createForm.lastName, lastName: createForm.lastName
}) })
createForm.firstName = '' createForm.firstName = ''
createForm.lastName = '' createForm.lastName = ''
@@ -150,7 +170,7 @@ const create = async () => {
} }
const remove = async (profileId) => { const remove = async (profileId) => {
if (!confirm('Supprimer ce profil ?')) return if (!confirm('Supprimer ce profil ?')) { return }
deleting.value = profileId deleting.value = profileId
try { try {
await deleteProfile(profileId) await deleteProfile(profileId)

View File

@@ -8,23 +8,29 @@
<div class="my-8"> <div class="my-8">
<div class="flex justify-between items-center mb-6"> <div class="flex justify-between items-center mb-6">
<h2 class="text-2xl font-bold">Sites</h2> <h2 class="text-2xl font-bold">
<button @click="openCreateSiteModal" class="btn btn-primary"> Sites
</h2>
<button class="btn btn-primary" @click="openCreateSiteModal">
<IconLucidePlus class="w-5 h-5 mr-2" aria-hidden="true" /> <IconLucidePlus class="w-5 h-5 mr-2" aria-hidden="true" />
Ajouter un site Ajouter un site
</button> </button>
</div> </div>
<div v-if="loading" class="flex justify-center items-center py-12"> <div v-if="loading" class="flex justify-center items-center py-12">
<span class="loading loading-spinner loading-lg"></span> <span class="loading loading-spinner loading-lg" />
</div> </div>
<div v-else-if="sites.length === 0" class="text-center py-12"> <div v-else-if="sites.length === 0" class="text-center py-12">
<div class="max-w-md mx-auto"> <div class="max-w-md mx-auto">
<IconLucideMapPin class="w-16 h-16 mx-auto text-gray-400 mb-4" aria-hidden="true" /> <IconLucideMapPin class="w-16 h-16 mx-auto text-gray-400 mb-4" aria-hidden="true" />
<h3 class="text-lg font-medium text-gray-900 mb-2">Aucun site trouvé</h3> <h3 class="text-lg font-medium text-gray-900 mb-2">
<p class="text-gray-500 mb-4">Commencez par ajouter votre premier site.</p> Aucun site trouvé
<button @click="openCreateSiteModal" class="btn btn-primary"> </h3>
<p class="text-gray-500 mb-4">
Commencez par ajouter votre premier site.
</p>
<button class="btn btn-primary" @click="openCreateSiteModal">
Ajouter un site Ajouter un site
</button> </button>
</div> </div>

View File

@@ -1,10 +1,11 @@
<template> <template>
<main class="container mx-auto px-6 py-8"> <main class="container mx-auto px-6 py-8">
<!-- Loading State --> <!-- Loading State -->
<div v-if="loading" class="my-8 text-center"> <div v-if="loading" class="my-8 text-center">
<div class="loading loading-spinner loading-lg"></div> <div class="loading loading-spinner loading-lg" />
<p class="mt-4 text-gray-600">Chargement du type...</p> <p class="mt-4 text-gray-600">
Chargement du type...
</p>
</div> </div>
<!-- Type Details --> <!-- Type Details -->
@@ -31,7 +32,9 @@
<!-- Familles de composants --> <!-- Familles de composants -->
<div v-if="componentRequirementCount > 0" class="mb-8 space-y-3"> <div v-if="componentRequirementCount > 0" class="mb-8 space-y-3">
<h3 class="text-lg font-semibold">Familles de composants</h3> <h3 class="text-lg font-semibold">
Familles de composants
</h3>
<div class="space-y-3"> <div class="space-y-3">
<div <div
v-for="requirement in type.componentRequirements" v-for="requirement in type.componentRequirements"
@@ -61,7 +64,9 @@
<!-- Groupes de pièces --> <!-- Groupes de pièces -->
<div v-if="pieceRequirementCount > 0" class="mb-8 space-y-3"> <div v-if="pieceRequirementCount > 0" class="mb-8 space-y-3">
<h3 class="text-lg font-semibold">Groupes de pièces</h3> <h3 class="text-lg font-semibold">
Groupes de pièces
</h3>
<div class="space-y-3"> <div class="space-y-3">
<div <div
v-for="requirement in type.pieceRequirements" v-for="requirement in type.pieceRequirements"
@@ -96,7 +101,9 @@
<div v-else class="my-8 text-center"> <div v-else class="my-8 text-center">
<div class="alert alert-error"> <div class="alert alert-error">
<div> <div>
<h3 class="font-bold">Type non trouvé</h3> <h3 class="font-bold">
Type non trouvé
</h3>
<p>Le type de machine demandé n'existe pas.</p> <p>Le type de machine demandé n'existe pas.</p>
</div> </div>
</div> </div>
@@ -155,4 +162,4 @@ onMounted(async () => {
console.log('Loading finished, loading.value:', loading.value) console.log('Loading finished, loading.value:', loading.value)
} }
}) })
</script> </script>

View File

@@ -1,11 +1,11 @@
<template> <template>
<main class="container mx-auto px-6 py-8"> <main class="container mx-auto px-6 py-8">
<!-- Loading State --> <!-- Loading State -->
<div v-if="loading" class="my-8 text-center"> <div v-if="loading" class="my-8 text-center">
<div class="loading loading-spinner loading-lg"></div> <div class="loading loading-spinner loading-lg" />
<p class="mt-4 text-gray-600">Chargement du type...</p> <p class="mt-4 text-gray-600">
Chargement du type...
</p>
</div> </div>
<!-- Edit Form --> <!-- Edit Form -->
@@ -21,8 +21,8 @@
</NuxtLink> </NuxtLink>
</div> </div>
<TypeEditForm <TypeEditForm
v-model="editedType" v-model="editedType"
:saving="saving" :saving="saving"
@submit="saveChanges" @submit="saveChanges"
/> />
@@ -34,7 +34,9 @@
<div v-else class="my-8 text-center"> <div v-else class="my-8 text-center">
<div class="alert alert-error"> <div class="alert alert-error">
<div> <div>
<h3 class="font-bold">Type non trouvé</h3> <h3 class="font-bold">
Type non trouvé
</h3>
<p>Le type de machine demandé n'existe pas.</p> <p>Le type de machine demandé n'existe pas.</p>
</div> </div>
</div> </div>
@@ -68,11 +70,11 @@ const editedType = ref({
maintenanceFrequency: '', maintenanceFrequency: '',
customFields: [], customFields: [],
componentRequirements: [], componentRequirements: [],
pieceRequirements: [], pieceRequirements: []
}) })
const parseOptions = (field = {}) => { const parseOptions = (field = {}) => {
if (field.type !== 'select') return [] if (field.type !== 'select') { return [] }
if (field.optionsText && typeof field.optionsText === 'string') { if (field.optionsText && typeof field.optionsText === 'string') {
return field.optionsText return field.optionsText
.split('\n') .split('\n')
@@ -114,7 +116,7 @@ const normalizeComponentRequirements = (requirements = []) =>
minCount: toIntegerOrNull(req.minCount, 1), minCount: toIntegerOrNull(req.minCount, 1),
maxCount: toIntegerOrNull(req.maxCount, null), maxCount: toIntegerOrNull(req.maxCount, null),
required: req.required ?? true, required: req.required ?? true,
allowNewModels: req.allowNewModels ?? true, allowNewModels: req.allowNewModels ?? true
})) }))
const normalizePieceRequirements = (requirements = []) => const normalizePieceRequirements = (requirements = []) =>
@@ -126,7 +128,7 @@ const normalizePieceRequirements = (requirements = []) =>
minCount: toIntegerOrNull(req.minCount, 0), minCount: toIntegerOrNull(req.minCount, 0),
maxCount: toIntegerOrNull(req.maxCount, null), maxCount: toIntegerOrNull(req.maxCount, null),
required: req.required ?? false, required: req.required ?? false,
allowNewModels: req.allowNewModels ?? true, allowNewModels: req.allowNewModels ?? true
})) }))
const saveChanges = async () => { const saveChanges = async () => {
@@ -140,11 +142,11 @@ const saveChanges = async () => {
...currentEditedType, ...currentEditedType,
customFields: normalizeCustomFields(currentEditedType.customFields), customFields: normalizeCustomFields(currentEditedType.customFields),
componentRequirements: normalizeComponentRequirements(currentEditedType.componentRequirements), componentRequirements: normalizeComponentRequirements(currentEditedType.componentRequirements),
pieceRequirements: normalizePieceRequirements(currentEditedType.pieceRequirements), pieceRequirements: normalizePieceRequirements(currentEditedType.pieceRequirements)
} }
const result = await updateMachineType(type.value.id, updatedType) const result = await updateMachineType(type.value.id, updatedType)
if (result.success) { if (result.success) {
showSuccess('Type mis à jour avec succès !') showSuccess('Type mis à jour avec succès !')
router.push('/machine-skeleton') router.push('/machine-skeleton')
@@ -165,14 +167,14 @@ onMounted(async () => {
const typeId = route.params.id const typeId = route.params.id
console.log('=== EDIT TYPE PAGE LOADING ===') console.log('=== EDIT TYPE PAGE LOADING ===')
console.log('Loading type with ID:', typeId) console.log('Loading type with ID:', typeId)
const result = await getMachineTypeById(typeId) const result = await getMachineTypeById(typeId)
console.log('API Result:', result) console.log('API Result:', result)
if (result.success) { if (result.success) {
type.value = result.data type.value = result.data
console.log('Type loaded successfully:', type.value) console.log('Type loaded successfully:', type.value)
// Initialiser les données éditées // Initialiser les données éditées
editedType.value = { editedType.value = {
name: type.value.name || '', name: type.value.name || '',
@@ -181,7 +183,7 @@ onMounted(async () => {
maintenanceFrequency: type.value.maintenanceFrequency || '', maintenanceFrequency: type.value.maintenanceFrequency || '',
customFields: type.value.customFields || [], customFields: type.value.customFields || [],
componentRequirements: type.value.componentRequirements || [], componentRequirements: type.value.componentRequirements || [],
pieceRequirements: type.value.pieceRequirements || [], pieceRequirements: type.value.pieceRequirements || []
} }
} else { } else {
console.error('Failed to load type:', result.error) console.error('Failed to load type:', result.error)
@@ -195,4 +197,4 @@ onMounted(async () => {
console.log('Loading finished, loading.value:', loading.value) console.log('Loading finished, loading.value:', loading.value)
} }
}) })
</script> </script>

View File

@@ -67,18 +67,6 @@ const sanitizePieces = (pieces: any[]): any[] => {
return pieces return pieces
.map((piece) => { .map((piece) => {
const name = typeof piece?.name === 'string' ? piece.name.trim() : ''
if (!name) {
return null
}
const reference = typeof piece?.reference === 'string' && piece.reference.trim().length > 0
? piece.reference.trim()
: undefined
const quantity = Number(piece?.quantity)
const normalizedQuantity = Number.isFinite(quantity) && quantity > 0 ? quantity : undefined
const rawTypePieceId = typeof piece?.typePieceId === 'string' const rawTypePieceId = typeof piece?.typePieceId === 'string'
? piece.typePieceId.trim() ? piece.typePieceId.trim()
: typeof piece?.typePiece?.id === 'string' : typeof piece?.typePiece?.id === 'string'
@@ -93,6 +81,19 @@ const sanitizePieces = (pieces: any[]): any[] => {
: '' : ''
const typePieceLabel = rawTypePieceLabel.length > 0 ? rawTypePieceLabel : undefined const typePieceLabel = rawTypePieceLabel.length > 0 ? rawTypePieceLabel : undefined
const rawName = typeof piece?.name === 'string' ? piece.name.trim() : ''
const name = rawName || typePieceLabel || ''
if (!name) {
return null
}
const reference = typeof piece?.reference === 'string' && piece.reference.trim().length > 0
? piece.reference.trim()
: undefined
const quantity = Number(piece?.quantity)
const normalizedQuantity = Number.isFinite(quantity) && quantity > 0 ? quantity : undefined
const result: Record<string, unknown> = { name } const result: Record<string, unknown> = { name }
if (reference !== undefined) { if (reference !== undefined) {
result.reference = reference result.reference = reference
@@ -118,7 +119,22 @@ const sanitizeSubComponents = (components: any[]): any[] => {
return components return components
.map((component) => { .map((component) => {
const name = typeof component?.name === 'string' ? component.name.trim() : '' const rawTypeComposantId = typeof component?.typeComposantId === 'string'
? component.typeComposantId.trim()
: typeof component?.typeComposant?.id === 'string'
? component.typeComposant.id.trim()
: ''
const typeComposantId = rawTypeComposantId.length > 0 ? rawTypeComposantId : undefined
const rawTypeComposantLabel = typeof component?.typeComposantLabel === 'string'
? component.typeComposantLabel.trim()
: typeof component?.typeComposant?.name === 'string'
? component.typeComposant.name.trim()
: ''
const typeComposantLabel = rawTypeComposantLabel.length > 0 ? rawTypeComposantLabel : undefined
const rawName = typeof component?.name === 'string' ? component.name.trim() : ''
const name = rawName || typeComposantLabel || ''
if (!name) { if (!name) {
return null return null
} }
@@ -147,6 +163,12 @@ const sanitizeSubComponents = (components: any[]): any[] => {
if (normalizedQuantity !== undefined) { if (normalizedQuantity !== undefined) {
result.quantity = normalizedQuantity result.quantity = normalizedQuantity
} }
if (typeComposantId) {
result.typeComposantId = typeComposantId
}
if (typeComposantLabel) {
result.typeComposantLabel = typeComposantLabel
}
return result return result
}) })
@@ -183,9 +205,9 @@ const hydratePieces = (pieces: any[]): any[] => {
} }
return pieces.map((piece) => ({ return pieces.map((piece) => ({
name: piece?.name ?? '', name: piece?.name ?? piece?.typePiece?.name ?? piece?.typePieceLabel ?? '',
reference: piece?.reference ?? '', reference: piece?.reference ?? '',
quantity: piece?.quantity ?? undefined, quantity: piece?.quantity ?? piece?.quantite ?? undefined,
typePieceId: piece?.typePieceId ?? piece?.typePiece?.id ?? '', typePieceId: piece?.typePieceId ?? piece?.typePiece?.id ?? '',
typePieceLabel: piece?.typePieceLabel ?? piece?.typePiece?.name ?? '', typePieceLabel: piece?.typePieceLabel ?? piece?.typePiece?.name ?? '',
})) }))
@@ -197,9 +219,11 @@ const hydrateSubComponents = (components: any[]): any[] => {
} }
return components.map((component) => ({ return components.map((component) => ({
name: component?.name ?? '', name: component?.name ?? component?.typeComposant?.name ?? component?.typeComposantLabel ?? '',
description: component?.description ?? '', description: component?.description ?? '',
quantity: component?.quantity ?? undefined, quantity: component?.quantity ?? component?.quantite ?? undefined,
typeComposantId: component?.typeComposantId ?? component?.typeComposant?.id ?? '',
typeComposantLabel: component?.typeComposantLabel ?? component?.typeComposant?.name ?? '',
customFields: hydrateCustomFields(component?.customFields), customFields: hydrateCustomFields(component?.customFields),
pieces: hydratePieces(component?.pieces), pieces: hydratePieces(component?.pieces),
subComponents: hydrateSubComponents(component?.subComponents), subComponents: hydrateSubComponents(component?.subComponents),
@@ -243,9 +267,11 @@ const mapComponentPieces = (pieces: any[]) => {
return [] return []
} }
return pieces.map((piece) => ({ return pieces.map((piece) => ({
name: piece?.name ?? '', name: piece?.name ?? piece?.typePiece?.name ?? '',
reference: piece?.reference ?? '', reference: piece?.reference ?? '',
quantity: piece?.quantity ?? piece?.quantite ?? undefined, quantity: piece?.quantity ?? piece?.quantite ?? undefined,
typePieceId: piece?.typePieceId ?? piece?.typePiece?.id ?? '',
typePieceLabel: piece?.typePieceLabel ?? piece?.typePiece?.name ?? '',
})) }))
} }
@@ -254,9 +280,11 @@ const mapSubComponents = (components: any[]): any[] => {
return [] return []
} }
return components.map((component) => ({ return components.map((component) => ({
name: component?.name ?? '', name: component?.name ?? component?.typeComposant?.name ?? '',
description: component?.description ?? '', description: component?.description ?? '',
quantity: component?.quantity ?? component?.quantite ?? undefined, quantity: component?.quantity ?? component?.quantite ?? undefined,
typeComposantId: component?.typeComposantId ?? component?.typeComposant?.id ?? '',
typeComposantLabel: component?.typeComposantLabel ?? component?.typeComposant?.name ?? '',
customFields: mapComponentCustomFields(component?.customFields), customFields: mapComponentCustomFields(component?.customFields),
pieces: mapComponentPieces(component?.pieces), pieces: mapComponentPieces(component?.pieces),
subComponents: mapSubComponents(component?.subComponents), subComponents: mapSubComponents(component?.subComponents),

View File

@@ -1,24 +1,24 @@
import { getFileIcon } from './fileIcons' import { getFileIcon } from './fileIcons'
export const getPreviewType = (document) => { export const getPreviewType = (document) => {
if (!document) return null if (!document) { return null }
const mime = (document.mimeType || '').toLowerCase() const mime = (document.mimeType || '').toLowerCase()
const path = document.path || '' const path = document.path || ''
const check = (prefix) => mime.startsWith(prefix) || path.startsWith(`data:${prefix}`) const check = prefix => mime.startsWith(prefix) || path.startsWith(`data:${prefix}`)
if (check('image/')) return 'image' if (check('image/')) { return 'image' }
if (mime === 'application/pdf' || path.startsWith('data:application/pdf')) return 'pdf' if (mime === 'application/pdf' || path.startsWith('data:application/pdf')) { return 'pdf' }
if (check('audio/')) return 'audio' if (check('audio/')) { return 'audio' }
if (check('video/')) return 'video' if (check('video/')) { return 'video' }
if (check('text/') || mime.includes('json') || mime.includes('xml') || path.startsWith('data:application/json')) return 'text' if (check('text/') || mime.includes('json') || mime.includes('xml') || path.startsWith('data:application/json')) { return 'text' }
return null return null
} }
export const canPreviewDocument = (document = {}) => !!getPreviewType(document) export const canPreviewDocument = (document = {}) => !!getPreviewType(document)
export const describeDocument = (document) => { export const describeDocument = (document) => {
if (!document) return '' if (!document) { return '' }
const name = document.filename || document.name || '' const name = document.filename || document.name || ''
const icon = getFileIcon({ name, mime: document.mimeType }) const icon = getFileIcon({ name, mime: document.mimeType })
return icon.label return icon.label

View File

@@ -1,8 +1,6 @@
const formatSize = (size) => { const formatSize = (size) => {
if (size === undefined || size === null) return '—' if (size === undefined || size === null) { return '—' }
if (size === 0) return '0 B' if (size === 0) { return '0 B' }
const units = ['B', 'KB', 'MB', 'GB'] const units = ['B', 'KB', 'MB', 'GB']
const index = Math.min(units.length - 1, Math.floor(Math.log(size) / Math.log(1024))) const index = Math.min(units.length - 1, Math.floor(Math.log(size) / Math.log(1024)))
const formatted = size / Math.pow(1024, index) const formatted = size / Math.pow(1024, index)
@@ -15,9 +13,9 @@ const renderPrintField = (label, value, fallback = '—') => {
} }
const renderPrintCustomFields = (fields = [], title, sectionClass = 'print-section') => { const renderPrintCustomFields = (fields = [], title, sectionClass = 'print-section') => {
if (!fields.length) return '' if (!fields.length) { return '' }
const items = fields const items = fields
.map((field) => `<div class="print-field"><label>${field.label}</label><span>${field.value || '—'}</span></div>`) .map(field => `<div class="print-field"><label>${field.label}</label><span>${field.value || '—'}</span></div>`)
.join('') .join('')
return ` return `
<div class="${sectionClass}"> <div class="${sectionClass}">
@@ -30,9 +28,9 @@ const renderPrintCustomFields = (fields = [], title, sectionClass = 'print-secti
} }
const renderPrintDocuments = (documents = [], title, sectionClass = 'print-section') => { const renderPrintDocuments = (documents = [], title, sectionClass = 'print-section') => {
if (!documents.length) return '' if (!documents.length) { return '' }
const rows = documents const rows = documents
.map((doc) => `<tr><td>${doc.name}</td><td>${doc.type}</td><td>${doc.size}</td></tr>`) .map(doc => `<tr><td>${doc.name}</td><td>${doc.type}</td><td>${doc.size}</td></tr>`)
.join('') .join('')
return ` return `
<div class="${sectionClass}"> <div class="${sectionClass}">
@@ -54,9 +52,9 @@ const renderPrintDocuments = (documents = [], title, sectionClass = 'print-secti
const renderPrintPieces = ( const renderPrintPieces = (
pieces = [], pieces = [],
title = 'Pièces indépendantes', title = 'Pièces indépendantes',
sectionClass = 'print-section print-section--pieces', sectionClass = 'print-section print-section--pieces'
) => { ) => {
if (!pieces.length) return '' if (!pieces.length) { return '' }
const cards = pieces const cards = pieces
.map((piece, idx) => { .map((piece, idx) => {
@@ -66,14 +64,14 @@ const renderPrintPieces = (
: '' : ''
const customFields = (piece.customFields || []) const customFields = (piece.customFields || [])
.filter((field) => field.value && field.value !== '—' && field.value !== '') .filter(field => field.value && field.value !== '—' && field.value !== '')
.map( .map(
(field) => ` field => `
<li> <li>
<span class="print-list-label">${field.label}</span> <span class="print-list-label">${field.label}</span>
<span class="print-list-value">${field.value}</span> <span class="print-list-value">${field.value}</span>
</li> </li>
`, `
) )
.join('') .join('')
@@ -83,7 +81,7 @@ const renderPrintPieces = (
const documentsBlock = (piece.documents || []).length const documentsBlock = (piece.documents || []).length
? `<div class="print-piece-section"><h4>Documents</h4><ul class="print-list">${piece.documents ? `<div class="print-piece-section"><h4>Documents</h4><ul class="print-list">${piece.documents
.map((doc) => `<li>${doc.name} <span class="print-list-hint">(${doc.type}${doc.size})</span></li>`) .map(doc => `<li>${doc.name} <span class="print-list-hint">(${doc.type}${doc.size})</span></li>`)
.join('')}</ul></div>` .join('')}</ul></div>`
: '' : ''
@@ -126,7 +124,7 @@ const renderPrintPieces = (
} }
const renderPrintComponents = (components = [], depth = 0, indexPath = []) => { const renderPrintComponents = (components = [], depth = 0, indexPath = []) => {
if (!components.length) return '' if (!components.length) { return '' }
return components return components
.map((component, idx) => { .map((component, idx) => {
const badges = [] const badges = []
@@ -143,21 +141,21 @@ const renderPrintComponents = (components = [], depth = 0, indexPath = []) => {
<span>Composant&nbsp;: ${component.name}</span> <span>Composant&nbsp;: ${component.name}</span>
</h3> </h3>
${component.description ? `<p class="print-muted">${component.description}</p>` : ''} ${component.description ? `<p class="print-muted">${component.description}</p>` : ''}
${badges.length ? `<div class="badge-group">${badges.map((badge) => `<span class="print-badge">${badge}</span>`).join('')}</div>` : ''} ${badges.length ? `<div class="badge-group">${badges.map(badge => `<span class="print-badge">${badge}</span>`).join('')}</div>` : ''}
${renderPrintCustomFields( ${renderPrintCustomFields(
component.customFields, component.customFields,
'Champs personnalisés', 'Champs personnalisés',
'print-section print-subsection print-section--custom-fields', 'print-section print-subsection print-section--custom-fields'
)} )}
${renderPrintPieces( ${renderPrintPieces(
(component.pieces || []).map((piece, pieceIdx) => ({ ...piece, indexPath: [...currentIndex, pieceIdx + 1] })), (component.pieces || []).map((piece, pieceIdx) => ({ ...piece, indexPath: [...currentIndex, pieceIdx + 1] })),
'Pièces du composant', 'Pièces du composant',
'print-section print-subsection print-section--pieces', 'print-section print-subsection print-section--pieces'
)} )}
${renderPrintDocuments( ${renderPrintDocuments(
component.documents, component.documents,
'Documents du composant', 'Documents du composant',
'print-section print-subsection print-section--documents', 'print-section print-subsection print-section--documents'
)} )}
${renderPrintComponents(component.subComponents || [], depth + 1, currentIndex)} ${renderPrintComponents(component.subComponents || [], depth + 1, currentIndex)}
</div> </div>
@@ -167,31 +165,31 @@ const renderPrintComponents = (components = [], depth = 0, indexPath = []) => {
} }
const normalizeDocuments = (docs = []) => { const normalizeDocuments = (docs = []) => {
return docs.map((doc) => ({ return docs.map(doc => ({
id: doc.id, id: doc.id,
name: doc.name || doc.filename || 'Document', name: doc.name || doc.filename || 'Document',
type: doc.mimeType || doc.type || '—', type: doc.mimeType || doc.type || '—',
size: formatSize(doc.size), size: formatSize(doc.size)
})) }))
} }
const normalizeCustomFields = (values = []) => { const normalizeCustomFields = (values = []) => {
return values.map((value) => ({ return values.map(value => ({
id: value.id, id: value.id,
label: value.customField?.name || 'Champ', label: value.customField?.name || 'Champ',
value: value.value || '—', value: value.value || '—'
})) }))
} }
const normalizeConstructeur = (constructeur) => { const normalizeConstructeur = (constructeur) => {
if (!constructeur) return null if (!constructeur) { return null }
return { return {
name: constructeur.name || '—', name: constructeur.name || '—',
contact: [constructeur.email, constructeur.phone].filter(Boolean).join(' • ') || '—', contact: [constructeur.email, constructeur.phone].filter(Boolean).join(' • ') || '—'
} }
} }
const normalizePiece = (piece) => ({ const normalizePiece = piece => ({
id: piece.id, id: piece.id,
name: piece.name || 'Pièce sans nom', name: piece.name || 'Pièce sans nom',
description: piece.description || '', description: piece.description || '',
@@ -199,10 +197,10 @@ const normalizePiece = (piece) => ({
customFields: normalizeCustomFields(piece.customFieldValues || []), customFields: normalizeCustomFields(piece.customFieldValues || []),
documents: normalizeDocuments(piece.documents || []), documents: normalizeDocuments(piece.documents || []),
constructeur: normalizeConstructeur(piece.constructeur), constructeur: normalizeConstructeur(piece.constructeur),
indexPath: piece.indexPath || null, indexPath: piece.indexPath || null
}) })
const normalizeComponent = (component) => ({ const normalizeComponent = component => ({
id: component.id, id: component.id,
name: component.name || 'Composant sans nom', name: component.name || 'Composant sans nom',
description: component.description || '', description: component.description || '',
@@ -210,7 +208,7 @@ const normalizeComponent = (component) => ({
documents: normalizeDocuments(component.documents || []), documents: normalizeDocuments(component.documents || []),
pieces: (component.pieces || []).map(normalizePiece), pieces: (component.pieces || []).map(normalizePiece),
subComponents: (component.sousComposants || component.subComponents || []).map(normalizeComponent), subComponents: (component.sousComposants || component.subComponents || []).map(normalizeComponent),
constructeur: normalizeConstructeur(component.constructeur), constructeur: normalizeConstructeur(component.constructeur)
}) })
export const buildMachinePrintContext = ({ export const buildMachinePrintContext = ({
@@ -219,7 +217,7 @@ export const buildMachinePrintContext = ({
machineReference, machineReference,
machinePieces = [], machinePieces = [],
components = [], components = [],
selection, selection
}) => { }) => {
const selectionState = selection || {} const selectionState = selection || {}
const machineSelection = selectionState.machine || {} const machineSelection = selectionState.machine || {}
@@ -231,7 +229,7 @@ export const buildMachinePrintContext = ({
const includeMachineDocuments = machineSelection.documents !== false const includeMachineDocuments = machineSelection.documents !== false
const isComponentSelected = (id) => { const isComponentSelected = (id) => {
if (!id) return true if (!id) { return true }
if (Object.prototype.hasOwnProperty.call(componentSelection, id)) { if (Object.prototype.hasOwnProperty.call(componentSelection, id)) {
return componentSelection[id] return componentSelection[id]
} }
@@ -239,7 +237,7 @@ export const buildMachinePrintContext = ({
} }
const isPieceSelected = (id) => { const isPieceSelected = (id) => {
if (!id) return true if (!id) { return true }
if (Object.prototype.hasOwnProperty.call(pieceSelection, id)) { if (Object.prototype.hasOwnProperty.call(pieceSelection, id)) {
return pieceSelection[id] return pieceSelection[id]
} }
@@ -259,16 +257,16 @@ export const buildMachinePrintContext = ({
const normalizedPieces = machinePieces const normalizedPieces = machinePieces
.map(normalizePiece) .map(normalizePiece)
.filter((piece) => isPieceSelected(piece.id)) .filter(piece => isPieceSelected(piece.id))
.map((piece, idx) => ({ .map((piece, idx) => ({
...piece, ...piece,
indexPath: [idx + 1], indexPath: [idx + 1]
})) }))
const normalizedComponents = components.map(normalizeComponent) const normalizedComponents = components.map(normalizeComponent)
const filterComponentTree = (component) => { const filterComponentTree = (component) => {
const filteredPieces = (component.pieces || []).filter((piece) => isPieceSelected(piece.id)) const filteredPieces = (component.pieces || []).filter(piece => isPieceSelected(piece.id))
const filteredSubComponents = (component.subComponents || []) const filteredSubComponents = (component.subComponents || [])
.map(filterComponentTree) .map(filterComponentTree)
.filter(Boolean) .filter(Boolean)
@@ -283,7 +281,7 @@ export const buildMachinePrintContext = ({
return { return {
...component, ...component,
pieces: filteredPieces, pieces: filteredPieces,
subComponents: filteredSubComponents, subComponents: filteredSubComponents
} }
} }
@@ -309,17 +307,17 @@ export const buildMachinePrintContext = ({
: [], : [],
documents: includeMachineDocuments documents: includeMachineDocuments
? normalizeDocuments(machine?.documents || []) ? normalizeDocuments(machine?.documents || [])
: [], : []
}, },
components: filteredComponents, components: filteredComponents,
pieces: normalizedPieces, pieces: normalizedPieces
} }
} }
export const buildMachinePrintHtml = (context, styles) => { export const buildMachinePrintHtml = (context, styles) => {
const title = context.machine.name ? `Impression - ${context.machine.name}` : 'Impression machine' const title = context.machine.name ? `Impression - ${context.machine.name}` : 'Impression machine'
const badgesHtml = context.machine.badges const badgesHtml = context.machine.badges
.map((badge) => `<span class="print-badge">${badge}</span>`) .map(badge => `<span class="print-badge">${badge}</span>`)
.join('') .join('')
const sections = [] const sections = []
@@ -357,7 +355,7 @@ export const buildMachinePrintHtml = (context, styles) => {
const customFieldsSection = renderPrintCustomFields( const customFieldsSection = renderPrintCustomFields(
context.machine.customFields, context.machine.customFields,
'Champs personnalisés de la machine', 'Champs personnalisés de la machine',
'print-section print-section--custom-fields', 'print-section print-section--custom-fields'
) )
if (customFieldsSection) { if (customFieldsSection) {
sections.push(customFieldsSection) sections.push(customFieldsSection)
@@ -366,7 +364,7 @@ export const buildMachinePrintHtml = (context, styles) => {
const documentsSection = renderPrintDocuments( const documentsSection = renderPrintDocuments(
context.machine.documents, context.machine.documents,
'Documents liés à la machine', 'Documents liés à la machine',
'print-section print-section--documents', 'print-section print-section--documents'
) )
if (documentsSection) { if (documentsSection) {
sections.push(documentsSection) sections.push(documentsSection)
@@ -380,7 +378,7 @@ export const buildMachinePrintHtml = (context, styles) => {
const piecesSection = renderPrintPieces( const piecesSection = renderPrintPieces(
context.pieces, context.pieces,
'Pièces indépendantes', 'Pièces indépendantes',
'print-section print-section--pieces', 'print-section print-section--pieces'
) )
if (piecesSection) { if (piecesSection) {
sections.push(piecesSection) sections.push(piecesSection)

70
eslint.config.mjs Normal file
View File

@@ -0,0 +1,70 @@
import nuxt from '@nuxt/eslint-config';
const globals = {
defineNuxtConfig: 'readonly',
defineNuxtRouteMiddleware: 'readonly',
navigateTo: 'readonly',
useRuntimeConfig: 'readonly',
$fetch: 'readonly',
};
const relaxedRules = {
'vue/no-parsing-error': 'off',
'vue/no-required-prop-with-default': 'off',
'vue/no-mutating-props': 'off',
'vue/html-self-closing': 'off',
'vue/singleline-html-element-content-newline': 'off',
'vue/multiline-html-element-content-newline': 'off',
'vue/html-indent': 'off',
'vue/attributes-order': 'off',
'vue/multi-word-component-names': 'off',
'vue/no-multiple-template-root': 'off',
'vue/v-on-event-hyphenation': 'off',
'vue/require-default-prop': 'off',
'no-console': 'off',
'no-unused-vars': 'off',
'require-await': 'off',
'comma-dangle': 'off',
curly: 'off',
'operator-linebreak': 'off',
'space-before-function-paren': 'off',
'arrow-parens': 'off',
semi: 'off',
quotes: 'off',
'func-call-spacing': 'off',
'no-trailing-spaces': 'off',
indent: 'off',
'no-multiple-empty-lines': 'off',
'import/order': 'off',
'no-irregular-whitespace': 'off',
'no-useless-escape': 'off',
'nuxt/prefer-import-meta': 'off',
'@typescript-eslint/no-unused-vars': 'off',
'@typescript-eslint/semi': 'off',
'@typescript-eslint/no-explicit-any': 'off',
'@typescript-eslint/no-dynamic-delete': 'off',
'@typescript-eslint/no-invalid-void-type': 'off',
'@typescript-eslint/unified-signatures': 'off',
};
export default await nuxt(
{
features: {
stylistic: false,
typescript: true,
nuxt: {
sortConfigKeys: false,
},
},
dirs: {
root: ['.', './app'],
},
},
{
name: 'project/custom-overrides',
languageOptions: {
globals,
},
rules: relaxedRules,
}
);

View File

@@ -1,9 +1,9 @@
import tailwindcss from "@tailwindcss/vite"; import tailwindcss from '@tailwindcss/vite'
export default defineNuxtConfig({ export default defineNuxtConfig({
compatibilityDate: '2025-07-15', compatibilityDate: '2025-07-15',
devtools: { enabled: true }, devtools: { enabled: true },
devServer: { devServer: {
port: 3001, port: 3001
}, },
modules: [ modules: [
[ [
@@ -12,10 +12,10 @@ export default defineNuxtConfig({
componentPrefix: 'Icon', componentPrefix: 'Icon',
warn: process.env.NODE_ENV === 'development', warn: process.env.NODE_ENV === 'development',
collections: { collections: {
lucide: () => import('@iconify-json/lucide/icons.json').then((i) => i.default), lucide: () => import('@iconify-json/lucide/icons.json').then(i => i.default)
}, }
}, }
], ]
], ],
runtimeConfig: { runtimeConfig: {
public: { public: {
@@ -27,13 +27,13 @@ export default defineNuxtConfig({
enableDebug: process.env.NUXT_PUBLIC_ENABLE_DEBUG || 'true', enableDebug: process.env.NUXT_PUBLIC_ENABLE_DEBUG || 'true',
enableAnalytics: process.env.NUXT_PUBLIC_ENABLE_ANALYTICS || 'false', enableAnalytics: process.env.NUXT_PUBLIC_ENABLE_ANALYTICS || 'false',
csrfToken: process.env.NUXT_PUBLIC_CSRF_TOKEN || '', csrfToken: process.env.NUXT_PUBLIC_CSRF_TOKEN || '',
logLevel: process.env.NUXT_PUBLIC_LOG_LEVEL || 'debug', logLevel: process.env.NUXT_PUBLIC_LOG_LEVEL || 'debug'
}, }
}, },
vite: { vite: {
plugins: [tailwindcss()], plugins: [tailwindcss()]
}, },
css: ["~/assets/app.css"], css: ['~/assets/app.css'],
router: { router: {
options: { options: {
strict: false strict: false

7531
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -8,7 +8,9 @@
"generate": "nuxt generate", "generate": "nuxt generate",
"preview": "nuxt preview", "preview": "nuxt preview",
"postinstall": "nuxt prepare", "postinstall": "nuxt prepare",
"start": "nuxt start" "start": "nuxt start",
"lint": "eslint . --ext .js,.ts,.vue",
"lint:fix": "npm run lint -- --fix"
}, },
"dependencies": { "dependencies": {
"@nuxtjs/tailwindcss": "^6.14.0", "@nuxtjs/tailwindcss": "^6.14.0",
@@ -21,6 +23,14 @@
}, },
"devDependencies": { "devDependencies": {
"@iconify-json/lucide": "^1.2.68", "@iconify-json/lucide": "^1.2.68",
"unplugin-icons": "^0.19.3" "@nuxt/eslint-config": "^1.9.0",
"@rushstack/eslint-patch": "^1.12.0",
"@typescript-eslint/eslint-plugin": "^8.44.1",
"@typescript-eslint/parser": "^8.44.1",
"eslint": "^9.36.0",
"eslint-plugin-vue": "^10.5.0",
"typescript": "^5.7.3",
"unplugin-icons": "^0.19.3",
"vue-eslint-parser": "^10.2.0"
} }
} }