1137 lines
38 KiB
Vue
1137 lines
38 KiB
Vue
<template>
|
|
<DocumentPreviewModal
|
|
:document="previewDocument"
|
|
:visible="previewVisible"
|
|
@close="closePreview"
|
|
/>
|
|
<main class="container mx-auto px-6 py-10">
|
|
<div v-if="loading" class="flex flex-col items-center gap-4 py-20 text-center">
|
|
<span class="loading loading-spinner loading-lg" aria-hidden="true" />
|
|
<p class="text-sm text-base-content/70">Chargement du composant…</p>
|
|
</div>
|
|
|
|
<div v-else-if="!component" class="max-w-xl mx-auto">
|
|
<div class="alert alert-error shadow-lg">
|
|
<div>
|
|
<h2 class="font-semibold text-lg">Composant introuvable</h2>
|
|
<p class="text-sm text-base-content/80">
|
|
Nous n'avons pas pu retrouver le composant demandé. Il a peut-être été supprimé.
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<NuxtLink to="/component-catalog" class="btn btn-primary mt-6">
|
|
Retour au catalogue
|
|
</NuxtLink>
|
|
</div>
|
|
|
|
<section v-else class="card border border-base-200 bg-base-100 shadow-sm max-w-5xl mx-auto">
|
|
<div class="card-body space-y-6">
|
|
<header class="flex flex-col gap-2 md:flex-row md:items-center md:justify-between">
|
|
<div>
|
|
<h1 class="text-3xl font-semibold text-base-content">Modifier le composant</h1>
|
|
<p class="text-sm text-base-content/70">
|
|
Mettez à jour les informations du composant et ses champs personnalisés.
|
|
</p>
|
|
</div>
|
|
<NuxtLink to="/component-catalog" class="btn btn-ghost btn-sm md:btn-md self-start">
|
|
Retour au catalogue
|
|
</NuxtLink>
|
|
</header>
|
|
|
|
<div class="grid grid-cols-1 gap-4 md:grid-cols-2">
|
|
<div class="form-control">
|
|
<label class="label">
|
|
<span class="label-text">Catégorie de composant</span>
|
|
</label>
|
|
<select
|
|
v-model="selectedTypeId"
|
|
class="select select-bordered select-sm md:select-md"
|
|
disabled
|
|
>
|
|
<option value="">Sélectionner une catégorie</option>
|
|
<option
|
|
v-for="type in componentTypeList"
|
|
:key="type.id"
|
|
:value="type.id"
|
|
>
|
|
{{ type.name }}
|
|
</option>
|
|
</select>
|
|
<p class="text-xs text-base-content/60 mt-1">
|
|
La catégorie d'origine ne peut pas être modifiée depuis cette page.
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="grid grid-cols-1 gap-4 md:grid-cols-2">
|
|
<div class="form-control">
|
|
<label class="label">
|
|
<span class="label-text">Nom du composant</span>
|
|
</label>
|
|
<input
|
|
v-model="editionForm.name"
|
|
type="text"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:disabled="saving"
|
|
placeholder="Nom affiché dans le catalogue"
|
|
required
|
|
>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="grid grid-cols-1 gap-4 md:grid-cols-2">
|
|
<div class="form-control">
|
|
<label class="label">
|
|
<span class="label-text">Référence</span>
|
|
</label>
|
|
<input
|
|
v-model="editionForm.reference"
|
|
type="text"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:disabled="saving"
|
|
placeholder="Référence interne ou constructeur"
|
|
>
|
|
</div>
|
|
|
|
<div class="form-control">
|
|
<label class="label">
|
|
<span class="label-text">Constructeur</span>
|
|
</label>
|
|
<ConstructeurSelect
|
|
v-model="editionForm.constructeurIds"
|
|
class="w-full"
|
|
:disabled="saving"
|
|
placeholder="Rechercher un ou plusieurs constructeurs..."
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="grid grid-cols-1 gap-4 md:grid-cols-2">
|
|
<div class="form-control">
|
|
<label class="label">
|
|
<span class="label-text">Prix indicatif (€)</span>
|
|
</label>
|
|
<input
|
|
v-model="editionForm.prix"
|
|
type="number"
|
|
step="0.01"
|
|
min="0"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:disabled="saving"
|
|
placeholder="Valeur indicatrice"
|
|
>
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="selectedType" class="space-y-3 rounded-lg border border-base-200 bg-base-200/40 p-4">
|
|
<div class="flex items-center justify-between gap-4">
|
|
<div>
|
|
<h2 class="font-semibold text-base-content">Squelette sélectionné</h2>
|
|
<p class="text-xs text-base-content/70">
|
|
{{ selectedType.description || 'Ce squelette définit la structure et les contraintes du composant.' }}
|
|
</p>
|
|
</div>
|
|
<span class="badge badge-outline">{{ formatStructurePreview(selectedTypeStructure) }}</span>
|
|
</div>
|
|
|
|
<details v-if="selectedTypeStructure" class="collapse collapse-arrow bg-base-100">
|
|
<summary class="collapse-title text-sm font-medium">
|
|
Consulter le détail du squelette
|
|
</summary>
|
|
<div class="collapse-content space-y-4 text-sm text-base-content/80">
|
|
<div v-if="getStructureCustomFields(selectedTypeStructure).length" class="space-y-2">
|
|
<h3 class="font-semibold text-sm text-base-content">Champs personnalisés</h3>
|
|
<ul class="space-y-2">
|
|
<li
|
|
v-for="field in getStructureCustomFields(selectedTypeStructure)"
|
|
:key="field.customFieldId || field.id || field.name"
|
|
class="rounded bg-base-200/60 px-3 py-2"
|
|
>
|
|
<p class="font-medium text-sm text-base-content">
|
|
{{ field.name || field.key }}
|
|
</p>
|
|
<p class="text-xs text-base-content/70 mt-1">
|
|
Type : {{ field.type || 'text' }}<span v-if="field.required"> • Obligatoire</span>
|
|
<span v-if="Array.isArray(field.options) && field.options.length">
|
|
• Options : {{ field.options.join(', ') }}
|
|
</span>
|
|
<span v-if="field.defaultValue">
|
|
• Défaut : {{ field.defaultValue }}
|
|
</span>
|
|
</p>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
|
|
<div v-if="getStructurePieces(selectedTypeStructure).length" class="space-y-2">
|
|
<h3 class="font-semibold text-sm text-base-content">Pièces imposées</h3>
|
|
<ul class="list-disc list-inside space-y-1">
|
|
<li
|
|
v-for="(piece, index) in getStructurePieces(selectedTypeStructure)"
|
|
:key="piece.role || piece.typePieceId || piece.familyCode || index"
|
|
>
|
|
{{ resolvePieceLabel(piece) }}
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
|
|
<div v-if="getStructureSubcomponents(selectedTypeStructure).length" class="space-y-2">
|
|
<h3 class="font-semibold text-sm text-base-content">Sous-composants</h3>
|
|
<ul class="list-disc list-inside space-y-1">
|
|
<li
|
|
v-for="(subcomponent, index) in getStructureSubcomponents(selectedTypeStructure)"
|
|
:key="subcomponent.alias || subcomponent.typeComposantId || subcomponent.familyCode || index"
|
|
>
|
|
{{ resolveSubcomponentLabel(subcomponent) }}
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
|
|
<p
|
|
v-if="!getStructureCustomFields(selectedTypeStructure).length && !getStructurePieces(selectedTypeStructure).length && !getStructureSubcomponents(selectedTypeStructure).length"
|
|
class="text-xs text-gray-500"
|
|
>
|
|
Ce squelette ne définit pas encore de pièces, sous-composants ou valeurs par défaut.
|
|
</p>
|
|
</div>
|
|
</details>
|
|
</div>
|
|
|
|
<div v-if="customFieldInputs.length" class="space-y-4 rounded-lg border border-base-200 bg-base-200/40 p-4">
|
|
<header class="space-y-1">
|
|
<h2 class="font-semibold text-base-content">Champs personnalisés</h2>
|
|
<p class="text-xs text-base-content/70">
|
|
Mettez à jour les valeurs propres à ce composant.
|
|
</p>
|
|
</header>
|
|
<div class="grid grid-cols-1 gap-4 md:grid-cols-2">
|
|
<div
|
|
v-for="(field, index) in customFieldInputs"
|
|
:key="fieldKey(field, index)"
|
|
class="form-control"
|
|
>
|
|
<label class="label">
|
|
<span class="label-text">{{ field.name }}</span>
|
|
<span v-if="field.required" class="label-text-alt text-error">*</span>
|
|
</label>
|
|
<input
|
|
v-if="field.type === 'text'"
|
|
v-model="field.value"
|
|
type="text"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:required="field.required"
|
|
:disabled="saving"
|
|
>
|
|
<input
|
|
v-else-if="field.type === 'number'"
|
|
v-model="field.value"
|
|
type="number"
|
|
step="0.01"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:required="field.required"
|
|
:disabled="saving"
|
|
>
|
|
<select
|
|
v-else-if="field.type === 'select'"
|
|
v-model="field.value"
|
|
class="select select-bordered select-sm md:select-md"
|
|
:required="field.required"
|
|
:disabled="saving"
|
|
>
|
|
<option value="">Sélectionner...</option>
|
|
<option
|
|
v-for="option in field.options"
|
|
:key="option"
|
|
:value="option"
|
|
>
|
|
{{ option }}
|
|
</option>
|
|
</select>
|
|
<div v-else-if="field.type === 'boolean'" class="flex items-center gap-2">
|
|
<input
|
|
v-model="field.value"
|
|
type="checkbox"
|
|
class="checkbox checkbox-sm"
|
|
true-value="true"
|
|
false-value="false"
|
|
:disabled="saving"
|
|
>
|
|
<span class="text-sm">{{ field.value === 'true' ? 'Oui' : 'Non' }}</span>
|
|
</div>
|
|
<input
|
|
v-else-if="field.type === 'date'"
|
|
v-model="field.value"
|
|
type="date"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:required="field.required"
|
|
:disabled="saving"
|
|
>
|
|
<input
|
|
v-else
|
|
v-model="field.value"
|
|
type="text"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:required="field.required"
|
|
:disabled="saving"
|
|
>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="space-y-4 rounded-lg border border-base-200 bg-base-200/40 p-4">
|
|
<header class="flex flex-col gap-1 md:flex-row md:items-center md:justify-between">
|
|
<div>
|
|
<h2 class="font-semibold text-base-content">Documents</h2>
|
|
<p class="text-xs text-base-content/70">
|
|
Gérez les documents associés à ce composant.
|
|
</p>
|
|
</div>
|
|
<span v-if="selectedFiles.length" class="badge badge-outline">
|
|
{{ selectedFiles.length }} document{{ selectedFiles.length > 1 ? 's' : '' }} prêt{{ selectedFiles.length > 1 ? 's' : '' }} à être ajouté{{ selectedFiles.length > 1 ? 's' : '' }}
|
|
</span>
|
|
</header>
|
|
<div :class="{ 'pointer-events-none opacity-60': saving || uploadingDocuments }">
|
|
<DocumentUpload
|
|
v-model="selectedFiles"
|
|
title="Déposer vos fichiers"
|
|
subtitle="Formats acceptés : PDF, images, documents…"
|
|
@files-added="handleFilesAdded"
|
|
/>
|
|
</div>
|
|
<p v-if="uploadingDocuments" class="text-xs text-base-content/70">
|
|
Téléversement des documents en cours…
|
|
</p>
|
|
<p v-else-if="loadingDocuments" class="text-xs text-base-content/70">
|
|
Chargement des documents en cours…
|
|
</p>
|
|
<div v-else-if="componentDocuments.length" class="space-y-2">
|
|
<div
|
|
v-for="document in componentDocuments"
|
|
:key="document.id || document.path || document.name"
|
|
class="flex items-center justify-between rounded border border-base-200 bg-base-100 px-3 py-2"
|
|
>
|
|
<div class="flex items-center gap-3 text-sm">
|
|
<div
|
|
class="flex-shrink-0 overflow-hidden rounded-md border border-base-200 bg-base-200/70 flex items-center justify-center"
|
|
:class="documentThumbnailClass(document)"
|
|
>
|
|
<img
|
|
v-if="isImageDocument(document) && document.path"
|
|
:src="document.path"
|
|
class="h-full w-full object-cover"
|
|
:alt="`Aperçu de ${document.name}`"
|
|
>
|
|
<iframe
|
|
v-else-if="shouldInlinePdf(document)"
|
|
:src="documentPreviewSrc(document)"
|
|
class="h-full w-full border-0 bg-white"
|
|
title="Aperçu PDF"
|
|
/>
|
|
<component
|
|
v-else
|
|
:is="documentIcon(document).component"
|
|
class="h-6 w-6"
|
|
:class="documentIcon(document).colorClass"
|
|
aria-hidden="true"
|
|
/>
|
|
</div>
|
|
<div>
|
|
<div class="font-medium">
|
|
{{ document.name }}
|
|
</div>
|
|
<div class="text-xs text-base-content/70">
|
|
{{ document.mimeType || 'Inconnu' }} • {{ formatSize(document.size) }}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="flex items-center gap-2">
|
|
<button
|
|
type="button"
|
|
class="btn btn-ghost btn-xs"
|
|
:disabled="!canPreviewDocument(document)"
|
|
:title="canPreviewDocument(document) ? 'Consulter le document' : 'Aucun aperçu disponible pour ce type'"
|
|
@click="openPreview(document)"
|
|
>
|
|
Consulter
|
|
</button>
|
|
<button
|
|
type="button"
|
|
class="btn btn-ghost btn-xs"
|
|
@click="downloadDocument(document)"
|
|
>
|
|
Télécharger
|
|
</button>
|
|
<button
|
|
type="button"
|
|
class="btn btn-error btn-xs"
|
|
:disabled="uploadingDocuments"
|
|
@click="removeDocument(document.id)"
|
|
>
|
|
Supprimer
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<p v-else class="text-xs text-base-content/70">
|
|
Aucun document n'est associé à ce composant pour le moment.
|
|
</p>
|
|
</div>
|
|
|
|
<div class="flex flex-col gap-3 md:flex-row md:justify-end">
|
|
<NuxtLink to="/component-catalog" class="btn btn-ghost" :class="{ 'btn-disabled': saving }">
|
|
Annuler
|
|
</NuxtLink>
|
|
<button type="button" class="btn btn-primary" :disabled="!canSubmit" @click="submitEdition">
|
|
<span v-if="saving" class="loading loading-spinner loading-sm mr-2" />
|
|
Enregistrer les modifications
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
</main>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { computed, onMounted, reactive, ref, watch } from 'vue'
|
|
import { useRoute, useRouter } from '#imports'
|
|
import ConstructeurSelect from '~/components/ConstructeurSelect.vue'
|
|
import DocumentUpload from '~/components/DocumentUpload.vue'
|
|
import DocumentPreviewModal from '~/components/DocumentPreviewModal.vue'
|
|
import { useComponentTypes } from '~/composables/useComponentTypes'
|
|
import { useComposants } from '~/composables/useComposants'
|
|
import { useCustomFields } from '~/composables/useCustomFields'
|
|
import { useApi } from '~/composables/useApi'
|
|
import { useToast } from '~/composables/useToast'
|
|
import { useDocuments } from '~/composables/useDocuments'
|
|
import { formatStructurePreview, normalizeStructureForEditor } from '~/shared/modelUtils'
|
|
import { uniqueConstructeurIds } from '~/shared/constructeurUtils'
|
|
import type { ComponentModelStructure } from '~/shared/types/inventory'
|
|
import type { ModelType } from '~/services/modelTypes'
|
|
import { getFileIcon } from '~/utils/fileIcons'
|
|
import { canPreviewDocument, isImageDocument, isPdfDocument } from '~/utils/documentPreview'
|
|
|
|
interface ComponentCatalogType extends ModelType {
|
|
structure: ComponentModelStructure | null
|
|
customFields?: Array<Record<string, any>>
|
|
}
|
|
|
|
interface CustomFieldInput {
|
|
id: string | null
|
|
name: string
|
|
type: string
|
|
required: boolean
|
|
options: string[]
|
|
value: string
|
|
customFieldId: string | null
|
|
customFieldValueId: string | null
|
|
}
|
|
|
|
const route = useRoute()
|
|
const router = useRouter()
|
|
const { get } = useApi()
|
|
const { componentTypes, loadComponentTypes } = useComponentTypes()
|
|
const { updateComposant } = useComposants()
|
|
const { upsertCustomFieldValue, updateCustomFieldValue } = useCustomFields()
|
|
const toast = useToast()
|
|
const { loadDocumentsByComponent, uploadDocuments, deleteDocument } = useDocuments()
|
|
|
|
const component = ref<any | null>(null)
|
|
const loading = ref(true)
|
|
const saving = ref(false)
|
|
const selectedFiles = ref<File[]>([])
|
|
const uploadingDocuments = ref(false)
|
|
const loadingDocuments = ref(false)
|
|
const componentDocuments = ref<any[]>([])
|
|
const previewDocument = ref<any | null>(null)
|
|
const previewVisible = ref(false)
|
|
|
|
const selectedTypeId = ref<string>('')
|
|
const editionForm = reactive({
|
|
name: '' as string,
|
|
reference: '' as string,
|
|
constructeurIds: [] as string[],
|
|
prix: '' as string,
|
|
})
|
|
|
|
const customFieldInputs = ref<CustomFieldInput[]>([])
|
|
const documentIcon = (doc: any) =>
|
|
getFileIcon({ name: doc?.filename || doc?.name, mime: doc?.mimeType })
|
|
const formatSize = (size: number | null | undefined) => {
|
|
if (size === null || size === undefined) {
|
|
return '—'
|
|
}
|
|
if (size === 0) {
|
|
return '0 B'
|
|
}
|
|
const units = ['B', 'KB', 'MB', 'GB']
|
|
const index = Math.min(units.length - 1, Math.floor(Math.log(size) / Math.log(1024)))
|
|
const formatted = size / Math.pow(1024, index)
|
|
return `${formatted.toFixed(1)} ${units[index]}`
|
|
}
|
|
const PDF_PREVIEW_MAX_BYTES = 5 * 1024 * 1024
|
|
const shouldInlinePdf = (document: any) => {
|
|
if (!document || !isPdfDocument(document) || !document.path) {
|
|
return false
|
|
}
|
|
if (typeof document.size === 'number' && document.size > PDF_PREVIEW_MAX_BYTES) {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
const appendPdfViewerParams = (src: string) => {
|
|
if (!src || src.startsWith('data:')) {
|
|
return src || ''
|
|
}
|
|
if (src.includes('#')) {
|
|
return `${src}&toolbar=0&navpanes=0`
|
|
}
|
|
return `${src}#toolbar=0&navpanes=0`
|
|
}
|
|
const documentPreviewSrc = (document: any) => {
|
|
if (!document?.path) {
|
|
return ''
|
|
}
|
|
if (isPdfDocument(document)) {
|
|
return appendPdfViewerParams(document.path)
|
|
}
|
|
return document.path
|
|
}
|
|
const documentThumbnailClass = (document: any) => {
|
|
if (shouldInlinePdf(document) || (isImageDocument(document) && document?.path)) {
|
|
return 'h-24 w-20'
|
|
}
|
|
return 'h-16 w-16'
|
|
}
|
|
const openPreview = (doc: any) => {
|
|
if (!doc || !canPreviewDocument(doc)) {
|
|
return
|
|
}
|
|
previewDocument.value = doc
|
|
previewVisible.value = true
|
|
}
|
|
const closePreview = () => {
|
|
previewVisible.value = false
|
|
previewDocument.value = null
|
|
}
|
|
const downloadDocument = (doc: any) => {
|
|
if (!doc?.path) {
|
|
return
|
|
}
|
|
const target = String(doc.path)
|
|
if (target.startsWith('data:')) {
|
|
const link = document.createElement('a')
|
|
link.href = target
|
|
link.download = doc.filename || doc.name || 'document'
|
|
link.click()
|
|
return
|
|
}
|
|
window.open(target, '_blank')
|
|
}
|
|
const removeDocument = async (documentId: string | number | null | undefined) => {
|
|
if (!documentId) {
|
|
return
|
|
}
|
|
const result = await deleteDocument(documentId, { updateStore: false })
|
|
if (result.success) {
|
|
componentDocuments.value = componentDocuments.value.filter((doc) => doc.id !== documentId)
|
|
}
|
|
}
|
|
const handleFilesAdded = async (files: File[]) => {
|
|
if (!files?.length || !component.value?.id) {
|
|
return
|
|
}
|
|
uploadingDocuments.value = true
|
|
try {
|
|
const result = await uploadDocuments(
|
|
{
|
|
files,
|
|
context: { composantId: component.value.id },
|
|
},
|
|
{ updateStore: false },
|
|
)
|
|
if (result.success) {
|
|
selectedFiles.value = []
|
|
await refreshDocuments()
|
|
}
|
|
} finally {
|
|
uploadingDocuments.value = false
|
|
}
|
|
}
|
|
const refreshDocuments = async () => {
|
|
if (!component.value?.id) {
|
|
componentDocuments.value = []
|
|
return
|
|
}
|
|
loadingDocuments.value = true
|
|
try {
|
|
const result = await loadDocumentsByComponent(component.value.id, { updateStore: false })
|
|
if (result.success) {
|
|
componentDocuments.value = result.data || []
|
|
}
|
|
} finally {
|
|
loadingDocuments.value = false
|
|
}
|
|
}
|
|
|
|
const componentTypeList = computed<ComponentCatalogType[]>(() =>
|
|
(componentTypes.value || [])
|
|
.filter((item: any) => item?.category === 'COMPONENT') as ComponentCatalogType[],
|
|
)
|
|
|
|
const selectedType = computed(() => {
|
|
if (!selectedTypeId.value) {
|
|
return null
|
|
}
|
|
return componentTypeList.value.find((type) => type.id === selectedTypeId.value) ?? null
|
|
})
|
|
|
|
const selectedTypeStructure = computed<ComponentModelStructure | null>(() => {
|
|
const structure = selectedType.value?.structure ?? null
|
|
return structure ? normalizeStructureForEditor(structure) : null
|
|
})
|
|
|
|
const requiredCustomFieldsFilled = computed(() =>
|
|
customFieldInputs.value.every((field) => {
|
|
if (!field.required) {
|
|
return true
|
|
}
|
|
if (field.type === 'boolean') {
|
|
return field.value === 'true' || field.value === 'false'
|
|
}
|
|
return toFieldString(field.value).trim() !== ''
|
|
}),
|
|
)
|
|
|
|
const canSubmit = computed(() => Boolean(
|
|
component.value &&
|
|
editionForm.name &&
|
|
requiredCustomFieldsFilled.value &&
|
|
!saving.value,
|
|
))
|
|
|
|
const toFieldString = (value: unknown): string => {
|
|
if (value === null || value === undefined) {
|
|
return ''
|
|
}
|
|
if (typeof value === 'string') {
|
|
return value
|
|
}
|
|
if (typeof value === 'number' || typeof value === 'boolean') {
|
|
return String(value)
|
|
}
|
|
return ''
|
|
}
|
|
|
|
const fetchComponent = async () => {
|
|
const id = route.params.id
|
|
if (!id || typeof id !== 'string') {
|
|
component.value = null
|
|
componentDocuments.value = []
|
|
return
|
|
}
|
|
const result = await get(`/composants/${id}`)
|
|
if (result.success) {
|
|
component.value = result.data
|
|
componentDocuments.value = Array.isArray(result.data?.documents) ? result.data.documents : []
|
|
} else {
|
|
component.value = null
|
|
componentDocuments.value = []
|
|
}
|
|
}
|
|
|
|
let initialized = false
|
|
|
|
watch(
|
|
[component, selectedTypeStructure],
|
|
([currentComponent, currentStructure]) => {
|
|
if (!currentComponent || initialized) {
|
|
return
|
|
}
|
|
|
|
selectedTypeId.value = currentComponent.typeComposantId || ''
|
|
|
|
editionForm.name = currentComponent.name || ''
|
|
editionForm.reference = currentComponent.reference || ''
|
|
editionForm.constructeurIds = uniqueConstructeurIds(
|
|
currentComponent,
|
|
Array.isArray(currentComponent.constructeurs) ? currentComponent.constructeurs : [],
|
|
currentComponent.constructeur ? [currentComponent.constructeur] : [],
|
|
)
|
|
editionForm.prix = currentComponent.prix !== null && currentComponent.prix !== undefined ? String(currentComponent.prix) : ''
|
|
|
|
customFieldInputs.value = buildCustomFieldInputs(
|
|
currentStructure,
|
|
currentComponent.customFieldValues,
|
|
)
|
|
|
|
initialized = true
|
|
},
|
|
{ immediate: true },
|
|
)
|
|
|
|
watch(selectedTypeStructure, (currentStructure) => {
|
|
if (!component.value) {
|
|
return
|
|
}
|
|
customFieldInputs.value = buildCustomFieldInputs(
|
|
currentStructure,
|
|
component.value.customFieldValues,
|
|
)
|
|
})
|
|
|
|
const submitEdition = async () => {
|
|
if (!component.value) {
|
|
return
|
|
}
|
|
|
|
const rawPrice = typeof editionForm.prix === 'string'
|
|
? editionForm.prix.trim()
|
|
: editionForm.prix === null || editionForm.prix === undefined
|
|
? ''
|
|
: String(editionForm.prix).trim()
|
|
|
|
const payload: Record<string, any> = {
|
|
name: editionForm.name.trim(),
|
|
}
|
|
|
|
const reference = editionForm.reference.trim()
|
|
payload.reference = reference ? reference : null
|
|
payload.constructeurIds = uniqueConstructeurIds(editionForm.constructeurIds)
|
|
|
|
if (rawPrice) {
|
|
const parsed = Number(rawPrice)
|
|
if (!Number.isNaN(parsed)) {
|
|
payload.prix = parsed
|
|
}
|
|
} else {
|
|
payload.prix = null
|
|
}
|
|
|
|
saving.value = true
|
|
try {
|
|
const result = await updateComposant(component.value.id, payload)
|
|
if (result.success) {
|
|
const updatedComponent = result.data
|
|
await saveCustomFieldValues(updatedComponent)
|
|
await router.push('/component-catalog')
|
|
}
|
|
} catch (error: any) {
|
|
toast.showError(error?.message || 'Erreur lors de la mise à jour du composant')
|
|
} finally {
|
|
saving.value = false
|
|
}
|
|
}
|
|
|
|
const buildCustomFieldInputs = (
|
|
structure: ComponentModelStructure | null,
|
|
values: any[] | null,
|
|
): CustomFieldInput[] => {
|
|
const normalizedStructure = structure ? normalizeStructureForEditor(structure) : null
|
|
const definitions = normalizeCustomFieldInputs(normalizedStructure)
|
|
const valueList = Array.isArray(values) ? values : []
|
|
|
|
const mapById = new Map<string, any>()
|
|
const mapByName = new Map<string, any>()
|
|
|
|
valueList.forEach((entry) => {
|
|
if (!entry || typeof entry !== 'object') {
|
|
return
|
|
}
|
|
const fieldId = entry.customField?.id || entry.customFieldId || null
|
|
if (fieldId) {
|
|
mapById.set(fieldId, entry)
|
|
}
|
|
const fieldName = entry.customField?.name || entry.name || entry.key || null
|
|
if (fieldName) {
|
|
mapByName.set(fieldName, entry)
|
|
}
|
|
})
|
|
|
|
const resolved: CustomFieldInput[] = definitions.map((definition) => {
|
|
const definitionId = definition.customFieldId || definition.id || null
|
|
const matched = (definitionId ? mapById.get(definitionId) : null) || mapByName.get(definition.name)
|
|
|
|
if (!matched) {
|
|
return {
|
|
...definition,
|
|
customFieldId: definition.customFieldId || definition.id,
|
|
customFieldValueId: null,
|
|
}
|
|
}
|
|
|
|
const resolvedValue = extractStoredCustomFieldValue(matched)
|
|
return {
|
|
...definition,
|
|
customFieldId: matched.customField?.id || definition.customFieldId || definition.id,
|
|
customFieldValueId: matched.id ?? null,
|
|
value: formatDefaultValue(definition.type, resolvedValue),
|
|
}
|
|
})
|
|
|
|
return resolved
|
|
}
|
|
|
|
const fieldKey = (field: CustomFieldInput, index: number) =>
|
|
field.customFieldValueId || field.id || `${field.name}-${index}`
|
|
|
|
const normalizeCustomFieldInputs = (structure: ComponentModelStructure | null): CustomFieldInput[] => {
|
|
if (!structure || typeof structure !== 'object') {
|
|
return []
|
|
}
|
|
const fields = Array.isArray(structure.customFields) ? structure.customFields : []
|
|
return fields
|
|
.map((field) => normalizeCustomField(field))
|
|
.filter((field): field is CustomFieldInput => field !== null)
|
|
}
|
|
|
|
const normalizeCustomField = (rawField: any): CustomFieldInput | null => {
|
|
if (!rawField || typeof rawField !== 'object') {
|
|
return null
|
|
}
|
|
const name = resolveFieldName(rawField)
|
|
if (!name) {
|
|
return null
|
|
}
|
|
const type = resolveFieldType(rawField)
|
|
const required = resolveRequiredFlag(rawField)
|
|
const options = resolveOptions(rawField)
|
|
const defaultSource = resolveDefaultValue(rawField)
|
|
const value = formatDefaultValue(type, defaultSource)
|
|
const id = typeof rawField.id === 'string' ? rawField.id : null
|
|
const customFieldId = typeof rawField.customFieldId === 'string' ? rawField.customFieldId : id
|
|
const customFieldValueId = typeof rawField.customFieldValueId === 'string'
|
|
? rawField.customFieldValueId
|
|
: null
|
|
return { id, name, type, required, options, value, customFieldId, customFieldValueId }
|
|
}
|
|
|
|
const resolveFieldName = (field: any): string => {
|
|
if (typeof field?.name === 'string' && field.name.trim()) {
|
|
return field.name.trim()
|
|
}
|
|
if (typeof field?.key === 'string' && field.key.trim()) {
|
|
return field.key.trim()
|
|
}
|
|
if (typeof field?.label === 'string' && field.label.trim()) {
|
|
return field.label.trim()
|
|
}
|
|
return ''
|
|
}
|
|
|
|
const resolveFieldType = (field: any): string => {
|
|
const allowed = ['text', 'number', 'select', 'boolean', 'date']
|
|
const rawType =
|
|
typeof field?.type === 'string'
|
|
? field.type
|
|
: typeof field?.value?.type === 'string'
|
|
? field.value.type
|
|
: ''
|
|
const value = rawType.toLowerCase()
|
|
return allowed.includes(value) ? value : 'text'
|
|
}
|
|
|
|
const resolveDefaultValue = (field: any): any => {
|
|
if (!field || typeof field !== 'object') {
|
|
return null
|
|
}
|
|
if (field.defaultValue !== undefined && field.defaultValue !== null) {
|
|
return field.defaultValue
|
|
}
|
|
if (field.value !== undefined && field.value !== null && typeof field.value !== 'object') {
|
|
return field.value
|
|
}
|
|
if (field.default !== undefined && field.default !== null) {
|
|
return field.default
|
|
}
|
|
if (field.value && typeof field.value === 'object') {
|
|
if ((field.value as any).defaultValue !== undefined && (field.value as any).defaultValue !== null) {
|
|
return (field.value as any).defaultValue
|
|
}
|
|
if ((field.value as any).value !== undefined && (field.value as any).value !== null && typeof (field.value as any).value !== 'object') {
|
|
return (field.value as any).value
|
|
}
|
|
}
|
|
return null
|
|
}
|
|
|
|
const formatDefaultValue = (type: string, defaultValue: any): string => {
|
|
if (defaultValue === null || defaultValue === undefined) {
|
|
return ''
|
|
}
|
|
if (typeof defaultValue === 'object') {
|
|
if (defaultValue === null) {
|
|
return ''
|
|
}
|
|
if ('defaultValue' in (defaultValue as Record<string, any>)) {
|
|
return formatDefaultValue(type, (defaultValue as Record<string, any>).defaultValue)
|
|
}
|
|
if ('value' in (defaultValue as Record<string, any>)) {
|
|
return formatDefaultValue(type, (defaultValue as Record<string, any>).value)
|
|
}
|
|
return ''
|
|
}
|
|
if (type === 'boolean') {
|
|
const normalized = String(defaultValue).toLowerCase()
|
|
if (normalized === 'true' || normalized === '1') {
|
|
return 'true'
|
|
}
|
|
if (normalized === 'false' || normalized === '0') {
|
|
return 'false'
|
|
}
|
|
return ''
|
|
}
|
|
return String(defaultValue)
|
|
}
|
|
|
|
const resolveRequiredFlag = (field: any): boolean => {
|
|
if (typeof field?.required === 'boolean') {
|
|
return field.required
|
|
}
|
|
const nestedRequired = field?.value?.required
|
|
if (typeof nestedRequired === 'boolean') {
|
|
return nestedRequired
|
|
}
|
|
if (typeof nestedRequired === 'string') {
|
|
const normalized = nestedRequired.toLowerCase()
|
|
return normalized === 'true' || normalized === '1'
|
|
}
|
|
return false
|
|
}
|
|
|
|
const resolveOptions = (field: any): string[] => {
|
|
const sources = [field?.options, field?.value?.options, field?.value?.choices]
|
|
for (const source of sources) {
|
|
if (Array.isArray(source)) {
|
|
const mapped = source
|
|
.map((option: unknown) => {
|
|
if (option === null || option === undefined) {
|
|
return ''
|
|
}
|
|
if (typeof option === 'string') {
|
|
return option.trim()
|
|
}
|
|
if (typeof option === 'object') {
|
|
const record = option || {}
|
|
const keys = ['value', 'label', 'name']
|
|
for (const key of keys) {
|
|
const candidate = record[key]
|
|
if (typeof candidate === 'string' && candidate.trim().length > 0) {
|
|
return candidate.trim()
|
|
}
|
|
}
|
|
}
|
|
const fallback = String(option).trim()
|
|
return fallback === '[object Object]' ? '' : fallback
|
|
})
|
|
.filter((option) => option.length > 0)
|
|
if (mapped.length) {
|
|
return mapped
|
|
}
|
|
}
|
|
}
|
|
return []
|
|
}
|
|
|
|
const extractStoredCustomFieldValue = (entry: any): any => {
|
|
if (entry === null || entry === undefined) {
|
|
return ''
|
|
}
|
|
if (typeof entry === 'string' || typeof entry === 'number' || typeof entry === 'boolean') {
|
|
return entry
|
|
}
|
|
if (typeof entry !== 'object') {
|
|
return String(entry)
|
|
}
|
|
const direct = entry.value
|
|
if (direct !== undefined && direct !== null) {
|
|
if (typeof direct === 'object') {
|
|
if (direct === null) {
|
|
return ''
|
|
}
|
|
if ('value' in direct && direct.value !== undefined && direct.value !== null) {
|
|
return direct.value
|
|
}
|
|
if ('defaultValue' in direct && direct.defaultValue !== undefined && direct.defaultValue !== null) {
|
|
return direct.defaultValue
|
|
}
|
|
return ''
|
|
}
|
|
return direct
|
|
}
|
|
if (entry.defaultValue !== undefined && entry.defaultValue !== null) {
|
|
return entry.defaultValue
|
|
}
|
|
if (entry.customFieldValue?.value !== undefined && entry.customFieldValue?.value !== null) {
|
|
return entry.customFieldValue.value
|
|
}
|
|
return ''
|
|
}
|
|
|
|
const getStructureCustomFields = (structure: ComponentModelStructure | null) => {
|
|
return Array.isArray(structure?.customFields) ? structure.customFields : []
|
|
}
|
|
|
|
const getStructurePieces = (structure: ComponentModelStructure | null) => {
|
|
return Array.isArray(structure?.pieces) ? structure.pieces : []
|
|
}
|
|
|
|
const getStructureSubcomponents = (structure: ComponentModelStructure | null) => {
|
|
if (Array.isArray(structure?.subcomponents)) {
|
|
return structure.subcomponents
|
|
}
|
|
const legacy = (structure as any)?.subComponents
|
|
return Array.isArray(legacy) ? legacy : []
|
|
}
|
|
|
|
const resolvePieceLabel = (piece: Record<string, any>) => {
|
|
const parts: string[] = []
|
|
if (piece.role) {
|
|
parts.push(piece.role)
|
|
}
|
|
if (piece.typePiece?.name) {
|
|
parts.push(piece.typePiece.name)
|
|
} else if (piece.typePieceLabel) {
|
|
parts.push(piece.typePieceLabel)
|
|
} else if (piece.typePiece?.code) {
|
|
parts.push(`Famille ${piece.typePiece.code}`)
|
|
} else if (piece.familyCode) {
|
|
parts.push(`Famille ${piece.familyCode}`)
|
|
} else if (piece.typePieceId) {
|
|
parts.push(`#${piece.typePieceId}`)
|
|
}
|
|
return parts.length ? parts.join(' • ') : 'Pièce'
|
|
}
|
|
|
|
const resolveSubcomponentLabel = (node: Record<string, any>) => {
|
|
const parts: string[] = []
|
|
if (node.alias) {
|
|
parts.push(node.alias)
|
|
}
|
|
if (node.typeComposant?.name) {
|
|
parts.push(node.typeComposant.name)
|
|
} else if (node.typeComposantLabel) {
|
|
parts.push(node.typeComposantLabel)
|
|
} else if (node.familyCode) {
|
|
parts.push(node.familyCode)
|
|
} else if (node.typeComposantId) {
|
|
parts.push(`#${node.typeComposantId}`)
|
|
}
|
|
|
|
const childCount = Array.isArray(node.subcomponents)
|
|
? node.subcomponents.length
|
|
: Array.isArray(node.subComponents)
|
|
? node.subComponents.length
|
|
: 0
|
|
if (childCount) {
|
|
parts.push(`${childCount} sous-composant(s)`)
|
|
}
|
|
return parts.length ? parts.join(' • ') : 'Sous-composant'
|
|
}
|
|
|
|
const buildCustomFieldMetadata = (field: CustomFieldInput) => ({
|
|
customFieldName: field.name,
|
|
customFieldType: field.type,
|
|
customFieldRequired: field.required,
|
|
customFieldOptions: field.options,
|
|
})
|
|
|
|
const shouldPersistField = (field: CustomFieldInput) => {
|
|
if (field.type === 'boolean') {
|
|
return field.value === 'true' || field.value === 'false'
|
|
}
|
|
return toFieldString(field.value).trim() !== ''
|
|
}
|
|
|
|
const formatValueForPersistence = (field: CustomFieldInput) => {
|
|
if (field.type === 'boolean') {
|
|
return field.value === 'true' ? 'true' : 'false'
|
|
}
|
|
return toFieldString(field.value).trim()
|
|
}
|
|
|
|
const saveCustomFieldValues = async (updatedComponent: any) => {
|
|
if (!updatedComponent || !updatedComponent.id) {
|
|
return
|
|
}
|
|
|
|
const definitionMap = new Map<string, string>()
|
|
const registerDefinitions = (fields: any[]) => {
|
|
if (!Array.isArray(fields)) {
|
|
return
|
|
}
|
|
fields.forEach((field) => {
|
|
if (!field || typeof field !== 'object') {
|
|
return
|
|
}
|
|
const name = typeof field.name === 'string' ? field.name : null
|
|
const id = typeof field.id === 'string' ? field.id : null
|
|
if (name && id && !definitionMap.has(name)) {
|
|
definitionMap.set(name, id)
|
|
}
|
|
})
|
|
}
|
|
|
|
registerDefinitions(updatedComponent?.typeComposant?.customFields)
|
|
registerDefinitions(updatedComponent?.typeMachineComponentRequirement?.typeComposant?.customFields)
|
|
|
|
const resolveDefinitionId = (field: CustomFieldInput) => {
|
|
if (field.customFieldId) {
|
|
return field.customFieldId
|
|
}
|
|
if (field.id) {
|
|
return field.id
|
|
}
|
|
return definitionMap.get(field.name) ?? null
|
|
}
|
|
|
|
for (const field of customFieldInputs.value) {
|
|
if (!shouldPersistField(field)) {
|
|
continue
|
|
}
|
|
|
|
const definitionId = resolveDefinitionId(field)
|
|
const metadata = definitionId ? undefined : buildCustomFieldMetadata(field)
|
|
const value = formatValueForPersistence(field)
|
|
|
|
if (field.customFieldValueId) {
|
|
const result = await updateCustomFieldValue(field.customFieldValueId, { value })
|
|
if (!result.success) {
|
|
toast.showError(`Impossible de mettre à jour le champ personnalisé "${field.name}"`)
|
|
} else if (definitionId && !field.customFieldId) {
|
|
field.customFieldId = definitionId
|
|
}
|
|
continue
|
|
}
|
|
|
|
const result = await upsertCustomFieldValue(
|
|
definitionId,
|
|
'composant',
|
|
updatedComponent.id,
|
|
value,
|
|
metadata,
|
|
)
|
|
|
|
if (!result.success) {
|
|
toast.showError(`Impossible d'enregistrer le champ personnalisé "${field.name}"`)
|
|
} else {
|
|
const createdValue = result.data
|
|
if (createdValue?.id) {
|
|
field.customFieldValueId = createdValue.id
|
|
}
|
|
const resolvedId = createdValue?.customField?.id || definitionId
|
|
if (resolvedId) {
|
|
field.customFieldId = resolvedId
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
onMounted(async () => {
|
|
await Promise.allSettled([loadComponentTypes(), fetchComponent()])
|
|
loading.value = false
|
|
if (component.value?.id) {
|
|
await refreshDocuments()
|
|
}
|
|
})
|
|
</script>
|