<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>3D Occupancy 可视化工具</title>
    <script src="https://d3js.org/d3.v7.min.js"></script>
    <style>
        * {
            box-sizing: border-box;
            margin: 0;
            padding: 0;
        }
        body {
            font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
            background: linear-gradient(135deg, #1a2a6c, #b21f1f, #fdbb2d);
            color: white;
            min-height: 100vh;
            display: flex;
            flex-direction: column;
            align-items: center;
            padding: 20px;
            overflow-x: hidden;
        }
        .container {
            max-width: 1400px;
            width: 100%;
            background: rgba(0, 0, 0, 0.7);
            border-radius: 15px;
            padding: 25px;
            box-shadow: 0 10px 30px rgba(0, 0, 0, 0.5);
            backdrop-filter: blur(10px);
        }
        header {
            text-align: center;
            margin-bottom: 30px;
            padding-bottom: 20px;
            border-bottom: 1px solid rgba(255, 255, 255, 0.2);
        }
        h1 {
            font-size: 2.8rem;
            margin-bottom: 10px;
            text-shadow: 2px 2px 4px rgba(0, 0, 0, 0.5);
        }
        .description {
            font-size: 1.2rem;
            opacity: 0.9;
            max-width: 800px;
            margin: 0 auto;
        }
        .visualization-area {
            display: flex;
            flex-wrap: wrap;
            gap: 30px;
            margin: 20px 0;
        }
        .canvas-container {
            flex: 2;
            min-width: 700px;
            height: 600px;
            background: #000;
            border-radius: 10px;
            overflow: hidden;
            position: relative;
            box-shadow: 0 5px 15px rgba(0, 0, 0, 0.5);
        }
        #occupancy-canvas {
            width: 100%;
            height: 100%;
            cursor: grab;
        }
        #occupancy-canvas:active {
            cursor: grabbing;
        }
        .controls-panel {
            flex: 1;
            min-width: 300px;
            background: rgba(30, 30, 40, 0.8);
            border-radius: 10px;
            padding: 20px;
            display: flex;
            flex-direction: column;
            gap: 20px;
        }
        .control-group {
            margin-bottom: 15px;
        }
        h3 {
            font-size: 1.3rem;
            margin-bottom: 15px;
            color: #fdbb2d;
        }
        .slider-container {
            margin-bottom: 15px;
        }
        .slider-label {
            display: flex;
            justify-content: space-between;
            margin-bottom: 5px;
        }
        input[type="range"] {
            width: 100%;
            height: 8px;
            -webkit-appearance: none;
            background: #4a4a6d;
            border-radius: 5px;
            outline: none;
        }
        input[type="range"]::-webkit-slider-thumb {
            -webkit-appearance: none;
            width: 20px;
            height: 20px;
            border-radius: 50%;
            background: #fdbb2d;
            cursor: pointer;
        }
        button {
            background: #1a2a6c;
            color: white;
            border: none;
            padding: 12px 20px;
            border-radius: 8px;
            cursor: pointer;
            font-size: 1rem;
            transition: background 0.3s;
            width: 100%;
            margin-top: 5px;
        }
        button:hover {
            background: #2a3a9c;
        }
        button.active {
            background: #fdbb2d;
            color: #1a2a6c;
        }
        button.save-btn {
            background: #27ae60;
        }
        button.save-btn:hover {
            background: #2ecc71;
        }
        button.delete-btn {
            background: #c0392b;
        }
        button.delete-btn:hover {
            background: #e74c3c;
        }
        .color-legend {
            display: grid;
            grid-template-columns: 1fr 1fr;
            gap: 10px;
            margin-top: 10px;
        }
        .color-item {
            display: flex;
            align-items: center;
        }
        .color-box {
            width: 20px;
            height: 20px;
            margin-right: 10px;
            border: 1px solid rgba(255, 255, 255, 0.3);
            border-radius: 4px;
        }
        .instructions {
            background: rgba(30, 30, 40, 0.8);
            padding: 20px;
            border-radius: 10px;
            margin-top: 30px;
        }
        .instructions h3 {
            color: #fdbb2d;
            margin-bottom: 15px;
        }
        .instructions ul {
            padding-left: 20px;
        }
        .instructions li {
            margin-bottom: 10px;
            line-height: 1.5;
        }
        .status-bar {
            display: flex;
            justify-content: space-between;
            margin-top: 15px;
            font-size: 0.9rem;
            color: #ccc;
        }
        footer {
            margin-top: 30px;
            text-align: center;
            font-size: 0.9rem;
            color: #ccc;
        }
        .loading {
            position: absolute;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            font-size: 1.2rem;
            color: white;
            text-align: center;
        }
        .error {
            color: #ff5252;
            text-align: center;
            padding: 20px;
        }
        .upload-area {
            border: 2px dashed rgba(255, 255, 255, 0.3);
            border-radius: 8px;
            padding: 15px;
            text-align: center;
            margin-top: 15px;
            cursor: pointer;
            transition: background 0.3s;
        }
        .upload-area:hover {
            background: rgba(255, 255, 255, 0.1);
        }
        .upload-area p {
            margin-bottom: 10px;
        }
        #file-input, #points-file-input {
            display: none;
        }
        .data-source {
            display: flex;
            gap: 10px;
            margin-bottom: 15px;
        }
        .data-source button {
            flex: 1;
        }
        .active-source {
            background: #fdbb2d !important;
            color: #1a2a6c !important;
        }
        .rotation-controls {
            display: grid;
            grid-template-columns: 1fr 1fr;
            gap: 10px;
        }
        .rotation-control {
            background: rgba(255, 255, 255, 0.1);
            padding: 10px;
            border-radius: 8px;
            text-align: center;
        }
        .rotation-value {
            font-weight: bold;
            color: #fdbb2d;
            margin-top: 5px;
        }
        .edit-info {
            background: rgba(52, 152, 219, 0.2);
            padding: 10px;
            border-radius: 8px;
            margin-top: 10px;
            font-size: 0.9rem;
        }
        .points-source {
            margin-top: 10px;
            display: flex;
            flex-direction: column;
            gap: 10px;
        }
        .points-source-btn {
            background: #8e44ad !important;
        }
        .points-source-btn.active-source {
            background: #fdbb2d !important;
        }
        @media (max-width: 1100px) {
            .visualization-area {
                flex-direction: column;
            }
            .canvas-container {
                min-width: 100%;
                height: 500px;
            }
        }
        @media (max-width: 768px) {
            .canvas-container {
                height: 400px;
            }
            .container {
                padding: 15px;
            }
            h1 {
                font-size: 2.2rem;
            }
        }
        .selected-center {
            animation: pulse 1.5s infinite;
        }
        @keyframes pulse {
            0% { transform: scale(1); }
            50% { transform: scale(1.2); }
            100% { transform: scale(1); }
        }
    </style>
