<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Geometric Lattice Visualization</title>
    <style>
        body, html {
            margin: 0;
            padding: 0;
            height: 100%;
            overflow: hidden;
            font-family: sans-serif;
        }
        
        #main-container {
            display: flex;
            flex-direction: column;
            height: 100vh;
            background-color: #f5f5f5;
        }
        
        #control-panel {
            padding: 15px;
            background-color: #fff;
            border-bottom: 1px solid #ddd;
            box-shadow: 0 2px 5px rgba(0,0,0,0.1);
        }
        
        #control-panel > div {
            margin-bottom: 10px;
            display: flex;
            align-items: center;
        }
        
        label {
            margin-right: 10px;
            font-weight: bold;
            min-width: 150px;
        }
        
        #slider-trim {
            width: 200px;
            margin-right: 10px;
        }
        
        #canvas-container {
            position: relative;
            flex-grow: 1;
        }
        
        #main-canvas {
            display: block;
            width: 100%;
            height: 100%;
        }
        
        #btn-reset-view {
            position: absolute;
            top: 15px;
            right: 15px;
            width: 30px;
            height: 30px;
            border-radius: 50%;
            background: white;
            border: 1px solid #ccc;
            font-size: 16px;
            cursor: pointer;
            z-index: 10;
            box-shadow: 0 2px 5px rgba(0,0,0,0.1);
        }
        
        #btn-reset-view:hover {
            background-color: #f0f0f0;
        }
    </style>
