<!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>
    <link rel="stylesheet" href="styles.css">
</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>点击选择JSON文件</p>
                        <div class="file-type-selector">
                            <label for="file-type">文件类型:</label>
                            <select id="file-type">
                                <option value="voxels">体素数据 (occ.json格式)</option>
                                <option value="centers">中心点数据 (centers.json格式)</option>
                            </select>
                        </div>
                        <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>
                    <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>
                    <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">保存中心点</button>
                    <div class="category-selector">
                        <label for="category-select">选择类别:</label>
                        <select id="category-select">
                            <option value="all">全部类别</option>
                        </select>
                        <button id="show-centers-coordinates">显示中心点坐标</button>
                    </div>
                    <button id="delete-center-point">删除中心点</button>
                    <button id="delete-unselected-points">删除未选中点</button>
                    <button id="split-point">分割选中点</button>
                </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 id="edit-info" class="edit-info" style="display: none;">
                        <p><strong>编辑模式说明:</strong></p>
                        <p>启用编辑模式后，可以拖动中心点调整位置</p>
                    </div>
                </div>
                
                <div id="legend" class="legend"></div>
            </div>
        </div>
        
        <div class="instructions">
            <h3>交互说明</h3>
            <ul>
                <li><strong>旋转视图</strong>: 按住鼠标左键并拖动 - 控制X和Y轴旋转</li>
                <li><strong>Z轴旋转</strong>: 按住鼠标右键或Ctrl键并拖动 - 控制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>: 启用编辑模式后，可以拖动中心点调整位置</li>
                <li><strong>删除中心点</strong>: 启用编辑模式后，右键点击中心点可删除</li>
                <li><strong>选择中心点</strong>: 启用编辑模式后，按住Ctrl键并点击中心点可选择/取消选择</li>
                <li><strong>删除未选中点</strong>: 启用编辑模式后，点击"删除未选中点"按钮可删除所有未选中的点</li>
                <li><strong>按类别删除点</strong>: 启用编辑模式后，选择一个具体类别，然后点击"按类别删除点"按钮可删除该类别的所有点</li>
                <li><strong>分割选中点</strong>: 启用编辑模式后，选择一个点，然后点击"分割选中点"按钮可将该点分割为两个相同类别的点（其中一个保持选中状态）</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>
        </div>
        
        <footer>
            <p>基于D3.js实现的3D Occupancy可视化工具 | 交互设计类似Open3D</p>
        </footer>
    </div>
    
    <script src="config.js"></script>
    <script src="data.js"></script>
    <script src="projection.js"></script>
    <script src="ui.js"></script>
    <script>
        // 初始化函数
        function init() {
            initElements();
            setupCanvas();
            createLegend();
            initEvents();
            
            // 加载示例数据
            state.voxelData = generateDemoData();
            state.centerPointsFromFile = generateDemoCenters();
            calculateCenters();
            
            // 隐藏初始加载提示
            elements.loading.style.display = 'none';
            
            // 设置初始UI值
            elements.voxelSize.value = CONFIG.DEFAULT_VOXEL_SIZE;
            elements.voxelSizeValue.textContent = CONFIG.DEFAULT_VOXEL_SIZE;
            elements.opacity.value = CONFIG.DEFAULT_OPACITY;
            elements.opacityValue.textContent = `${CONFIG.DEFAULT_OPACITY}%`;
            elements.centerSize.value = CONFIG.DEFAULT_CENTER_SIZE;
            elements.centerSizeValue.textContent = CONFIG.DEFAULT_CENTER_SIZE;
            
            updateSelectedCategoriesText();
            
            // 启动动画循环
            animate();
        }

        // 页面加载完成后初始化
        window.addEventListener('load', init);
    </script>
</body>
</html>