/** * gioser-graph.js — Grafo semántico con Cytoscape.js * * Detecta automáticamente elementos agregados al DOM * (incluso los creados dinámicamente por el WASM) y monta el grafo. * * Efecto "wineandcheesemap": clic en nodo → centra + desvanece resto. * Doble clic → callback de navegación (window.__gioserGraphNavigate). */ (function () { 'use strict'; var COLORS = { logos: '#d0dbff', aire: '#d0dbff', nomos: '#f59056', fuego: '#f59056', kay: '#d49873', tierra: '#d49873', uku: '#6cd0f3', agua: '#6cd0f3', }; function caminoColor(c) { return COLORS[c] || '#888'; } function initGraph(container) { var apiUrl = container.getAttribute('data-api-url') || 'https://api.gioser.net'; var onNavigate = window.__gioserGraphNavigate || null; if (typeof cytoscape === 'undefined') { var check = setInterval(function () { if (typeof cytoscape !== 'undefined') { clearInterval(check); initGraph(container); } }, 100); return; } // Esperar un frame antes de fetch — el deck puede estar oculto // y Cytoscape necesita tamaño real para el layout cose. requestAnimationFrame(function () { fetch(apiUrl + '/graph?limit=500') .then(function (r) { return r.json(); }) .then(function (data) { if (!data.nodes || data.nodes.length === 0) return; var elements = []; for (var i = 0; i < data.nodes.length; i++) { var d = data.nodes[i].data; if (!d.doc_id) continue; var color = caminoColor(d.camino); elements.push({ group: 'nodes', data: { id: d.id, doc_id: d.doc_id, label: d.name.length > 24 ? d.name.slice(0, 22) + '\u2026' : d.name, camino: d.camino.toUpperCase(), color: color, }, }); } var nodeIds = {}; for (var i = 0; i < elements.length; i++) { nodeIds[elements[i].data.id] = true; } for (var i = 0; i < data.edges.length; i++) { var ed = data.edges[i].data; if (!nodeIds[ed.source] || !nodeIds[ed.target]) continue; elements.push({ group: 'edges', data: { id: ed.id, source: ed.source, target: ed.target, weight: ed.weight || 0.7, }, }); } var tipMap = {}; for (var i = 0; i < data.nodes.length; i++) { var d = data.nodes[i].data; if (d.doc_id) tipMap[d.id] = d; } // Layout: 'preset' primero con posiciones aleatorias para // que el contenedor tome tamaño, luego cose cuando visible. var cy = cytoscape({ container: container, elements: elements, style: [ { selector: 'edge', style: { 'width': 'mapData(weight, 0.5, 1.0, 0.5, 4.5)', 'line-color': 'rgba(255,255,255,0.16)', 'curve-style': 'haystack', 'haystack-radius': 0, 'opacity': 0.6, }, }, { selector: 'node', style: { 'shape': 'round-rectangle', 'width': 130, 'height': 34, 'background-color': 'data(color)', 'background-opacity': 0.18, 'border-color': 'data(color)', 'border-width': 1.8, 'border-opacity': 0.55, 'color': 'rgba(232,234,245,0.88)', 'font-size': 11, 'font-family': 'Inter, system-ui, sans-serif', 'font-weight': 500, 'text-valign': 'center', 'text-halign': 'center', 'label': 'data(label)', 'min-zoomed-font-size': 8, 'transition-property': 'background-opacity, border-opacity, border-width', 'transition-duration': 180, }, }, ], layout: { name: 'preset', fit: false, }, }); // Ahora aplicar layout cose con animación (se adaptará al tamaño real) cy.layout({ name: 'cose', animate: 'end', animationEasing: 'ease-out', animationDuration: 600, idealEdgeLength: 150, nodeRepulsion: 7000, gravity: 0.2, numIter: 800, fit: true, padding: 25, }).run(); // Tooltip var tooltipEl = document.createElement('div'); tooltipEl.className = 'cy-tooltip'; tooltipEl.style.cssText = 'position:absolute;z-index:10;pointer-events:none;' + 'background:rgba(6,5,13,0.90);color:#e8eaf5;' + 'padding:6px 10px;border-radius:8px;font-size:11px;' + 'font-family:Inter,sans-serif;line-height:1.4;' + 'border:1px solid rgba(255,255,255,0.10);' + 'backdrop-filter:blur(8px);max-width:220px;' + 'opacity:0;transition:opacity 180ms ease;'; container.style.position = 'relative'; container.appendChild(tooltipEl); cy.on('mouseover', 'node', function (ev) { var n = ev.target; n.style({ 'background-opacity': 0.45, 'border-opacity': 0.9, 'border-width': 2.2 }); var tipData = tipMap[n.id()]; if (tipData && tipData.preview) { tooltipEl.textContent = tipData.preview.slice(0, 130); tooltipEl.style.opacity = '1'; } }); cy.on('mouseout', 'node', function (ev) { var n = ev.target; n.style({ 'background-opacity': 0.18, 'border-opacity': 0.55, 'border-width': 1.8 }); tooltipEl.style.opacity = '0'; }); cy.on('mousemove', function (ev) { if (tooltipEl.style.opacity === '1') { var pos = ev.renderedPosition || { x: 0, y: 0 }; tooltipEl.style.left = (pos.x + 14) + 'px'; tooltipEl.style.top = (pos.y - 10) + 'px'; } }); cy.on('click', 'node', function (ev) { var node = ev.target; cy.nodes().not(node).not(node.neighborhood()).forEach(function (n) { n.style({ 'opacity': 0.12 }); }); cy.edges().forEach(function (e) { e.style({ 'opacity': 0.06 }); }); node.neighborhood().nodes().forEach(function (n) { n.style({ 'opacity': 1 }); }); node.style({ 'opacity': 1, 'background-opacity': 0.40, 'border-opacity': 1 }); node.connectedEdges().forEach(function (e) { e.style({ 'opacity': 0.7 }); }); cy.animate({ center: { eles: node }, zoom: 2.5, duration: 350, }); }); cy.on('dblclick', 'node', function (ev) { var docId = ev.target.data('doc_id'); if (onNavigate && docId) onNavigate(docId); }); cy.on('click', function (ev) { if (ev.target === cy) { cy.nodes().forEach(function (n) { n.style({ 'opacity': 1, 'background-opacity': 0.18, 'border-opacity': 0.55, 'border-width': 1.8 }); }); cy.edges().forEach(function (e) { e.style({ 'opacity': 0.6 }); }); cy.animate({ zoom: 1, pan: { x: 0, y: 0 }, duration: 300 }); } }); var ro = new ResizeObserver(function () { cy.resize().fit(25); }); ro.observe(container); }) .catch(function (err) { console.warn('gioser-graph: error:', err); container.innerHTML = '
' + '· grafo no disponible ·
'; }); }); // requestAnimationFrame } // MutationObserver: detecta agregados dinámicamente var observer = new MutationObserver(function (mutations) { for (var m = 0; m < mutations.length; m++) { var added = mutations[m].addedNodes; for (var i = 0; i < added.length; i++) { var el = added[i]; if (el.tagName && el.tagName.toLowerCase() === 'gioser-graph') { initGraph(el); } var graphs = el.querySelectorAll ? el.querySelectorAll('gioser-graph') : []; for (var j = 0; j < graphs.length; j++) { initGraph(graphs[j]); } } } }); observer.observe(document.documentElement, { childList: true, subtree: true, }); // Inicializar los que ya existen var existing = document.querySelectorAll('gioser-graph'); for (var i = 0; i < existing.length; i++) { initGraph(existing[i]); } })();