</head>
<body>
    <div id="main-container">
        <div id="control-panel">
            <div>
                <label>units in x direction</label>
                <input type="radio" name="unitsX" id="radio-x-0" value="0"> 0
                <input type="radio" name="unitsX" id="radio-x-1" value="1"> 1
                <input type="radio" name="unitsX" id="radio-x-2" value="2" checked> 2
                <input type="radio" name="unitsX" id="radio-x-3" value="3"> 3
                <input type="radio" name="unitsX" id="radio-x-4" value="4"> 4
            </div>
            <div>
                <label>units in y direction</label>
                <input type="radio" name="unitsY" id="radio-y-0" value="0"> 0
                <input type="radio" name="unitsY" id="radio-y-1" value="1"> 1
                <input type="radio" name="unitsY" id="radio-y-2" value="2" checked> 2
                <input type="radio" name="unitsY" id="radio-y-3" value="3"> 3
                <input type="radio" name="unitsY" id="radio-y-4" value="4"> 4
            </div>
            <div>
                <label>trim</label>
                <input type="range" id="slider-trim" min="0.65" max="0.9" step="0.001" value="0.8">
                <span id="label-trim-value">0.800</span>
            </div>
            <div>
                <input type="checkbox" id="checkbox-double"> <label for="checkbox-double">double</label>
                <input type="checkbox" id="checkbox-sphere"> <label for="checkbox-sphere">sphere</label>
                <input type="checkbox" id="checkbox-triangles"> <label for="checkbox-triangles">triangles</label>
            </div>
        </div>
        <div id="canvas-container">
            <canvas id="main-canvas"></canvas>
            <button id="btn-reset-view">+</button>
        </div>
    </div>

    <script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/r128/three.min.js"></script>
    <script src="https://cdn.jsdelivr.net/npm/three@0.128.0/examples/js/controls/OrbitControls.js"></script>
    
    <script>
        // Global variables
        let scene, camera, renderer, controls;
        let mainGroup = new THREE.Group();
        
        // Initialize the scene
        function init() {
            // Set up scene
            scene = new THREE.Scene();
            scene.background = new THREE.Color(0xf0f0f0);
            
            // Set up camera
            const container = document.getElementById('canvas-container');
            camera = new THREE.PerspectiveCamera(75, container.clientWidth / container.clientHeight, 0.1, 100);
            camera.position.set(0, 0, 5);
            
            // Set up renderer
            renderer = new THREE.WebGLRenderer({ 
                canvas: document.getElementById('main-canvas'),
                antialias: true 
            });
            renderer.setPixelRatio(window.devicePixelRatio);
            renderer.setSize(container.clientWidth, container.clientHeight);
            
            // Set up controls
            controls = new THREE.OrbitControls(camera, renderer.domElement);
            controls.enableDamping = true;
            controls.dampingFactor = 0.05;
            
            // Add lighting
            const hemisphereLight = new THREE.HemisphereLight(0xffffff, 0x444444, 0.8);
            scene.add(hemisphereLight);
            
            const directionalLight = new THREE.DirectionalLight(0xffffff, 0.8);
            directionalLight.position.set(1, 1, 1);
            scene.add(directionalLight);
            
            // Add main group to scene
            scene.add(mainGroup);
            
            // Set up event listeners
            setupEventListeners();
            
            // Initial scene update
            updateScene();
            
            // Start animation loop
            animate();
            
            // Handle window resize
            window.addEventListener('resize', onWindowResize);
        }
        
        // Set up all UI event listeners
        function setupEventListeners() {
            // Radio buttons
            const radioX = document.querySelectorAll('input[name="unitsX"]');
            const radioY = document.querySelectorAll('input[name="unitsY"]');
            
            radioX.forEach(radio => radio.addEventListener('change', updateScene));
            radioY.forEach(radio => radio.addEventListener('change', updateScene));
            
            // Checkboxes
            document.getElementById('checkbox-double').addEventListener('change', updateScene);
            document.getElementById('checkbox-sphere').addEventListener('change', updateScene);
            document.getElementById('checkbox-triangles').addEventListener('change', updateScene);
            
            // Trim slider
            const trimSlider = document.getElementById('slider-trim');
            const trimLabel = document.getElementById('label-trim-value');
            
            trimSlider.addEventListener('input', function() {
                trimLabel.textContent = parseFloat(this.value).toFixed(3);
                updateScene();
            });
            
            // Reset view button
            document.getElementById('btn-reset-view').addEventListener('click', function() {
                controls.reset();
            });
        }
        
        // Create a curved triangle geometry
        function createCurvedTriangle(trimValue) {
            const segments = 20;
            const geometry = new THREE.BufferGeometry();
            
            // Base positions for vertices
            const a = new THREE.Vector3(1, 0, trimValue).normalize();
            const b = new THREE.Vector3(-0.5, Math.sqrt(3)/2, trimValue).normalize();
            const c = new THREE.Vector3(-0.5, -Math.sqrt(3)/2, trimValue).normalize();
            
            // Vertex positions array
            const vertices = [a.clone()];
            const colors = [];
            
            // Generate edges
            function generateEdge(start, end) {
                for (let i = 1; i < segments; i++) {
                    const t = i / segments;
                    const point = new THREE.Vector3();
                    point.lerpVectors(start, end, t);
                    point.normalize();
                    vertices.push(point);
                }
            }
            
            // Create edges
            generateEdge(a, b);
            vertices.push(b.clone());
            generateEdge(b, c);
            vertices.push(c.clone());
            generateEdge(c, a);
            
            // Create faces and vertex colors
            const indices = [];
            const center = new THREE.Vector3(0, 0, 1);
            const centerIndex = vertices.length;
            vertices.push(center);
            
            // Create faces (triangle fan)
            for (let i = 0; i < vertices.length - 1; i++) {
                const next = (i + 1) % (vertices.length - 1);
                indices.push(centerIndex, i, next);
            }
            
            // Set vertex colors (gradient)
            const startColor = new THREE.Color(0xffc0cb);
            const endColor = new THREE.Color(0x904090);
            
            for (let i = 0; i < vertices.length; i++) {
                if (i === centerIndex) {
                    colors.push(startColor.r, startColor.g, startColor.b);
                } else {
                    colors.push(endColor.r, endColor.g, endColor.b);
                }
            }
            
            // Set geometry attributes
            geometry.setFromPoints(vertices);
            geometry.setIndex(indices);
            geometry.setAttribute('color', new THREE.Float32BufferAttribute(colors, 3));
            
            // Create outline
            const lineGeometry = new THREE.BufferGeometry();
            lineGeometry.setFromPoints(vertices.slice(0, -1));
            
            return { geometry, lineGeometry };
        }
        
        // Create a secondary curved triangle (double)
        function createSecondaryCurvedTriangle(trimValue) {
            const segments = 20;
            const geometry = new THREE.BufferGeometry();
            
            // Base positions for vertices
            const a = new THREE.Vector3(1, 0, trimValue).normalize();
            const b = new THREE.Vector3(-0.5, Math.sqrt(3)/2, trimValue).normalize();
            const c = new THREE.Vector3(-0.5, -Math.sqrt(3)/2, trimValue).normalize();
            
            // Vertex positions array
            const vertices = [a.clone()];
            const colors = [];
            
            // Generate edges
            function generateEdge(start, end) {
                for (let i = 1; i < segments; i++) {
                    const t = i / segments;
                    const point = new THREE.Vector3();
                    point.lerpVectors(start, end, t);
                    point.normalize();
                    vertices.push(point);
                }
            }
            
            // Create edges
            generateEdge(a, b);
            vertices.push(b.clone());
            generateEdge(b, c);
            vertices.push(c.clone());
            generateEdge(c, a);
            
            // Create faces and vertex colors
            const indices = [];
            const center = new THREE.Vector3(0, 0, 1);
            const centerIndex = vertices.length;
            vertices.push(center);
            
            // Create faces
            for (let i = 0; i < vertices.length - 1; i++) {
                const next = (i + 1) % (vertices.length - 1);
                indices.push(centerIndex, i, next);
            }
            
            // Set vertex colors (different gradient)
            const startColor = new THREE.Color(0x80ffff);
            const endColor = new THREE.Color(0x008080);
            
            for (let i = 0; i < vertices.length; i++) {
                if (i === centerIndex) {
                    colors.push(startColor.r, startColor.g, startColor.b);
                } else {
                    colors.push(endColor.r, endColor.g, endColor.b);
                }
            }
            
            // Set geometry attributes
            geometry.setFromPoints(vertices);
            geometry.setIndex(indices);
            geometry.setAttribute('color', new THREE.Float32BufferAttribute(colors, 3));
            
            return geometry;
        }
        
        // Update the scene based on UI controls
        function updateScene() {
            // Clear previous objects
            while (mainGroup.children.length > 0) {
                mainGroup.remove(mainGroup.children[0]);
            }
            
            // Get current values from UI
            const trimValue = parseFloat(document.getElementById('slider-trim').value);
            const showDouble = document.getElementById('checkbox-double').checked;
            const showSphere = document.getElementById('checkbox-sphere').checked;
            const showTriangles = document.getElementById('checkbox-triangles').checked;
            
            const unitsX = parseInt(document.querySelector('input[name="unitsX"]:checked').value);
            const unitsY = parseInt(document.querySelector('input[name="unitsY"]:checked').value);
            
            // Basis vectors for triangular lattice
            const v1 = new THREE.Vector3(1, 0, 0);
            const v2 = new THREE.Vector3(0.5, Math.sqrt(3)/2, 0);
            
            // Create the primary lattice
            for (let i = -unitsX; i <= unitsX; i++) {
                for (let j = -unitsY; j <= unitsY; j++) {
                    const position = new THREE.Vector3()
                        .addScaledVector(v1, i)
                        .addScaledVector(v2, j);
                    
                    createLatticeElement(position, trimValue, (i+j) % 2 === 1);
                }
            }
            
            // Create the secondary lattice if enabled
            if (showDouble) {
                for (let i = -unitsX; i <= unitsX; i++) {
                    for (let j = -unitsY; j <= unitsY; j++) {
                        const position = new THREE.Vector3()
                            .addScaledVector(v1, i + 0.5)
                            .addScaledVector(v2, j + 0.5);
                        
                        createSecondaryLatticeElement(position, trimValue);
                    }
                }
            }
            
            // Add sphere if enabled
            if (showSphere) {
                const sphereGeometry = new THREE.SphereGeometry(1, 32, 32);
                const sphereMaterial = new THREE.MeshPhongMaterial({ 
                    color: 0xdddddd,
                    shininess: 80
                });
                const sphere = new THREE.Mesh(sphereGeometry, sphereMaterial);
                mainGroup.add(sphere);
            }
            
            // Add planar triangles if enabled
            if (showTriangles) {
                createPlanarTriangles(unitsX, unitsY);
            }
        }
        
        // Create a lattice element with rotation
        function createLatticeElement(position, trimValue, rotate) {
            // Create triangle geometry
            const { geometry, lineGeometry } = createCurvedTriangle(trimValue);
            
            // Create materials
            const material = new THREE.MeshStandardMaterial({
                vertexColors: true,
                flatShading: true,
                side: THREE.DoubleSide
            });
            
            const lineMaterial = new THREE.LineBasicMaterial({ 
                color: 0x000000,
                linewidth: 2
            });
            
            // Create mesh and outline
            const mesh = new THREE.Mesh(geometry, material);
            const outline = new THREE.LineLoop(lineGeometry, lineMaterial);
            
            // Apply rotation if needed
            if (rotate) {
                mesh.rotation.y = Math.PI;
                outline.rotation.y = Math.PI;
            }
            
            // Position the element
            mesh.position.copy(position);
            outline.position.copy(position);
            
            // Add to group
            mainGroup.add(mesh);
            mainGroup.add(outline);
        }
        
        // Create secondary lattice element
        function createSecondaryLatticeElement(position, trimValue) {
            const geometry = createSecondaryCurvedTriangle(trimValue);
            
            // Create material
            const material = new THREE.MeshStandardMaterial({
                vertexColors: true,
                flatShading: true,
                side: THREE.DoubleSide
            });
            
            // Create mesh and apply rotation
            const mesh = new THREE.Mesh(geometry, material);
            mesh.rotation.y = Math.PI;
            
            // Position the element
            mesh.position.copy(position);
            
            // Add to group
            mainGroup.add(mesh);
        }
        
        // Create planar triangles and vertices
        function createPlanarTriangles(unitsX, unitsY) {
            // Basis vectors for triangular lattice
            const v1 = new THREE.Vector3(1, 0, 0);
            const v2 = new THREE.Vector3(0.5, Math.sqrt(3)/2, 0);
            const trimValue = parseFloat(document.getElementById('slider-trim').value);
            
            // Materials
            const triangleMaterial = new THREE.MeshBasicMaterial({
                color: 0x800080,
                transparent: true,
                opacity: 0.4,
                side: THREE.DoubleSide
            });
            
            const vertexMaterial = new THREE.PointsMaterial({
                color: 0x000000,
                size: 0.05
            });
            
            // Collect all vertices
            const allVertices = [];
            
            // Create primary lattice triangles
            for (let i = -unitsX; i <= unitsX; i++) {
                for (let j = -unitsY; j <= unitsY; j++) {
                    const position = new THREE.Vector3()
                        .addScaledVector(v1, i)
                        .addScaledVector(v2, j);
                    
                    // Base positions for vertices
                    const a = new THREE.Vector3(1, 0, trimValue).add(position);
                    const b = new THREE.Vector3(-0.5, Math.sqrt(3)/2, trimValue).add(position);
                    const c = new THREE.Vector3(-0.5, -Math.sqrt(3)/2, trimValue).add(position);
                    
                    // Create triangle geometry
                    const geometry = new THREE.BufferGeometry();
                    geometry.setFromPoints([a, b, c]);
                    geometry.computeVertexNormals();
                    
                    // Create triangle mesh
                    const triangle = new THREE.Mesh(geometry, triangleMaterial);
                    mainGroup.add(triangle);
                    
                    // Add to vertices collection
                    allVertices.push(a, b, c);
                }
            }
            
            // Create secondary lattice triangles if enabled
            if (document.getElementById('checkbox-double').checked) {
                for (let i = -unitsX; i <= unitsX; i++) {
                    for (let j = -unitsY; j <= unitsY; j++) {
                        const position = new THREE.Vector3()
                            .addScaledVector(v1, i + 0.5)
                            .addScaledVector(v2, j + 0.5);
                        
                        // Base positions for vertices
                        const a = new THREE.Vector3(1, 0, trimValue).add(position);
                        const b = new THREE.Vector3(-0.5, Math.sqrt(3)/2, trimValue).add(position);
                        const c = new THREE.Vector3(-0.5, -Math.sqrt(3)/2, trimValue).add(position);
                        
                        // Create triangle geometry
                        const geometry = new THREE.BufferGeometry();
                        geometry.setFromPoints([a, b, c]);
                        geometry.computeVertexNormals();
                        
                        // Create triangle mesh
                        const triangle = new THREE.Mesh(geometry, triangleMaterial);
                        mainGroup.add(triangle);
                        
                        // Add to vertices collection
                        allVertices.push(a, b, c);
                    }
                }
            }
            
            // Create vertex points
            if (allVertices.length > 0) {
                const vertexGeometry = new THREE.BufferGeometry();
                vertexGeometry.setFromPoints(allVertices);
                
                const vertices = new THREE.Points(vertexGeometry, vertexMaterial);
                mainGroup.add(vertices);
            }
        }
        
        // Handle window resize
        function onWindowResize() {
            const container = document.getElementById('canvas-container');
            camera.aspect = container.clientWidth / container.clientHeight;
            camera.updateProjectionMatrix();
            renderer.setSize(container.clientWidth, container.clientHeight);
        }
        
        // Animation loop
        function animate() {
            requestAnimationFrame(animate);
            controls.update();
            renderer.render(scene, camera);
        }
        
        // Initialize when the page loads
        window.addEventListener('load', init);
    </script>
</body>
</html>