<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width,initial-scale=1" />
<title>Curved Triangular Lattice - three.js</title>
<style>
    :root {
        --bg: #f0f0f0;
        --panel-bg: #ffffffcc;
        --accent: #673ab7;
        --text: #222;
    }
    html, body {
        height: 100%;
        margin: 0;
        background: var(--bg);
        color: var(--text);
        font-family: system-ui, -apple-system, Segoe UI, Roboto, Helvetica, Arial, "Apple Color Emoji", "Segoe UI Emoji";
    }
    #main-container {
        display: grid;
        grid-template-rows: auto 1fr;
        height: 100vh;
        width: 100vw;
    }
    #control-panel {
        display: grid;
        grid-template-columns: repeat(4, minmax(200px, 1fr));
        gap: 10px 16px;
        padding: 10px 12px;
        background: var(--panel-bg);
        backdrop-filter: blur(4px);
        border-bottom: 1px solid #ddd;
    }
    #control-panel > div {
        display: flex;
        align-items: center;
        flex-wrap: wrap;
        gap: 8px;
    }
    #control-panel label {
        font-weight: 600;
        margin-right: 6px;
    }
    #canvas-container {
        position: relative;
        overflow: hidden;
    }
    #main-canvas {
        display: block;
        width: 100%;
        height: 100%;
        outline: none;
    }
    #btn-reset-view {
        position: absolute;
        top: 10px;
        right: 10px;
        z-index: 5;
        width: 32px;
        height: 32px;
        border-radius: 50%;
        border: 1px solid #bbb;
        background: #fff;
        color: #333;
        font-weight: bold;
        cursor: pointer;
        box-shadow: 0 1px 4px rgba(0,0,0,0.15);
        user-select: none;
    }
    #btn-reset-view:hover {
        background: #fafafa;
    }
    input[type="range"] {
        width: 180px;
        accent-color: var(--accent);
    }
    input[type="radio"], input[type="checkbox"] {
        accent-color: var(--accent);
    }
    .note {
        font-size: 12px;
        color: #666;
        margin-left: 6px;
    }
</style>
</head>
<body>
<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>trim</label>
            <input type="range" id="slider-trim">
            <span id="label-trim-value"></span>
            <span class="note">(concave ⇦ 0.65 … 0.90 ⇨ round)</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>
