<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Procedural Pattern Visualization</title>
    <style>
        body {
            margin: 0;
            font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
            background-color: #f0f0f0;
            color: #333;
            overflow: hidden;
        }

        #main-container {
            display: flex;
            flex-direction: column;
            height: 100vh;
        }

        #control-panel {
            padding: 10px 15px;
            background-color: #ffffff;
            border-bottom: 1px solid #ccc;
            display: flex;
            flex-wrap: wrap;
            justify-content: center;
            align-items: center;
            gap: 24px;
            box-shadow: 0 2px 4px rgba(0,0,0,0.1);
            z-index: 10;
        }
        
        #control-panel > div {
            display: flex;
            align-items: center;
            gap: 8px;
        }
        
        label {
            font-weight: bold;
            font-size: 14px;
            color: #555;
        }

        input[type="radio"], input[type="checkbox"] {
            margin-right: 2px;
        }
        
        input[type="range"] {
            width: 150px;
            vertical-align: middle;
        }
        
        #label-trim-value {
            font-family: monospace;
            background-color: #eee;
            padding: 2px 6px;
            border-radius: 4px;
            min-width: 40px;
            text-align: center;
        }

        #canvas-container {
            flex-grow: 1;
            position: relative;
        }

        #main-canvas {
            display: block;
            width: 100%;
            height: 100%;
        }

        #btn-reset-view {
            position: absolute;
            top: 10px;
            right: 10px;
            width: 32px;
            height: 32px;
            font-size: 24px;
            font-weight: bold;
            border: 1px solid #999;
            background-color: rgba(255, 255, 255, 0.8);
            border-radius: 50%;
            cursor: pointer;
            line-height: 28px;
            text-align: center;
            color: #555;
            padding: 0;
            padding-bottom: 2px;
        }
        #btn-reset-view:hover {
            background-color: white;
            color: #000;
        }
    </style>
