<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<style>
  *{margin:0;padding:0;box-sizing:border-box}
  html,body{width:100%;height:100%;overflow:hidden;font-family:'Helvetica Neue',Arial,sans-serif;background:#fafafa}
  .container{display:flex;flex-direction:column;height:100%;padding:12px 16px}
  .controls{display:flex;flex-wrap:wrap;gap:10px;align-items:center;padding:8px 0;border-bottom:1px solid #e0e0e0;margin-bottom:8px}
  .controls label{font-size:12px;color:#555;font-weight:500}
  .controls select,.controls button{font-size:12px;padding:4px 10px;border:1px solid #ccc;border-radius:4px;background:#fff;cursor:pointer}
  .controls button{background:#2c3e50;color:#fff;border:none;font-weight:600;padding:5px 14px}
  .controls button:hover{background:#34495e}
  .controls button.running{background:#c0392b}
  .ctrl-group{display:flex;align-items:center;gap:5px}
  .legend{display:flex;gap:14px;align-items:center;font-size:11px;color:#666;margin-left:auto}
  .legend-item{display:flex;align-items:center;gap:4px}
  .legend-dot{width:10px;height:10px;border-radius:50%;border:1px solid #999}
  .canvas-wrap{flex:1;position:relative;min-height:0}
  canvas{width:100%;height:100%;display:block}
  .stats{display:flex;gap:16px;padding:8px 0;border-top:1px solid #e0e0e0;font-size:12px;color:#444}
  .stat-bar{display:flex;align-items:center;gap:4px}
  .stat-fill{height:8px;border-radius:4px;transition:width 0.3s}
  .stat-label{min-width:28px;text-align:right;font-variant-numeric:tabular-nums}
</style>
</head>
<body>
<div class="container">
  <div class="controls">
    <div class="ctrl-group">
      <label>Network:</label>
      <select id="topoSel">
        <option value="powerlaw" selected>Power-Law (γ=2.7)</option>
        <option value="powerlaw_low">Power-Law (γ=2.1)</option>
        <option value="chain">Chain</option>
        <option value="tree">Tree</option>
      </select>
    </div>
    <div class="ctrl-group">
      <label>Nodes:</label>
      <select id="nodeSel">
        <option value="30">30</option>
        <option value="50" selected>50</option>
        <option value="80">80</option>
      </select>
    </div>
    <div class="ctrl-group">
      <label>Data placement:</label>
      <select id="placeSel">
        <option value="influential" selected>Correct → Influential</option>
        <option value="random">Random</option>
        <option value="peripheral">Correct → Peripheral</option>
      </select>
    </div>
    <button id="runBtn" onclick="toggleRun()">▶ Run</button>
    <button onclick="resetSim()">↻ Reset</button>
    <div class="legend">
      <span style="font-weight:600">Round: <span id="roundLabel">0</span></span>
      <div class="legend-item"><div class="legend-dot" style="background:#27ae60"></div>Truthful</div>
      <div class="legend-item"><div class="legend-dot" style="background:#e74c3c"></div>Hallucinating</div>
      <div class="legend-item"><div class="legend-dot" style="background:#bdc3c7"></div>Don't Know</div>
    </div>
  </div>
  <div class="canvas-wrap"><canvas id="cv"></canvas></div>
  <div class="stats">
    <div class="stat-bar"><span style="color:#27ae60;font-weight:600">ρ<sub>T</sub></span> <div style="width:120px;height:8px;background:#eee;border-radius:4px;overflow:hidden"><div id="barT" class="stat-fill" style="background:#27ae60;width:35%"></div></div><span class="stat-label" id="valT">0.35</span></div>
    <div class="stat-bar"><span style="color:#e74c3c;font-weight:600">ρ<sub>H</sub></span> <div style="width:120px;height:8px;background:#eee;border-radius:4px;overflow:hidden"><div id="barH" class="stat-fill" style="background:#e74c3c;width:35%"></div></div><span class="stat-label" id="valH">0.35</span></div>
    <div class="stat-bar"><span style="color:#95a5a6;font-weight:600">ρ<sub>D</sub></span> <div style="width:120px;height:8px;background:#eee;border-radius:4px;overflow:hidden"><div id="barD" class="stat-fill" style="background:#bdc3c7;width:30%"></div></div><span class="stat-label" id="valD">0.30</span></div>
  </div>
</div>

<script>
const cv=document.getElementById('cv'),ctx=cv.getContext('2d');
const COLORS={T:'#27ae60',H:'#e74c3c',D:'#bdc3c7'};
const COLORS_LIGHT={T:'#2ecc7140',H:'#e74c3c40',D:'#bdc3c740'};
let nodes=[],edges=[],running=false,round=0,animId=null,stepTimer=null;

function resize(){const r=cv.parentElement.getBoundingClientRect();cv.width=r.width*devicePixelRatio;cv.height=r.height*devicePixelRatio;cv.style.width=r.width+'px';cv.style.height=r.height+'px';ctx.setTransform(devicePixelRatio,0,0,devicePixelRatio,0,0)}
window.addEventListener('resize',()=>{resize();if(nodes.length)layoutForce(60)});

function genPowerLaw(N,gamma){
  let adj=Array.from({length:N},()=>[]);
  for(let i=1;i<N;i++){
    let degs=Array.from({length:i},(_,j)=>(adj[j].length+1));
    let weights=degs.map(d=>Math.pow(d,-gamma+2));
    let sum=weights.reduce((a,b)=>a+b,0);
    let nEdges=Math.max(1,Math.min(3,Math.floor(Math.pow(Math.random(),gamma-1)*5)+1));
    let targets=new Set();
    for(let e=0;e<nEdges&&targets.size<i;e++){
      let r=Math.random()*sum,acc=0;
      for(let j=0;j<i;j++){acc+=weights[j];if(acc>=r){targets.add(j);break}}
    }
    targets.forEach(j=>{adj[j].push(i);adj[i].push(j)});
  }
  return adj;
}
function genChain(N){let adj=Array.from({length:N},()=>[]);for(let i=0;i<N-1;i++){adj[i].push(i+1);adj[i+1].push(i)}return adj}
function genTree(N){
  let adj=Array.from({length:N},()=>[]);
  for(let i=1;i<N;i++){let p=Math.floor((i-1)/3);adj[p].push(i);adj[i].push(p)}
  return adj;
}

function initSim(){
  round=0;
  const N=+document.getElementById('nodeSel').value;
  const topo=document.getElementById('topoSel').value;
  const place=document.getElementById('placeSel').value;
  let adj;
  if(topo==='chain')adj=genChain(N);
  else if(topo==='tree')adj=genTree(N);
  else{const g=topo==='powerlaw_low'?2.1:2.7;adj=genPowerLaw(N,g)}
  
  const W=cv.width/devicePixelRatio,H=cv.height/devicePixelRatio;
  nodes=adj.map((_,i)=>({id:i,x:W/2+(Math.random()-0.5)*W*0.6,y:H/2+(Math.random()-0.5)*H*0.6,vx:0,vy:0,neighbors:adj[i],state:'D',deg:adj[i].length}));
  edges=[];
  adj.forEach((nb,i)=>nb.forEach(j=>{if(i<j)edges.push({s:i,t:j})}));
  
  // Sort by degree for placement
  let sorted=[...nodes].sort((a,b)=>b.deg-a.deg);
  let nCorrect=Math.floor(N*0.35);
  let targets;
  if(place==='influential')targets=sorted.slice(0,nCorrect).map(n=>n.id);
  else if(place==='peripheral')targets=sorted.slice(-nCorrect).map(n=>n.id);
  else{let shuf=[...Array(N).keys()];for(let i=N-1;i>0;i--){let j=Math.floor(Math.random()*(i+1));[shuf[i],shuf[j]]=[shuf[j],shuf[i]]}targets=shuf.slice(0,nCorrect)}
  
  targets.forEach(id=>{nodes[id].state='T'});
  // Assign some hallucinating
  let remaining=nodes.filter(n=>n.state==='D');
  let nHalluc=Math.floor(remaining.length*0.45);
  for(let i=0;i<nHalluc;i++)remaining[i].state='H';
  
  layoutForce(120);
  updateStats();
  document.getElementById('roundLabel').textContent='0';
}

function layoutForce(iters){
  const W=cv.width/devicePixelRatio,H=cv.height/devicePixelRatio;
  const N=nodes.length;
  for(let iter=0;iter<iters;iter++){
    // Repulsion
    for(let i=0;i<N;i++){nodes[i].vx=0;nodes[i].vy=0}
    for(let i=0;i<N;i++)for(let j=i+1;j<N;j++){
      let dx=nodes[i].x-nodes[j].x,dy=nodes[i].y-nodes[j].y;
      let d2=dx*dx+dy*dy+1;let f=800/d2;
      nodes[i].vx+=dx*f;nodes[i].vy+=dy*f;
      nodes[j].vx-=dx*f;nodes[j].vy-=dy*f;
    }
    // Attraction
    edges.forEach(e=>{
      let dx=nodes[e.t].x-nodes[e.s].x,dy=nodes[e.t].y-nodes[e.s].y;
      let d=Math.sqrt(dx*dx+dy*dy)+0.1;let f=(d-60)*0.02;
      nodes[e.s].vx+=dx/d*f;nodes[e.s].vy+=dy/d*f;
      nodes[e.t].vx-=dx/d*f;nodes[e.t].vy-=dy/d*f;
    });
    // Center gravity
    nodes.forEach(n=>{n.vx+=(W/2-n.x)*0.01;n.vy+=(H/2-n.y)*0.01});
    // Apply
    let damping=0.85;
    nodes.forEach(n=>{
      n.x+=n.vx*damping;n.y+=n.vy*damping;
      n.x=Math.max(20,Math.min(W-20,n.x));
      n.y=Math.max(20,Math.min(H-20,n.y));
    });
  }
  draw();
}

function draw(){
  const W=cv.width/devicePixelRatio,H=cv.height/devicePixelRatio;
  ctx.clearRect(0,0,W,H);
  // Edges
  ctx.strokeStyle='#d5d8dc';ctx.lineWidth=1;
  edges.forEach(e=>{
    ctx.beginPath();ctx.moveTo(nodes[e.s].x,nodes[e.s].y);ctx.lineTo(nodes[e.t].x,nodes[e.t].y);ctx.stroke();
  });
  // Nodes
  const R=nodes.length>60?6:nodes.length>40?8:10;
  nodes.forEach(n=>{
    ctx.beginPath();ctx.arc(n.x,n.y,R+Math.min(n.deg*0.8,6),0,Math.PI*2);
    ctx.fillStyle=COLORS[n.state];ctx.fill();
    ctx.strokeStyle='#fff';ctx.lineWidth=1.5;ctx.stroke();
  });
}

function step(){
  round++;
  document.getElementById('roundLabel').textContent=round;
  let newStates=nodes.map(n=>n.state);
  
  nodes.forEach((n,i)=>{
    let nbs=n.neighbors;if(!nbs.length)return;
    let counts={T:0,H:0,D:0};
    nbs.forEach(j=>counts[nodes[j].state]++);
    let total=nbs.length;
    let fracT=counts.T/total,fracH=counts.H/total;
    
    // RUM-inspired transition: utility depends on neighbor consensus + own state persistence
    let pStay=0.3; // inertia
    if(n.state==='T'){
      // Truthful node: can be swayed to H if many neighbors hallucinate
      if(Math.random()>pStay){
        let utilH=-0.5+1.8*fracH;let utilT=0.5+1.2*fracT;let utilD=-1.0;
        let eH=Math.exp(utilH),eT=Math.exp(utilT),eD=Math.exp(utilD);let s=eH+eT+eD;
        let r=Math.random()*s;
        newStates[i]=r<eT?'T':r<eT+eH?'H':'D';
      }
    }else if(n.state==='H'){
      if(Math.random()>pStay){
        let utilH=0.3+1.4*fracH;let utilT=-0.3+2.0*fracT;let utilD=-1.2;
        let eH=Math.exp(utilH),eT=Math.exp(utilT),eD=Math.exp(utilD);let s=eH+eT+eD;
        let r=Math.random()*s;
        newStates[i]=r<eH?'H':r<eH+eT?'T':'D';
      }
    }else{// D
      let utilH=-0.2+1.5*fracH;let utilT=-0.2+1.8*fracT;let utilD=0.1;
      let eH=Math.exp(utilH),eT=Math.exp(utilT),eD=Math.exp(utilD);let s=eH+eT+eD;
      let r=Math.random()*s;
      newStates[i]=r<eT?'T':r<eT+eH?'H':'D';
    }
  });
  nodes.forEach((n,i)=>n.state=newStates[i]);
  draw();updateStats();
}

function updateStats(){
  let N=nodes.length;
  let c={T:0,H:0,D:0};nodes.forEach(n=>c[n.state]++);
  let rT=(c.T/N),rH=(c.H/N),rD=(c.D/N);
  document.getElementById('barT').style.width=(rT*100)+'%';
  document.getElementById('barH').style.width=(rH*100)+'%';
  document.getElementById('barD').style.width=(rD*100)+'%';
  document.getElementById('valT').textContent=rT.toFixed(2);
  document.getElementById('valH').textContent=rH.toFixed(2);
  document.getElementById('valD').textContent=rD.toFixed(2);
}

function toggleRun(){
  running=!running;
  document.getElementById('runBtn').textContent=running?'⏸ Pause':'▶ Run';
  document.getElementById('runBtn').classList.toggle('running',running);
  if(running)runLoop();else clearTimeout(stepTimer);
}
function runLoop(){if(!running||round>=30)return toggleRun();step();stepTimer=setTimeout(runLoop,400)}
function resetSim(){running=false;document.getElementById('runBtn').textContent='▶ Run';document.getElementById('runBtn').classList.remove('running');clearTimeout(stepTimer);initSim()}

resize();initSim();
document.getElementById('topoSel').onchange=resetSim;
document.getElementById('nodeSel').onchange=resetSim;
document.getElementById('placeSel').onchange=resetSim;
</script>
</body>
</html>
