<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>3D Pattern Visualization</title>
    <style>
        body { margin: 0; overflow: hidden; font-family: Arial, sans-serif; }
        #main-container {
            display: grid;
            grid-template-rows: auto 1fr;
            height: 100vh;
        }
        #control-panel {
            display: flex;
            flex-direction: column;
            gap: 10px;
            padding: 10px;
            background-color: #222;
            color: white;
        }
        #control-panel label {
            margin-bottom: 5px;
        }
        #canvas-container {
            position: relative;
            width: 100%;
            height: 100%;
        }
        #btn-reset-view {
            position: absolute;
            top: 10px;
            right: 10px;
            background-color: #444;
            border: none;
            color: white;
            padding: 8px 12px;
            font-size: 16px;
            cursor: pointer;
            border-radius: 5px;
        }
        #btn-reset-view:hover {
            background-color: #666;
        }
        #slider-trim {
            width: 100%;
        }
    </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.8</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>
        const scene = new THREE.Scene();
        scene.background = new THREE.Color(0xf0f0f0);

        const camera = new THREE.PerspectiveCamera(75, window.innerWidth/window.innerHeight, 0.1, 1000);
        camera.position.set(0, 0, 5);

        const renderer = new THREE.WebGLRenderer({ canvas: document.getElementById('main-canvas') });
        renderer.setSize(window.innerWidth, window.innerHeight);
        renderer.setPixelRatio(window.devicePixelRatio);

        const controls = new THREE.OrbitControls(camera, renderer.domElement);
        controls.enableDamping = true;
        controls.enablePan = false;
        controls.enableZoom = true;

        const hemiLight = new THREE.HemisphereLight(0xffffff, 0x444444, 1);
        scene.add(hemiLight);

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

        const group = new THREE.Group();
        scene.add(group);

        document.querySelectorAll('input[name="unitsX"]').forEach(radio => {
            radio.addEventListener('change', updateScene);
        });
        document.querySelectorAll('input[name="unitsY"]').forEach(radio => {
            radio.addEventListener('change', updateScene);
        });
        document.getElementById('slider-trim').addEventListener('input', function() {
            document.getElementById('label-trim-value').textContent = this.value;
            updateScene();
        });
        document.getElementById('checkbox-double').addEventListener('change', updateScene);
        document.getElementById('checkbox-sphere').addEventListener('change', updateScene);
        document.getElementById('checkbox-triangles').addEventListener('change', updateScene);
        document.getElementById('btn-reset-view').addEventListener('click', () => controls.reset());

        function updateScene() {
            group.children.forEach(child => group.remove(child));

            const unitsX = parseInt(document.querySelector('input[name="unitsX"]:checked').value);
            const unitsY = parseInt(document.querySelector('input[name="unitsY"]:checked').value);
            const trimValue = parseFloat(document.getElementById('slider-trim').value);
            const doubleEnabled = document.getElementById('checkbox-double').checked;
            const sphereEnabled = document.getElementById('checkbox-sphere').checked;
            const trianglesEnabled = document.getElementById('checkbox-triangles').checked;

            const basisX = new THREE.Vector3(1, 0, 0);
            const basisY = new THREE.Vector3(0, 1, 0);

            for (let i = 0; i <= unitsX; i++) {
                for (let j = 0; j <= unitsY; j++) {
                    const x = i * basisX.x + j * basisY.x;
                    const y = i * basisX.y + j * basisY.y;
                    const z = 0;

                    const shape = createCurvedTriangle(trimValue);
                    shape.position.set(x, y, z);
                    group.add(shape);

                    if (doubleEnabled) {
                        const doubleShape = createCurvedTriangle(trimValue);
                        doubleShape.position.set(x + basisX.x + basisY.x, y + basisX.y + basisY.y);
                        doubleShape.rotation.y = Math.PI;
                        doubleShape.material.color.set(0x80ffff);
                        group.add(doubleShape);
                    }
                }
            }

            if (sphereEnabled) {
                const sphere = createSphere();
                sphere.position.set(0, 0, 0);
                group.add(sphere);
            }

            if (trianglesEnabled) {
                const triangles = createTriangles();
                group.add(triangles);
            }
        }

        function createCurvedTriangle(trimValue) {
            const radius = 1;
            const pointA = new THREE.Vector3(1, 0, 0);
            const pointB = new THREE.Vector3(Math.cos(Math.PI/3), Math.sin(Math.PI/3), 0);
            const pointC = new THREE.Vector3(Math.cos(Math.PI/3), -Math.sin(Math.PI/3), 0);
            pointA.normalize();
            pointB.normalize();
            pointC.normalize();

            const numSegments = 10;
            const edgePoints = [];

            for (let i = 0; i < 3; i++) {
                const start = [pointA, pointB, pointC][i];
                const end = [pointA, pointB, pointC][(i + 1) % 3];
                const direction = end.clone().sub(start).normalize();
                const angle = Math.acos(start.dot(end));
                const arcRadius = radius * trimValue;

                for (let j = 0; j <= numSegments; j++) {
                    const t = j / numSegments;
                    const angleAt = angle * t;
                    const point = start.clone().add(direction.clone().multiplyScalar(arcRadius * Math.sin(angleAt)));
                    edgePoints.push(point);
                }
            }

            const geometry = new THREE.BufferGeometry();
            const vertices = [];
            const colors = [];

            for (let i = 0; i < edgePoints.length; i++) {
                vertices.push(edgePoints[i].x, edgePoints[i].y, edgePoints[i].z);
            }

            const indices = [];
            for (let i = 1; i < edgePoints.length; i++) {
                indices.push(0, i, i + 1);
            }

            geometry.setAttribute('position', new THREE.Float32BufferAttribute(vertices, 3));
            geometry.setIndex(indices);

            const center = new THREE.Vector3().addVectors(pointA, pointB).add(pointC).divideScalar(3);
            for (let i = 0; i < vertices.length / 3; i++) {
                const distance = edgePoints[i].distanceTo(center);
                const t = 1 - distance / radius;
                const color = new THREE.Color(0xffc0cb).lerp(new THREE.Color(0x904090), t);
                colors.push(color.r, color.g, color.b);
            }

            geometry.setAttribute('color', new THREE.Float32BufferAttribute(colors, 3));

            const material = new THREE.MeshStandardMaterial({ vertexColors: true });
            const mesh = new THREE.Mesh(geometry, material);

            const outlineGeometry = new THREE.BufferGeometry().setFromPoints(edgePoints);
            const outlineMaterial = new THREE.LineBasicMaterial({ color: 0x000000, linewidth: 2 });
            const outline = new THREE.LineSegments(outlineGeometry, outlineMaterial);

            return { mesh, outline };
        }

        function createSphere() {
            const geometry = new THREE.SphereGeometry(1, 32, 32);
            const material = new THREE.MeshPhongMaterial({ color: 0xdddddd, shininess: 80 });
            return new THREE.Mesh(geometry, material);
        }

        function createTriangles() {
            const points = [];
            const triangles = [];
            const colors = [];
            const geometry = new THREE.BufferGeometry();

            for (let i = 0; i < 3; i++) {
                const point = [new THREE.Vector3(1, 0, 0), new THREE.Vector3(Math.cos(Math.PI/3), Math.sin(Math.PI/3), 0), new THREE.Vector3(Math.cos(Math.PI/3), -Math.sin(Math.PI/3), 0)][i];
                points.push(point);
            }

            for (let i = 0; i < 3; i++) {
                triangles.push(i, (i + 1) % 3, (i + 2) % 3);
            }

            geometry.setAttribute('position', new THREE.Float32BufferAttribute(points.flatMap(p => [p.x, p.y, p.z]), 3));
            geometry.setIndex(triangles);

            const material = new THREE.MeshBasicMaterial({ color: 0x800080, transparent: true, opacity: 0.4 });
            const mesh = new THREE.Mesh(geometry, material);

            const vertexGeometry = new THREE.BufferGeometry().setFromPoints(points);
            const vertexMaterial = new THREE.PointsMaterial({ color: 0x000000, size: 3 });
            const vertexPoints = new THREE.Points(vertexGeometry, vertexMaterial);

            return { mesh, vertexPoints };
        }

        window.addEventListener('resize', () => {
            camera.aspect = window.innerWidth / window.innerHeight;
            camera.updateProjectionMatrix();
            renderer.setSize(window.innerWidth, window.innerHeight);
        });

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