930 lines
30 KiB
Vue
930 lines
30 KiB
Vue
<template>
|
|
<main class="mx-auto flex w-full max-w-5xl flex-col gap-8 px-4 py-8 sm:px-6 lg:px-8">
|
|
<header class="flex flex-col gap-4 md:flex-row md:items-center md:justify-between">
|
|
<div class="space-y-1">
|
|
<h1 class="text-3xl font-semibold text-base-content">Nouvel composant</h1>
|
|
<p class="text-sm text-base-content/70">
|
|
Sélectionnez la catégorie cible puis complétez les informations du composant.
|
|
</p>
|
|
</div>
|
|
<NuxtLink to="/component-catalog" class="btn btn-ghost btn-sm md:btn-md self-start">
|
|
Retour au catalogue
|
|
</NuxtLink>
|
|
</header>
|
|
|
|
<section class="card border border-base-200 bg-base-100 shadow-sm">
|
|
<div class="card-body space-y-6">
|
|
<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="loadingTypes || submitting"
|
|
required
|
|
>
|
|
<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 v-if="loadingTypes" class="text-xs text-gray-500 mt-1">
|
|
Chargement des catégories…
|
|
</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="creationForm.name"
|
|
type="text"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:disabled="submitting || !selectedType"
|
|
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="creationForm.reference"
|
|
type="text"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:disabled="submitting || !selectedType"
|
|
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="creationForm.constructeurId"
|
|
class="w-full"
|
|
:disabled="submitting || !selectedType"
|
|
placeholder="Rechercher un constructeur..."
|
|
/>
|
|
</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="creationForm.prix"
|
|
type="number"
|
|
step="0.01"
|
|
min="0"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:disabled="submitting || !selectedType"
|
|
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(selectedType.structure) }}</span>
|
|
</div>
|
|
|
|
<details v-if="selectedType.structure" 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(selectedType.structure).length" class="space-y-2">
|
|
<h3 class="font-semibold text-sm text-base-content">Champs personnalisés</h3>
|
|
<ul class="list-disc list-inside space-y-1">
|
|
<li
|
|
v-for="field in getStructureCustomFields(selectedType.structure)"
|
|
:key="field.key || field.name"
|
|
>
|
|
<span class="font-medium">{{ field.key || field.name }}</span>
|
|
<span v-if="field.value !== undefined && field.value !== null"> : {{ field.value }}</span>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
|
|
<div v-if="getStructurePieces(selectedType.structure).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(selectedType.structure)"
|
|
:key="piece.role || piece.typePieceId || piece.familyCode || index"
|
|
>
|
|
{{ resolvePieceLabel(piece) }}
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
|
|
<div v-if="getStructureSubcomponents(selectedType.structure).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(selectedType.structure)"
|
|
:key="subcomponent.alias || subcomponent.typeComposantId || subcomponent.familyCode || index"
|
|
>
|
|
{{ resolveSubcomponentLabel(subcomponent) }}
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
</details>
|
|
</div>
|
|
|
|
<div
|
|
v-if="structureHasRequirements"
|
|
class="space-y-4 rounded-lg border border-primary/30 bg-primary/5 p-4"
|
|
>
|
|
<div class="flex items-start justify-between gap-4">
|
|
<div>
|
|
<h2 class="font-semibold text-base-content">
|
|
Sélection des éléments du squelette
|
|
</h2>
|
|
<p class="text-xs text-base-content/70">
|
|
Affectez les pièces et sous-composants concrets correspondant à la catégorie choisie.
|
|
</p>
|
|
</div>
|
|
<span
|
|
class="badge"
|
|
:class="structureSelectionsComplete ? 'badge-success' : structureDataLoading ? 'badge-info' : 'badge-warning'"
|
|
>
|
|
{{ structureSelectionsComplete ? 'Complet' : structureDataLoading ? 'Chargement…' : 'Incomplet' }}
|
|
</span>
|
|
</div>
|
|
|
|
<div
|
|
v-if="structureDataLoading"
|
|
class="flex items-center gap-3 rounded-md border border-base-200 bg-base-100 p-3 text-sm text-base-content/70"
|
|
>
|
|
<span class="loading loading-spinner loading-sm" aria-hidden="true"></span>
|
|
Chargement du catalogue de pièces et de composants…
|
|
</div>
|
|
<ComponentStructureAssignmentNode
|
|
v-else-if="structureAssignments"
|
|
:assignment="structureAssignments"
|
|
:pieces="availablePieces"
|
|
:components="availableComponents"
|
|
/>
|
|
<p v-else class="text-xs text-error">
|
|
Impossible de générer les emplacements définis par le squelette.
|
|
</p>
|
|
</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">
|
|
Renseignez les valeurs propres à ce composant selon le squelette choisi.
|
|
</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="submitting"
|
|
>
|
|
<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="submitting"
|
|
>
|
|
<select
|
|
v-else-if="field.type === 'select'"
|
|
v-model="field.value"
|
|
class="select select-bordered select-sm md:select-md"
|
|
:required="field.required"
|
|
:disabled="submitting"
|
|
>
|
|
<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="submitting"
|
|
>
|
|
<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="submitting"
|
|
>
|
|
<input
|
|
v-else
|
|
v-model="field.value"
|
|
type="text"
|
|
class="input input-bordered input-sm md:input-md"
|
|
:required="field.required"
|
|
:disabled="submitting"
|
|
>
|
|
</div>
|
|
</div>
|
|
</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': submitting }">
|
|
Annuler
|
|
</NuxtLink>
|
|
<button type="button" class="btn btn-primary" :disabled="!canSubmit" @click="submitCreation">
|
|
<span v-if="submitting" class="loading loading-spinner loading-sm mr-2"></span>
|
|
Créer le composant
|
|
</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 ComponentStructureAssignmentNode, {
|
|
type StructureAssignmentNode,
|
|
} from '~/components/ComponentStructureAssignmentNode.vue'
|
|
import { useComponentTypes } from '~/composables/useComponentTypes'
|
|
import { useComposants } from '~/composables/useComposants'
|
|
import { usePieces } from '~/composables/usePieces'
|
|
import { useToast } from '~/composables/useToast'
|
|
import { useCustomFields } from '~/composables/useCustomFields'
|
|
import { formatStructurePreview } from '~/shared/modelUtils'
|
|
import type {
|
|
ComponentModelPiece,
|
|
ComponentModelStructure,
|
|
ComponentModelStructureNode,
|
|
} from '~/shared/types/inventory'
|
|
import type { ModelType } from '~/services/modelTypes'
|
|
|
|
interface ComponentCatalogType extends ModelType {
|
|
structure: ComponentModelStructure | null
|
|
customFields?: Array<Record<string, any>>
|
|
}
|
|
|
|
const route = useRoute()
|
|
const router = useRouter()
|
|
|
|
const { componentTypes, loadComponentTypes, loadingComponentTypes } = useComponentTypes()
|
|
const {
|
|
createComposant,
|
|
composants: componentCatalogRef,
|
|
loadComposants,
|
|
loading: componentsLoading,
|
|
} = useComposants()
|
|
const {
|
|
pieces: pieceCatalogRef,
|
|
loadPieces,
|
|
loading: piecesLoading,
|
|
} = usePieces()
|
|
const toast = useToast()
|
|
const { upsertCustomFieldValue, updateCustomFieldValue } = useCustomFields()
|
|
|
|
const initialTypeId = ref<string>(typeof route.query.typeId === 'string' ? route.query.typeId : '')
|
|
const selectedTypeId = ref<string>(initialTypeId.value)
|
|
const submitting = ref(false)
|
|
const creationForm = reactive({
|
|
name: '' as string,
|
|
reference: '' as string,
|
|
constructeurId: null as string | null,
|
|
prix: '' as string,
|
|
})
|
|
const lastSuggestedName = ref('')
|
|
const customFieldInputs = ref<CustomFieldInput[]>([])
|
|
const structureAssignments = ref<StructureAssignmentNode | null>(null)
|
|
|
|
const availablePieces = computed(() => pieceCatalogRef.value ?? [])
|
|
const availableComponents = computed(() => componentCatalogRef.value ?? [])
|
|
const structureDataLoading = computed(
|
|
() => piecesLoading.value || componentsLoading.value,
|
|
)
|
|
|
|
watch(
|
|
() => route.query.typeId,
|
|
(value) => {
|
|
if (typeof value === 'string') {
|
|
selectedTypeId.value = value
|
|
}
|
|
},
|
|
)
|
|
|
|
watch(selectedTypeId, (id) => {
|
|
const current = typeof route.query.typeId === 'string' ? route.query.typeId : ''
|
|
if ((id || '') === current) {
|
|
return
|
|
}
|
|
const nextQuery = { ...route.query }
|
|
if (id) {
|
|
nextQuery.typeId = id
|
|
} else {
|
|
delete nextQuery.typeId
|
|
}
|
|
router.replace({ path: route.path, query: nextQuery }).catch(() => {})
|
|
})
|
|
|
|
const loadingTypes = computed(() => loadingComponentTypes.value)
|
|
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
|
|
})
|
|
|
|
watch(selectedType, (type) => {
|
|
if (!type) {
|
|
clearCreationForm()
|
|
customFieldInputs.value = []
|
|
structureAssignments.value = null
|
|
return
|
|
}
|
|
if (!creationForm.name || creationForm.name === lastSuggestedName.value) {
|
|
creationForm.name = type.name
|
|
}
|
|
lastSuggestedName.value = creationForm.name
|
|
customFieldInputs.value = normalizeCustomFieldInputs(type.structure)
|
|
structureAssignments.value = initializeStructureAssignments(type.structure)
|
|
})
|
|
|
|
const extractSubcomponents = (
|
|
definition: ComponentModelStructure | ComponentModelStructureNode | null | undefined,
|
|
): ComponentModelStructureNode[] => {
|
|
if (!definition || typeof definition !== 'object') {
|
|
return []
|
|
}
|
|
const raw = Array.isArray((definition as any).subcomponents)
|
|
? (definition as any).subcomponents
|
|
: Array.isArray((definition as any).subComponents)
|
|
? (definition as any).subComponents
|
|
: []
|
|
return raw.filter(
|
|
(item: unknown): item is ComponentModelStructureNode =>
|
|
!!item && typeof item === 'object',
|
|
)
|
|
}
|
|
|
|
const extractPiecesFromNode = (
|
|
definition: ComponentModelStructure | ComponentModelStructureNode | null | undefined,
|
|
): ComponentModelPiece[] => {
|
|
if (!definition || typeof definition !== 'object') {
|
|
return []
|
|
}
|
|
const raw = Array.isArray((definition as any).pieces)
|
|
? (definition as any).pieces
|
|
: []
|
|
return raw.filter(
|
|
(item: unknown): item is ComponentModelPiece =>
|
|
!!item && typeof item === 'object',
|
|
)
|
|
}
|
|
|
|
const buildAssignmentNode = (
|
|
definition: ComponentModelStructureNode | ComponentModelStructure,
|
|
path: string,
|
|
): StructureAssignmentNode => {
|
|
const pieces = extractPiecesFromNode(definition).map((piece, index) => ({
|
|
path: `${path}:piece-${index}`,
|
|
definition: piece,
|
|
selectedPieceId: '',
|
|
}))
|
|
|
|
const subcomponents = extractSubcomponents(definition).map(
|
|
(child, index) => buildAssignmentNode(child, `${path}:sub-${index}`),
|
|
)
|
|
|
|
return {
|
|
path,
|
|
definition,
|
|
selectedComponentId: '',
|
|
pieces,
|
|
subcomponents,
|
|
}
|
|
}
|
|
|
|
const initializeStructureAssignments = (
|
|
structure: ComponentModelStructure | null,
|
|
): StructureAssignmentNode | null => {
|
|
if (!structure || typeof structure !== 'object') {
|
|
return null
|
|
}
|
|
return buildAssignmentNode(structure, 'root')
|
|
}
|
|
|
|
const hasAssignments = (node: StructureAssignmentNode | null): boolean => {
|
|
if (!node) {
|
|
return false
|
|
}
|
|
if (node.pieces.length > 0 || node.subcomponents.length > 0) {
|
|
return true
|
|
}
|
|
return node.subcomponents.some((child) => hasAssignments(child))
|
|
}
|
|
|
|
const structureHasRequirements = computed(() =>
|
|
hasAssignments(structureAssignments.value),
|
|
)
|
|
|
|
const isAssignmentNodeComplete = (
|
|
node: StructureAssignmentNode,
|
|
isRootNode = false,
|
|
): boolean => {
|
|
const piecesComplete = node.pieces.every(
|
|
(piece) => !!piece.selectedPieceId && piece.selectedPieceId.length > 0,
|
|
)
|
|
const subcomponentsComplete = node.subcomponents.every(
|
|
(child) =>
|
|
!!child.selectedComponentId &&
|
|
child.selectedComponentId.length > 0 &&
|
|
isAssignmentNodeComplete(child, false),
|
|
)
|
|
return piecesComplete && subcomponentsComplete && (isRootNode || !!node.selectedComponentId)
|
|
}
|
|
|
|
const structureSelectionsComplete = computed(() => {
|
|
if (!structureHasRequirements.value) {
|
|
return true
|
|
}
|
|
if (structureDataLoading.value) {
|
|
return false
|
|
}
|
|
if (!structureAssignments.value) {
|
|
return false
|
|
}
|
|
return isAssignmentNodeComplete(structureAssignments.value, true)
|
|
})
|
|
|
|
const stripNullish = (input: Record<string, any>) =>
|
|
Object.fromEntries(
|
|
Object.entries(input).filter(
|
|
([, value]) => value !== null && value !== undefined && value !== '',
|
|
),
|
|
)
|
|
|
|
const sanitizeStructureDefinition = (
|
|
definition: ComponentModelStructureNode,
|
|
) =>
|
|
stripNullish({
|
|
alias: definition.alias ?? null,
|
|
typeComposantId: definition.typeComposantId ?? null,
|
|
typeComposantLabel: definition.typeComposantLabel ?? null,
|
|
modelId: definition.modelId ?? null,
|
|
familyCode: (definition as any).familyCode ?? null,
|
|
})
|
|
|
|
const sanitizePieceDefinition = (definition: ComponentModelPiece) =>
|
|
stripNullish({
|
|
role: (definition as any).role ?? null,
|
|
typePieceId: definition.typePieceId ?? null,
|
|
typePieceLabel: definition.typePieceLabel ?? null,
|
|
reference: definition.reference ?? null,
|
|
})
|
|
|
|
const serializeStructureAssignments = (
|
|
root: StructureAssignmentNode | null,
|
|
) => {
|
|
if (!root) {
|
|
return null
|
|
}
|
|
|
|
const serializeNode = (
|
|
assignment: StructureAssignmentNode,
|
|
isRootNode = false,
|
|
): Record<string, any> => {
|
|
const serializedPieces = assignment.pieces
|
|
.filter((piece) => !!piece.selectedPieceId)
|
|
.map((piece) =>
|
|
stripNullish({
|
|
path: piece.path,
|
|
definition: sanitizePieceDefinition(piece.definition),
|
|
selectedPieceId: piece.selectedPieceId,
|
|
}),
|
|
)
|
|
|
|
const serializedSubcomponents = assignment.subcomponents
|
|
.map((child) => serializeNode(child, false))
|
|
.filter((child) => Object.keys(child).length > 0)
|
|
|
|
const base: Record<string, any> = {
|
|
path: assignment.path,
|
|
definition: sanitizeStructureDefinition(assignment.definition),
|
|
}
|
|
|
|
if (!isRootNode) {
|
|
base.selectedComponentId = assignment.selectedComponentId
|
|
}
|
|
if (serializedPieces.length) {
|
|
base.pieces = serializedPieces
|
|
}
|
|
if (serializedSubcomponents.length) {
|
|
base.subcomponents = serializedSubcomponents
|
|
}
|
|
|
|
return stripNullish(base)
|
|
}
|
|
|
|
const serializedRoot = serializeNode(root, true)
|
|
if (
|
|
(!serializedRoot.pieces || serializedRoot.pieces.length === 0) &&
|
|
(!serializedRoot.subcomponents || serializedRoot.subcomponents.length === 0)
|
|
) {
|
|
return null
|
|
}
|
|
return serializedRoot
|
|
}
|
|
|
|
const requiredCustomFieldsFilled = computed(() =>
|
|
customFieldInputs.value.every((field) => {
|
|
if (!field.required) {
|
|
return true
|
|
}
|
|
if (field.type === 'boolean') {
|
|
return field.value === 'true' || field.value === 'false'
|
|
}
|
|
return field.value !== ''
|
|
}),
|
|
)
|
|
|
|
const canSubmit = computed(() => Boolean(
|
|
selectedType.value &&
|
|
creationForm.name &&
|
|
requiredCustomFieldsFilled.value &&
|
|
structureSelectionsComplete.value &&
|
|
!submitting.value,
|
|
))
|
|
|
|
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.familyCode) {
|
|
parts.push(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 clearCreationForm = () => {
|
|
creationForm.name = ''
|
|
creationForm.reference = ''
|
|
creationForm.constructeurId = null
|
|
creationForm.prix = ''
|
|
lastSuggestedName.value = ''
|
|
structureAssignments.value = null
|
|
}
|
|
|
|
const submitCreation = async () => {
|
|
if (!selectedType.value) {
|
|
toast.showError('Sélectionnez une catégorie de composant.')
|
|
return
|
|
}
|
|
const payload: Record<string, any> = {
|
|
name: creationForm.name.trim(),
|
|
typeComposantId: selectedType.value.id,
|
|
}
|
|
|
|
const reference = creationForm.reference.trim()
|
|
if (reference) {
|
|
payload.reference = reference
|
|
}
|
|
|
|
if (creationForm.constructeurId) {
|
|
payload.constructeurId = creationForm.constructeurId
|
|
}
|
|
|
|
const rawPrice = typeof creationForm.prix === 'string'
|
|
? creationForm.prix.trim()
|
|
: creationForm.prix === null || creationForm.prix === undefined
|
|
? ''
|
|
: String(creationForm.prix).trim()
|
|
|
|
if (rawPrice) {
|
|
const parsed = Number(rawPrice)
|
|
if (!Number.isNaN(parsed)) {
|
|
payload.prix = parsed
|
|
}
|
|
}
|
|
|
|
if (structureHasRequirements.value && !structureSelectionsComplete.value) {
|
|
toast.showError('Complétez la sélection des pièces et sous-composants.')
|
|
return
|
|
}
|
|
|
|
const serializedStructure = structureHasRequirements.value
|
|
? serializeStructureAssignments(structureAssignments.value)
|
|
: null
|
|
|
|
if (serializedStructure) {
|
|
payload.structure = serializedStructure
|
|
}
|
|
|
|
submitting.value = true
|
|
try {
|
|
const result = await createComposant(payload)
|
|
if (result.success) {
|
|
await saveCustomFieldValues(result.data)
|
|
toast.showSuccess('Composant créé avec succès')
|
|
await router.push('/component-catalog')
|
|
} else if (result.error) {
|
|
toast.showError(result.error)
|
|
}
|
|
} catch (error: any) {
|
|
toast.showError(error?.message || 'Erreur lors de la création du composant')
|
|
} finally {
|
|
submitting.value = false
|
|
}
|
|
}
|
|
|
|
onMounted(async () => {
|
|
await Promise.allSettled([
|
|
loadComponentTypes(),
|
|
loadPieces(),
|
|
loadComposants(),
|
|
])
|
|
})
|
|
|
|
interface CustomFieldInput {
|
|
id: string | null
|
|
name: string
|
|
type: string
|
|
required: boolean
|
|
options: string[]
|
|
value: string
|
|
customFieldId: string | null
|
|
customFieldValueId: string | null
|
|
}
|
|
|
|
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 = !!rawField.required
|
|
const options = Array.isArray(rawField.options)
|
|
? rawField.options.filter((option: unknown): option is string => typeof option === 'string')
|
|
: []
|
|
const value = formatDefaultValue(type, rawField.value)
|
|
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 value = typeof field?.type === 'string' ? field.type.toLowerCase() : ''
|
|
return allowed.includes(value) ? value : 'text'
|
|
}
|
|
|
|
const formatDefaultValue = (type: string, defaultValue: any): string => {
|
|
if (defaultValue === null || defaultValue === undefined) {
|
|
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 buildCustomFieldMetadata = (field: CustomFieldInput) => ({
|
|
customFieldName: field.name,
|
|
customFieldType: field.type,
|
|
customFieldRequired: field.required,
|
|
customFieldOptions: field.options,
|
|
})
|
|
|
|
const saveCustomFieldValues = async (createdComponent: any) => {
|
|
if (!createdComponent || !createdComponent.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(createdComponent?.typeComposant?.customFields)
|
|
registerDefinitions(createdComponent?.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',
|
|
createdComponent.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
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const shouldPersistField = (field: CustomFieldInput) => {
|
|
if (field.type === 'boolean') {
|
|
return field.value === 'true' || field.value === 'false'
|
|
}
|
|
return field.value.trim() !== ''
|
|
}
|
|
|
|
const formatValueForPersistence = (field: CustomFieldInput) => {
|
|
if (field.type === 'boolean') {
|
|
return field.value === 'true' ? 'true' : 'false'
|
|
}
|
|
return field.value.trim()
|
|
}
|
|
</script>
|