</head>
<body>

    <!-- Main container -->
    <div id="main-container">

        <!-- Control Panel -->
        <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 for="slider-trim">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>

        <!-- Visualization Area -->
        <div id="canvas-container">
            <canvas id="main-canvas"></canvas>
            <button id="btn-reset-view">+</button>
        </div>
    </div>

    <!-- Scripts -->
    <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>
        let scene, camera, renderer, controls, mainGroup;

        const canvas = document.getElementById('main-canvas');
        const resetButton = document.getElementById('btn-reset-view');

        // --- CONTROLS ---
        const controlsMap = {
            unitsX: document.querySelectorAll('input[name="unitsX"]'),
            unitsY: document.querySelectorAll('input[name="unitsY"]'),
            trim: document.getElementById('slider-trim'),
            trimValueLabel: document.getElementById('label-trim-value'),
            double: document.getElementById('checkbox-double'),
            sphere: document.getElementById('checkbox-sphere'),
            triangles: document.getElementById('checkbox-triangles'),
        };

        function getControlState() {
            return {
                unitsX: parseInt(document.querySelector('input[name="unitsX"]:checked').value),
                unitsY: parseInt(document.querySelector('input[name="unitsY"]:checked').value),
                trim: parseFloat(controlsMap.trim.value),
                showDouble: controlsMap.double.checked,
                showSphere: controlsMap.sphere.checked,
                showTriangles: controlsMap.triangles.checked,
            };
        }

        // --- SCENE INITIALIZATION ---
        function init() {
            // Scene
            scene = new THREE.Scene();
            scene.background = new THREE.Color(0xf0f0f0);

            // Camera
            const fov = 50;
            const aspect = canvas.clientWidth / canvas.clientHeight;
            const near = 0.1;
            const far = 1000;
            camera = new THREE.PerspectiveCamera(fov, aspect, near, far);
            camera.position.set(0, 0, 35);
            
            // Renderer
            renderer = new THREE.WebGLRenderer({
                canvas: canvas,
                antialias: true
            });
            renderer.setSize(canvas.clientWidth, canvas.clientHeight);
            renderer.setPixelRatio(window.devicePixelRatio);
            
            // Controls
            controls = new THREE.OrbitControls(camera, renderer.domElement);
            controls.enableDamping = true;
            controls.dampingFactor = 0.05;
            controls.minDistance = 5;
            controls.maxDistance = 100;

            // Lights
            const hemisphereLight = new THREE.HemisphereLight(0xffffff, 0x444444, 1.0);
            scene.add(hemisphereLight);

            const directionalLight = new THREE.DirectionalLight(0xffffff, 0.8);
            directionalLight.position.set(5, 5, 10);
            scene.add(directionalLight);

            // Main group to hold all generated geometry
            mainGroup = new THREE.Group();
            scene.add(mainGroup);

            // Event Listeners
            setupEventListeners();
            
            // Initial scene generation
            updateScene();

            // Start animation loop
            animate();
        }

        // --- EVENT LISTENERS ---
        function setupEventListeners() {
            Object.values(controlsMap).flat().forEach(control => {
                if (control.id === 'slider-trim') {
                    control.addEventListener('input', () => {
                        controlsMap.trimValueLabel.textContent = parseFloat(control.value).toFixed(3);
                        updateScene();
                    });
                } else if(control.type === 'radio' || control.type === 'checkbox') {
                    control.addEventListener('change', updateScene);
                }
            });

            resetButton.addEventListener('click', () => controls.reset());

            window.addEventListener('resize', onWindowResize);
        }

        function onWindowResize() {
            camera.aspect = canvas.clientWidth / canvas.clientHeight;
            camera.updateProjectionMatrix();
            renderer.setSize(canvas.clientWidth, canvas.clientHeight);
        }

        // --- ANIMATION LOOP ---
        function animate() {
            requestAnimationFrame(animate);
            controls.update();
            renderer.render(scene, camera);
        }

        // --- SCENE UPDATE LOGIC ---
        function updateScene() {
            // Clear previous objects
            mainGroup.clear();

            const state = getControlState();
            const R = 2.0; // Base radius for a single triangle
            const tileWidth = R * Math.sqrt(3);
            const tileHeight = R * 1.5;

            const primaryColors = { center: new THREE.Color(0xffc0cb), edge: new THREE.Color(0x904090) };
            const doubleColors = { center: new THREE.Color(0x80ffff), edge: new THREE.Color(0x008080) };

            // Generate lattice
            for (let j = -state.unitsY; j <= state.unitsY; j++) {
                for (let i = -state.unitsX; i <= state.unitsX; i++) {
                    const isEvenRow = Math.abs(j % 2) === 0;
                    const offsetX = isEvenRow ? 0 : tileWidth / 2;
                    const posX = i * tileWidth + offsetX;
                    const posY = j * tileHeight;

                    // Primary Pattern (alternating up/down to tessellate)
                    const rotZ = isEvenRow ? 0 : Math.PI;
                    const primaryShape = createCurvedTriangle(R, state.trim, primaryColors);
                    primaryShape.position.set(posX, posY, 0);
                    primaryShape.rotation.z = rotZ;
                    mainGroup.add(primaryShape);

                    // Doubled Pattern (interwoven, rotated 180 deg from primary)
                    if (state.showDouble) {
                        const rotZDouble = isEvenRow ? Math.PI : 0;
                        const doubleShape = createCurvedTriangle(R, state.trim, doubleColors);
                        doubleShape.position.set(posX, posY, 0);
                        doubleShape.rotation.z = rotZDouble;
                        mainGroup.add(doubleShape);
                    }

                    // Planar Triangles
                    if (state.showTriangles) {
                        const triangleGroup = createPlanarTriangle(R);
                        triangleGroup.position.set(posX, posY, 0);
                        triangleGroup.rotation.z = rotZ;
                        mainGroup.add(triangleGroup);
                    }
                }
            }
            
            // Reference Sphere
            if (state.showSphere) {
                const sphereGeom = new THREE.SphereGeometry(R, 32, 32);
                const sphereMat = new THREE.MeshPhongMaterial({ color: 0xdddddd, shininess: 80 });
                const sphereMesh = new THREE.Mesh(sphereGeom, sphereMat);
                // Position under the central triangle at origin
                sphereMesh.position.z = 0;
                mainGroup.add(sphereMesh);
            }
        }
        
        // --- GEOMETRY GENERATION ---

        function createCurvedTriangle(radius, trim, colors) {
            const group = new THREE.Group();
            const subdivisions = 20;

            // 1. Define base vertices of equilateral triangle in XY plane
            const p1 = new THREE.Vector3(0, radius, 0);
            const p2 = new THREE.Vector3(-radius * Math.sqrt(3) / 2, -radius / 2, 0);
            const p3 = new THREE.Vector3(radius * Math.sqrt(3) / 2, -radius / 2, 0);
            
            // 2. Define Bezier control points (pulled towards origin by 'trim')
            const c12 = new THREE.Vector3().addVectors(p1, p2).multiplyScalar(0.5 * trim);
            const c23 = new THREE.Vector3().addVectors(p2, p3).multiplyScalar(0.5 * trim);
            const c31 = new THREE.Vector3().addVectors(p3, p1).multiplyScalar(0.5 * trim);

            const curve1 = new THREE.QuadraticBezierCurve3(p1, c12, p2);
            const curve2 = new THREE.QuadraticBezierCurve3(p2, c23, p3);
            const curve3 = new THREE.QuadraticBezierCurve3(p3, c31, p1);

            // 3. Generate points on curves and project to sphere surface
            const boundaryPoints = [];
            [curve1, curve2, curve3].forEach(curve => {
                const points = curve.getPoints(subdivisions);
                for (let i = 0; i < points.length -1; i++) { 
                    const sphericalPoint = points[i].clone();
                    sphericalPoint.z = Math.sqrt(Math.max(0, radius*radius - sphericalPoint.x*sphericalPoint.x - sphericalPoint.y*sphericalPoint.y));
                    boundaryPoints.push(sphericalPoint);
                }
            });
             const lastPoint = curve3.getPoint(1).clone();
             lastPoint.z = Math.sqrt(Math.max(0, radius*radius - lastPoint.x*lastPoint.x - lastPoint.y*lastPoint.y));
             boundaryPoints.push(lastPoint);

            // 4. Create the mesh geometry as a cone (triangle fan from origin)
            const geometry = new THREE.BufferGeometry();
            const positions = [];
            const vertColors = [];
            
            // Center point (origin, will be tip of cone)
            positions.push(0, 0, 0);
            vertColors.push(colors.center.r, colors.center.g, colors.center.b);

            // Boundary points
            boundaryPoints.forEach(p => {
                positions.push(p.x, p.y, p.z);
                vertColors.push(colors.edge.r, colors.edge.g, colors.edge.b);
            });

            // Indices for triangle fan
            const indices = [];
            for (let i = 1; i < boundaryPoints.length; i++) {
                indices.push(0, i, i + 1);
            }
            indices.push(0, boundaryPoints.length, 1); // close the fan

            geometry.setAttribute('position', new THREE.Float32BufferAttribute(positions, 3));
            geometry.setAttribute('color', new THREE.Float32BufferAttribute(vertColors, 3));
            geometry.setIndex(indices);
            geometry.computeVertexNormals();

            const material = new THREE.MeshStandardMaterial({
                vertexColors: true,
                side: THREE.DoubleSide
            });
            const mesh = new THREE.Mesh(geometry, material);
            group.add(mesh);
            
            // 5. Create the outline
            const lineGeom = new THREE.BufferGeometry().setFromPoints(boundaryPoints);
            const lineMat = new THREE.LineBasicMaterial({ color: 0x000000, linewidth: 2 });
            const line = new THREE.LineLoop(lineGeom, lineMat);
            group.add(line);

            return group;
        }
        
        function createPlanarTriangle(radius) {
            const group = new THREE.Group();
            
            const p1 = new THREE.Vector3(0, radius, 0);
            const p2 = new THREE.Vector3(-radius * Math.sqrt(3) / 2, -radius / 2, 0);
            const p3 = new THREE.Vector3(radius * Math.sqrt(3) / 2, -radius / 2, 0);
            const pointsArray = [p1, p2, p3];

            // Triangle mesh
            const triGeom = new THREE.BufferGeometry().setFromPoints(pointsArray);
            const triMat = new THREE.MeshBasicMaterial({
                color: 0x800080,
                transparent: true,
                opacity: 0.4,
                side: THREE.DoubleSide
            });
            const triangle = new THREE.Mesh(triGeom, triMat);
            group.add(triangle);

            // Vertex points
            const pointsGeom = new THREE.BufferGeometry().setFromPoints(pointsArray);
            const pointsMat = new THREE.PointsMaterial({ color: 0x000000, size: 3, sizeAttenuation: true });
            const points = new THREE.Points(pointsGeom, pointsMat);
            points.position.z = 0.01; // slight offset to prevent z-fighting
            group.add(points);
            
            return group;
        }

        // --- START ---
        init();
    </script>
</body>
</html>