`;
return stepDiv;
}
function renderModuleParameters(step, index, module) {
let html = '';
// Toujours afficher le dropdown LLM Provider en premier
const currentProvider = step.parameters?.llmProvider || module.defaultLLM || '';
const defaultProvider = module.defaultLLM || 'claude';
html += `
`;
// ✅ NOUVEAU: Afficher dropdown personnalité pour SmartTouch
if (step.module === 'smarttouch' && state.personalities.length > 0) {
const currentPersonality = step.parameters?.personalityName || '';
html += `
`;
}
// Autres paramètres du module (sauf llmProvider et personalityName qui sont déjà affichés)
if (module.parameters && Object.keys(module.parameters).length > 0) {
Object.entries(module.parameters).forEach(([paramName, paramConfig]) => {
// Skip llmProvider et personalityName car déjà affichés ci-dessus
if (paramName === 'llmProvider' || paramName === 'personalityName') return;
const value = step.parameters?.[paramName] || paramConfig.default || '';
if (paramConfig.enum) {
html += `
`;
} else if (paramConfig.type === 'number') {
html += `
`;
}
});
}
return html;
}
// ====================
// PIPELINE OPERATIONS
// ====================
function addStep(moduleId, mode = null) {
const module = state.modules.find(m => m.id === moduleId);
if (!module) return;
const newStep = {
step: state.nextStepNumber++,
module: moduleId,
mode: mode || module.modes[0],
intensity: module.defaultIntensity || 1.0,
parameters: {},
enabled: true
};
state.pipeline.pipeline.push(newStep);
reorderSteps();
renderPipeline();
}
function deleteStep(index) {
state.pipeline.pipeline.splice(index, 1);
reorderSteps();
renderPipeline();
}
function duplicateStep(index) {
const step = state.pipeline.pipeline[index];
const duplicated = JSON.parse(JSON.stringify(step));
duplicated.step = state.nextStepNumber++;
state.pipeline.pipeline.splice(index + 1, 0, duplicated);
reorderSteps();
renderPipeline();
}
function moveStepUp(index) {
if (index === 0) return;
const temp = state.pipeline.pipeline[index];
state.pipeline.pipeline[index] = state.pipeline.pipeline[index - 1];
state.pipeline.pipeline[index - 1] = temp;
reorderSteps();
renderPipeline();
}
function moveStepDown(index) {
if (index === state.pipeline.pipeline.length - 1) return;
const temp = state.pipeline.pipeline[index];
state.pipeline.pipeline[index] = state.pipeline.pipeline[index + 1];
state.pipeline.pipeline[index + 1] = temp;
reorderSteps();
renderPipeline();
}
function updateStepMode(index, mode) {
state.pipeline.pipeline[index].mode = mode;
updatePreview();
}
function updateStepIntensity(index, intensity) {
state.pipeline.pipeline[index].intensity = intensity;
updatePreview();
}
function updateStepParameter(index, paramName, value) {
if (!state.pipeline.pipeline[index].parameters) {
state.pipeline.pipeline[index].parameters = {};
}
// Si value est vide/null/undefined, supprimer la clé pour utiliser le default
if (value === '' || value === null || value === undefined) {
delete state.pipeline.pipeline[index].parameters[paramName];
} else {
state.pipeline.pipeline[index].parameters[paramName] = value;
}
updatePreview();
}
function reorderSteps() {
state.pipeline.pipeline.forEach((step, index) => {
step.step = index + 1;
});
state.nextStepNumber = state.pipeline.pipeline.length + 1;
}
function clearPipeline() {
if (!confirm('Effacer tout le pipeline ?')) return;
state.pipeline.pipeline = [];
state.nextStepNumber = 1;
document.getElementById('pipelineName').value = '';
document.getElementById('pipelineDesc').value = '';
renderPipeline();
}
// ====================
// DRAG & DROP
// ====================
let draggedElement = null;
function handleDragStart(e) {
draggedElement = e.target;
e.target.classList.add('dragging');
e.dataTransfer.effectAllowed = 'copy';
e.dataTransfer.setData('moduleId', e.target.dataset.moduleId);
}
function handleDragEnd(e) {
e.target.classList.remove('dragging');
}
// Setup drop zone
document.addEventListener('DOMContentLoaded', () => {
const canvas = document.getElementById('pipelineCanvas');
canvas.addEventListener('dragover', (e) => {
e.preventDefault();
e.dataTransfer.dropEffect = 'copy';
});
canvas.addEventListener('drop', (e) => {
e.preventDefault();
const moduleId = e.dataTransfer.getData('moduleId');
if (moduleId) {
addStep(moduleId);
}
});
// Add step button
document.getElementById('addStepBtn').addEventListener('click', () => {
const firstModule = state.modules[0];
if (firstModule) {
addStep(firstModule.id);
}
});
});
// ====================
// TEMPLATES
// ====================
async function loadTemplate(templateId) {
try {
const response = await fetch(`/api/pipeline/templates/${templateId}`);
const data = await response.json();
if (data.success) {
state.pipeline = data.template;
state.nextStepNumber = data.template.pipeline.length + 1;
document.getElementById('pipelineName').value = data.template.name;
document.getElementById('pipelineDesc').value = data.template.description || '';
renderPipeline();
showStatus(`Template "${data.template.name}" chargé`, 'success');
}
} catch (error) {
showStatus(`Erreur chargement template: ${error.message}`, 'error');
}
}
// ====================
// SAVE / VALIDATE / TEST
// ====================
async function savePipeline() {
const name = document.getElementById('pipelineName').value.trim();
const description = document.getElementById('pipelineDesc').value.trim();
if (!name) {
showStatus('Nom du pipeline requis', 'error');
return;
}
if (state.pipeline.pipeline.length === 0) {
showStatus('Pipeline vide, ajoutez au moins une étape', 'error');
return;
}
state.pipeline.name = name;
state.pipeline.description = description;
state.pipeline.metadata.saved = new Date().toISOString();
try {
const response = await fetch('/api/pipeline/save', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ pipelineDefinition: state.pipeline })
});
const data = await response.json();
if (data.success) {
showStatus(`✅ Pipeline "${name}" sauvegardé`, 'success');
} else {
showStatus(`Erreur: ${data.error}`, 'error');
}
} catch (error) {
showStatus(`Erreur sauvegarde: ${error.message}`, 'error');
}
}
async function validatePipeline() {
const name = document.getElementById('pipelineName').value.trim();
if (!name) {
state.pipeline.name = 'Unnamed Pipeline';
} else {
state.pipeline.name = name;
}
try {
const response = await fetch('/api/pipeline/validate', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ pipelineDefinition: state.pipeline })
});
const data = await response.json();
if (data.valid) {
showStatus('✅ Pipeline valide', 'success');
} else {
showStatus(`❌ Erreurs: ${data.errors.join(', ')}`, 'error');
}
} catch (error) {
showStatus(`Erreur validation: ${error.message}`, 'error');
}
}
async function testPipeline() {
const name = document.getElementById('pipelineName').value.trim();
if (!name) {
showStatus('Nom du pipeline requis pour le test', 'error');
return;
}
state.pipeline.name = name;
state.pipeline.description = document.getElementById('pipelineDesc').value.trim();
const rowNumber = prompt('Numéro de ligne Google Sheets à tester ?', '2');
if (!rowNumber) return;
showStatus('🚀 Test en cours...', 'info');
try {
const response = await fetch('/api/pipeline/execute', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({
pipelineConfig: state.pipeline,
rowNumber: parseInt(rowNumber)
})
});
const data = await response.json();
if (data.success) {
showStatus(`✅ Test réussi! Durée: ${data.result.stats.totalDuration}ms`, 'success');
console.log('Test result:', data.result);
} else {
showStatus(`❌ Test échoué: ${data.error}`, 'error');
}
} catch (error) {
showStatus(`Erreur test: ${error.message}`, 'error');
}
}
// ====================
// PREVIEW & HELPERS
// ====================
function updatePreview() {
const preview = document.getElementById('previewJson');
preview.textContent = JSON.stringify(state.pipeline, null, 2);
}
function showStatus(message, type) {
const status = document.getElementById('status');
status.textContent = message;
status.className = `status ${type}`;
status.style.display = 'block';
setTimeout(() => {
status.style.display = 'none';
}, 5000);
}