refactor: Separate concerns and add components dropdown
- Add ComponentsDropdown.vue with save/load/delete functionality - Create components store (Pinia) for state management - Extract WebMCP initialization to services/webmcp.ts - Extract MCP tools registration to services/canvasTools.ts - Simplify Canvas.vue (~340 lines -> ~105 lines) - Update App.vue header with components dropdown Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
@@ -2,12 +2,16 @@
|
|||||||
import Canvas from './components/Canvas.vue'
|
import Canvas from './components/Canvas.vue'
|
||||||
import StatusBar from './components/StatusBar.vue'
|
import StatusBar from './components/StatusBar.vue'
|
||||||
import Toolbar from './components/Toolbar.vue'
|
import Toolbar from './components/Toolbar.vue'
|
||||||
|
import ComponentsDropdown from './components/ComponentsDropdown.vue'
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="app-container">
|
<div class="app-container">
|
||||||
<header class="app-header">
|
<header class="app-header">
|
||||||
<h1>Agent UI</h1>
|
<div class="header-left">
|
||||||
|
<h1>Agent UI</h1>
|
||||||
|
<ComponentsDropdown />
|
||||||
|
</div>
|
||||||
<StatusBar />
|
<StatusBar />
|
||||||
</header>
|
</header>
|
||||||
<main class="app-main">
|
<main class="app-main">
|
||||||
@@ -34,6 +38,12 @@ import Toolbar from './components/Toolbar.vue'
|
|||||||
border-bottom: 1px solid var(--border-color);
|
border-bottom: 1px solid var(--border-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.header-left {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 1.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
.app-header h1 {
|
.app-header h1 {
|
||||||
font-size: 1.25rem;
|
font-size: 1.25rem;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
|
|||||||
@@ -1,339 +1,49 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { onMounted } from 'vue'
|
import { onMounted, onUnmounted } from 'vue'
|
||||||
|
import { initWebMCP } from '../services/webmcp'
|
||||||
|
import { registerCanvasTools } from '../services/canvasTools'
|
||||||
|
import { renderInlineComponent, type VueComponentDefinition } from '../services/dynamicComponents'
|
||||||
import { useCanvasStore } from '../stores/canvas'
|
import { useCanvasStore } from '../stores/canvas'
|
||||||
import {
|
|
||||||
renderInlineComponent,
|
|
||||||
componentsApi,
|
|
||||||
type VueComponentDefinition
|
|
||||||
} from '../services/dynamicComponents'
|
|
||||||
|
|
||||||
const canvasStore = useCanvasStore()
|
const canvasStore = useCanvasStore()
|
||||||
|
|
||||||
onMounted(async () => {
|
function handleLoadComponent(e: Event) {
|
||||||
// Importar webmcp - esto crea el widget automáticamente
|
const detail = (e as CustomEvent).detail
|
||||||
const WebMCPModule = await import('@nucleoriofrio/webmcp/src/webmcp.js')
|
if (!detail) return
|
||||||
const WebMCP = WebMCPModule.default || WebMCPModule
|
|
||||||
|
|
||||||
const webmcp = new WebMCP({
|
const container = document.getElementById('canvas-content')
|
||||||
color: '#6366f1',
|
if (!container) return
|
||||||
position: 'bottom-right',
|
|
||||||
inactivityTimeout: 60 * 60 * 1000 // 1 hora
|
|
||||||
})
|
|
||||||
|
|
||||||
// Escuchar eventos de conexión
|
const placeholder = container.querySelector('.canvas-placeholder')
|
||||||
webmcp.on?.('connected', () => {
|
if (placeholder) placeholder.remove()
|
||||||
canvasStore.setConnected(true)
|
|
||||||
})
|
|
||||||
webmcp.on?.('disconnected', () => {
|
|
||||||
canvasStore.setConnected(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
// Registrar herramientas para el canvas
|
const definition: VueComponentDefinition = {
|
||||||
registerCanvasTools(webmcp)
|
id: detail.id,
|
||||||
|
name: detail.name,
|
||||||
// Exponer webmcp globalmente para debug
|
template: detail.template,
|
||||||
;(window as any).webmcp = webmcp
|
setup: detail.setup,
|
||||||
|
style: detail.style,
|
||||||
// Verificar si ya está conectado
|
props: detail.props,
|
||||||
if (webmcp.isConnected) {
|
imports: detail.imports || ['ref', 'reactive', 'computed']
|
||||||
canvasStore.setConnected(true)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const result = renderInlineComponent(definition, container, {}, false)
|
||||||
|
;(window as any).__vueComponentUnmount = result.unmount
|
||||||
|
|
||||||
|
window.dispatchEvent(new CustomEvent('vue-component-rendered', { detail }))
|
||||||
|
canvasStore.addToHistory({ tool: 'load_vue_component', args: detail, timestamp: Date.now() })
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
window.addEventListener('load-vue-component', handleLoadComponent)
|
||||||
|
|
||||||
|
await initWebMCP()
|
||||||
|
registerCanvasTools()
|
||||||
})
|
})
|
||||||
|
|
||||||
function registerCanvasTools(mcp: any) {
|
onUnmounted(() => {
|
||||||
// render_html: Renderiza HTML en el canvas con soporte para scripts inline
|
window.removeEventListener('load-vue-component', handleLoadComponent)
|
||||||
mcp.registerTool(
|
})
|
||||||
'render_html',
|
|
||||||
'Renderiza HTML en el canvas. Soporta <script> tags que se ejecutan automáticamente y <style> tags.',
|
|
||||||
{
|
|
||||||
type: 'object',
|
|
||||||
properties: {
|
|
||||||
html: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'El código HTML a renderizar (puede incluir <script> y <style> tags)'
|
|
||||||
},
|
|
||||||
mode: {
|
|
||||||
type: 'string',
|
|
||||||
enum: ['replace', 'append', 'prepend'],
|
|
||||||
description: 'Modo: replace (reemplaza), append (agrega al final), prepend (al inicio)'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
required: ['html']
|
|
||||||
},
|
|
||||||
(args: { html: string; mode?: string }) => {
|
|
||||||
const container = document.getElementById('canvas-content')
|
|
||||||
if (!container) return 'Error: canvas no encontrado'
|
|
||||||
|
|
||||||
// Quitar placeholder si existe
|
|
||||||
const placeholder = container.querySelector('.canvas-placeholder')
|
|
||||||
if (placeholder) placeholder.remove()
|
|
||||||
|
|
||||||
const mode = args.mode || 'replace'
|
|
||||||
if (mode === 'replace') {
|
|
||||||
container.innerHTML = args.html
|
|
||||||
} else if (mode === 'append') {
|
|
||||||
container.insertAdjacentHTML('beforeend', args.html)
|
|
||||||
} else if (mode === 'prepend') {
|
|
||||||
container.insertAdjacentHTML('afterbegin', args.html)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ejecutar scripts inline
|
|
||||||
const scripts = container.querySelectorAll('script')
|
|
||||||
scripts.forEach((oldScript) => {
|
|
||||||
const newScript = document.createElement('script')
|
|
||||||
Array.from(oldScript.attributes).forEach(attr => {
|
|
||||||
newScript.setAttribute(attr.name, attr.value)
|
|
||||||
})
|
|
||||||
newScript.textContent = oldScript.textContent
|
|
||||||
oldScript.parentNode?.replaceChild(newScript, oldScript)
|
|
||||||
})
|
|
||||||
|
|
||||||
canvasStore.addToHistory({ tool: 'render_html', args, timestamp: Date.now() })
|
|
||||||
return 'HTML renderizado'
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
// render_vue_component: Renderiza un componente Vue 3 dinámico
|
|
||||||
mcp.registerTool(
|
|
||||||
'render_vue_component',
|
|
||||||
'Renderiza un componente Vue 3 completo con acceso a ref, reactive, computed, watch, Pinia stores, etc.',
|
|
||||||
{
|
|
||||||
type: 'object',
|
|
||||||
properties: {
|
|
||||||
id: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'ID único del componente'
|
|
||||||
},
|
|
||||||
name: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'Nombre del componente (ej: MyCounter)'
|
|
||||||
},
|
|
||||||
template: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'Template HTML del componente con sintaxis Vue'
|
|
||||||
},
|
|
||||||
setup: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'Código de la función setup (debe retornar un objeto con las propiedades reactivas)'
|
|
||||||
},
|
|
||||||
style: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'CSS del componente (opcional)'
|
|
||||||
},
|
|
||||||
props: {
|
|
||||||
type: 'array',
|
|
||||||
items: { type: 'string' },
|
|
||||||
description: 'Lista de props que acepta el componente'
|
|
||||||
},
|
|
||||||
imports: {
|
|
||||||
type: 'array',
|
|
||||||
items: { type: 'string' },
|
|
||||||
description: 'Funciones de Vue a importar: ref, reactive, computed, watch, watchEffect, onMounted, onUnmounted, nextTick, h'
|
|
||||||
},
|
|
||||||
componentProps: {
|
|
||||||
type: 'object',
|
|
||||||
description: 'Valores para las props del componente'
|
|
||||||
},
|
|
||||||
mode: {
|
|
||||||
type: 'string',
|
|
||||||
enum: ['replace', 'append'],
|
|
||||||
description: 'replace: limpia el canvas, append: agrega al final'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
required: ['id', 'name', 'template']
|
|
||||||
},
|
|
||||||
(args: {
|
|
||||||
id: string
|
|
||||||
name: string
|
|
||||||
template: string
|
|
||||||
setup?: string
|
|
||||||
style?: string
|
|
||||||
props?: string[]
|
|
||||||
imports?: string[]
|
|
||||||
componentProps?: Record<string, any>
|
|
||||||
mode?: string
|
|
||||||
}) => {
|
|
||||||
const container = document.getElementById('canvas-content')
|
|
||||||
if (!container) return 'Error: canvas no encontrado'
|
|
||||||
|
|
||||||
// Quitar placeholder
|
|
||||||
const placeholder = container.querySelector('.canvas-placeholder')
|
|
||||||
if (placeholder) placeholder.remove()
|
|
||||||
|
|
||||||
const definition: VueComponentDefinition = {
|
|
||||||
id: args.id,
|
|
||||||
name: args.name,
|
|
||||||
template: args.template,
|
|
||||||
setup: args.setup,
|
|
||||||
style: args.style,
|
|
||||||
props: args.props,
|
|
||||||
imports: args.imports || ['ref', 'reactive', 'computed']
|
|
||||||
}
|
|
||||||
|
|
||||||
const isAppend = args.mode === 'append'
|
|
||||||
const result = renderInlineComponent(definition, container, args.componentProps || {}, isAppend)
|
|
||||||
|
|
||||||
// Guardar referencia para cleanup
|
|
||||||
;(window as any).__vueComponentUnmount = result.unmount
|
|
||||||
|
|
||||||
canvasStore.addToHistory({ tool: 'render_vue_component', args, timestamp: Date.now() })
|
|
||||||
return `Componente Vue "${args.name}" renderizado correctamente`
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
// save_vue_component: Guarda un componente en la base de datos
|
|
||||||
mcp.registerTool(
|
|
||||||
'save_vue_component',
|
|
||||||
'Guarda un componente Vue en la base de datos para reutilizarlo después',
|
|
||||||
{
|
|
||||||
type: 'object',
|
|
||||||
properties: {
|
|
||||||
id: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'ID único del componente (se genera automáticamente si no se proporciona)'
|
|
||||||
},
|
|
||||||
name: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'Nombre del componente'
|
|
||||||
},
|
|
||||||
template: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'Template HTML del componente'
|
|
||||||
},
|
|
||||||
setup: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'Código de la función setup'
|
|
||||||
},
|
|
||||||
style: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'CSS del componente'
|
|
||||||
},
|
|
||||||
props: {
|
|
||||||
type: 'array',
|
|
||||||
items: { type: 'string' },
|
|
||||||
description: 'Lista de props'
|
|
||||||
},
|
|
||||||
imports: {
|
|
||||||
type: 'array',
|
|
||||||
items: { type: 'string' },
|
|
||||||
description: 'Funciones de Vue necesarias'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
required: ['name', 'template']
|
|
||||||
},
|
|
||||||
async (args: Omit<VueComponentDefinition, 'id'> & { id?: string }) => {
|
|
||||||
try {
|
|
||||||
const result = await componentsApi.save({
|
|
||||||
id: args.id || `comp-${Date.now()}`,
|
|
||||||
name: args.name,
|
|
||||||
template: args.template,
|
|
||||||
setup: args.setup,
|
|
||||||
style: args.style,
|
|
||||||
props: args.props,
|
|
||||||
imports: args.imports
|
|
||||||
})
|
|
||||||
canvasStore.addToHistory({ tool: 'save_vue_component', args, timestamp: Date.now() })
|
|
||||||
return `Componente "${args.name}" guardado con ID: ${result.id}`
|
|
||||||
} catch (e: any) {
|
|
||||||
return `Error al guardar: ${e.message}`
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
// load_vue_component: Carga y renderiza un componente guardado
|
|
||||||
mcp.registerTool(
|
|
||||||
'load_vue_component',
|
|
||||||
'Carga un componente Vue guardado desde la base de datos y lo renderiza',
|
|
||||||
{
|
|
||||||
type: 'object',
|
|
||||||
properties: {
|
|
||||||
id: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'ID del componente a cargar'
|
|
||||||
},
|
|
||||||
componentProps: {
|
|
||||||
type: 'object',
|
|
||||||
description: 'Props para pasar al componente'
|
|
||||||
},
|
|
||||||
mode: {
|
|
||||||
type: 'string',
|
|
||||||
enum: ['replace', 'append'],
|
|
||||||
description: 'replace: limpia el canvas, append: agrega al final'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
required: ['id']
|
|
||||||
},
|
|
||||||
async (args: { id: string; componentProps?: Record<string, any>; mode?: string }) => {
|
|
||||||
try {
|
|
||||||
const definition = await componentsApi.getById(args.id)
|
|
||||||
if (!definition) {
|
|
||||||
return `Error: Componente con ID "${args.id}" no encontrado`
|
|
||||||
}
|
|
||||||
|
|
||||||
const container = document.getElementById('canvas-content')
|
|
||||||
if (!container) return 'Error: canvas no encontrado'
|
|
||||||
|
|
||||||
const placeholder = container.querySelector('.canvas-placeholder')
|
|
||||||
if (placeholder) placeholder.remove()
|
|
||||||
|
|
||||||
const isAppend = args.mode === 'append'
|
|
||||||
const result = renderInlineComponent(definition, container, args.componentProps || {}, isAppend)
|
|
||||||
;(window as any).__vueComponentUnmount = result.unmount
|
|
||||||
|
|
||||||
canvasStore.addToHistory({ tool: 'load_vue_component', args, timestamp: Date.now() })
|
|
||||||
return `Componente "${definition.name}" cargado y renderizado`
|
|
||||||
} catch (e: any) {
|
|
||||||
return `Error: ${e.message}`
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
// list_vue_components: Lista los componentes guardados
|
|
||||||
mcp.registerTool(
|
|
||||||
'list_vue_components',
|
|
||||||
'Lista todos los componentes Vue guardados en la base de datos',
|
|
||||||
{
|
|
||||||
type: 'object',
|
|
||||||
properties: {}
|
|
||||||
},
|
|
||||||
async () => {
|
|
||||||
try {
|
|
||||||
const components = await componentsApi.getAll()
|
|
||||||
if (components.length === 0) {
|
|
||||||
return 'No hay componentes guardados'
|
|
||||||
}
|
|
||||||
const list = components.map(c => `- ${c.id}: ${c.name}`).join('\n')
|
|
||||||
return `Componentes guardados:\n${list}`
|
|
||||||
} catch (e: any) {
|
|
||||||
return `Error: ${e.message}`
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
// delete_vue_component: Elimina un componente
|
|
||||||
mcp.registerTool(
|
|
||||||
'delete_vue_component',
|
|
||||||
'Elimina un componente Vue de la base de datos',
|
|
||||||
{
|
|
||||||
type: 'object',
|
|
||||||
properties: {
|
|
||||||
id: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'ID del componente a eliminar'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
required: ['id']
|
|
||||||
},
|
|
||||||
async (args: { id: string }) => {
|
|
||||||
try {
|
|
||||||
await componentsApi.delete(args.id)
|
|
||||||
return `Componente "${args.id}" eliminado`
|
|
||||||
} catch (e: any) {
|
|
||||||
return `Error: ${e.message}`
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
|
|||||||
357
frontend/src/components/ComponentsDropdown.vue
Normal file
357
frontend/src/components/ComponentsDropdown.vue
Normal file
@@ -0,0 +1,357 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import { ref, onMounted, onUnmounted } from 'vue'
|
||||||
|
import { useComponentsStore } from '../stores/components'
|
||||||
|
|
||||||
|
const store = useComponentsStore()
|
||||||
|
const isOpen = ref(false)
|
||||||
|
|
||||||
|
async function handleLoadComponent(comp: any) {
|
||||||
|
window.dispatchEvent(new CustomEvent('load-vue-component', {
|
||||||
|
detail: {
|
||||||
|
id: comp.id,
|
||||||
|
name: comp.name,
|
||||||
|
template: comp.template,
|
||||||
|
setup: comp.setup,
|
||||||
|
style: comp.style,
|
||||||
|
props: typeof comp.props === 'string' ? JSON.parse(comp.props) : comp.props || [],
|
||||||
|
imports: typeof comp.imports === 'string' ? JSON.parse(comp.imports) : comp.imports || []
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
isOpen.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleDelete(id: string, e: Event) {
|
||||||
|
e.stopPropagation()
|
||||||
|
if (!confirm('¿Eliminar este componente?')) return
|
||||||
|
await store.deleteComponent(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleDropdown() {
|
||||||
|
isOpen.value = !isOpen.value
|
||||||
|
if (isOpen.value) {
|
||||||
|
store.fetchComponents()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeDropdown(e: MouseEvent) {
|
||||||
|
const target = e.target as HTMLElement
|
||||||
|
if (!target.closest('.dropdown-container')) {
|
||||||
|
isOpen.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onComponentRendered(e: Event) {
|
||||||
|
store.handleComponentRendered((e as CustomEvent).detail)
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
document.addEventListener('click', closeDropdown)
|
||||||
|
window.addEventListener('vue-component-rendered', onComponentRendered)
|
||||||
|
})
|
||||||
|
|
||||||
|
onUnmounted(() => {
|
||||||
|
document.removeEventListener('click', closeDropdown)
|
||||||
|
window.removeEventListener('vue-component-rendered', onComponentRendered)
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="dropdown-container">
|
||||||
|
<button class="dropdown-trigger" @click.stop="toggleDropdown" title="Componentes guardados">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<path d="M21 16V8a2 2 0 0 0-1-1.73l-7-4a2 2 0 0 0-2 0l-7 4A2 2 0 0 0 3 8v8a2 2 0 0 0 1 1.73l7 4a2 2 0 0 0 2 0l7-4A2 2 0 0 0 21 16z"/>
|
||||||
|
<polyline points="3.29 7 12 12 20.71 7"/>
|
||||||
|
<line x1="12" y1="22" x2="12" y2="12"/>
|
||||||
|
</svg>
|
||||||
|
<span>Componentes</span>
|
||||||
|
<span v-if="store.hasCurrentComponent" class="badge">1</span>
|
||||||
|
<svg class="chevron" :class="{ open: isOpen }" xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<polyline points="6 9 12 15 18 9"/>
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<div v-if="isOpen" class="dropdown-menu" @click.stop>
|
||||||
|
<!-- Componente actual -->
|
||||||
|
<div v-if="store.currentComponent" class="section current-section">
|
||||||
|
<div class="section-header">
|
||||||
|
<span class="section-title">Componente actual</span>
|
||||||
|
</div>
|
||||||
|
<div class="current-component">
|
||||||
|
<div class="item-info">
|
||||||
|
<span class="item-name">{{ store.currentComponent.name }}</span>
|
||||||
|
<span class="item-id">{{ store.currentComponent.id }}</span>
|
||||||
|
</div>
|
||||||
|
<div class="item-actions">
|
||||||
|
<button
|
||||||
|
class="action-btn save-btn"
|
||||||
|
@click="store.saveCurrentComponent()"
|
||||||
|
:disabled="store.saving"
|
||||||
|
title="Guardar componente"
|
||||||
|
>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<path d="M19 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h11l5 5v11a2 2 0 0 1-2 2z"/>
|
||||||
|
<polyline points="17 21 17 13 7 13 7 21"/>
|
||||||
|
<polyline points="7 3 7 8 15 8"/>
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
class="action-btn delete-btn"
|
||||||
|
@click="store.clearCurrentComponent()"
|
||||||
|
title="Descartar"
|
||||||
|
>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<line x1="18" y1="6" x2="6" y2="18"/>
|
||||||
|
<line x1="6" y1="6" x2="18" y2="18"/>
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Separador -->
|
||||||
|
<div v-if="store.currentComponent && store.savedCount > 0" class="divider"></div>
|
||||||
|
|
||||||
|
<!-- Componentes guardados -->
|
||||||
|
<div class="section saved-section">
|
||||||
|
<div class="section-header">
|
||||||
|
<span class="section-title">Guardados</span>
|
||||||
|
<span class="section-count">{{ store.savedCount }}</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="store.loading" class="dropdown-loading">
|
||||||
|
Cargando...
|
||||||
|
</div>
|
||||||
|
<div v-else-if="store.savedCount === 0" class="dropdown-empty">
|
||||||
|
No hay componentes guardados
|
||||||
|
</div>
|
||||||
|
<template v-else>
|
||||||
|
<div
|
||||||
|
v-for="comp in store.savedComponents"
|
||||||
|
:key="comp.id"
|
||||||
|
class="dropdown-item"
|
||||||
|
>
|
||||||
|
<div class="item-info" @click="handleLoadComponent(comp)">
|
||||||
|
<span class="item-name">{{ comp.name }}</span>
|
||||||
|
<span class="item-id">{{ comp.id }}</span>
|
||||||
|
</div>
|
||||||
|
<div class="item-actions">
|
||||||
|
<button
|
||||||
|
class="action-btn load-btn"
|
||||||
|
@click="handleLoadComponent(comp)"
|
||||||
|
title="Cargar en canvas"
|
||||||
|
>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<polyline points="15 10 20 15 15 20"/>
|
||||||
|
<path d="M4 4v7a4 4 0 0 0 4 4h12"/>
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
class="action-btn delete-btn"
|
||||||
|
@click="handleDelete(comp.id, $event)"
|
||||||
|
title="Eliminar"
|
||||||
|
>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<path d="M3 6h18"/>
|
||||||
|
<path d="M19 6v14c0 1-1 2-2 2H7c-1 0-2-1-2-2V6"/>
|
||||||
|
<path d="M8 6V4c0-1 1-2 2-2h4c1 0 2 1 2 2v2"/>
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.dropdown-container {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-trigger {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
padding: 0.5rem 0.75rem;
|
||||||
|
background: var(--bg-hover);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: 8px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
font-size: 0.875rem;
|
||||||
|
font-weight: 500;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.15s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-trigger:hover {
|
||||||
|
background: var(--bg-tertiary, rgba(255,255,255,0.1));
|
||||||
|
color: var(--text-primary);
|
||||||
|
border-color: var(--border-hover, var(--border-color));
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge {
|
||||||
|
background: #6366f1;
|
||||||
|
color: white;
|
||||||
|
font-size: 0.7rem;
|
||||||
|
font-weight: 600;
|
||||||
|
padding: 0.125rem 0.375rem;
|
||||||
|
border-radius: 9999px;
|
||||||
|
min-width: 18px;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chevron {
|
||||||
|
transition: transform 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chevron.open {
|
||||||
|
transform: rotate(180deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-menu {
|
||||||
|
position: absolute;
|
||||||
|
top: calc(100% + 4px);
|
||||||
|
left: 0;
|
||||||
|
min-width: 300px;
|
||||||
|
max-height: 400px;
|
||||||
|
overflow-y: auto;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: 12px;
|
||||||
|
box-shadow: 0 4px 20px rgba(0, 0, 0, 0.4);
|
||||||
|
z-index: 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section {
|
||||||
|
padding: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 0.5rem 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section-title {
|
||||||
|
font-size: 0.7rem;
|
||||||
|
font-weight: 600;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.05em;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.section-count {
|
||||||
|
font-size: 0.7rem;
|
||||||
|
background: var(--bg-hover);
|
||||||
|
padding: 0.125rem 0.5rem;
|
||||||
|
border-radius: 9999px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.divider {
|
||||||
|
height: 1px;
|
||||||
|
background: var(--border-color);
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.current-section {
|
||||||
|
background: rgba(99, 102, 241, 0.1);
|
||||||
|
border-radius: 8px;
|
||||||
|
margin: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.current-component {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 0.5rem 0.75rem;
|
||||||
|
gap: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-loading,
|
||||||
|
.dropdown-empty {
|
||||||
|
padding: 1rem;
|
||||||
|
text-align: center;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
font-size: 0.875rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 0.625rem 0.75rem;
|
||||||
|
border-radius: 8px;
|
||||||
|
margin: 0.25rem 0;
|
||||||
|
transition: background 0.15s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-item:hover {
|
||||||
|
background: var(--bg-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-info {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.125rem;
|
||||||
|
cursor: pointer;
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-name {
|
||||||
|
font-size: 0.875rem;
|
||||||
|
font-weight: 500;
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-id {
|
||||||
|
font-size: 0.7rem;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
font-family: monospace;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-actions {
|
||||||
|
display: flex;
|
||||||
|
gap: 0.375rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-btn {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 28px;
|
||||||
|
height: 28px;
|
||||||
|
border: none;
|
||||||
|
border-radius: 6px;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.15s ease;
|
||||||
|
background: transparent;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-btn:hover {
|
||||||
|
transform: scale(1.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-btn:disabled {
|
||||||
|
opacity: 0.5;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
|
||||||
|
.save-btn:hover {
|
||||||
|
background: rgba(34, 197, 94, 0.2);
|
||||||
|
color: #22c55e;
|
||||||
|
}
|
||||||
|
|
||||||
|
.load-btn:hover {
|
||||||
|
background: rgba(99, 102, 241, 0.2);
|
||||||
|
color: #6366f1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.delete-btn:hover {
|
||||||
|
background: rgba(239, 68, 68, 0.2);
|
||||||
|
color: #ef4444;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
257
frontend/src/services/canvasTools.ts
Normal file
257
frontend/src/services/canvasTools.ts
Normal file
@@ -0,0 +1,257 @@
|
|||||||
|
import { useCanvasStore } from '../stores/canvas'
|
||||||
|
import { registerTool } from './webmcp'
|
||||||
|
import {
|
||||||
|
renderInlineComponent,
|
||||||
|
componentsApi,
|
||||||
|
type VueComponentDefinition
|
||||||
|
} from './dynamicComponents'
|
||||||
|
|
||||||
|
function getCanvasContainer() {
|
||||||
|
return document.getElementById('canvas-content')
|
||||||
|
}
|
||||||
|
|
||||||
|
function removePlaceholder(container: HTMLElement) {
|
||||||
|
const placeholder = container.querySelector('.canvas-placeholder')
|
||||||
|
if (placeholder) placeholder.remove()
|
||||||
|
}
|
||||||
|
|
||||||
|
function emitComponentRendered(args: any) {
|
||||||
|
window.dispatchEvent(new CustomEvent('vue-component-rendered', {
|
||||||
|
detail: {
|
||||||
|
id: args.id,
|
||||||
|
name: args.name,
|
||||||
|
template: args.template,
|
||||||
|
setup: args.setup,
|
||||||
|
style: args.style,
|
||||||
|
props: args.props,
|
||||||
|
imports: args.imports
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
export function registerCanvasTools() {
|
||||||
|
const canvasStore = useCanvasStore()
|
||||||
|
|
||||||
|
// render_html
|
||||||
|
registerTool(
|
||||||
|
'render_html',
|
||||||
|
'Renderiza HTML en el canvas. Soporta <script> tags que se ejecutan automáticamente y <style> tags.',
|
||||||
|
{
|
||||||
|
type: 'object',
|
||||||
|
properties: {
|
||||||
|
html: {
|
||||||
|
type: 'string',
|
||||||
|
description: 'El código HTML a renderizar (puede incluir <script> y <style> tags)'
|
||||||
|
},
|
||||||
|
mode: {
|
||||||
|
type: 'string',
|
||||||
|
enum: ['replace', 'append', 'prepend'],
|
||||||
|
description: 'Modo: replace (reemplaza), append (agrega al final), prepend (al inicio)'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
required: ['html']
|
||||||
|
},
|
||||||
|
(args: { html: string; mode?: string }) => {
|
||||||
|
const container = getCanvasContainer()
|
||||||
|
if (!container) return 'Error: canvas no encontrado'
|
||||||
|
|
||||||
|
removePlaceholder(container)
|
||||||
|
|
||||||
|
const mode = args.mode || 'replace'
|
||||||
|
if (mode === 'replace') {
|
||||||
|
container.innerHTML = args.html
|
||||||
|
} else if (mode === 'append') {
|
||||||
|
container.insertAdjacentHTML('beforeend', args.html)
|
||||||
|
} else if (mode === 'prepend') {
|
||||||
|
container.insertAdjacentHTML('afterbegin', args.html)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ejecutar scripts inline
|
||||||
|
const scripts = container.querySelectorAll('script')
|
||||||
|
scripts.forEach((oldScript) => {
|
||||||
|
const newScript = document.createElement('script')
|
||||||
|
Array.from(oldScript.attributes).forEach(attr => {
|
||||||
|
newScript.setAttribute(attr.name, attr.value)
|
||||||
|
})
|
||||||
|
newScript.textContent = oldScript.textContent
|
||||||
|
oldScript.parentNode?.replaceChild(newScript, oldScript)
|
||||||
|
})
|
||||||
|
|
||||||
|
canvasStore.addToHistory({ tool: 'render_html', args, timestamp: Date.now() })
|
||||||
|
return 'HTML renderizado'
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// render_vue_component
|
||||||
|
registerTool(
|
||||||
|
'render_vue_component',
|
||||||
|
'Renderiza un componente Vue 3 completo con acceso a ref, reactive, computed, watch, Pinia stores, etc.',
|
||||||
|
{
|
||||||
|
type: 'object',
|
||||||
|
properties: {
|
||||||
|
id: { type: 'string', description: 'ID único del componente' },
|
||||||
|
name: { type: 'string', description: 'Nombre del componente (ej: MyCounter)' },
|
||||||
|
template: { type: 'string', description: 'Template HTML del componente con sintaxis Vue' },
|
||||||
|
setup: { type: 'string', description: 'Código de la función setup (debe retornar un objeto con las propiedades reactivas)' },
|
||||||
|
style: { type: 'string', description: 'CSS del componente (opcional)' },
|
||||||
|
props: { type: 'array', items: { type: 'string' }, description: 'Lista de props que acepta el componente' },
|
||||||
|
imports: { type: 'array', items: { type: 'string' }, description: 'Funciones de Vue a importar: ref, reactive, computed, watch, watchEffect, onMounted, onUnmounted, nextTick, h' },
|
||||||
|
componentProps: { type: 'object', description: 'Valores para las props del componente' },
|
||||||
|
mode: { type: 'string', enum: ['replace', 'append'], description: 'replace: limpia el canvas, append: agrega al final' }
|
||||||
|
},
|
||||||
|
required: ['id', 'name', 'template']
|
||||||
|
},
|
||||||
|
(args: {
|
||||||
|
id: string
|
||||||
|
name: string
|
||||||
|
template: string
|
||||||
|
setup?: string
|
||||||
|
style?: string
|
||||||
|
props?: string[]
|
||||||
|
imports?: string[]
|
||||||
|
componentProps?: Record<string, any>
|
||||||
|
mode?: string
|
||||||
|
}) => {
|
||||||
|
const container = getCanvasContainer()
|
||||||
|
if (!container) return 'Error: canvas no encontrado'
|
||||||
|
|
||||||
|
removePlaceholder(container)
|
||||||
|
|
||||||
|
const definition: VueComponentDefinition = {
|
||||||
|
id: args.id,
|
||||||
|
name: args.name,
|
||||||
|
template: args.template,
|
||||||
|
setup: args.setup,
|
||||||
|
style: args.style,
|
||||||
|
props: args.props,
|
||||||
|
imports: args.imports || ['ref', 'reactive', 'computed']
|
||||||
|
}
|
||||||
|
|
||||||
|
const isAppend = args.mode === 'append'
|
||||||
|
const result = renderInlineComponent(definition, container, args.componentProps || {}, isAppend)
|
||||||
|
|
||||||
|
;(window as any).__vueComponentUnmount = result.unmount
|
||||||
|
|
||||||
|
emitComponentRendered(args)
|
||||||
|
|
||||||
|
canvasStore.addToHistory({ tool: 'render_vue_component', args, timestamp: Date.now() })
|
||||||
|
return `Componente Vue "${args.name}" renderizado correctamente`
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// save_vue_component
|
||||||
|
registerTool(
|
||||||
|
'save_vue_component',
|
||||||
|
'Guarda un componente Vue en la base de datos para reutilizarlo después',
|
||||||
|
{
|
||||||
|
type: 'object',
|
||||||
|
properties: {
|
||||||
|
id: { type: 'string', description: 'ID único del componente (se genera automáticamente si no se proporciona)' },
|
||||||
|
name: { type: 'string', description: 'Nombre del componente' },
|
||||||
|
template: { type: 'string', description: 'Template HTML del componente' },
|
||||||
|
setup: { type: 'string', description: 'Código de la función setup' },
|
||||||
|
style: { type: 'string', description: 'CSS del componente' },
|
||||||
|
props: { type: 'array', items: { type: 'string' }, description: 'Lista de props' },
|
||||||
|
imports: { type: 'array', items: { type: 'string' }, description: 'Funciones de Vue necesarias' }
|
||||||
|
},
|
||||||
|
required: ['name', 'template']
|
||||||
|
},
|
||||||
|
async (args: Omit<VueComponentDefinition, 'id'> & { id?: string }) => {
|
||||||
|
try {
|
||||||
|
const result = await componentsApi.save({
|
||||||
|
id: args.id || `comp-${Date.now()}`,
|
||||||
|
name: args.name,
|
||||||
|
template: args.template,
|
||||||
|
setup: args.setup,
|
||||||
|
style: args.style,
|
||||||
|
props: args.props,
|
||||||
|
imports: args.imports
|
||||||
|
})
|
||||||
|
canvasStore.addToHistory({ tool: 'save_vue_component', args, timestamp: Date.now() })
|
||||||
|
return `Componente "${args.name}" guardado con ID: ${result.id}`
|
||||||
|
} catch (e: any) {
|
||||||
|
return `Error al guardar: ${e.message}`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// load_vue_component
|
||||||
|
registerTool(
|
||||||
|
'load_vue_component',
|
||||||
|
'Carga un componente Vue guardado desde la base de datos y lo renderiza',
|
||||||
|
{
|
||||||
|
type: 'object',
|
||||||
|
properties: {
|
||||||
|
id: { type: 'string', description: 'ID del componente a cargar' },
|
||||||
|
componentProps: { type: 'object', description: 'Props para pasar al componente' },
|
||||||
|
mode: { type: 'string', enum: ['replace', 'append'], description: 'replace: limpia el canvas, append: agrega al final' }
|
||||||
|
},
|
||||||
|
required: ['id']
|
||||||
|
},
|
||||||
|
async (args: { id: string; componentProps?: Record<string, any>; mode?: string }) => {
|
||||||
|
try {
|
||||||
|
const definition = await componentsApi.getById(args.id)
|
||||||
|
if (!definition) {
|
||||||
|
return `Error: Componente con ID "${args.id}" no encontrado`
|
||||||
|
}
|
||||||
|
|
||||||
|
const container = getCanvasContainer()
|
||||||
|
if (!container) return 'Error: canvas no encontrado'
|
||||||
|
|
||||||
|
removePlaceholder(container)
|
||||||
|
|
||||||
|
const isAppend = args.mode === 'append'
|
||||||
|
const result = renderInlineComponent(definition, container, args.componentProps || {}, isAppend)
|
||||||
|
;(window as any).__vueComponentUnmount = result.unmount
|
||||||
|
|
||||||
|
canvasStore.addToHistory({ tool: 'load_vue_component', args, timestamp: Date.now() })
|
||||||
|
return `Componente "${definition.name}" cargado y renderizado`
|
||||||
|
} catch (e: any) {
|
||||||
|
return `Error: ${e.message}`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// list_vue_components
|
||||||
|
registerTool(
|
||||||
|
'list_vue_components',
|
||||||
|
'Lista todos los componentes Vue guardados en la base de datos',
|
||||||
|
{
|
||||||
|
type: 'object',
|
||||||
|
properties: {}
|
||||||
|
},
|
||||||
|
async () => {
|
||||||
|
try {
|
||||||
|
const components = await componentsApi.getAll()
|
||||||
|
if (components.length === 0) {
|
||||||
|
return 'No hay componentes guardados'
|
||||||
|
}
|
||||||
|
const list = components.map(c => `- ${c.id}: ${c.name}`).join('\n')
|
||||||
|
return `Componentes guardados:\n${list}`
|
||||||
|
} catch (e: any) {
|
||||||
|
return `Error: ${e.message}`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// delete_vue_component
|
||||||
|
registerTool(
|
||||||
|
'delete_vue_component',
|
||||||
|
'Elimina un componente Vue de la base de datos',
|
||||||
|
{
|
||||||
|
type: 'object',
|
||||||
|
properties: {
|
||||||
|
id: { type: 'string', description: 'ID del componente a eliminar' }
|
||||||
|
},
|
||||||
|
required: ['id']
|
||||||
|
},
|
||||||
|
async (args: { id: string }) => {
|
||||||
|
try {
|
||||||
|
await componentsApi.delete(args.id)
|
||||||
|
return `Componente "${args.id}" eliminado`
|
||||||
|
} catch (e: any) {
|
||||||
|
return `Error: ${e.message}`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
52
frontend/src/services/webmcp.ts
Normal file
52
frontend/src/services/webmcp.ts
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
import { useCanvasStore } from '../stores/canvas'
|
||||||
|
|
||||||
|
let webmcpInstance: any = null
|
||||||
|
|
||||||
|
export async function initWebMCP() {
|
||||||
|
if (webmcpInstance) return webmcpInstance
|
||||||
|
|
||||||
|
const WebMCPModule = await import('@nucleoriofrio/webmcp/src/webmcp.js')
|
||||||
|
const WebMCP = WebMCPModule.default || WebMCPModule
|
||||||
|
|
||||||
|
webmcpInstance = new WebMCP({
|
||||||
|
color: '#6366f1',
|
||||||
|
position: 'bottom-right',
|
||||||
|
inactivityTimeout: 60 * 60 * 1000 // 1 hora
|
||||||
|
})
|
||||||
|
|
||||||
|
const canvasStore = useCanvasStore()
|
||||||
|
|
||||||
|
webmcpInstance.on?.('connected', () => {
|
||||||
|
canvasStore.setConnected(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
webmcpInstance.on?.('disconnected', () => {
|
||||||
|
canvasStore.setConnected(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
if (webmcpInstance.isConnected) {
|
||||||
|
canvasStore.setConnected(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Exponer globalmente para debug
|
||||||
|
;(window as any).webmcp = webmcpInstance
|
||||||
|
|
||||||
|
return webmcpInstance
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getWebMCP() {
|
||||||
|
return webmcpInstance
|
||||||
|
}
|
||||||
|
|
||||||
|
export function registerTool(
|
||||||
|
name: string,
|
||||||
|
description: string,
|
||||||
|
schema: object,
|
||||||
|
handler: Function
|
||||||
|
) {
|
||||||
|
if (!webmcpInstance) {
|
||||||
|
console.warn('[WebMCP] Instance not initialized')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
webmcpInstance.registerTool(name, description, schema, handler)
|
||||||
|
}
|
||||||
124
frontend/src/stores/components.ts
Normal file
124
frontend/src/stores/components.ts
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
import { defineStore } from 'pinia'
|
||||||
|
import { ref, computed } from 'vue'
|
||||||
|
import { componentsApi, type VueComponentDefinition } from '../services/dynamicComponents'
|
||||||
|
|
||||||
|
export interface ComponentState extends VueComponentDefinition {
|
||||||
|
created_at?: string
|
||||||
|
updated_at?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useComponentsStore = defineStore('components', () => {
|
||||||
|
// State
|
||||||
|
const savedComponents = ref<ComponentState[]>([])
|
||||||
|
const currentComponent = ref<ComponentState | null>(null)
|
||||||
|
const loading = ref(false)
|
||||||
|
const saving = ref(false)
|
||||||
|
|
||||||
|
// Getters
|
||||||
|
const savedCount = computed(() => savedComponents.value.length)
|
||||||
|
const hasCurrentComponent = computed(() => currentComponent.value !== null)
|
||||||
|
|
||||||
|
// Actions
|
||||||
|
async function fetchComponents() {
|
||||||
|
loading.value = true
|
||||||
|
try {
|
||||||
|
savedComponents.value = await componentsApi.getAll()
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Error fetching components:', e)
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function saveComponent(component: ComponentState) {
|
||||||
|
saving.value = true
|
||||||
|
try {
|
||||||
|
await componentsApi.save({
|
||||||
|
id: component.id,
|
||||||
|
name: component.name,
|
||||||
|
template: component.template,
|
||||||
|
setup: component.setup,
|
||||||
|
style: component.style,
|
||||||
|
props: component.props,
|
||||||
|
imports: component.imports
|
||||||
|
})
|
||||||
|
await fetchComponents()
|
||||||
|
return true
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Error saving component:', e)
|
||||||
|
return false
|
||||||
|
} finally {
|
||||||
|
saving.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function saveCurrentComponent() {
|
||||||
|
if (!currentComponent.value) return false
|
||||||
|
return saveComponent(currentComponent.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
async function deleteComponent(id: string) {
|
||||||
|
try {
|
||||||
|
await componentsApi.delete(id)
|
||||||
|
await fetchComponents()
|
||||||
|
return true
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Error deleting component:', e)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function deleteAllComponents() {
|
||||||
|
try {
|
||||||
|
await componentsApi.deleteAll()
|
||||||
|
savedComponents.value = []
|
||||||
|
return true
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Error deleting all components:', e)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function setCurrentComponent(component: ComponentState | null) {
|
||||||
|
currentComponent.value = component
|
||||||
|
}
|
||||||
|
|
||||||
|
function clearCurrentComponent() {
|
||||||
|
currentComponent.value = null
|
||||||
|
}
|
||||||
|
|
||||||
|
// Listener para cuando se renderiza un componente
|
||||||
|
function handleComponentRendered(detail: any) {
|
||||||
|
if (detail) {
|
||||||
|
currentComponent.value = {
|
||||||
|
id: detail.id || `comp-${Date.now()}`,
|
||||||
|
name: detail.name || 'Unnamed',
|
||||||
|
template: detail.template || '',
|
||||||
|
setup: detail.setup || '',
|
||||||
|
style: detail.style || '',
|
||||||
|
props: detail.props || [],
|
||||||
|
imports: detail.imports || []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
// State
|
||||||
|
savedComponents,
|
||||||
|
currentComponent,
|
||||||
|
loading,
|
||||||
|
saving,
|
||||||
|
// Getters
|
||||||
|
savedCount,
|
||||||
|
hasCurrentComponent,
|
||||||
|
// Actions
|
||||||
|
fetchComponents,
|
||||||
|
saveComponent,
|
||||||
|
saveCurrentComponent,
|
||||||
|
deleteComponent,
|
||||||
|
deleteAllComponents,
|
||||||
|
setCurrentComponent,
|
||||||
|
clearCurrentComponent,
|
||||||
|
handleComponentRendered
|
||||||
|
}
|
||||||
|
})
|
||||||
Reference in New Issue
Block a user