<!DOCTYPE html>
<html>
<head>
  <meta charset="utf-8">
  <title>3D Graph Viewer</title>
  <style>
    body { margin: 0; font-family: sans-serif; overflow: hidden; }
    #graph { position: absolute; top: 0; left: 0; width: 100vw; height: 100vh; }
  </style>
</head>
<body>
  <div id="graph"></div>
  
  <script src="https://unpkg.com/three@0.149.0/build/three.min.js"></script>
  <script src="https://unpkg.com/3d-force-graph@1.73"></script>
  
  <script>
    // 嵌入图谱数据
    const graphData = {
  "nodes": [
    {
      "id": "end of road",
      "x": 0,
      "y": 0,
      "z": 0,
      "fx": 0,
      "fy": 0,
      "fz": 0
    },
    {
      "id": "inside building",
      "x": 55,
      "y": 0,
      "z": 0,
      "fx": 55,
      "fy": 0,
      "fz": 0
    },
    {
      "id": "at \"y2\"",
      "x": 71.5,
      "y": 16.5,
      "z": 16.5,
      "fx": 71.5,
      "fy": 16.5,
      "fz": 16.5
    },
    {
      "id": "low n/s passage",
      "x": 71.5,
      "y": -38.5,
      "z": 16.5,
      "fx": 71.5,
      "fy": -38.5,
      "fz": 16.5
    },
    {
      "id": "hall of the mountain king",
      "x": 71.5,
      "y": -93.5,
      "z": 16.5,
      "fx": 71.5,
      "fy": -93.5,
      "fz": 16.5
    },
    {
      "id": "secret e/w canyon above tight canyon",
      "x": 16.5,
      "y": -148.5,
      "z": 16.5,
      "fx": 16.5,
      "fy": -148.5,
      "fz": 16.5
    },
    {
      "id": "secret canyon",
      "x": -38.5,
      "y": -148.5,
      "z": 16.5,
      "fx": -38.5,
      "fy": -148.5,
      "fz": 16.5
    },
    {
      "id": "n/s canyon",
      "x": 16.5,
      "y": -148.5,
      "z": -38.5,
      "fx": 16.5,
      "fy": -148.5,
      "fz": -38.5
    },
    {
      "id": "in tall e/w canyon",
      "x": 16.5,
      "y": -93.5,
      "z": -38.5,
      "fx": 16.5,
      "fy": -93.5,
      "fz": -38.5
    },
    {
      "id": "in swiss cheese room",
      "x": 16.5,
      "y": -38.5,
      "z": -38.5,
      "fx": 16.5,
      "fy": -38.5,
      "fz": -38.5
    },
    {
      "id": "at east end of twopit room",
      "x": -38.5,
      "y": -38.5,
      "z": -38.5,
      "fx": -38.5,
      "fy": -38.5,
      "fz": -38.5
    },
    {
      "id": "at west end of twopit room",
      "x": -93.5,
      "y": -38.5,
      "z": -38.5,
      "fx": -93.5,
      "fy": -38.5,
      "fz": -38.5
    },
    {
      "id": "in west pit",
      "x": -93.5,
      "y": -38.5,
      "z": -93.5,
      "fx": -93.5,
      "fy": -38.5,
      "fz": -93.5
    },
    {
      "id": "oriental room",
      "x": 33,
      "y": -22,
      "z": -22,
      "fx": 33,
      "fy": -22,
      "fz": -22
    },
    {
      "id": "misty cavern",
      "x": 33,
      "y": 33,
      "z": -22,
      "fx": 33,
      "fy": 33,
      "fz": -22
    },
    {
      "id": "alcove",
      "x": -22,
      "y": 33,
      "z": -22,
      "fx": -22,
      "fy": 33,
      "fz": -22
    },
    {
      "id": "plover room",
      "x": 33,
      "y": 33,
      "z": -22,
      "fx": 33,
      "fy": 33,
      "fz": -22
    },
    {
      "id": "in soft room",
      "x": 71.5,
      "y": -38.5,
      "z": -38.5,
      "fx": 71.5,
      "fy": -38.5,
      "fz": -38.5
    },
    {
      "id": "bedquilt",
      "x": 71.5,
      "y": 16.5,
      "z": -38.5,
      "fx": 71.5,
      "fy": 16.5,
      "fz": -38.5
    },
    {
      "id": "at complex junction",
      "x": 126.5,
      "y": 16.5,
      "z": -38.5,
      "fx": 126.5,
      "fy": 16.5,
      "fz": -38.5
    },
    {
      "id": "in a valley",
      "x": 0,
      "y": -55,
      "z": 0,
      "fx": 0,
      "fy": -55,
      "fz": 0
    },
    {
      "id": "at slit in streambed",
      "x": 0,
      "y": -110,
      "z": 0,
      "fx": 0,
      "fy": -110,
      "fz": 0
    },
    {
      "id": "outside grate",
      "x": 0,
      "y": -165,
      "z": 0,
      "fx": 0,
      "fy": -165,
      "fz": 0
    },
    {
      "id": "below the grate",
      "x": 0,
      "y": -165,
      "z": -55,
      "fx": 0,
      "fy": -165,
      "fz": -55
    },
    {
      "id": "in cobble",
      "x": -55,
      "y": -165,
      "z": -55,
      "fx": -55,
      "fy": -165,
      "fz": -55
    },
    {
      "id": "in debris room",
      "x": -110,
      "y": -165,
      "z": -55,
      "fx": -110,
      "fy": -165,
      "fz": -55
    },
    {
      "id": "sloping e/w canyon",
      "x": -165,
      "y": -165,
      "z": -55,
      "fx": -165,
      "fy": -165,
      "fz": -55
    },
    {
      "id": "orange river chamber",
      "x": -220,
      "y": -165,
      "z": -55,
      "fx": -220,
      "fy": -165,
      "fz": -55
    },
    {
      "id": "at top of small pit",
      "x": -275,
      "y": -165,
      "z": -55,
      "fx": -275,
      "fy": -165,
      "fz": -55
    },
    {
      "id": "in hall of mists",
      "x": -275,
      "y": -165,
      "z": -110,
      "fx": -275,
      "fy": -165,
      "fz": -110
    },
    {
      "id": "low room",
      "x": -275,
      "y": -220,
      "z": -110,
      "fx": -275,
      "fy": -220,
      "fz": -110
    }
  ],
  "links": [
    {
      "source": "end of road",
      "target": "inside building",
      "action": "east",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "end of road",
      "target": "in a valley",
      "action": "south",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in a valley",
      "target": "end of road",
      "action": "north (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "inside building",
      "target": "end of road",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "inside building",
      "target": "at \"y2\"",
      "action": "plugh",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in a valley",
      "target": "at slit in streambed",
      "action": "south",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "at slit in streambed",
      "target": "in a valley",
      "action": "north (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "at \"y2\"",
      "target": "inside building",
      "action": "plugh",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "at \"y2\"",
      "target": "low n/s passage",
      "action": "south",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "at slit in streambed",
      "target": "outside grate",
      "action": "south",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "outside grate",
      "target": "at slit in streambed",
      "action": "north (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "outside grate",
      "target": "below the grate",
      "action": "down",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "below the grate",
      "target": "outside grate",
      "action": "up (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "below the grate",
      "target": "in cobble",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in cobble",
      "target": "below the grate",
      "action": "east (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "in cobble",
      "target": "in debris room",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in debris room",
      "target": "in cobble",
      "action": "east (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "in debris room",
      "target": "sloping e/w canyon",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "sloping e/w canyon",
      "target": "in debris room",
      "action": "east (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "sloping e/w canyon",
      "target": "orange river chamber",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "orange river chamber",
      "target": "sloping e/w canyon",
      "action": "east (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "orange river chamber",
      "target": "at top of small pit",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "at top of small pit",
      "target": "orange river chamber",
      "action": "east (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "at top of small pit",
      "target": "in hall of mists",
      "action": "down",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in hall of mists",
      "target": "at top of small pit",
      "action": "up (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "in hall of mists",
      "target": "low room",
      "action": "south",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in hall of mists",
      "target": "hall of the mountain king",
      "action": "north",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "hall of the mountain king",
      "target": "in hall of mists",
      "action": "south (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "low room",
      "target": "in hall of mists",
      "action": "north",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "hall of the mountain king",
      "target": "low n/s passage",
      "action": "north",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "hall of the mountain king",
      "target": "secret e/w canyon above tight canyon",
      "action": "southwest",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "secret e/w canyon above tight canyon",
      "target": "hall of the mountain king",
      "action": "northeast (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "low n/s passage",
      "target": "at \"y2\"",
      "action": "north",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "low n/s passage",
      "target": "hall of the mountain king",
      "action": "south",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "secret e/w canyon above tight canyon",
      "target": "secret canyon",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "secret e/w canyon above tight canyon",
      "target": "n/s canyon",
      "action": "down",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "n/s canyon",
      "target": "secret e/w canyon above tight canyon",
      "action": "up (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "secret canyon",
      "target": "secret e/w canyon above tight canyon",
      "action": "east",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "n/s canyon",
      "target": "in tall e/w canyon",
      "action": "north",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in tall e/w canyon",
      "target": "n/s canyon",
      "action": "south (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "in tall e/w canyon",
      "target": "in swiss cheese room",
      "action": "north",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in swiss cheese room",
      "target": "in tall e/w canyon",
      "action": "south (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "in swiss cheese room",
      "target": "at east end of twopit room",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in swiss cheese room",
      "target": "in swiss cheese room",
      "action": "northwest",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in swiss cheese room",
      "target": "oriental room",
      "action": "northwest",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in swiss cheese room",
      "target": "in soft room",
      "action": "east",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in swiss cheese room",
      "target": "bedquilt",
      "action": "northeast",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "bedquilt",
      "target": "in swiss cheese room",
      "action": "southwest (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "at east end of twopit room",
      "target": "at west end of twopit room",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "at east end of twopit room",
      "target": "in swiss cheese room",
      "action": "east",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "oriental room",
      "target": "misty cavern",
      "action": "north",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "oriental room",
      "target": "in swiss cheese room",
      "action": "southeast",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in soft room",
      "target": "in swiss cheese room",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "bedquilt",
      "target": "at complex junction",
      "action": "east",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "at complex junction",
      "target": "bedquilt",
      "action": "west (auto)",
      "color": "red",
      "width": 2
    },
    {
      "source": "at west end of twopit room",
      "target": "in west pit",
      "action": "down",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "at west end of twopit room",
      "target": "at east end of twopit room",
      "action": "east",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "in west pit",
      "target": "at west end of twopit room",
      "action": "up",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "misty cavern",
      "target": "alcove",
      "action": "west",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "misty cavern",
      "target": "oriental room",
      "action": "south",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "alcove",
      "target": "plover room",
      "action": "east",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "alcove",
      "target": "misty cavern",
      "action": "northwest",
      "color": "#000080",
      "width": 1
    },
    {
      "source": "plover room",
      "target": "alcove",
      "action": "west",
      "color": "#000080",
      "width": 1
    }
  ]
};
    
    // 创建3D图谱
    const graph = ForceGraph3D()(document.getElementById('graph'))
      .backgroundColor('#ffffff')
      .graphData(graphData)
      .nodeLabel('id')
      .linkLabel('action')
      .linkColor(link => link.color || '#000080')
      .linkWidth(link => link.width || 1)
      .nodeColor(node => {
        // 检测重叠节点
        const posMap = {};
        graphData.nodes.forEach(n => {
          const key = n.x + ',' + n.y + ',' + n.z;
          if (!posMap[key]) posMap[key] = [];
          posMap[key].push(n);
        });
        const nodeKey = node.x + ',' + node.y + ',' + node.z;
        const overlap = posMap[nodeKey] && posMap[nodeKey].length > 1;
        return overlap ? '#800080' : '#000080';
      })
      .nodeThreeObject(node => {
        const group = new THREE.Group();
        
        // 球体
        const sphereGeometry = new THREE.SphereGeometry(5);
        const sphereMaterial = new THREE.MeshLambertMaterial({
          color: node.__overlap ? '#800080' : '#000080'
        });
        const sphere = new THREE.Mesh(sphereGeometry, sphereMaterial);
        group.add(sphere);
        
        // 文字标签
        const sprite = new THREE.Sprite(new THREE.SpriteMaterial({
          map: new THREE.CanvasTexture((() => {
            const canvas = document.createElement('canvas');
            const ctx = canvas.getContext('2d');
            
            ctx.font = '32px Arial';
            const textMetrics = ctx.measureText(node.id);
            const textWidth = textMetrics.width;
            
            canvas.width = Math.max(512, textWidth + 80);
            canvas.height = 96;
            
            ctx.font = '32px Arial';
            ctx.fillStyle = 'rgba(255, 255, 255, 0.7)';
            ctx.fillRect(0, 0, canvas.width, canvas.height);
            
            ctx.fillStyle = 'rgba(0, 0, 0, 0.8)';
            ctx.textAlign = 'center';
            ctx.textBaseline = 'middle';
            ctx.fillText(node.id, canvas.width / 2, canvas.height / 2);
            
            return canvas;
          })()),
          transparent: true,
          opacity: 0.8
        }));
        const canvasWidth = Math.max(512, node.id.length * 30 + 80);
        sprite.scale.set(canvasWidth / 10, 9.6, 1);
        sprite.position.y = 15;
        group.add(sprite);
        
        return group;
      })
      .linkCurvature(0.25)
      .linkDirectionalArrowLength(4)
      .linkDirectionalArrowRelPos(1);
  </script>
</body>
</html>