feat(vwb-v3): Architecture Thin Client fonctionnelle

API = Source de vérité unique (SQLite + Flask)
- Backend: API v3 avec session, workflow, capture, execute
- Frontend: Vanilla TypeScript, pas de state local
- Contrats stricts pour les actions RPA
- Drag & drop pour réorganiser les étapes
- Insertion d'étapes entre deux existantes
- Bibliothèque de captures (sessionStorage)
- Exécution avec coordonnées statiques (pyautogui)

Fonctionne mais fragile (coordonnées fixes, pas de détection visuelle)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
Dom
2026-01-23 12:07:13 +01:00
parent a9a53991bc
commit 858e6007f9
23 changed files with 6813 additions and 6 deletions

View File

@@ -0,0 +1,96 @@
<!DOCTYPE html>
<html lang="fr">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>VWB v3 - Thin Client</title>
<link rel="stylesheet" href="/styles.css">
</head>
<body>
<header>
<h1>Visual Workflow Builder v3</h1>
<span class="badge">Thin Client - API = Verite</span>
</header>
<main>
<!-- Colonne gauche: Workflows -->
<aside class="sidebar">
<section class="panel">
<h2>Workflows</h2>
<form id="create-workflow-form">
<input type="text" placeholder="Nouveau workflow..." required />
<button type="submit">+</button>
</form>
<div id="workflow-list"></div>
</section>
</aside>
<!-- Zone centrale: Canvas/Etapes -->
<section class="main-content">
<div class="toolbar">
<h3 id="workflow-name">Aucun workflow</h3>
<div id="action-buttons"></div>
</div>
<!-- Zone scrollable pour les étapes -->
<div class="steps-container">
<h4>Étapes du workflow</h4>
<div id="steps-list"></div>
</div>
</section>
<!-- Zone de capture (colonne séparée) -->
<section class="capture-section">
<div class="capture-zone">
<div class="capture-header">
<h4>Capture d'écran</h4>
<button id="btn-toggle-capture" class="toggle-btn">Réduire</button>
</div>
<div id="capture-content">
<div class="capture-controls">
<button id="btn-capture" class="capture-btn">Capturer</button>
<div class="timer-control">
<select id="capture-delay">
<option value="0">Immédiat</option>
<option value="3">3s</option>
<option value="5">5s</option>
<option value="10">10s</option>
</select>
<button id="btn-capture-timer" class="capture-btn secondary">Avec délai</button>
</div>
</div>
<div id="timer-countdown" class="timer-countdown hidden"></div>
<div id="capture-preview"></div>
<button id="btn-fullscreen" class="fullscreen-btn hidden">Ouvrir en plein écran</button>
<!-- Bibliothèque de captures -->
<div class="capture-library">
<h5>Bibliothèque <span id="library-count">(0)</span></h5>
<div id="capture-library-list"></div>
</div>
</div>
</div>
</section>
<!-- Colonne droite: Propriétés et Execution -->
<aside class="sidebar right">
<section class="panel">
<h2>Étape sélectionnée</h2>
<div id="selected-step">
<p class="empty">Sélectionnez une étape</p>
</div>
</section>
<section class="panel">
<h2>Exécution</h2>
<div id="execution-status">
<p>Prêt à exécuter</p>
<button id="btn-start" class="primary">Démarrer</button>
</div>
</section>
</aside>
</main>
<script type="module" src="/src/main.ts"></script>
</body>
</html>