</head>
<body>
    <div class="container">
        <header>
            <h1>3D Occupancy 可视化工具</h1>
            <p class="description">使用类Open3D交互方式可视化12个类别的3D occupancy数据</p>
        </header>
        
        <div class="visualization-area">
            <div class="canvas-container">
                <canvas id="occupancy-canvas"></canvas>
                <div id="loading" class="loading">请上传或加载数据</div>
            </div>
            
            <div class="controls-panel">
                <div class="control-group">
                    <h3>数据源</h3>
                    <div class="data-source">
                        <button id="load-demo" class="active-source">示例数据</button>
                        <button id="load-file">上传数据</button>
                    </div>
                    <div id="upload-area" class="upload-area" style="display: none;">
                        <p>点击选择occ.json文件</p>
                        <button>选择文件</button>
                        <input type="file" id="file-input" accept=".json">
                    </div>

                    <div class="points-source">
                        <h3>中心点数据源</h3>
                        <div class="data-source">
                            <button id="calc-centers" class="points-source-btn active-source">自动计算</button>
                            <button id="load-points-file" class="points-source-btn">从文件加载</button>
                        </div>
                        <div id="upload-points-area" class="upload-area" style="display: none;">
                            <p>点击选择points.json文件</p>
                            <button>选择文件</button>
                            <input type="file" id="points-file-input" accept=".json">
                        </div>
                    </div>
                </div>
                
                <div class="control-group">
                    <h3>视角控制</h3>
                    <button id="reset-view">重置视角</button>
                    <button id="toggle-rotation">自动旋转: 关闭</button>
                    <button id="toggle-centers">显示中心点: 关闭</button>
                    <button id="toggle-edit-mode">编辑模式: 关闭</button>
                    <button id="save-centers" class="save-btn">保存中心点</button>
                    <button id="delete-center" class="delete-btn">删除选中中心点</button>
                    
                    <div class="edit-info" id="edit-info" style="display: none;">
                        <p>编辑模式已启用：</p>
                        <p>• 点击中心点可以选择</p>
                        <p>• 拖动中心点可以调整位置</p>
                        <p>• 按Delete键或点击删除按钮删除选中中心点</p>
                        <p>• 完成后点击"保存中心点"</p>
                    </div>
                    
                    <div class="rotation-controls">
                        <div class="rotation-control">
                            <div>X旋转</div>
                            <div id="rotation-x-value" class="rotation-value">0°</div>
                        </div>
                        <div class="rotation-control">
                            <div>Y旋转</div>
                            <div id="rotation-y-value" class="rotation-value">0°</div>
                        </div>
                        <div class="rotation-control">
                            <div>Z旋转</div>
                            <div id="rotation-z-value" class="rotation-value">0°</div>
                        </div>
                        <div class="rotation-control">
                            <div>缩放</div>
                            <div id="scale-value" class="rotation-value">1.0x</div>
                        </div>
                    </div>
                </div>
                
                <div class="control-group">
                    <h3>显示选项</h3>
                    <div class="slider-container">
                        <div class="slider-label">
                            <span>体素大小</span>
                            <span id="voxel-size-value">5</span>
                        </div>
                        <input type="range" id="voxel-size" min="1" max="10" value="5">
                    </div>
                    
                    <div class="slider-container">
                        <div class="slider-label">
                            <span>透明度</span>
                            <span id="opacity-value">80%</span>
                        </div>
                        <input type="range" id="opacity" min="10" max="100" value="80">
                    </div>
                    
                    <div class="slider-container">
                        <div class="slider-label">
                            <span>中心点大小</span>
                            <span id="center-size-value">8</span>
                        </div>
                        <input type="range" id="center-size" min="3" max="15" value="8">
                    </div>
                </div>
                
                <div class="control-group">
                    <h3>类别筛选</h3>
                    <div class="color-legend" id="legend"></div>
                </div>
            </div>
        </div>
        
        <div class="instructions">
            <h3>使用说明</h3>
            <ul>
                <li><strong>旋转视图</strong>: 按住鼠标左键并拖动 - 控制X和Y轴旋转</li>
                <li><strong>Z轴旋转</strong>: 按住鼠标右键并拖动 - 控制Z轴旋转</li>
                <li><strong>缩放视图</strong>: 使用鼠标滚轮</li>
                <li><strong>平移视图</strong>: 按住Shift键+鼠标拖动</li>
                <li><strong>选择/取消选择类别</strong>: 点击右侧图例中的颜色块</li>
                <li><strong>重置视图</strong>: 点击"重置视角"按钮</li>
                <li><strong>显示中心点</strong>: 点击"显示中心点"按钮</li>
                <li><strong>编辑中心点</strong>: 启用编辑模式后，可以点击选择中心点，拖动调整位置，或按Delete键删除</li>
                <li><strong>保存中心点</strong>: 编辑完成后点击"保存中心点"按钮</li>
                <li><strong>上传数据</strong>: 点击"上传数据"按钮，然后选择本地的occ.json文件</li>
                <li><strong>中心点数据</strong>: 可以选择自动计算中心点或从points.json文件加载</li>
            </ul>
        </div>
        
        <div class="status-bar">
            <span id="selected-categories">显示类别: 全部</span>
            <span id="voxel-count">体素数量: 0</span>
            <span id="center-count">中心点: 0</span>
            <span id="center-source">中心点来源: 自动计算</span>
            <span id="rotation-status">旋转: 关闭</span>
            <span id="edit-status">编辑: 关闭</span>
            <span id="selected-center">选中中心点: 无</span>
        </div>
        
        <footer>
            <p>基于D3.js实现的3D Occupancy可视化工具 | 交互设计类似Open3D</p>
        </footer>
    </div>

    <script>
        // 配置参数
        const GRID_SIZE = 20;
        const COLORS = [
            "#FF5252", "#FF4081", "#E040FB", "#7C4DFF", 
            "#536DFE", "#448AFF", "#40C4FF", "#18FFFF",
            "#64FFDA", "#69F0AE", "#B2FF59", "#EEFF41"
        ];
        const CENTER_COLOR = "#FFFFFF"; // 中心点颜色
        
        // 状态变量
        let rotationX = 0;
        let rotationY = 0;
        let rotationZ = 0;
        let scale = 1;
        let offsetX = 0;
        let offsetY = 0;
        let isDragging = false;
        let isEditing = false;
        let isDraggingCenter = false;
        let lastX, lastY;
        let autoRotate = false;
        let showCenters = false;
        let animationId;
        let voxelData = [];
        let centerPoints = [];
        let selectedCategories = new Set([...Array(12).keys()]);
        let draggedCenterIndex = -1;
        let selectedCenterIndex = -1;
        let originalCenterPos = null;
        let centerPointsFromFile = [];
        let useFileCenters = false;
        
        // 初始化画布
        const canvas = document.getElementById('occupancy-canvas');
        const ctx = canvas.getContext('2d');
        const loadingElement = document.getElementById('loading');
        
        // 设置画布尺寸
        function setupCanvas() {
            const container = canvas.parentElement;
            canvas.width = container.clientWidth;
            canvas.height = container.clientHeight;
        }
        
        // 生成示例数据
        function generateDemoData() {
            const data = [];
            
            // 生成一些随机形状
            const shapes = [
                {type: 'sphere', x: 5, y: 5, z: 5, r: 4, category: 0},
                {type: 'cube', x: 15, y: 5, z: 5, size: 5, category: 3},
                {type: 'sphere', x: 5, y: 15, z: 15, r: 5, category: 6},
                {type: 'cube', x: 15, y: 15, z: 15, size: 4, category: 9},
            ];
            
            for (let x = 0; x < GRID_SIZE; x++) {
                for (let y = 0; y < GRID_SIZE; y++) {
                    for (let z = 0; z < GRID_SIZE; z++) {
                        let category = 0;
                        
                        // 检查点是否在任何形状内
                        for (const shape of shapes) {
                            if (shape.type === 'sphere') {
                                const dx = x - shape.x;
                                const dy = y - shape.y;
                                const dz = z - shape.z;
                                const distance = Math.sqrt(dx*dx + dy*dy + dz*dz);
                                
                                if (distance <= shape.r) {
                                    category = shape.category;
                                    break;
                                }
                            } else if (shape.type === 'cube') {
                                if (Math.abs(x - shape.x) <= shape.size/2 &&
                                    Math.abs(y - shape.y) <= shape.size/2 &&
                                    Math.abs(z - shape.z) <= shape.size/2) {
                                    category = shape.category;
                                    break;
                                }
                            }
                        }
                        
                        // 随机添加一些噪点
                        if (category === 0 && Math.random() < 0.01) {
                            category = Math.floor(Math.random() * 11) + 1;
                        }
                        
                        if (category > 0) {
                            data.push({x, y, z, category});
                        }
                    }
                }
            }
            
            return data;
        }
        
        // 生成示例中心点数据
        function generateDemoCenters() {
            return [
                {x: 5, y: 5, z: 5, category: 0},
                {x: 15, y: 5, z: 5, category: 3},
                {x: 5, y: 15, z: 15, category: 6},
                {x: 15, y: 15, z: 15, category: 9}
            ];
        }
        
        // 计算中心点
        function calculateCenters() {
            // 按类别分组
            const groups = {};
            voxelData.forEach(voxel => {
                if (!groups[voxel.category]) {
                    groups[voxel.category] = [];
                }
                groups[voxel.category].push(voxel);
            });
            
            // 计算每个组的中心点
            centerPoints = [];
            Object.keys(groups).forEach(category => {
                const points = groups[category];
                const sum = points.reduce((acc, point) => {
                    acc.x += point.x;
                    acc.y += point.y;
                    acc.z += point.z;
                    return acc;
                }, {x: 0, y: 0, z: 0});
                
                centerPoints.push({
                    x: sum.x / points.length,
                    y: sum.y / points.length,
                    z: sum.z / points.length,
                    category: parseInt(category)
                });
            });
            
            document.getElementById('center-count').textContent = `中心点: ${centerPoints.length}`;
            updateSelectedCenterText();
        }
        
        // 处理上传的JSON文件
        function handleJsonFile(file, isPointsFile = false) {
            loadingElement.style.display = 'block';
            loadingElement.textContent = '正在解析数据...';
            
            const reader = new FileReader();
            reader.onload = function(e) {
                try {
                    const jsonData = JSON.parse(e.target.result);
                    
                    // 验证数据格式
                    if (!Array.isArray(jsonData)) {
                        throw new Error('数据格式错误: 应为数组');
                    }
                    
                    if (jsonData.length > 0) {
                        const firstItem = jsonData[0];
                        const requiredFields = isPointsFile ? 
                            ['x', 'y', 'z', 'category'] : 
                            ['x', 'y', 'z', 'category'];
                            
                        for (const field of requiredFields) {
                            if (!firstItem.hasOwnProperty(field)) {
                                throw new Error(`数据格式错误: 每个项应包含${requiredFields.join(', ')}字段`);
                            }
                        }
                    }
                    
                    loadingElement.style.display = 'none';
                    
                    if (isPointsFile) {
                        centerPointsFromFile = jsonData;
                        if (useFileCenters) {
                            centerPoints = centerPointsFromFile;
                            document.getElementById('center-source').textContent = '中心点来源: 文件';
                            drawScene();
                        }
                    } else {
                        voxelData = jsonData;
                        if (!useFileCenters) {
                            calculateCenters();
                        }
                        drawScene();
                    }
                } catch (error) {
                    loadingElement.innerHTML = `<div class="error">解析JSON失败: ${error.message}</div>`;
                }
            };
            
            reader.onerror = function() {
                loadingElement.innerHTML = '<div class="error">读取文件失败</div>';
            };
            
            reader.readAsText(file);
        }
        
        // 3D到2D投影 - 使用四元数避免万向节锁
        function project(x, y, z) {
            // 将欧拉角转换为四元数
            const cy = Math.cos(rotationY * 0.5);
            const sy = Math.sin(rotationY * 0.5);
            const cp = Math.cos(rotationX * 0.5);
            const sp = Math.sin(rotationX * 0.5);
            const cr = Math.cos(rotationZ * 0.5);
            const sr = Math.sin(rotationZ * 0.5);
            
            // 四元数
            const w = cr * cp * cy + sr * sp * sy;
            const xq = sr * cp * cy - cr * sp * sy;
            const yq = cr * sp * cy + sr * cp * sy;
            const zq = cr * cp * sy - sr * sp * cy;
            
            // 使用四元数旋转点
            const ix = w * x + yq * z - zq * y;
            const iy = w * y + zq * x - xq * z;
            const iz = w * z + xq * y - yq * x;
            const iw = -xq * x - yq * y - zq * z;
            
            // 应用旋转
            const xr = ix * w + iw * -xq + iy * -zq - iz * -yq;
            const yr = iy * w + iw * -yq + iz * -xq - ix * -zq;
            const zr = iz * w + iw * -zq + ix * -yq - iy * -xq;
            
            // 应用缩放和平移
            const scaleFactor = 10 * scale;
            const screenX = (xr * scaleFactor) + canvas.width/2 + offsetX;
            const screenY = (yr * scaleFactor) + canvas.height/2 + offsetY;
            
            // 返回深度信息用于排序
            return { x: screenX, y: screenY, z: zr };
        }
        
        // 反向投影：从屏幕坐标转换为3D坐标（近似）
        function unproject(screenX, screenY, depth) {
            // 转换为相对于画布中心的坐标
            const x = (screenX - canvas.width/2 - offsetX) / (10 * scale);
            const y = (screenY - canvas.height/2 - offsetY) / (10 * scale);
            const z = depth;
            
            // 反向旋转（使用四元数的逆）
            const cy = Math.cos(-rotationY * 0.5);
            const sy = Math.sin(-rotationY * 0.5);
            const cp = Math.cos(-rotationX * 0.5);
            const sp = Math.sin(-rotationX * 0.5);
            const cr = Math.cos(-rotationZ * 0.5);
            const sr = Math.sin(-rotationZ * 0.5);
            
            const w = cr * cp * cy + sr * sp * sy;
            const xq = sr * cp * cy - cr * sp * sy;
            const yq = cr * sp * cy + sr * cp * sy;
            const zq = cr * cp * sy - sr * sp * cy;
            
            // 使用四元数反向旋转点
            const ix = w * x + yq * z - zq * y;
            const iy = w * y + zq * x - xq * z;
            const iz = w * z + xq * y - yq * x;
            const iw = -xq * x - yq * y - zq * z;
            
            const xr = ix * w + iw * -xq + iy * -zq - iz * -yq;
            const yr = iy * w + iw * -yq + iz * -xq - ix * -zq;
            const zr = iz * w + iw * -zq + ix * -yq - iy * -xq;
            
            return { x: xr, y: yr, z: zr };
        }
        
        // 绘制场景
        function drawScene() {
            // 清除画布
            ctx.clearRect(0, 0, canvas.width, canvas.height);
            
            if (voxelData.length === 0) return;
            
            // 绘制背景网格
            drawGrid();
            
            // 对体素按深度排序（从远到近）
            const sortedVoxels = voxelData
                .filter(voxel => selectedCategories.has(voxel.category))
                .map(voxel => {
                    const pos = project(voxel.x, voxel.y, voxel.z);
                    return { ...voxel, screenX: pos.x, screenY: pos.y, depth: pos.z };
                })
                .sort((a, b) => b.depth - a.depth);
            
            // 绘制体素
            const voxelSize = parseInt(document.getElementById('voxel-size').value);
            const opacity = parseInt(document.getElementById('opacity').value) / 100;
            
            sortedVoxels.forEach(voxel => {
                const color = COLORS[voxel.category % COLORS.length];
                const alpha = opacity * (0.7 + 0.3 * (voxel.depth + GRID_SIZE) / (2 * GRID_SIZE));
                
                ctx.fillStyle = color.replace(')', `, ${alpha})`).replace('rgb', 'rgba');
                ctx.fillRect(
                    voxel.screenX - voxelSize/2,
                    voxel.screenY - voxelSize/2,
                    voxelSize,
                    voxelSize
                );
                
                // 绘制边框
                ctx.strokeStyle = 'rgba(255, 255, 255, 0.2)';
                ctx.lineWidth = 1;
                ctx.strokeRect(
                    voxel.screenX - voxelSize/2,
                    voxel.screenY - voxelSize/2,
                    voxelSize,
                    voxelSize
                );
            });
            
            // 绘制中心点
            if (showCenters && centerPoints.length > 0) {
                const centerSize = parseInt(document.getElementById('center-size').value);
                
                centerPoints
                    .filter(center => selectedCategories.has(center.category))
                    .forEach((center, index) => {
                        const pos = project(center.x, center.y, center.z);
                        
                        // 绘制中心点
                        ctx.fillStyle = CENTER_COLOR;
                        ctx.beginPath();
                        ctx.arc(pos.x, pos.y, centerSize, 0, Math.PI * 2);
                        ctx.fill();
                        
                        // 绘制外圈
                        ctx.strokeStyle = COLORS[center.category % COLORS.length];
                        ctx.lineWidth = 2;
                        ctx.beginPath();
                        ctx.arc(pos.x, pos.y, centerSize + 2, 0, Math.PI * 2);
                        ctx.stroke();
                        
                        // 如果这个中心点被选中，绘制特殊效果
                        if (selectedCenterIndex === index) {
                            ctx.strokeStyle = '#FFD700';
                            ctx.lineWidth = 3;
                            ctx.beginPath();
                            ctx.arc(pos.x, pos.y, centerSize + 4, 0, Math.PI * 2);
                            ctx.stroke();
                        }
                        
                        // 如果正在拖动这个中心点，绘制特殊效果
                        if (isDraggingCenter && draggedCenterIndex === index) {
                            ctx.strokeStyle = '#FF6B00';
                            ctx.lineWidth = 3;
                            ctx.beginPath();
                            ctx.arc(pos.x, pos.y, centerSize + 4, 0, Math.PI * 2);
                            ctx.stroke();
                        }
                    });
            }
            
            // 更新状态信息
            document.getElementById('voxel-count').textContent = `体素数量: ${sortedVoxels.length}`;
            updateRotationValues();
        }
        
        // 更新旋转值显示
        function updateRotationValues() {
            document.getElementById('rotation-x-value').textContent = `${Math.round(rotationX * 180 / Math.PI)}°`;
            document.getElementById('rotation-y-value').textContent = `${Math.round(rotationY * 180 / Math.PI)}°`;
            document.getElementById('rotation-z-value').textContent = `${Math.round(rotationZ * 180 / Math.PI)}°`;
            document.getElementById('scale-value').textContent = `${scale.toFixed(1)}x`;
        }
        
        // 更新选中中心点文本
        function updateSelectedCenterText() {
            if (selectedCenterIndex === -1) {
                document.getElementById('selected-center').textContent = '选中中心点: 无';
            } else {
                const center = centerPoints[selectedCenterIndex];
                document.getElementById('selected-center').textContent = 
                    `选中中心点: 类别 ${center.category} (${center.x.toFixed(1)}, ${center.y.toFixed(1)}, ${center.z.toFixed(1)})`;
            }
        }
        
        // 绘制背景网格
        function drawGrid() {
            ctx.strokeStyle = 'rgba(255, 255, 255, 0.1)';
            ctx.lineWidth = 1;
            
            // 绘制XY平面网格
            for (let i = -GRID_SIZE; i <= GRID_SIZE; i += 5) {
                // X轴
                const startX = project(-GRID_SIZE, i, 0);
                const endX = project(GRID_SIZE, i, 0);
                ctx.beginPath();
                ctx.moveTo(startX.x, startX.y);
                ctx.lineTo(endX.x, endX.y);
                ctx.stroke();
                
                // Y轴
                const startY = project(i, -GRID_SIZE, 0);
                const endY = project(i, GRID_SIZE, 0);
                ctx.beginPath();
                ctx.moveTo(startY.x, startY.y);
                ctx.lineTo(endY.x, endY.y);
                ctx.stroke();
            }
            
            // 绘制坐标轴
            const origin = project(0, 0, 0);
            const xAxis = project(GRID_SIZE/2, 0, 0);
            const yAxis = project(0, GRID_SIZE/2, 0);
            const zAxis = project(0, 0, GRID_SIZE/2);
            
            // X轴 (红色)
            ctx.strokeStyle = '#FF5252';
            ctx.lineWidth = 2;
            ctx.beginPath();
            ctx.moveTo(origin.x, origin.y);
            ctx.lineTo(xAxis.x, xAxis.y);
            ctx.stroke();
            
            // Y轴 (绿色)
            ctx.strokeStyle = '#69F0AE';
            ctx.lineWidth = 2;
            ctx.beginPath();
            ctx.moveTo(origin.x, origin.y);
            ctx.lineTo(yAxis.x, yAxis.y);
            ctx.stroke();
            
            // Z轴 (蓝色)
            ctx.strokeStyle = '#448AFF';
            ctx.lineWidth = 2;
            ctx.beginPath();
            ctx.moveTo(origin.x, origin.y);
            ctx.lineTo(zAxis.x, zAxis.y);
            ctx.stroke();
        }
        
        // 创建图例
        function createLegend() {
            const legend = d3.select('#legend');
            legend.html(''); // 清空现有内容
            
            for (let i = 0; i < 12; i++) {
                const item = legend.append('div')
                    .attr('class', 'color-item')
                    .style('cursor', 'pointer')
                    .on('click', function() {
                        if (selectedCategories.has(i)) {
                            selectedCategories.delete(i);
                            d3.select(this).style('opacity', 0.4);
                        } else {
                            selectedCategories.add(i);
                            d3.select(this).style('opacity', 1);
                        }
                        updateSelectedCategoriesText();
                        drawScene();
                    });
                
                item.append('div')
                    .attr('class', 'color-box')
                    .style('background-color', COLORS[i]);
                
                item.append('span')
                    .text(`类别 ${i}`);
            }
        }
        
        // 更新选中的类别文本
        function updateSelectedCategoriesText() {
            const selected = document.getElementById('selected-categories');
            if (selectedCategories.size === 12) {
                selected.textContent = '显示类别: 全部';
            } else if (selectedCategories.size === 0) {
                selected.textContent = '显示类别: 无';
            } else {
                selected.textContent = `显示类别: ${Array.from(selectedCategories).join(', ')}`;
            }
        }
        
        // 保存中心点数据
        function saveCenterPoints() {
            // 创建要保存的数据结构
            const dataToSave = centerPoints.map(center => ({
                x: parseFloat(center.x.toFixed(2)),
                y: parseFloat(center.y.toFixed(2)),
                z: parseFloat(center.z.toFixed(2)),
                category: center.category
            }));
            
            // 创建JSON字符串
            const jsonString = JSON.stringify(dataToSave, null, 2);
            
            // 创建Blob对象
            const blob = new Blob([jsonString], { type: 'application/json' });
            
            // 创建下载链接
            const url = URL.createObjectURL(blob);
            const a = document.createElement('a');
            a.href = url;
            a.download = 'centers.json';
            a.click();
            
            // 清理URL
            URL.revokeObjectURL(url);
            
            alert('中心点数据已保存为centers.json文件');
        }
        
        // 删除选中的中心点
        function deleteSelectedCenter() {
            if (selectedCenterIndex === -1) {
                alert('请先选择一个中心点');
                return;
            }
            
            if (confirm('确定要删除这个中心点吗？')) {
                centerPoints.splice(selectedCenterIndex, 1);
                selectedCenterIndex = -1;
                updateSelectedCenterText();
                document.getElementById('center-count').textContent = `中心点: ${centerPoints.length}`;
                drawScene();
            }
        }
        
        // 动画循环
        function animate() {
            if (autoRotate) {
                rotationY += 0.005;
                rotationX += 0.002;
                rotationZ += 0.003;
            }
            
            drawScene();
            animationId = requestAnimationFrame(animate);
        }
        
        // 切换中心点数据源
        function toggleCenterSource(useFile) {
            useFileCenters = useFile;
            
            if (useFileCenters) {
                document.getElementById('calc-centers').classList.remove('active-source');
                document.getElementById('load-points-file').classList.add('active-source');
                document.getElementById('upload-points-area').style.display = 'block';
                
                if (centerPointsFromFile.length > 0) {
                    centerPoints = centerPointsFromFile;
                    document.getElementById('center-source').textContent = '中心点来源: 文件';
                } else {
                    document.getElementById('center-source').textContent = '中心点来源: 文件(未上传)';
                }
            } else {
                document.getElementById('calc-centers').classList.add('active-source');
                document.getElementById('load-points-file').classList.remove('active-source');
                document.getElementById('upload-points-area').style.display = 'none';
                
                calculateCenters();
                document.getElementById('center-source').textContent = '中心点来源: 自动计算';
            }
            
            drawScene();
        }
        
        // 初始化事件监听
        function initEvents() {
            let isRightClick = false;
            let isShiftPressed = false;
            
            // 鼠标拖动旋转
            canvas.addEventListener('mousedown', (e) => {
                const mouseX = e.clientX - canvas.getBoundingClientRect().left;
                const mouseY = e.clientY - canvas.getBoundingClientRect().top;
                
                // 检查是否点击了中心点（在编辑模式下）
                if (isEditing && showCenters) {
                    const centerSize = parseInt(document.getElementById('center-size').value);
                    
                    for (let i = 0; i < centerPoints.length; i++) {
                        const center = centerPoints[i];
                        const pos = project(center.x, center.y, center.z);
                        const distance = Math.sqrt(Math.pow(mouseX - pos.x, 2) + Math.pow(mouseY - pos.y, 2));
                        
                        if (distance <= centerSize + 5) {
                            isDraggingCenter = true;
                            draggedCenterIndex = i;
                            selectedCenterIndex = i;
                            originalCenterPos = {x: center.x, y: center.y, z: center.z};
                            canvas.style.cursor = 'grabbing';
                            updateSelectedCenterText();
                            e.preventDefault();
                            return;
                        }
                    }
                }
                
                // 如果不是在拖动中心点，则进行正常的视角操作
                isDragging = true;
                lastX = e.clientX;
                lastY = e.clientY;
                canvas.style.cursor = 'grabbing';
                isRightClick = e.button === 2; // 只有鼠标右键才设置为true
                
                // 防止事件冒泡
                e.preventDefault();
            });
            
            canvas.addEventListener('mousemove', (e) => {
                const mouseX = e.clientX - canvas.getBoundingClientRect().left;
                const mouseY = e.clientY - canvas.getBoundingClientRect().top;
                
                // 处理中心点拖动
                if (isDraggingCenter && draggedCenterIndex >= 0) {
                    const center = centerPoints[draggedCenterIndex];
                    const originalPos = project(originalCenterPos.x, originalCenterPos.y, originalCenterPos.z);
                    
                    // 计算鼠标移动距离
                    const dx = mouseX - originalPos.x;
                    const dy = mouseY - originalPos.y;
                    
                    // 转换为3D空间中的移动（简化处理，假设在XY平面上移动）
                    const moveX = dx / (10 * scale);
                    const moveY = -dy / (10 * scale); // Y轴方向相反
                    
                    // 更新中心点位置
                    center.x = originalCenterPos.x + moveX;
                    center.y = originalCenterPos.y + moveY;
                    
                    e.preventDefault();
                    return;
                }
                
                if (!isDragging) return;
                
                const deltaX = e.clientX - lastX;
                const deltaY = e.clientY - lastY;
                
                if (isShiftPressed) {
                    // 平移视图
                    offsetX += deltaX;
                    offsetY += deltaY;
                } else if (isRightClick) {
                    // 右键拖动 - Z轴旋转
                    rotationZ += deltaX * 0.01;
                    drawScene(); // 重新绘制场景以反映Z轴旋转
                } else {
                    // 左键拖动 - X和Y轴旋转
                    rotationY += deltaX * 0.01;
                    rotationX += deltaY * 0.01;
                    
                    // 限制X旋转角度在-π/2到π/2之间
                    rotationX = Math.max(-Math.PI/2, Math.min(Math.PI/2, rotationX));
                    drawScene(); // 重新绘制场景以反映X/Y轴旋转
                }
                
                lastX = e.clientX;
                lastY = e.clientY;
            });
            
            canvas.addEventListener('mouseup', () => {
                if (isDraggingCenter) {
                    isDraggingCenter = false;
                    draggedCenterIndex = -1;
                    originalCenterPos = null;
                    canvas.style.cursor = isEditing ? 'pointer' : 'grab';
                    return;
                }
                
                isDragging = false;
                canvas.style.cursor = isEditing ? 'pointer' : 'grab';
                isRightClick = false;
            });
            
            canvas.addEventListener('mouseleave', () => {
                if (isDraggingCenter) {
                    isDraggingCenter = false;
                    draggedCenterIndex = -1;
                    originalCenterPos = null;
                }
                
                isDragging = false;
                canvas.style.cursor = isEditing ? 'pointer' : 'grab';
                isRightClick = false;
            });
            
            // 阻止右键菜单
            canvas.addEventListener('contextmenu', (e) => {
                e.preventDefault();
            });
            
            // 键盘事件 - 检测Shift键
            window.addEventListener('keydown', (e) => {
                if (e.key === 'Shift') {
                    isShiftPressed = true;
                    canvas.style.cursor = 'move';
                }
                
                // 删除键 - 删除选中的中心点
                if ((e.key === 'Delete' || e.key === 'Backspace') && isEditing && selectedCenterIndex !== -1) {
                    deleteSelectedCenter();
                }
            });
            
            window.addEventListener('keyup', (e) => {
                if (e.key === 'Shift') {
                    isShiftPressed = false;
                    canvas.style.cursor = isEditing ? 'pointer' : 'grab';
                }
            });
            
            // 鼠标滚轮缩放
            canvas.addEventListener('wheel', (e) => {
                e.preventDefault();
                scale += e.deltaY * -0.001;
                scale = Math.max(0.5, Math.min(5, scale));
            });
            
            // 重置视图按钮
            document.getElementById('reset-view').addEventListener('click', () => {
                rotationX = 0;
                rotationY = 0;
                rotationZ = 0;
                scale = 1;
                offsetX = 0;
                offsetY = 0;
                selectedCenterIndex = -1;
                updateSelectedCenterText();
            });
            
            // 自动旋转切换
            document.getElementById('toggle-rotation').addEventListener('click', function() {
                autoRotate = !autoRotate;
                this.textContent = `自动旋转: ${autoRotate ? '开启' : '关闭'}`;
                document.getElementById('rotation-status').textContent = `旋转: ${autoRotate ? '开启' : '关闭'}`;
            });
            
            // 显示中心点切换
            document.getElementById('toggle-centers').addEventListener('click', function() {
                showCenters = !showCenters;
                this.textContent = `显示中心点: ${showCenters ? '开启' : '关闭'}`;
                drawScene();
            });
            
            // 编辑模式切换
            document.getElementById('toggle-edit-mode').addEventListener('click', function() {
                isEditing = !isEditing;
                this.textContent = `编辑模式: ${isEditing ? '开启' : '关闭'}`;
                document.getElementById('edit-status').textContent = `编辑: ${isEditing ? '开启' : '关闭'}`;
                document.getElementById('edit-info').style.display = isEditing ? 'block' : 'none';
                canvas.style.cursor = isEditing ? 'pointer' : 'grab';
                
                // 退出编辑模式时取消选中中心点
                if (!isEditing) {
                    selectedCenterIndex = -1;
                    updateSelectedCenterText();
                }
            });
            
            // 保存中心点按钮
            document.getElementById('save-centers').addEventListener('click', saveCenterPoints);
            
            // 删除中心点按钮
            document.getElementById('delete-center').addEventListener('click', deleteSelectedCenter);
            
            // 体素大小滑块
            document.getElementById('voxel-size').addEventListener('input', function() {
                document.getElementById('voxel-size-value').textContent = this.value;
                drawScene();
            });
            
            // 透明度滑块
            document.getElementById('opacity').addEventListener('input', function() {
                document.getElementById('opacity-value').textContent = `${this.value}%`;
                drawScene();
            });
            
            // 中心点大小滑块
            document.getElementById('center-size').addEventListener('input', function() {
                document.getElementById('center-size-value').textContent = this.value;
                drawScene();
            });
            
            // 数据源切换
            document.getElementById('load-demo').addEventListener('click', function() {
                document.getElementById('load-demo').classList.add('active-source');
                document.getElementById('load-file').classList.remove('active-source');
                document.getElementById('upload-area').style.display = 'none';
                
                loadingElement.style.display = 'block';
                loadingElement.textContent = '正在生成示例数据...';
                
                setTimeout(() => {
                    voxelData = generateDemoData();
                    centerPointsFromFile = generateDemoCenters();
                    
                    if (useFileCenters) {
                        centerPoints = centerPointsFromFile;
                        document.getElementById('center-source').textContent = '中心点来源: 文件';
                    } else {
                        calculateCenters();
                        document.getElementById('center-source').textContent = '中心点来源: 自动计算';
                    }
                    
                    loadingElement.style.display = 'none';
                    drawScene();
                }, 500);
            });
            
            document.getElementById('load-file').addEventListener('click', function() {
                document.getElementById('load-demo').classList.remove('active-source');
                document.getElementById('load-file').classList.add('active-source');
                document.getElementById('upload-area').style.display = 'block';
            });
            
            // 文件上传处理
            document.getElementById('upload-area').addEventListener('click', function() {
                document.getElementById('file-input').click();
            });
            
            document.getElementById('file-input').addEventListener('change', function(e) {
                if (e.target.files.length > 0) {
                    handleJsonFile(e.target.files[0], false);
                }
            });
            
            // 中心点数据源切换
            document.getElementById('calc-centers').addEventListener('click', function() {
                toggleCenterSource(false);
            });
            
            document.getElementById('load-points-file').addEventListener('click', function() {
                toggleCenterSource(true);
            });
            
            // 中心点文件上传处理
            document.getElementById('upload-points-area').addEventListener('click', function() {
                document.getElementById('points-file-input').click();
            });
            
            document.getElementById('points-file-input').addEventListener('change', function(e) {
                if (e.target.files.length > 0) {
                    handleJsonFile(e.target.files[0], true);
                }
            });
        }
        
        // 初始化函数
        function init() {
            setupCanvas();
            createLegend();
            initEvents();
            updateSelectedCategoriesText();
            updateSelectedCenterText();
            
            // 默认加载示例数据
            document.getElementById('load-demo').click();
            
            // 启动动画循环
            animate();
            
            // 响应窗口大小变化
            window.addEventListener('resize', () => {
                setupCanvas();
                drawScene();
            });
        }
        
        // 启动应用
        init();
    </script>
</body>
</html>