(() => {
    // ========== Scene Setup ==========
    const canvas = document.getElementById('main-canvas');

    const scene = new THREE.Scene();
    scene.background = new THREE.Color(0xf0f0f0);

    const camera = new THREE.PerspectiveCamera(45, window.innerWidth/window.innerHeight, 0.01, 1000);
    camera.position.set(4.2, 4.0, 5.2);

    const renderer = new THREE.WebGLRenderer({ canvas, antialias: true });
    renderer.setPixelRatio(Math.min(window.devicePixelRatio || 1, 2));
    renderer.setSize(window.innerWidth, window.innerHeight - document.getElementById('control-panel').offsetHeight);

    const controls = new THREE.OrbitControls(camera, renderer.domElement);
    controls.enableDamping = true;
    controls.dampingFactor = 0.08;
    controls.minDistance = 1.2;
    controls.maxDistance = 50;
    controls.target.set(0, 0, 0);

    // Lighting
    const hemiLight = new THREE.HemisphereLight(0xffffff, 0x444444, 0.9);
    hemiLight.position.set(0, 2, 0);
    scene.add(hemiLight);

    const dirLight = new THREE.DirectionalLight(0xffffff, 0.8);
    dirLight.position.set(3, 5, 2);
    scene.add(dirLight);

    // Main group that is regenerated
    const mainGroup = new THREE.Group();
    scene.add(mainGroup);

    // ========== Controls ==========
    const radiosX = Array.from(document.querySelectorAll('input[name="unitsX"]'));
    const radiosY = Array.from(document.querySelectorAll('input[name="unitsY"]'));
    const sliderTrim = document.getElementById('slider-trim');
    const labelTrim = document.getElementById('label-trim-value');
    const chkDouble = document.getElementById('checkbox-double');
    const chkSphere = document.getElementById('checkbox-sphere');
    const chkTriangles = document.getElementById('checkbox-triangles');

    // Slider configuration
    sliderTrim.min = "0.65";
    sliderTrim.max = "0.9";
    sliderTrim.step = "0.001";
    sliderTrim.value = "0.8";
    labelTrim.textContent = "0.800";

    function getUnitsX() {
        const el = document.querySelector('input[name="unitsX"]:checked');
        return el ? parseInt(el.value, 10) : 2;
    }
    function getUnitsY() {
        const el = document.querySelector('input[name="unitsY"]:checked');
        return el ? parseInt(el.value, 10) : 2;
    }

    // ========== Geometry helpers ==========

    // Spherical linear interpolation between two vectors on a unit sphere
    function slerpVec3(a, b, t) {
        const ax = a.x, ay = a.y, az = a.z;
        const bx = b.x, by = b.y, bz = b.z;
        const dot = Math.max(-1, Math.min(1, ax*bx + ay*by + az*bz));
        const omega = Math.acos(dot);
        if (omega < 1e-6) {
            return new THREE.Vector3(ax, ay, az).lerp(new THREE.Vector3(bx, by, bz), t).normalize();
        }
        const sinom = Math.sin(omega);
        const s0 = Math.sin((1 - t) * omega) / sinom;
        const s1 = Math.sin(t * omega) / sinom;
        return new THREE.Vector3(
            s0 * ax + s1 * bx,
            s0 * ay + s1 * by,
            s0 * az + s1 * bz
        );
    }

    function colorLerpHex(c1, c2, t) {
        const c1r = (c1 >> 16) & 255, c1g = (c1 >> 8) & 255, c1b = c1 & 255;
        const c2r = (c2 >> 16) & 255, c2g = (c2 >> 8) & 255, c2b = c2 & 255;
        const r = c1r + (c2r - c1r) * t;
        const g = c1g + (c2g - c1g) * t;
        const b = c1b + (c2b - c1b) * t;
        return [r/255, g/255, b/255];
    }

    // Build the curved triangle geometry: mesh geometry + outline geometry + local references
    function buildCurvedTriangleGeometry(trim, colorScheme = "magenta") {
        // Config
        const segmentsPerEdge = 64; // smoothness of edge arcs
        // Equilateral arrangement on a unit sphere with XY side length = 1 for tiling
        const rXY = 1 / Math.sqrt(3); // ring radius so XY chord between 120°-separated points is 1
        const z = Math.sqrt(1 - rXY * rXY);

        const v0 = new THREE.Vector3(rXY, 0, z);
        const v1 = new THREE.Vector3(rXY * Math.cos(2 * Math.PI / 3), rXY * Math.sin(2 * Math.PI / 3), z);
        const v2 = new THREE.Vector3(rXY * Math.cos(4 * Math.PI / 3), rXY * Math.sin(4 * Math.PI / 3), z);

        // Center point along the average normal, scaled by trim
        const centerDir = new THREE.Vector3().addVectors(v0, v1).add(v2).normalize();
        const center = centerDir.clone().multiplyScalar(trim);

        // Build perimeter polyline points along spherical arcs v0->v1->v2->v0
        const edgePoints = [];
        const edges = [[v0, v1], [v1, v2], [v2, v0]];
        for (let e = 0; e < 3; e++) {
            const [a, b] = edges[e];
            for (let i = 0; i < segmentsPerEdge; i++) {
                const t = i / segmentsPerEdge;
                edgePoints.push(slerpVec3(a, b, t));
            }
        }
        const N = edgePoints.length;

        // Vertex colors: center to edge gradient
        const centerColorHex = (colorScheme === "cyan") ? 0x80ffff : 0xffc0cb; // light cyan or light pink
        const edgeColorHex   = (colorScheme === "cyan") ? 0x008080 : 0x904090; // teal or dark magenta

        // Build triangle fan geometry: center -> edge[i] -> edge[i+1]
        const positions = new Float32Array(N * 3 * 3);
        const colors = new Float32Array(N * 3 * 3);
        let pOff = 0, cOff = 0;

        const [cr, cg, cb] = colorLerpHex(centerColorHex, centerColorHex, 1); // just center color
        const [er, eg, eb] = colorLerpHex(edgeColorHex, edgeColorHex, 1);     // edge color

        for (let i = 0; i < N; i++) {
            const a = edgePoints[i];
            const b = edgePoints[(i + 1) % N];

            // center
            positions[pOff++] = center.x;
            positions[pOff++] = center.y;
            positions[pOff++] = center.z;
            colors[cOff++] = cr; colors[cOff++] = cg; colors[cOff++] = cb;

            // a
            positions[pOff++] = a.x;
            positions[pOff++] = a.y;
            positions[pOff++] = a.z;
            colors[cOff++] = er; colors[cOff++] = eg; colors[cOff++] = eb;

            // b
            positions[pOff++] = b.x;
            positions[pOff++] = b.y;
            positions[pOff++] = b.z;
            colors[cOff++] = er; colors[cOff++] = eg; colors[cOff++] = eb;
        }

        const geom = new THREE.BufferGeometry();
        geom.setAttribute('position', new THREE.BufferAttribute(positions, 3));
        geom.setAttribute('color', new THREE.BufferAttribute(colors, 3));
        geom.computeVertexNormals();

        // Outline geometry (closed loop)
        const outlinePos = new Float32Array(N * 2 * 3);
        let oOff = 0;
        for (let i = 0; i < N; i++) {
            const a = edgePoints[i];
            const b = edgePoints[(i + 1) % N];
            outlinePos[oOff++] = a.x; outlinePos[oOff++] = a.y; outlinePos[oOff++] = a.z;
            outlinePos[oOff++] = b.x; outlinePos[oOff++] = b.y; outlinePos[oOff++] = b.z;
        }
        const outlineGeom = new THREE.BufferGeometry();
        outlineGeom.setAttribute('position', new THREE.BufferAttribute(outlinePos, 3));

        // Local planar triangle vertices (underlying flat triangle, z=0)
        const t0 = new THREE.Vector3(rXY, 0, 0);
        const t1 = new THREE.Vector3(rXY * Math.cos(2 * Math.PI / 3), rXY * Math.sin(2 * Math.PI / 3), 0);
        const t2 = new THREE.Vector3(rXY * Math.cos(4 * Math.PI / 3), rXY * Math.sin(4 * Math.PI / 3), 0);

        return {
            geom,
            outlineGeom,
            localTriangle: [t0, t1, t2],
            localCorners3D: [v0.clone(), v1.clone(), v2.clone()] // if needed
        };
    }

    function createCurvedTriangleObject(sharedGeometry, colorScheme = "magenta") {
        const material = new THREE.MeshStandardMaterial({
            vertexColors: true,
            roughness: 0.35,
            metalness: 0.0,
            side: THREE.DoubleSide
        });
        // Outline
        const lineMat = new THREE.LineBasicMaterial({
            color: 0x000000,
            linewidth: 2
        });

        const mesh = new THREE.Mesh(sharedGeometry.geom, material);
        const outline = new THREE.LineSegments(sharedGeometry.outlineGeom, lineMat);
        const grp = new THREE.Group();
        grp.add(mesh);
        grp.add(outline);
        return grp;
    }

    function createFlatTriangleObjects(localTriangle, color = 0x800080) {
        // Triangle mesh
        const triGeom = new THREE.BufferGeometry();
        const triPos = new Float32Array([
            localTriangle[0].x, localTriangle[0].y, localTriangle[0].z,
            localTriangle[1].x, localTriangle[1].y, localTriangle[1].z,
            localTriangle[2].x, localTriangle[2].y, localTriangle[2].z
        ]);
        triGeom.setAttribute('position', new THREE.BufferAttribute(triPos, 3));
        triGeom.computeVertexNormals();
        const triMat = new THREE.MeshBasicMaterial({
            color,
            transparent: true,
            opacity: 0.4,
            side: THREE.DoubleSide,
            depthWrite: false
        });
        const triMesh = new THREE.Mesh(triGeom, triMat);

        // Points at vertices
        const ptsGeom = triGeom.clone();
        const ptsMat = new THREE.PointsMaterial({ color: 0x000000, size: 6, sizeAttenuation: false });
        const pts = new THREE.Points(ptsGeom, ptsMat);

        const grp = new THREE.Group();
        grp.add(triMesh);
        grp.add(pts);
        return grp;
    }

    // ========== Update / Regenerate Scene ==========
    const basis = {
        // centers lattice basis vectors for s = 1
        b1: new THREE.Vector3(1.5, Math.sqrt(3)/2, 0),
        b2: new THREE.Vector3(1.5, -Math.sqrt(3)/2, 0),
        // half shift to interleave the "double" lattice
        offset: new THREE.Vector3(0.75, Math.sqrt(3)/4, 0)
    };

    function disposeObject3D(obj) {
        obj.traverse(child => {
            if (child.geometry) {
                child.geometry.dispose?.();
            }
            if (child.material) {
                if (Array.isArray(child.material)) {
                    child.material.forEach(m => m.dispose?.());
                } else {
                    child.material.dispose?.();
                }
            }
        });
    }

    function clearMainGroup() {
        while (mainGroup.children.length > 0) {
            const c = mainGroup.children.pop();
            disposeObject3D(c);
        }
    }

    function updateScene() {
        clearMainGroup();

        const unitsX = getUnitsX();
        const unitsY = getUnitsY();
        const trim = parseFloat(sliderTrim.value);
        // Build base geometries for primary and double (different color schemes)
        const basePrimary = buildCurvedTriangleGeometry(trim, "magenta");
        const baseDouble  = buildCurvedTriangleGeometry(trim, "cyan");

        const shapesPrimary = new THREE.Group();
        const shapesDouble = new THREE.Group();
        const flatsGroup = new THREE.Group();

        // Pre-create reusable objects (we will clone transforms, but reuse geometries)
        // For performance, we create new Mesh/LineSegments per instance that share BufferGeometry.
        function addInstance(targetGroup, baseGeom, pos, rotZ) {
            const obj = createCurvedTriangleObject(baseGeom);
            obj.position.copy(pos);
            obj.rotation.z = rotZ;
            targetGroup.add(obj);
            return obj;
        }
        function addFlatInstance(targetGroup, localTriangle, pos, rotZ) {
            const obj = createFlatTriangleObjects(localTriangle);
            obj.position.copy(pos);
            obj.rotation.z = rotZ;
            targetGroup.add(obj);
            return obj;
        }

        for (let i = -unitsX; i <= unitsX; i++) {
            for (let j = -unitsY; j <= unitsY; j++) {
                const pos = new THREE.Vector3().addScaledVector(basis.b1, i).addScaledVector(basis.b2, j);
                const parity = (i + j) & 1;
                const rotZ = parity === 0 ? 0 : Math.PI;

                // Primary lattice
                addInstance(shapesPrimary, basePrimary, pos, rotZ);
                if (chkTriangles.checked) {
                    addFlatInstance(flatsGroup, basePrimary.localTriangle, pos, rotZ);
                }

                // Double lattice (interwoven)
                if (chkDouble.checked) {
                    const pos2 = new THREE.Vector3().copy(pos).add(basis.offset);
                    const rotZ2 = rotZ + Math.PI; // opposite orientation
                    addInstance(shapesDouble, baseDouble, pos2, rotZ2);
                    if (chkTriangles.checked) {
                        addFlatInstance(flatsGroup, basePrimary.localTriangle, pos2, rotZ2);
                    }
                }
            }
        }

        mainGroup.add(shapesPrimary);
        if (chkDouble.checked) mainGroup.add(shapesDouble);
        if (chkTriangles.checked) mainGroup.add(flatsGroup);

        if (chkSphere.checked) {
            const sphGeom = new THREE.SphereGeometry(1, 48, 32);
            const sphMat = new THREE.MeshPhongMaterial({ color: 0xdddddd, shininess: 80, specular: 0x999999 });
            const sphere = new THREE.Mesh(sphGeom, sphMat);
            sphere.position.set(0, 0, 0);
            mainGroup.add(sphere);
        }
    }

    // ========== Events ==========
    [...radiosX, ...radiosY].forEach(r => r.addEventListener('change', updateScene));
    sliderTrim.addEventListener('input', () => {
        labelTrim.textContent = parseFloat(sliderTrim.value).toFixed(3);
        updateScene();
    });
    chkDouble.addEventListener('change', updateScene);
    chkSphere.addEventListener('change', updateScene);
    chkTriangles.addEventListener('change', updateScene);

    document.getElementById('btn-reset-view').addEventListener('click', () => {
        controls.reset();
    });

    // Initial build
    updateScene();

    // Resize handling
    function onResize() {
        const panelH = document.getElementById('control-panel').offsetHeight;
        const w = window.innerWidth;
        const h = window.innerHeight - panelH;
        camera.aspect = w / h;
        camera.updateProjectionMatrix();
        renderer.setSize(w, h);
    }
    window.addEventListener('resize', onResize);

    // Render loop
    function animate() {
        requestAnimationFrame(animate);
        controls.update();
        renderer.render(scene, camera);
    }
    animate();

})();
</script>
</body>
</html>