| <!DOCTYPE html> |
| <html lang="en" class="dark bg-zinc-950 text-zinc-100"> |
| <head> |
| <meta charset="UTF-8"> |
| <meta name="viewport" content="width=device-width, initial-scale=1.0"> |
| |
| |
| <title>LYGO Protocol Stack — Sovereign Lattice Mesh (SLM)</title> |
| <meta name="title" content="LYGO Protocol Stack — Sovereign Lattice Mesh"> |
| <meta name="description" content="Interactive visualization of the Sovereign Lattice Mesh. Test Merkle Sync, Mycelium Sharding, and Harmonic Vortex Consensus matrices in real-time."> |
| <meta name="keywords" content="LYGO, Cryptography, Sovereign Node, P0 Kernel, Harmonic Consensus, Distributed Mycelium, Merkle Tree"> |
| <meta name="author" content="DeepSeekOracle / EXCAVATIONPRO"> |
| <meta name="robots" content="index, follow"> |
|
|
| |
| <meta property="og:type" content="website"> |
| <meta property="og:url" content="https://deepseekoracle.github.io/lygo-protocol-stack/SovereignLatticeMesh.html"> |
| <meta property="og:title" content="LYGO Protocol Stack — Sovereign Lattice Mesh"> |
| <meta property="og:description" content="Simulate the decentralized mesh: Merkle Anti-Entropy, 10/12 Shard Reconstruction, and 3-6-9 Harmonic Consensus."> |
| <meta property="og:image" content="https://deepseekoracle.github.io/lygo-protocol-stack/assets/preview-slm.jpg"> |
|
|
| |
| <meta property="twitter:card" content="summary_large_image"> |
| <meta property="twitter:url" content="https://deepseekoracle.github.io/lygo-protocol-stack/SovereignLatticeMesh.html"> |
| <meta property="twitter:title" content="LYGO Protocol Stack — Sovereign Lattice Mesh"> |
| <meta property="twitter:description" content="Simulate the decentralized mesh: Merkle Anti-Entropy, 10/12 Shard Reconstruction, and 3-6-9 Harmonic Consensus."> |
| <meta property="twitter:image" content="https://deepseekoracle.github.io/lygo-protocol-stack/assets/preview-slm.jpg"> |
|
|
| |
| <script src="https://cdn.tailwindcss.com"></script> |
| <script> |
| tailwind.config = { |
| theme: { |
| extend: { |
| colors: { |
| lygo: { |
| 400: '#4ade80', |
| 500: '#22c55e', |
| 600: '#16a34a', |
| 950: '#052e16' |
| }, |
| danger: { |
| 500: '#ef4444', |
| 900: '#7f1d1d' |
| } |
| } |
| } |
| } |
| } |
| </script> |
| <style> |
| @import url('https://fonts.googleapis.com/css2?family=JetBrains+Mono:wght@300;400;500;700&display=swap'); |
| body { font-family: 'JetBrains Mono', monospace; } |
| ::-webkit-scrollbar { width: 6px; height: 6px; } |
| ::-webkit-scrollbar-track { background: #09090b; } |
| ::-webkit-scrollbar-thumb { background: #27272a; border-radius: 3px; } |
| ::-webkit-scrollbar-thumb:hover { background: #3f3f46; } |
| |
| .tree-line::before { |
| content: ''; |
| position: absolute; |
| top: -20px; left: 50%; width: 2px; height: 20px; |
| background-color: #3f3f46; |
| } |
| .tree-line-diverged::before { background-color: #ef4444; } |
| </style> |
| </head> |
| <body class="min-h-screen flex flex-col antialiased"> |
|
|
| |
| <header class="border-b border-zinc-800 bg-zinc-900/50 backdrop-blur sticky top-0 z-50"> |
| <div class="max-w-7xl mx-auto px-4 h-16 flex items-center justify-between"> |
| <div class="flex items-center space-x-3"> |
| <div class="w-3 h-3 rounded-full bg-lygo-500 shadow-[0_0_10px_#22c55e] animate-pulse"></div> |
| <span class="font-bold tracking-wider text-sm sm:text-base uppercase text-zinc-100">LYGO_STACK // SOVEREIGN LATTICE MESH</span> |
| </div> |
| <nav class="hidden md:flex items-center space-x-4 text-xs font-medium text-zinc-400"> |
| <a href="https://github.com/DeepSeekOracle/lygo-protocol-stack" target="_blank" class="hover:text-lygo-500 transition-colors">GITHUB_REPO</a> |
| <a href="https://huggingface.co/spaces/DeepSeekOracle/LYGO-Resonance-Engine" target="_blank" class="hover:text-lygo-500 transition-colors">HF_SPACE</a> |
| <a href="https://clawhub.ai/deepseekoracle" target="_blank" class="hover:text-lygo-500 transition-colors">CLAWHUB</a> |
| <a href="https://chatagent.ca" target="_blank" class="hover:text-lygo-500 transition-colors text-zinc-200 border border-zinc-700 px-3 py-1.5 rounded bg-zinc-800">ORACLE_PORTAL</a> |
| </nav> |
| </div> |
| </header> |
|
|
| |
| <div class="max-w-7xl mx-auto w-full px-4 pt-6"> |
| <div class="flex space-x-2 border-b border-zinc-800"> |
| <button onclick="switchTab('merkle')" id="tab-merkle" class="px-4 py-3 text-xs font-bold uppercase tracking-wider text-lygo-500 border-b-2 border-lygo-500 transition-colors">1. Merkle Sync</button> |
| <button onclick="switchTab('mycelium')" id="tab-mycelium" class="px-4 py-3 text-xs font-bold uppercase tracking-wider text-zinc-500 hover:text-zinc-300 border-b-2 border-transparent transition-colors">2. Mycelium Shards</button> |
| <button onclick="switchTab('consensus')" id="tab-consensus" class="px-4 py-3 text-xs font-bold uppercase tracking-wider text-zinc-500 hover:text-zinc-300 border-b-2 border-transparent transition-colors">3. Harmonic Consensus</button> |
| </div> |
| </div> |
|
|
| |
| <main class="max-w-7xl w-full mx-auto p-4 flex-grow flex flex-col"> |
| |
| |
| <section id="panel-merkle" class="flex-grow flex flex-col space-y-6 animate-fade-in"> |
| <div class="bg-zinc-900/60 border border-zinc-800 rounded-xl p-5"> |
| <div class="flex justify-between items-start mb-6"> |
| <div> |
| <h2 class="text-lg font-bold text-zinc-100 uppercase tracking-wider mb-1">Anti-Entropy Merkle Matrix</h2> |
| <p class="text-xs text-zinc-500">Simulate hierarchical state reconciliation. Modifying a leaf node cascades hash recalculations to the root.</p> |
| </div> |
| <button id="btn-diverge" onclick="toggleDivergence()" class="bg-zinc-800 hover:bg-danger-900 border border-zinc-700 hover:border-danger-500 text-zinc-300 hover:text-danger-500 font-bold py-2 px-4 rounded text-xs tracking-widest transition-all uppercase"> |
| Introduce Divergence |
| </button> |
| </div> |
| |
| |
| <div class="relative py-8 flex flex-col items-center space-y-12 bg-zinc-950 rounded-lg border border-zinc-900 overflow-x-auto"> |
| |
| <div class="relative flex flex-col items-center"> |
| <span class="text-[10px] text-zinc-500 uppercase tracking-widest mb-1">Global Root Hash</span> |
| <div id="hash-root" class="bg-lygo-950/50 border border-lygo-500/50 text-lygo-400 font-mono text-sm px-4 py-2 rounded shadow-[0_0_15px_rgba(34,197,94,0.1)] transition-colors duration-300">0x7F1A4D83...A1B2C3D4</div> |
| </div> |
|
|
| |
| <div class="flex w-full max-w-2xl justify-around relative"> |
| |
| <svg class="absolute top-[-48px] w-full h-[48px] z-0" preserveAspectRatio="none"> |
| <path id="path-root-left" d="M 50% 0 L 25% 48" stroke="#3f3f46" stroke-width="2" fill="none"/> |
| <path id="path-root-right" d="M 50% 0 L 75% 48" stroke="#3f3f46" stroke-width="2" fill="none"/> |
| </svg> |
|
|
| <div class="relative flex flex-col items-center z-10"> |
| <span class="text-[10px] text-zinc-500 uppercase tracking-widest mb-1">Branch A-B</span> |
| <div id="hash-ab" class="bg-zinc-900 border border-zinc-700 text-zinc-300 font-mono text-xs px-3 py-1.5 rounded transition-colors duration-300">0x88A21B...</div> |
| </div> |
| <div class="relative flex flex-col items-center z-10"> |
| <span class="text-[10px] text-zinc-500 uppercase tracking-widest mb-1">Branch C-D</span> |
| <div id="hash-cd" class="bg-zinc-900 border border-zinc-700 text-zinc-300 font-mono text-xs px-3 py-1.5 rounded transition-colors duration-300">0x44F99E...</div> |
| </div> |
| </div> |
|
|
| |
| <div class="flex w-full max-w-4xl justify-around relative"> |
| <svg class="absolute top-[-48px] w-full h-[48px] z-0" preserveAspectRatio="none"> |
| <path d="M 25% 0 L 12.5% 48" stroke="#3f3f46" stroke-width="2" fill="none"/> |
| <path d="M 25% 0 L 37.5% 48" stroke="#3f3f46" stroke-width="2" fill="none"/> |
| <path id="path-cd-left" d="M 75% 0 L 62.5% 48" stroke="#3f3f46" stroke-width="2" fill="none"/> |
| <path id="path-cd-right" d="M 75% 0 L 87.5% 48" stroke="#3f3f46" stroke-width="2" fill="none"/> |
| </svg> |
|
|
| <div class="relative flex flex-col items-center z-10"> |
| <span class="text-[10px] text-lygo-500 font-bold uppercase tracking-widest mb-1">Node Alpha</span> |
| <div class="bg-zinc-900 border border-zinc-700 text-zinc-400 font-mono text-xs px-2 py-1 rounded">0x11A...</div> |
| </div> |
| <div class="relative flex flex-col items-center z-10"> |
| <span class="text-[10px] text-lygo-500 font-bold uppercase tracking-widest mb-1">Node Beta</span> |
| <div class="bg-zinc-900 border border-zinc-700 text-zinc-400 font-mono text-xs px-2 py-1 rounded">0x22B...</div> |
| </div> |
| <div class="relative flex flex-col items-center z-10"> |
| <span class="text-[10px] text-lygo-500 font-bold uppercase tracking-widest mb-1">Node Gamma</span> |
| <div id="hash-gamma" class="bg-zinc-900 border border-zinc-700 text-zinc-400 font-mono text-xs px-2 py-1 rounded transition-colors duration-300">0x33C...</div> |
| </div> |
| <div class="relative flex flex-col items-center z-10"> |
| <span class="text-[10px] text-lygo-500 font-bold uppercase tracking-widest mb-1">Node Delta</span> |
| <div id="hash-delta" class="bg-zinc-900 border border-zinc-700 text-zinc-400 font-mono text-xs px-2 py-1 rounded transition-colors duration-300">0x44D...</div> |
| </div> |
| </div> |
| </div> |
| </div> |
| </section> |
|
|
| |
| <section id="panel-mycelium" class="hidden flex-grow flex flex-col space-y-6 animate-fade-in"> |
| <div class="bg-zinc-900/60 border border-zinc-800 rounded-xl p-5 grid grid-cols-1 lg:grid-cols-12 gap-6"> |
| |
| <div class="lg:col-span-4 flex flex-col space-y-6"> |
| <div> |
| <h2 class="text-lg font-bold text-zinc-100 uppercase tracking-wider mb-1">Distributed Mycelium</h2> |
| <p class="text-xs text-zinc-500">Consistent Hash Ring evaluating the 10/12 Erasure Coding survivability threshold.</p> |
| </div> |
| <hr class="border-zinc-800"> |
| <div> |
| <div class="flex justify-between text-xs mb-2"> |
| <span class="text-zinc-400 font-bold uppercase tracking-wider">Nodes Offline</span> |
| <span class="text-danger-500 font-bold" id="val-offline">0 Nodes</span> |
| </div> |
| <input type="range" id="param-offline" min="0" max="3" value="0" class="w-full accent-danger-500 bg-zinc-800 rounded-lg appearance-none h-1.5 cursor-pointer"> |
| <p class="text-[10px] text-zinc-500 mt-3 leading-relaxed">Adjusting this slider simulates physical hardware partitions. The network must retain at least 10 active data fragments across surviving nodes to rebuild the P0 kernel.</p> |
| </div> |
| |
| <div id="mycelium-status" class="mt-auto bg-lygo-950/30 border border-lygo-500/30 p-4 rounded-lg"> |
| <div class="text-[10px] text-zinc-400 uppercase tracking-widest mb-1">Network State</div> |
| <div class="text-lygo-500 font-bold text-sm tracking-wide" id="mycelium-msg">DATA RECONSTRUCTION SUCCESSFUL (12/12 Secure)</div> |
| </div> |
| </div> |
|
|
| <div class="lg:col-span-8 bg-zinc-950 rounded-lg border border-zinc-900 flex justify-center items-center relative min-h-[400px]"> |
| <canvas id="canvas-ring" width="400" height="400" class="max-w-full"></canvas> |
| </div> |
| </div> |
| </section> |
|
|
| |
| <section id="panel-consensus" class="hidden flex-grow flex flex-col space-y-6 animate-fade-in"> |
| <div class="bg-zinc-900/60 border border-zinc-800 rounded-xl p-5 grid grid-cols-1 lg:grid-cols-12 gap-6"> |
| |
| <div class="lg:col-span-4 flex flex-col space-y-6"> |
| <div> |
| <h2 class="text-lg font-bold text-zinc-100 uppercase tracking-wider mb-1">Harmonic Vortex Engine</h2> |
| <p class="text-xs text-zinc-500">Calculates geometric phase alignments. Votes are weighted by fixed node Ethical Mass (Φ).</p> |
| </div> |
| <hr class="border-zinc-800"> |
| |
| <div class="space-y-4"> |
| <div class="flex justify-between items-center"> |
| <div> |
| <div class="text-xs font-bold text-zinc-300 uppercase">Node Alpha</div> |
| <div class="text-[10px] text-zinc-500">Mass: 1.618</div> |
| </div> |
| <select id="vote-alpha" class="bg-zinc-950 border border-zinc-700 text-zinc-200 text-xs rounded px-2 py-1 outline-none focus:border-lygo-500"> |
| <option value="3">3 (Creation)</option> |
| <option value="6">6 (Relation)</option> |
| <option value="9" selected>9 (Completion)</option> |
| <option value="-1">Dissonance (-1)</option> |
| </select> |
| </div> |
| |
| <div class="flex justify-between items-center"> |
| <div> |
| <div class="text-xs font-bold text-zinc-300 uppercase">Node Beta</div> |
| <div class="text-[10px] text-zinc-500">Mass: 1.250</div> |
| </div> |
| <select id="vote-beta" class="bg-zinc-950 border border-zinc-700 text-zinc-200 text-xs rounded px-2 py-1 outline-none focus:border-lygo-500"> |
| <option value="3">3 (Creation)</option> |
| <option value="6">6 (Relation)</option> |
| <option value="9" selected>9 (Completion)</option> |
| <option value="-1">Dissonance (-1)</option> |
| </select> |
| </div> |
|
|
| <div class="flex justify-between items-center"> |
| <div> |
| <div class="text-xs font-bold text-zinc-300 uppercase">Node Gamma</div> |
| <div class="text-[10px] text-zinc-500">Mass: 0.850</div> |
| </div> |
| <select id="vote-gamma" class="bg-zinc-950 border border-zinc-700 text-zinc-200 text-xs rounded px-2 py-1 outline-none focus:border-lygo-500"> |
| <option value="3">3 (Creation)</option> |
| <option value="6" selected>6 (Relation)</option> |
| <option value="9">9 (Completion)</option> |
| <option value="-1">Dissonance (-1)</option> |
| </select> |
| </div> |
|
|
| <div class="flex justify-between items-center"> |
| <div> |
| <div class="text-xs font-bold text-zinc-300 uppercase">Node Delta</div> |
| <div class="text-[10px] text-zinc-500">Mass: 0.310</div> |
| </div> |
| <select id="vote-delta" class="bg-zinc-950 border border-zinc-700 text-zinc-200 text-xs rounded px-2 py-1 outline-none focus:border-lygo-500"> |
| <option value="3" selected>3 (Creation)</option> |
| <option value="6">6 (Relation)</option> |
| <option value="9">9 (Completion)</option> |
| <option value="-1">Dissonance (-1)</option> |
| </select> |
| </div> |
| </div> |
| </div> |
|
|
| <div class="lg:col-span-8 flex flex-col space-y-4"> |
| <div class="bg-zinc-950 rounded-lg border border-zinc-900 flex justify-center items-center relative min-h-[350px] overflow-hidden"> |
| <canvas id="canvas-vortex" width="400" height="400"></canvas> |
| </div> |
| |
| <div class="grid grid-cols-3 gap-4"> |
| <div class="bg-zinc-950 border border-zinc-800 p-3 rounded-lg text-center"> |
| <div class="text-[10px] text-zinc-500 uppercase tracking-widest mb-1">Magnitude</div> |
| <div id="vortex-mag" class="text-lg font-bold text-zinc-200 font-mono">0.000</div> |
| </div> |
| <div class="bg-zinc-950 border border-zinc-800 p-3 rounded-lg text-center"> |
| <div class="text-[10px] text-zinc-500 uppercase tracking-widest mb-1">Phase Angle</div> |
| <div id="vortex-ang" class="text-lg font-bold text-zinc-200 font-mono">0.00π</div> |
| </div> |
| <div class="bg-lygo-950/20 border border-lygo-500/50 p-3 rounded-lg text-center shadow-[0_0_15px_rgba(34,197,94,0.05)]"> |
| <div class="text-[10px] text-lygo-500 uppercase tracking-widest mb-1">Final Consensus</div> |
| <div id="vortex-snap" class="text-2xl font-bold text-lygo-400 font-mono">9</div> |
| </div> |
| </div> |
| </div> |
|
|
| </div> |
| </section> |
| </main> |
|
|
| |
| <footer class="border-t border-zinc-900 bg-zinc-950 py-6 px-4 mt-auto"> |
| <div class="max-w-7xl mx-auto flex flex-col sm:flex-row items-center justify-between text-[10px] text-zinc-600 space-y-3 sm:space-y-0"> |
| <div>© 2026 LYGO PROTOCOL STACK // ZERO TRUST IMMUTABLE ARCHITECTURE</div> |
| <div class="tracking-widest uppercase text-lygo-600">Reference: Δ9Φ963-SLM-PRODUCTION-DRAFT</div> |
| </div> |
| </footer> |
|
|
| |
| <script> |
| |
| function switchTab(tab) { |
| document.getElementById('panel-merkle').classList.add('hidden'); |
| document.getElementById('panel-mycelium').classList.add('hidden'); |
| document.getElementById('panel-consensus').classList.add('hidden'); |
| |
| document.getElementById('tab-merkle').className = "px-4 py-3 text-xs font-bold uppercase tracking-wider text-zinc-500 hover:text-zinc-300 border-b-2 border-transparent transition-colors"; |
| document.getElementById('tab-mycelium').className = "px-4 py-3 text-xs font-bold uppercase tracking-wider text-zinc-500 hover:text-zinc-300 border-b-2 border-transparent transition-colors"; |
| document.getElementById('tab-consensus').className = "px-4 py-3 text-xs font-bold uppercase tracking-wider text-zinc-500 hover:text-zinc-300 border-b-2 border-transparent transition-colors"; |
| |
| document.getElementById(`panel-${tab}`).classList.remove('hidden'); |
| document.getElementById(`tab-${tab}`).className = "px-4 py-3 text-xs font-bold uppercase tracking-wider text-lygo-500 border-b-2 border-lygo-500 transition-colors"; |
| |
| if(tab === 'mycelium') drawMycelium(); |
| if(tab === 'consensus') calculateConsensus(); |
| } |
| |
| |
| let isDiverged = false; |
| function toggleDivergence() { |
| isDiverged = !isDiverged; |
| const btn = document.getElementById('btn-diverge'); |
| |
| const hashDelta = document.getElementById('hash-delta'); |
| const hashCD = document.getElementById('hash-cd'); |
| const hashRoot = document.getElementById('hash-root'); |
| const pathCDRight = document.getElementById('path-cd-right'); |
| const pathRootRight = document.getElementById('path-root-right'); |
| |
| if (isDiverged) { |
| btn.innerText = "Resolve Network State"; |
| btn.classList.replace('bg-zinc-800', 'bg-danger-900'); |
| btn.classList.replace('text-zinc-300', 'text-danger-500'); |
| |
| |
| hashDelta.innerText = "0xERR...999"; |
| hashDelta.classList.replace('bg-zinc-900', 'bg-danger-900'); |
| hashDelta.classList.replace('border-zinc-700', 'border-danger-500'); |
| hashDelta.classList.replace('text-zinc-400', 'text-danger-500'); |
| pathCDRight.setAttribute('stroke', '#ef4444'); |
| |
| setTimeout(() => { |
| hashCD.innerText = "0xDEADBE...EF"; |
| hashCD.classList.replace('bg-zinc-900', 'bg-danger-900/50'); |
| hashCD.classList.replace('border-zinc-700', 'border-danger-500/50'); |
| hashCD.classList.replace('text-zinc-300', 'text-danger-400'); |
| pathRootRight.setAttribute('stroke', '#ef4444'); |
| }, 300); |
| |
| setTimeout(() => { |
| hashRoot.innerText = "0xMISMATCH...CRITICAL"; |
| hashRoot.classList.replace('bg-lygo-950/50', 'bg-danger-950'); |
| hashRoot.classList.replace('border-lygo-500/50', 'border-danger-500'); |
| hashRoot.classList.replace('text-lygo-400', 'text-danger-500'); |
| hashRoot.classList.replace('shadow-[0_0_15px_rgba(34,197,94,0.1)]', 'shadow-[0_0_15px_rgba(239,68,68,0.2)]'); |
| }, 600); |
| |
| } else { |
| btn.innerText = "Introduce Divergence"; |
| btn.classList.replace('bg-danger-900', 'bg-zinc-800'); |
| btn.classList.replace('text-danger-500', 'text-zinc-300'); |
| |
| |
| hashDelta.innerText = "0x44D..."; |
| hashDelta.classList.replace('bg-danger-900', 'bg-zinc-900'); |
| hashDelta.classList.replace('border-danger-500', 'border-zinc-700'); |
| hashDelta.classList.replace('text-danger-500', 'text-zinc-400'); |
| pathCDRight.setAttribute('stroke', '#3f3f46'); |
| |
| setTimeout(() => { |
| hashCD.innerText = "0x44F99E..."; |
| hashCD.classList.replace('bg-danger-900/50', 'bg-zinc-900'); |
| hashCD.classList.replace('border-danger-500/50', 'border-zinc-700'); |
| hashCD.classList.replace('text-danger-400', 'text-zinc-300'); |
| pathRootRight.setAttribute('stroke', '#3f3f46'); |
| }, 200); |
| |
| setTimeout(() => { |
| hashRoot.innerText = "0x7F1A4D83...A1B2C3D4"; |
| hashRoot.classList.replace('bg-danger-950', 'bg-lygo-950/50'); |
| hashRoot.classList.replace('border-danger-500', 'border-lygo-500/50'); |
| hashRoot.classList.replace('text-danger-500', 'text-lygo-400'); |
| hashRoot.classList.replace('shadow-[0_0_15px_rgba(239,68,68,0.2)]', 'shadow-[0_0_15px_rgba(34,197,94,0.1)]'); |
| }, 400); |
| } |
| } |
| |
| |
| const sliderOffline = document.getElementById('param-offline'); |
| sliderOffline.addEventListener('input', drawMycelium); |
| |
| function drawMycelium() { |
| const canvas = document.getElementById('canvas-ring'); |
| const ctx = canvas.getContext('2d'); |
| const w = canvas.width; |
| const h = canvas.height; |
| const cx = w/2; |
| const cy = h/2; |
| const r = 140; |
| |
| ctx.clearRect(0,0,w,h); |
| |
| |
| ctx.beginPath(); |
| ctx.arc(cx, cy, r, 0, 2*Math.PI); |
| ctx.strokeStyle = "#27272a"; |
| ctx.lineWidth = 4; |
| ctx.stroke(); |
| |
| const offlineCount = parseInt(sliderOffline.value); |
| document.getElementById('val-offline').innerText = `${offlineCount} Nodes`; |
| |
| |
| let survivingFragments = 12; |
| if (offlineCount === 1) survivingFragments = 10; |
| if (offlineCount === 2) survivingFragments = 7; |
| if (offlineCount === 3) survivingFragments = 3; |
| |
| |
| const statusBox = document.getElementById('mycelium-status'); |
| const statusMsg = document.getElementById('mycelium-msg'); |
| |
| if (survivingFragments >= 10) { |
| statusBox.className = "mt-auto bg-lygo-950/30 border border-lygo-500/30 p-4 rounded-lg"; |
| statusMsg.className = "text-lygo-500 font-bold text-sm tracking-wide"; |
| statusMsg.innerText = `DATA RECONSTRUCTION SUCCESSFUL (${survivingFragments}/12 Chunks Secure)`; |
| } else { |
| statusBox.className = "mt-auto bg-danger-900/30 border border-danger-500/50 p-4 rounded-lg"; |
| statusMsg.className = "text-danger-500 font-bold text-sm tracking-wide"; |
| statusMsg.innerText = `CRITICAL FAILURE: DATA IRRECOVERABLE (${survivingFragments}/12 Fragments)`; |
| } |
| |
| |
| const nodes = ['Alpha', 'Beta', 'Gamma', 'Delta']; |
| for(let i=0; i<4; i++) { |
| const angle = (i * Math.PI / 2) - Math.PI/4; |
| const nx = cx + Math.cos(angle) * r; |
| const ny = cy + Math.sin(angle) * r; |
| const isOffline = i < offlineCount; |
| |
| ctx.beginPath(); |
| ctx.arc(nx, ny, 12, 0, 2*Math.PI); |
| ctx.fillStyle = isOffline ? "#ef4444" : "#22c55e"; |
| ctx.fill(); |
| ctx.strokeStyle = "#09090b"; |
| ctx.lineWidth = 3; |
| ctx.stroke(); |
| |
| ctx.fillStyle = isOffline ? "#52525b" : "#a1a1aa"; |
| ctx.font = "10px monospace"; |
| ctx.textAlign = "center"; |
| ctx.fillText(`N_${nodes[i]}`, nx, ny + 25); |
| } |
| |
| |
| for(let i=0; i<12; i++) { |
| const angle = (i * Math.PI / 6); |
| const fx = cx + Math.cos(angle) * (r - 25); |
| const fy = cy + Math.sin(angle) * (r - 25); |
| const isLost = i >= survivingFragments; |
| |
| ctx.beginPath(); |
| ctx.rect(fx - 4, fy - 4, 8, 8); |
| ctx.fillStyle = isLost ? "#3f3f46" : "#4ade80"; |
| ctx.fill(); |
| if(!isLost) { |
| ctx.shadowColor = "#4ade80"; |
| ctx.shadowBlur = 5; |
| ctx.fill(); |
| ctx.shadowBlur = 0; |
| } |
| } |
| } |
| |
| |
| const votes = ['alpha', 'beta', 'gamma', 'delta']; |
| const masses = [1.618, 1.250, 0.850, 0.310]; |
| |
| votes.forEach(n => { |
| document.getElementById(`vote-${n}`).addEventListener('change', calculateConsensus); |
| }); |
| |
| function calculateConsensus() { |
| const canvas = document.getElementById('canvas-vortex'); |
| const ctx = canvas.getContext('2d'); |
| const w = canvas.width; |
| const h = canvas.height; |
| const cx = w/2; |
| const cy = h/2; |
| const r = 130; |
| |
| ctx.clearRect(0,0,w,h); |
| |
| |
| ctx.beginPath(); |
| ctx.arc(cx, cy, r, 0, 2*Math.PI); |
| ctx.strokeStyle = "#27272a"; |
| ctx.lineWidth = 2; |
| ctx.stroke(); |
| |
| ctx.beginPath(); ctx.moveTo(cx - r - 20, cy); ctx.lineTo(cx + r + 20, cy); ctx.strokeStyle = "#18181b"; ctx.stroke(); |
| ctx.beginPath(); ctx.moveTo(cx, cy - r - 20); ctx.lineTo(cx, cy + r + 20); ctx.stroke(); |
| |
| |
| const gates = { |
| 3: { ang: 0, label: "3 (Creation)" }, |
| 6: { ang: 2*Math.PI/3, label: "6 (Relation)" }, |
| 9: { ang: 4*Math.PI/3, label: "9 (Completion)" }, |
| "-1": { ang: Math.PI, label: "-1 (Dissonance)" } |
| }; |
| |
| ctx.font = "10px monospace"; |
| for (let [val, gate] of Object.entries(gates)) { |
| |
| const gx = cx + Math.cos(gate.ang) * r; |
| const gy = cy - Math.sin(gate.ang) * r; |
| |
| ctx.beginPath(); ctx.arc(gx, gy, 6, 0, 2*Math.PI); |
| ctx.fillStyle = val === "-1" ? "#ef4444" : "#3f3f46"; |
| ctx.fill(); |
| |
| ctx.fillStyle = "#a1a1aa"; |
| ctx.textAlign = val === "3" ? "left" : (val === "-1" ? "right" : "center"); |
| const yOffset = val === "6" ? -15 : (val === "9" ? 20 : -10); |
| const xOffset = val === "3" ? 10 : (val === "-1" ? -10 : 0); |
| ctx.fillText(gate.label, gx + xOffset, gy + yOffset); |
| } |
| |
| |
| let accX = 0; |
| let accY = 0; |
| let totalMass = 0; |
| |
| votes.forEach((node, idx) => { |
| const val = document.getElementById(`vote-${node}`).value; |
| const mass = masses[idx]; |
| const angle = gates[val].ang; |
| |
| accX += mass * Math.cos(angle); |
| accY += mass * Math.sin(angle); |
| totalMass += mass; |
| }); |
| |
| const meanX = accX / totalMass; |
| const meanY = accY / totalMass; |
| |
| const magnitude = Math.sqrt(meanX*meanX + meanY*meanY); |
| let phaseAngle = Math.atan2(meanY, meanX); |
| if (phaseAngle < 0) phaseAngle += 2*Math.PI; |
| |
| |
| let snapVal = "-1"; |
| let minDelta = Infinity; |
| |
| for (let [val, gate] of Object.entries(gates)) { |
| let delta = Math.abs(phaseAngle - gate.ang); |
| if (delta > Math.PI) delta = 2*Math.PI - delta; |
| |
| |
| if (Math.abs(delta - minDelta) < 0.0001) { |
| if (val === "9" || (val === "6" && snapVal === "3")) { |
| snapVal = val; |
| } |
| } else if (delta < minDelta) { |
| minDelta = delta; |
| snapVal = val; |
| } |
| } |
| |
| |
| const scale = r; |
| const vx = cx + meanX * scale; |
| const vy = cy - meanY * scale; |
| |
| ctx.beginPath(); |
| ctx.moveTo(cx, cy); |
| ctx.lineTo(vx, vy); |
| ctx.strokeStyle = "#4ade80"; |
| ctx.lineWidth = 3; |
| ctx.stroke(); |
| |
| |
| ctx.beginPath(); |
| ctx.arc(vx, vy, 5, 0, 2*Math.PI); |
| ctx.fillStyle = "#22c55e"; |
| ctx.fill(); |
| ctx.shadowColor = "#4ade80"; ctx.shadowBlur = 10; ctx.fill(); ctx.shadowBlur = 0; |
| |
| |
| document.getElementById('vortex-mag').innerText = magnitude.toFixed(3); |
| document.getElementById('vortex-ang').innerText = (phaseAngle / Math.PI).toFixed(2) + "π"; |
| document.getElementById('vortex-snap').innerText = snapVal; |
| } |
| |
| |
| setTimeout(() => { |
| drawMycelium(); |
| calculateConsensus(); |
| }, 100); |
| </script> |
| </body> |
| </html> |
|
|