static/media/js/wisp/terrain3.js @ 9dd13d5e7522

Pass on flask-lesscss
author Steve Losh <steve@stevelosh.com>
date Fri, 22 Dec 2017 18:31:05 -0500
parents e7bc59b9ebda
children (none)
{
    var _ns_ = {
        id: 'demo',
        doc: void 0
    };
    var ndarray = require('ndarray');
}
var width = exports.width = 610;
var height = exports.height = 400;
var wireframe = exports.wireframe = true;
var wireframeWidth = exports.wireframeWidth = 1.2;
var terrainHeight = exports.terrainHeight = 50;
var terrainSize = exports.terrainSize = 100;
void 0;
void 0;
void 0;
var inc = exports.inc = function inc(x) {
    return x + 1;
};
var dec = exports.dec = function dec(x) {
    return x - 1;
};
void 0;
void 0;
void 0;
void 0;
void 0;
void 0;
void 0;
void 0;
var midpoint = exports.midpoint = function midpoint(a, b) {
    return (a + b) / 2;
};
var average2 = exports.average2 = function average2(a, b) {
    return (a + b) / 2;
};
var average4 = exports.average4 = function average4(a, b, c, d) {
    return (a + b + c + d) / 4;
};
var safeAverage = exports.safeAverage = function safeAverage(a, b, c, d) {
    return function () {
        var totalø1 = 0;
        var countø1 = 0;
        a ? (function () {
            totalø1 = totalø1 + a;
            return countø1 = inc(countø1);
        })() : void 0;
        b ? (function () {
            totalø1 = totalø1 + b;
            return countø1 = inc(countø1);
        })() : void 0;
        c ? (function () {
            totalø1 = totalø1 + c;
            return countø1 = inc(countø1);
        })() : void 0;
        d ? (function () {
            totalø1 = totalø1 + d;
            return countø1 = inc(countø1);
        })() : void 0;
        return totalø1 / countø1;
    }.call(this);
};
var isEven = exports.isEven = function isEven(n) {
    return 0 == n % 2;
};
var isOdd = exports.isOdd = function isOdd(n) {
    return 1 == n % 2;
};
var rand = exports.rand = function rand() {
    return Math.random();
};
var randAroundZero = exports.randAroundZero = function randAroundZero(spread) {
    return spread * rand() * 2 - spread;
};
var jitter = exports.jitter = function jitter(value, spread) {
    return value + randAroundZero(spread);
};
var heightmapResolution = exports.heightmapResolution = function heightmapResolution(heightmap) {
    return heightmap.shape[0];
};
var heightmapLastIndex = exports.heightmapLastIndex = function heightmapLastIndex(heightmap) {
    return dec(heightmapResolution(heightmap));
};
var heightmapCenterIndex = exports.heightmapCenterIndex = function heightmapCenterIndex(heightmap) {
    return midpoint(0, heightmapLastIndex(heightmap));
};
var heightmapGet = exports.heightmapGet = function heightmapGet(heightmap, x, y) {
    return heightmap.get(x, y);
};
var heightmapGetSafe = exports.heightmapGetSafe = function heightmapGetSafe(heightmap, x, y) {
    return function () {
        var lastø1 = heightmapLastIndex(heightmap);
        return 0 <= x && x <= lastø1 && (0 <= y && y <= lastø1) ? (function () {
            return heightmapGet(heightmap, x, y);
        })() : void 0;
    }.call(this);
};
var heightmapSet = exports.heightmapSet = function heightmapSet(heightmap, x, y, val) {
    return heightmap.set(x, y, val);
};
var heightmapSetIfUnset = exports.heightmapSetIfUnset = function heightmapSetIfUnset(heightmap, x, y, val) {
    return 0 == heightmapGet(heightmap, x, y) ? (function () {
        return heightmapSet(heightmap, x, y, val);
    })() : void 0;
};
var normalize = exports.normalize = function normalize(heightmap) {
    return function () {
        var maxø1 = 0 - Infinity;
        var minø1 = Infinity;
        (function () {
            var array2ø1 = heightmap;
            return function () {
                var G__3ø1 = array2ø1.data.length;
                return function loop() {
                    var recur = loop;
                    var index1ø1 = 0;
                    do {
                        recur = index1ø1 < G__3ø1 ? (function () {
                            (function () {
                                var elø1 = array2ø1.data[index1ø1];
                                maxø1 < elø1 ? (function () {
                                    return maxø1 = elø1;
                                })() : void 0;
                                return minø1 > elø1 ? (function () {
                                    return minø1 = elø1;
                                })() : void 0;
                            }.call(this));
                            return loop[0] = inc(index1ø1), loop;
                        })() : void 0;
                    } while (index1ø1 = loop[0], recur === loop);
                    return recur;
                }.call(this);
            }.call(this);
        }.call(this));
        return function () {
            var spanø1 = maxø1 - minø1;
            return function () {
                var array4ø1 = heightmap;
                return function () {
                    var G__5ø1 = array4ø1.shape[0];
                    return function loop() {
                        var recur = loop;
                        var xø1 = 0;
                        do {
                            recur = xø1 < G__5ø1 ? (function () {
                                (function () {
                                    var G__6ø1 = array4ø1.shape[1];
                                    return function loop() {
                                        var recur = loop;
                                        var yø1 = 0;
                                        do {
                                            recur = yø1 < G__6ø1 ? (function () {
                                                (function () {
                                                    return heightmapSet(heightmap, xø1, yø1, (heightmapGet(heightmap, xø1, yø1) - minø1) / spanø1);
                                                })();
                                                return loop[0] = inc(yø1), loop;
                                            })() : void 0;
                                        } while (yø1 = loop[0], recur === loop);
                                        return recur;
                                    }.call(this);
                                }.call(this));
                                return loop[0] = inc(xø1), loop;
                            })() : void 0;
                        } while (xø1 = loop[0], recur === loop);
                        return recur;
                    }.call(this);
                }.call(this);
            }.call(this);
        }.call(this);
    }.call(this);
};
var makeHeightmap = exports.makeHeightmap = function makeHeightmap(exponent) {
    return function () {
        var resolutionø1 = Math.pow(2, exponent) + 1;
        return function () {
            var heightmapø1 = ndarray(new Float64Array(resolutionø1 * resolutionø1), [
                resolutionø1,
                resolutionø1
            ]);
            heightmapø1.exponent = exponent;
            heightmapø1.resolution = resolutionø1;
            heightmapø1.last = dec(resolutionø1);
            return heightmapø1;
        }.call(this);
    }.call(this);
};
var topLeftCorner = exports.topLeftCorner = function topLeftCorner(heightmap) {
    return function () {
        var centerø1 = heightmapCenterIndex(heightmap);
        return heightmap.lo(0, 0).hi(inc(centerø1), inc(centerø1));
    }.call(this);
};
var topRightCorner = exports.topRightCorner = function topRightCorner(heightmap) {
    return function () {
        var centerø1 = heightmapCenterIndex(heightmap);
        return heightmap.lo(centerø1, 0).hi(inc(centerø1), inc(centerø1));
    }.call(this);
};
var bottomLeftCorner = exports.bottomLeftCorner = function bottomLeftCorner(heightmap) {
    return function () {
        var centerø1 = heightmapCenterIndex(heightmap);
        return heightmap.lo(0, centerø1).hi(inc(centerø1), inc(centerø1));
    }.call(this);
};
var bottomRightCorner = exports.bottomRightCorner = function bottomRightCorner(heightmap) {
    return function () {
        var centerø1 = heightmapCenterIndex(heightmap);
        return heightmap.lo(centerø1, centerø1).hi(inc(centerø1), inc(centerø1));
    }.call(this);
};
var dsInitCorners = exports.dsInitCorners = function dsInitCorners(heightmap) {
    return function () {
        var lastø1 = heightmapLastIndex(heightmap);
        heightmapSet(heightmap, 0, 0, rand());
        heightmapSet(heightmap, 0, lastø1, rand());
        heightmapSet(heightmap, lastø1, 0, rand());
        return heightmapSet(heightmap, lastø1, lastø1, rand());
    }.call(this);
};
var dsSquare = exports.dsSquare = function dsSquare(heightmap, x, y, radius, spread) {
    return function () {
        var newHeightø1 = jitter(average4(heightmapGet(heightmap, x - radius, y - radius), heightmapGet(heightmap, x - radius, y + radius), heightmapGet(heightmap, x + radius, y - radius), heightmapGet(heightmap, x + radius, y + radius)), spread);
        return heightmapSet(heightmap, x, y, newHeightø1);
    }.call(this);
};
var dsDiamond = exports.dsDiamond = function dsDiamond(heightmap, x, y, radius, spread) {
    return function () {
        var newHeightø1 = jitter(safeAverage(heightmapGetSafe(heightmap, x - radius, y), heightmapGetSafe(heightmap, x + radius, y), heightmapGetSafe(heightmap, x, y - radius), heightmapGetSafe(heightmap, x, y + radius)), spread);
        return heightmapSet(heightmap, x, y, newHeightø1);
    }.call(this);
};
var dsSquares = exports.dsSquares = function dsSquares(heightmap, radius, spread) {
    return function () {
        var start8ø1 = radius;
        var end9ø1 = heightmapResolution(heightmap);
        var stride7ø1 = 2 * radius;
        return function loop() {
            var recur = loop;
            var xø1 = start8ø1;
            do {
                recur = xø1 < end9ø1 ? (function () {
                    (function loop() {
                        var recur = loop;
                        var yø1 = start8ø1;
                        do {
                            recur = yø1 < end9ø1 ? (function () {
                                (function () {
                                    return dsSquare(heightmap, xø1, yø1, radius, spread);
                                })();
                                return loop[0] = yø1 + stride7ø1, loop;
                            })() : void 0;
                        } while (yø1 = loop[0], recur === loop);
                        return recur;
                    }.call(this));
                    return loop[0] = xø1 + stride7ø1, loop;
                })() : void 0;
            } while (xø1 = loop[0], recur === loop);
            return recur;
        }.call(this);
    }.call(this);
};
var dsDiamonds = exports.dsDiamonds = function dsDiamonds(heightmap, radius, spread) {
    return function () {
        var sizeø1 = heightmapResolution(heightmap);
        return function () {
            var start11ø1 = 0;
            var end12ø1 = sizeø1;
            var stride10ø1 = radius;
            return function loop() {
                var recur = loop;
                var yø1 = start11ø1;
                do {
                    recur = yø1 < end12ø1 ? (function () {
                        (function () {
                            return function () {
                                var shiftø1 = isEven(yø1 / radius) ? radius : 0;
                                return function () {
                                    var start14ø1 = shiftø1;
                                    var end15ø1 = sizeø1;
                                    var stride13ø1 = 2 * radius;
                                    return function loop() {
                                        var recur = loop;
                                        var xø1 = start14ø1;
                                        do {
                                            recur = xø1 < end15ø1 ? (function () {
                                                (function () {
                                                    return dsDiamond(heightmap, xø1, yø1, radius, spread);
                                                })();
                                                return loop[0] = xø1 + stride13ø1, loop;
                                            })() : void 0;
                                        } while (xø1 = loop[0], recur === loop);
                                        return recur;
                                    }.call(this);
                                }.call(this);
                            }.call(this);
                        })();
                        return loop[0] = yø1 + stride10ø1, loop;
                    })() : void 0;
                } while (yø1 = loop[0], recur === loop);
                return recur;
            }.call(this);
        }.call(this);
    }.call(this);
};
var diamondSquare = exports.diamondSquare = function diamondSquare(heightmap) {
    return function () {
        var initialSpreadø1 = 0.3;
        var spreadReductionø1 = 0.5;
        var centerø1 = heightmapCenterIndex(heightmap);
        var sizeø1 = heightmap.shape[0];
        dsInitCorners(heightmap);
        (function loop() {
            var recur = loop;
            var radiusø1 = centerø1;
            var spreadø1 = initialSpreadø1;
            do {
                recur = radiusø1 >= 1 ? (function () {
                    dsSquares(heightmap, radiusø1, spreadø1);
                    dsDiamonds(heightmap, radiusø1, spreadø1);
                    return loop[0] = radiusø1 / 2, loop[1] = spreadø1 * spreadReductionø1, loop;
                })() : void 0;
            } while (radiusø1 = loop[0], spreadø1 = loop[1], recur === loop);
            return recur;
        }.call(this));
        return normalize(heightmap);
    }.call(this);
};
var diamondSquare1 = exports.diamondSquare1 = function diamondSquare1(heightmap) {
    dsInitCorners(heightmap);
    return normalize(heightmap);
};
var diamondSquare2 = exports.diamondSquare2 = function diamondSquare2(heightmap) {
    return function () {
        var initialSpreadø1 = 0.3;
        var spreadReductionø1 = 0.5;
        var centerø1 = heightmapCenterIndex(heightmap);
        var sizeø1 = heightmap.shape[0];
        dsInitCorners(heightmap);
        dsSquares(heightmap, centerø1, initialSpreadø1);
        return normalize(heightmap);
    }.call(this);
};
var diamondSquare3 = exports.diamondSquare3 = function diamondSquare3(heightmap) {
    return function () {
        var initialSpreadø1 = 0.3;
        var spreadReductionø1 = 0.5;
        var centerø1 = heightmapCenterIndex(heightmap);
        var sizeø1 = heightmap.shape[0];
        dsInitCorners(heightmap);
        dsSquares(heightmap, centerø1, initialSpreadø1);
        dsDiamonds(heightmap, centerø1, initialSpreadø1);
        dsSquares(heightmap, centerø1 / 2, spreadReductionø1 * initialSpreadø1);
        dsDiamonds(heightmap, centerø1 / 2, spreadReductionø1 * initialSpreadø1);
        return normalize(heightmap);
    }.call(this);
};
var makeDirectionalLight = exports.makeDirectionalLight = function makeDirectionalLight() {
    return function () {
        var lightø1 = new THREE.DirectionalLight(16777215, 1);
        lightø1.position.set(100, 0, 150);
        return lightø1;
    }.call(this);
};
var makeCamera = exports.makeCamera = function makeCamera() {
    return function () {
        var cameraø1 = new THREE.PerspectiveCamera(55, width / height, 0.1, 1000);
        cameraø1.position.set(0, -100, 150);
        return cameraø1;
    }.call(this);
};
var makeRenderer = exports.makeRenderer = function makeRenderer() {
    return function () {
        var rendererø1 = new THREE.WebGLRenderer({ 'antialias': false });
        rendererø1.setClearColor(16777215);
        rendererø1.setSize(width, height);
        rendererø1.setPixelRatio(2);
        return rendererø1;
    }.call(this);
};
var makeGeometry = exports.makeGeometry = function makeGeometry(heightmap) {
    return function () {
        var resolutionø1 = heightmap.shape[0];
        var geometryø1 = new THREE.PlaneGeometry(terrainSize, terrainSize, resolutionø1 - 1, resolutionø1 - 1);
        return geometryø1;
    }.call(this);
};
var makeControls = exports.makeControls = function makeControls(camera, renderer) {
    return function () {
        var controlsø1 = new THREE.TrackballControls(camera, renderer.domElement);
        controlsø1.rotateSpeed = 1.4;
        controlsø1.zoomSpeed = 0.5;
        controlsø1.staticMoving = true;
        controlsø1.dynamicDampingFactor = 0.3;
        return controlsø1;
    }.call(this);
};
var makePlane = exports.makePlane = function makePlane(geometry) {
    return function () {
        var materialø1 = new THREE.MeshLambertMaterial({
            'wireframe': wireframe,
            'wireframeLinewidth': wireframeWidth,
            'color': 47872
        });
        return new THREE.Mesh(geometry, materialø1);
    }.call(this);
};
var attachToDom = exports.attachToDom = function attachToDom(renderer, elName, refreshFn) {
    return function () {
        var containerø1 = document.getElementById(elName);
        var settingsø1 = document.createElement('div');
        var refreshButtonø1 = document.createElement('button');
        var buttonTextø1 = document.createTextNode('Refresh');
        var cancelScrollø1 = function (e) {
            return e.preventDefault();
        };
        refreshButtonø1.onclick = refreshFn;
        renderer.domElement.onmousewheel = cancelScrollø1;
        renderer.domElement.addEventListener('MozMousePixelScroll', cancelScrollø1, false);
        refreshButtonø1.appendChild(buttonTextø1);
        containerø1.appendChild(renderer.domElement);
        containerø1.appendChild(settingsø1);
        return settingsø1.appendChild(refreshButtonø1);
    }.call(this);
};
var updateGeometry = exports.updateGeometry = function updateGeometry(geometry, heightmap) {
    (function loop() {
        var recur = loop;
        var iø1 = 0;
        do {
            recur = iø1 < geometry.vertices.length ? (function () {
                geometry.vertices[iø1].z = terrainHeight * heightmap.data[iø1];
                return loop[0] = iø1 + 1, loop;
            })() : void 0;
        } while (iø1 = loop[0], recur === loop);
        return recur;
    }.call(this));
    geometry.computeVertexNormals();
    return geometry;
};
var makeDemo = exports.makeDemo = function makeDemo(elementId, algorithm, size) {
    var scene = new THREE.Scene();
    scene.add(new THREE.AxisHelper(100));
    var clock = new THREE.Clock();
    var camera = makeCamera();
    var renderer = makeRenderer();
    var geometry = void 0;
    var plane = void 0;
    scene.add(makeDirectionalLight());
    scene.add(new THREE.AmbientLight(16777215, 0.05));
    var refresh = function refresh() {
        return function () {
            var heightmapø1 = makeHeightmap(size);
            console.log('Generating terrain...');
            (function () {
                var G__16ø1 = new Date().getTime();
                var G__18ø1 = (function () {
                    return algorithm(heightmapø1);
                })();
                var G__17ø1 = new Date().getTime();
                console.log('Elapsed time: ' + (G__17ø1 - G__16ø1) + 'ms.');
                return G__18ø1;
            }.call(this));
            console.log('Rebuilding geometry...');
            (function () {
                var G__19ø1 = new Date().getTime();
                var G__21ø1 = (function () {
                    geometry = makeGeometry(heightmapø1);
                    return updateGeometry(geometry, heightmapø1);
                })();
                var G__20ø1 = new Date().getTime();
                console.log('Elapsed time: ' + (G__20ø1 - G__19ø1) + 'ms.');
                return G__21ø1;
            }.call(this));
            console.log('Rebuilding plane...');
            return function () {
                var G__22ø1 = new Date().getTime();
                var G__24ø1 = (function () {
                    scene.remove(plane);
                    plane = makePlane(geometry);
                    return scene.add(plane);
                })();
                var G__23ø1 = new Date().getTime();
                console.log('Elapsed time: ' + (G__23ø1 - G__22ø1) + 'ms.');
                return G__24ø1;
            }.call(this);
        }.call(this);
    };
    attachToDom(renderer, elementId, refresh);
    var controls = makeControls(camera, renderer);
    var render = function render() {
        return function () {
            var deltaø1 = clock.getDelta();
            requestAnimationFrame(render);
            controls.update(deltaø1);
            return renderer.render(scene, camera);
        }.call(this);
    };
    refresh();
    render();
    return void 0;
};
var makeFinal = exports.makeFinal = function makeFinal(elementId) {
    var scene = new THREE.Scene();
    scene.add(new THREE.AxisHelper(100));
    var clock = new THREE.Clock();
    var camera = makeCamera();
    var renderer = makeRenderer();
    var geometry = void 0;
    var plane = void 0;
    scene.add(makeDirectionalLight());
    scene.add(new THREE.AmbientLight(16777215, 0.05));
    var refresh = function refresh() {
        return function () {
            var heightmapø1 = makeHeightmap(6);
            console.log('Generating terrain...');
            (function () {
                var G__25ø1 = new Date().getTime();
                var G__27ø1 = (function () {
                    return diamondSquare(heightmapø1);
                })();
                var G__26ø1 = new Date().getTime();
                console.log('Elapsed time: ' + (G__26ø1 - G__25ø1) + 'ms.');
                return G__27ø1;
            }.call(this));
            console.log('Rebuilding geometry...');
            (function () {
                var G__28ø1 = new Date().getTime();
                var G__30ø1 = (function () {
                    geometry = makeGeometry(heightmapø1);
                    return updateGeometry(geometry, heightmapø1);
                })();
                var G__29ø1 = new Date().getTime();
                console.log('Elapsed time: ' + (G__29ø1 - G__28ø1) + 'ms.');
                return G__30ø1;
            }.call(this));
            console.log('Rebuilding plane...');
            return function () {
                var G__31ø1 = new Date().getTime();
                var G__33ø1 = (function () {
                    scene.remove(plane);
                    plane = makePlane(geometry);
                    return scene.add(plane);
                })();
                var G__32ø1 = new Date().getTime();
                console.log('Elapsed time: ' + (G__32ø1 - G__31ø1) + 'ms.');
                return G__33ø1;
            }.call(this);
        }.call(this);
    };
    attachToDom(renderer, elementId, refresh);
    var controls = makeControls(camera, renderer);
    var render = function render() {
        return function () {
            var deltaø1 = clock.getDelta();
            requestAnimationFrame(render);
            controls.update(deltaø1);
            return renderer.render(scene, camera);
        }.call(this);
    };
    refresh();
    render();
    return void 0;
};
var run = exports.run = function run() {
    makeDemo('demo-1', diamondSquare1, 2);
    makeDemo('demo-2', diamondSquare2, 4);
    makeDemo('demo-3', diamondSquare3, 4);
    return makeFinal('demo-final');
};
$(run);
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["anonymous.wisp"],"names":["_ns_","id","doc","width","exports","height","wireframe","wireframeWidth","terrainHeight","terrainSize","inc","x","dec","midpoint","a","b","average2","average4","c","d","safeAverage","totalø1","countø1","isEven","n","isOdd","rand","Math","random","randAroundZero","spread","jitter","value","heightmapResolution","heightmap","shape","heightmapLastIndex","heightmapCenterIndex","heightmapGet","y","get","heightmapGetSafe","lastø1","heightmapSet","val","set","heightmapSetIfUnset","normalize","maxø1","Infinity","minø1","elø1","spanø1","xø1","yø1","makeHeightmap","exponent","resolutionø1","pow","heightmapø1","ndarray","Float64Array","resolution","last","topLeftCorner","centerø1","lo","hi","topRightCorner","bottomLeftCorner","bottomRightCorner","dsInitCorners","dsSquare","radius","newHeightø1","dsDiamond","dsSquares","dsDiamonds","sizeø1","shiftø1","diamondSquare","initialSpreadø1","spreadReductionø1","radiusø1","spreadø1","diamondSquare1","diamondSquare2","diamondSquare3","makeDirectionalLight","lightø1","THREE","DirectionalLight","position.set","makeCamera","cameraø1","PerspectiveCamera","makeRenderer","rendererø1","WebGLRenderer","setClearColor","setSize","setPixelRatio","makeGeometry","geometryø1","PlaneGeometry","makeControls","camera","renderer","controlsø1","TrackballControls","domElement","rotateSpeed","zoomSpeed","staticMoving","dynamicDampingFactor","makePlane","geometry","materialø1","MeshLambertMaterial","Mesh","attachToDom","elName","refreshFn","containerø1","document","getElementById","settingsø1","createElement","refreshButtonø1","buttonTextø1","createTextNode","cancelScrollø1","e","preventDefault","onclick","domElement.onmousewheel","domElement.addEventListener","appendChild","updateGeometry","iø1","vertices.length","vertices","z","data","computeVertexNormals","makeDemo","elementId","algorithm","size","scene","Scene","add","AxisHelper","clock","Clock","plane","AmbientLight","refresh","remove","controls","render","deltaø1","getDelta","requestAnimationFrame","update","makeFinal","run","$"],"mappings":";IAAA,IAACA,I,GAAD;AAAA,QAAAC,E,EAAI,MAAJ;AAAA,QAAAC,G,EAAA,K,CAAA;AAAA,M;;;AAKA,IAAKC,KAAA,GAAAC,OAAA,CAAAD,KAAA,GAAM,GAAX,C;AACA,IAAKE,MAAA,GAAAD,OAAA,CAAAC,MAAA,GAAO,GAAZ,C;AACA,IAAKC,SAAA,GAAAF,OAAA,CAAAE,SAAA,G,IAAL,C;AACA,IAAKC,cAAA,GAAAH,OAAA,CAAAG,cAAA,GAAgB,GAArB,C;AACA,IAAKC,aAAA,GAAAJ,OAAA,CAAAI,aAAA,GAAe,EAApB,C;AACA,IAAKC,WAAA,GAAAL,OAAA,CAAAK,WAAA,GAAa,GAAlB,C;;;;AAoBA,IAAMC,GAAA,GAAAN,OAAA,CAAAM,GAAA,GAAN,SAAMA,GAAN,CAAWC,CAAX,EACE;AAAA,WAAGA,CAAH,GAAK,CAAL;AAAA,CADF,C;AAGA,IAAMC,GAAA,GAAAR,OAAA,CAAAQ,GAAA,GAAN,SAAMA,GAAN,CAAWD,CAAX,EACE;AAAA,WAAGA,CAAH,GAAK,CAAL;AAAA,CADF,C;;;;;;;;;AAwEA,IAAME,QAAA,GAAAT,OAAA,CAAAS,QAAA,GAAN,SAAMA,QAAN,CAAgBC,CAAhB,EAAkBC,CAAlB,EACE;AAAA,WAAG,CAAGD,CAAH,GAAKC,CAAL,CAAH,GAAW,CAAX;AAAA,CADF,C;AAGA,IAAMC,QAAA,GAAAZ,OAAA,CAAAY,QAAA,GAAN,SAAMA,QAAN,CAAgBF,CAAhB,EAAkBC,CAAlB,EACE;AAAA,WAAG,CAAGD,CAAH,GAAKC,CAAL,CAAH,GAAW,CAAX;AAAA,CADF,C;AAGA,IAAME,QAAA,GAAAb,OAAA,CAAAa,QAAA,GAAN,SAAMA,QAAN,CAAgBH,CAAhB,EAAkBC,CAAlB,EAAoBG,CAApB,EAAsBC,CAAtB,EACE;AAAA,WAAG,CAAGL,C,GAAEC,C,GAAEG,CAAP,GAASC,CAAT,CAAH,GAAe,CAAf;AAAA,CADF,C;AAGA,IAAMC,WAAA,GAAAhB,OAAA,CAAAgB,WAAA,GAAN,SAAMA,WAAN,CAAoBN,CAApB,EAAsBC,CAAtB,EAAwBG,CAAxB,EAA0BC,CAA1B,EACE;AAAA,W,YAAM;AAAA,YAAAE,O,GAAM,CAAN;AAAA,QAAQ,IAAAC,O,GAAM,CAAN,CAAR;AAAA,QACER,CAAN,G,aAAQ;AAAA,YAAMO,OAAN,GAAMA,O,GAAMP,CAAZ;AAAA,YAAe,OAAMQ,OAAN,G,IAAMA,O,CAAN,CAAf;AAAA,S,CAAA,EAAR,G,MAAA,CADI;AAAA,QAEEP,CAAN,G,aAAQ;AAAA,YAAMM,OAAN,GAAMA,O,GAAMN,CAAZ;AAAA,YAAe,OAAMO,OAAN,G,IAAMA,O,CAAN,CAAf;AAAA,S,CAAA,EAAR,G,MAAA,CAFI;AAAA,QAGEJ,CAAN,G,aAAQ;AAAA,YAAMG,OAAN,GAAMA,O,GAAMH,CAAZ;AAAA,YAAe,OAAMI,OAAN,G,IAAMA,O,CAAN,CAAf;AAAA,S,CAAA,EAAR,G,MAAA,CAHI;AAAA,QAIEH,CAAN,G,aAAQ;AAAA,YAAME,OAAN,GAAMA,O,GAAMF,CAAZ;AAAA,YAAe,OAAMG,OAAN,G,IAAMA,O,CAAN,CAAf;AAAA,S,CAAA,EAAR,G,MAAA,CAJI;AAAA,QAKJ,OAAGD,OAAH,GAASC,OAAT,CALI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AASA,IAAMC,MAAA,GAAAnB,OAAA,CAAAmB,MAAA,GAAN,SAAMA,MAAN,CAAaC,CAAb,EACE;AAAA,WAAI,CAAJ,IAAWA,CAAL,GAAO,CAAb;AAAA,CADF,C;AAGA,IAAMC,KAAA,GAAArB,OAAA,CAAAqB,KAAA,GAAN,SAAMA,KAAN,CAAYD,CAAZ,EACE;AAAA,WAAI,CAAJ,IAAWA,CAAL,GAAO,CAAb;AAAA,CADF,C;AAKA,IAAME,IAAA,GAAAtB,OAAA,CAAAsB,IAAA,GAAN,SAAMA,IAAN,GACE;AAAA,WAACC,IAAA,CAAKC,MAAN;AAAA,CADF,C;AAGA,IAAMC,cAAA,GAAAzB,OAAA,CAAAyB,cAAA,GAAN,SAAMA,cAAN,CAAwBC,MAAxB,EACE;AAAA,WAAMA,M,GAAQJ,IAAD,EAAV,GAAiB,CAApB,GAAuBI,MAAvB;AAAA,CADF,C;AAGA,IAAMC,MAAA,GAAA3B,OAAA,CAAA2B,MAAA,GAAN,SAAMA,MAAN,CAAcC,KAAd,EAAoBF,MAApB,EACE;AAAA,WAAGE,KAAH,GAAUH,cAAD,CAAkBC,MAAlB,CAAT;AAAA,CADF,C;AAKA,IAAMG,mBAAA,GAAA7B,OAAA,CAAA6B,mBAAA,GAAN,SAAMA,mBAAN,CAA4BC,SAA5B,EACE;AAAA,WAAMA,SAAA,CAAUC,KAAhB,CAAsB,CAAtB;AAAA,CADF,C;AAGA,IAAMC,kBAAA,GAAAhC,OAAA,CAAAgC,kBAAA,GAAN,SAAMA,kBAAN,CAA4BF,SAA5B,EACE;AAAA,WAACtB,GAAD,CAAMqB,mBAAD,CAAsBC,SAAtB,CAAL;AAAA,CADF,C;AAGA,IAAMG,oBAAA,GAAAjC,OAAA,CAAAiC,oBAAA,GAAN,SAAMA,oBAAN,CAA8BH,SAA9B,EACE;AAAA,WAACrB,QAAD,CAAU,CAAV,EAAauB,kBAAD,CAAsBF,SAAtB,CAAZ;AAAA,CADF,C;AAIA,IAAMI,YAAA,GAAAlC,OAAA,CAAAkC,YAAA,GAAN,SAAMA,YAAN,CAAqBJ,SAArB,EAA+BvB,CAA/B,EAAiC4B,CAAjC,EACE;AAAA,WAAML,SAAL,CAACM,GAAF,CAAgB7B,CAAhB,EAAkB4B,CAAlB;AAAA,CADF,C;AAGA,IAAME,gBAAA,GAAArC,OAAA,CAAAqC,gBAAA,GAAN,SAAMA,gBAAN,CAA0BP,SAA1B,EAAoCvB,CAApC,EAAsC4B,CAAtC,EACE;AAAA,W,YAAM;AAAA,YAAAG,M,GAAMN,kBAAD,CAAsBF,SAAtB,CAAL;AAAA,QACJ,OAAe,C,IAAEvB,CAAN,IAAMA,C,IAAE+B,MAAb,IACK,CAAI,C,IAAEH,CAAN,IAAMA,C,IAAEG,MAAR,CADX,G,aAEE;AAAA,mBAACJ,YAAD,CAAeJ,SAAf,EAAyBvB,CAAzB,EAA2B4B,CAA3B;AAAA,S,CAAA,EAFF,G,MAAA,CADI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAMA,IAAMI,YAAA,GAAAvC,OAAA,CAAAuC,YAAA,GAAN,SAAMA,YAAN,CAAsBT,SAAtB,EAAgCvB,CAAhC,EAAkC4B,CAAlC,EAAoCK,GAApC,EACE;AAAA,WAAMV,SAAL,CAACW,GAAF,CAAgBlC,CAAhB,EAAkB4B,CAAlB,EAAoBK,GAApB;AAAA,CADF,C;AAGA,IAAME,mBAAA,GAAA1C,OAAA,CAAA0C,mBAAA,GAAN,SAAMA,mBAAN,CAA+BZ,SAA/B,EAAyCvB,CAAzC,EAA2C4B,CAA3C,EAA6CK,GAA7C,EACE;AAAA,WAAU,CAAJ,IAAON,YAAD,CAAeJ,SAAf,EAAyBvB,CAAzB,EAA2B4B,CAA3B,CAAZ,G,aACE;AAAA,eAACI,YAAD,CAAgBT,SAAhB,EAA0BvB,CAA1B,EAA4B4B,CAA5B,EAA8BK,GAA9B;AAAA,K,CAAA,EADF,G,MAAA;AAAA,CADF,C;AAKA,IAAMG,SAAA,GAAA3C,OAAA,CAAA2C,SAAA,GAAN,SAAMA,SAAN,CAAiBb,SAAjB,EACE;AAAA,W,YAAM;AAAA,YAAAc,K,IAAI,GAAGC,QAAP;AAAA,QACA,IAAAC,K,GAAID,QAAJ,CADA;AAAA,QAEJ,C;2BAAkBf,S;;;;;;;;;oCAAHiB,I;gCACJH,KAAH,GAAOG,IAAb,G,aAAiB;AAAA,2CAAMH,KAAN,GAAUG,IAAV;AAAA,iC,CAAA,EAAjB,G,MAAA,C;gCACA,OAASD,KAAH,GAAOC,IAAb,G,aAAiB;AAAA,2CAAMD,KAAN,GAAUC,IAAV;AAAA,iC,CAAA,EAAjB,G,MAAA,C;;;;;;;;cAFF,C,IAAA,GAFI;AAAA,QAKJ,O,YAAM;AAAA,gBAAAC,M,GAAQJ,KAAH,GAAOE,KAAZ;AAAA,YACJ,O;+BAAkBhB,S;;;;;4BAALmB,G;;oCAAAA,G;;;;;4CAAEC,G;;oDAAAA,G;6DACb;AAAA,2DAACX,YAAD,CAAgBT,SAAhB,EAA0BmB,GAA1B,EAA4BC,GAA5B,EACmB,CAAIhB,YAAD,CAAeJ,SAAf,EAAyBmB,GAAzB,EAA2BC,GAA3B,CAAH,GAAiCJ,KAAjC,CAAH,GACGE,MAFnB;AAAA,iD,CAAA,G;qEADaE,G;;iDAAAA,G;;;;qDAAFD,G;;iCAAAA,G;;;;kBAAb,C,IAAA,EADI;AAAA,S,KAAN,C,IAAA,EALI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAaA,IAAME,aAAA,GAAAnD,OAAA,CAAAmD,aAAA,GAAN,SAAMA,aAAN,CAAsBC,QAAtB,EACE;AAAA,W,YAAM;AAAA,YAAAC,Y,GAAe9B,IAAA,CAAK+B,GAAN,CAAU,CAAV,EAAYF,QAAZ,CAAH,GAAyB,CAApC;AAAA,QACJ,O,YAAM;AAAA,gBAAAG,W,GAAWC,OAAD,CAAS,IAAKC,YAAL,CAAqBJ,YAAH,GAAcA,YAAhC,CAAT,EACS;AAAA,gBAACA,YAAD;AAAA,gBAAYA,YAAZ;AAAA,aADT,CAAV;AAAA,YAEEE,WAAA,CAAUH,QAAhB,GAAyBA,QAAzB,CAFI;AAAA,YAGEG,WAAA,CAAUG,UAAhB,GAA2BL,YAA3B,CAHI;AAAA,YAIEE,WAAA,CAAUI,IAAhB,GAAsBnD,GAAD,CAAK6C,YAAL,CAArB,CAJI;AAAA,YAKJ,OAAAE,WAAA,CALI;AAAA,S,KAAN,C,IAAA,EADI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAUA,IAAMK,aAAA,GAAA5D,OAAA,CAAA4D,aAAA,GAAN,SAAMA,aAAN,CAAuB9B,SAAvB,EACE;AAAA,W,YAAM;AAAA,YAAA+B,Q,GAAQ5B,oBAAD,CAAwBH,SAAxB,CAAP;AAAA,QACJ,OAAIA,SACD,CAACgC,E,CAAG,C,EAAE,C,CACN,CAACC,EAFJ,CAEQzD,GAAD,CAAKuD,QAAL,CAFP,EAEqBvD,GAAD,CAAKuD,QAAL,CAFpB,EADI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAMA,IAAMG,cAAA,GAAAhE,OAAA,CAAAgE,cAAA,GAAN,SAAMA,cAAN,CAAwBlC,SAAxB,EACE;AAAA,W,YAAM;AAAA,YAAA+B,Q,GAAQ5B,oBAAD,CAAwBH,SAAxB,CAAP;AAAA,QACJ,OAAIA,SACD,CAACgC,E,CAAGD,Q,EAAO,C,CACX,CAACE,EAFJ,CAEQzD,GAAD,CAAKuD,QAAL,CAFP,EAEqBvD,GAAD,CAAKuD,QAAL,CAFpB,EADI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAMA,IAAMI,gBAAA,GAAAjE,OAAA,CAAAiE,gBAAA,GAAN,SAAMA,gBAAN,CAA0BnC,SAA1B,EACE;AAAA,W,YAAM;AAAA,YAAA+B,Q,GAAQ5B,oBAAD,CAAwBH,SAAxB,CAAP;AAAA,QACJ,OAAIA,SACD,CAACgC,E,CAAG,C,EAAED,Q,CACN,CAACE,EAFJ,CAEQzD,GAAD,CAAKuD,QAAL,CAFP,EAEqBvD,GAAD,CAAKuD,QAAL,CAFpB,EADI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAMA,IAAMK,iBAAA,GAAAlE,OAAA,CAAAkE,iBAAA,GAAN,SAAMA,iBAAN,CAA2BpC,SAA3B,EACE;AAAA,W,YAAM;AAAA,YAAA+B,Q,GAAQ5B,oBAAD,CAAwBH,SAAxB,CAAP;AAAA,QACJ,OAAIA,SACD,CAACgC,E,CAAGD,Q,EAAOA,Q,CACX,CAACE,EAFJ,CAEQzD,GAAD,CAAKuD,QAAL,CAFP,EAEqBvD,GAAD,CAAKuD,QAAL,CAFpB,EADI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAQA,IAAMM,aAAA,GAAAnE,OAAA,CAAAmE,aAAA,GAAN,SAAMA,aAAN,CAAuBrC,SAAvB,EACE;AAAA,W,YAAM;AAAA,YAAAQ,M,GAAMN,kBAAD,CAAsBF,SAAtB,CAAL;AAAA,QACHS,YAAD,CAAgBT,SAAhB,EAA0B,CAA1B,EAA+B,CAA/B,EAAqCR,IAAD,EAApC,EADI;AAAA,QAEHiB,YAAD,CAAgBT,SAAhB,EAA0B,CAA1B,EAA+BQ,MAA/B,EAAqChB,IAAD,EAApC,EAFI;AAAA,QAGHiB,YAAD,CAAgBT,SAAhB,EAA0BQ,MAA1B,EAA+B,CAA/B,EAAqChB,IAAD,EAApC,EAHI;AAAA,QAIJ,OAACiB,YAAD,CAAgBT,SAAhB,EAA0BQ,MAA1B,EAA+BA,MAA/B,EAAqChB,IAAD,EAApC,EAJI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAOA,IAAM8C,QAAA,GAAApE,OAAA,CAAAoE,QAAA,GAAN,SAAMA,QAAN,CAAiBtC,SAAjB,EAA2BvB,CAA3B,EAA6B4B,CAA7B,EAA+BkC,MAA/B,EAAsC3C,MAAtC,EACE;AAAA,W,YAAM;AAAA,YAAA4C,W,GAAY3C,MAAD,CACGd,QAAD,CACGqB,YAAD,CAAeJ,SAAf,EAA4BvB,CAAH,GAAK8D,MAA9B,EAAyClC,CAAH,GAAKkC,MAA3C,CADF,EAEGnC,YAAD,CAAeJ,SAAf,EAA4BvB,CAAH,GAAK8D,MAA9B,EAAyClC,CAAH,GAAKkC,MAA3C,CAFF,EAGGnC,YAAD,CAAeJ,SAAf,EAA4BvB,CAAH,GAAK8D,MAA9B,EAAyClC,CAAH,GAAKkC,MAA3C,CAHF,EAIGnC,YAAD,CAAeJ,SAAf,EAA4BvB,CAAH,GAAK8D,MAA9B,EAAyClC,CAAH,GAAKkC,MAA3C,CAJF,CADF,EAME3C,MANF,CAAX;AAAA,QAOJ,OAACa,YAAD,CAAgBT,SAAhB,EAA0BvB,CAA1B,EAA4B4B,CAA5B,EAA8BmC,WAA9B,EAPI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAUA,IAAMC,SAAA,GAAAvE,OAAA,CAAAuE,SAAA,GAAN,SAAMA,SAAN,CAAkBzC,SAAlB,EAA4BvB,CAA5B,EAA8B4B,CAA9B,EAAgCkC,MAAhC,EAAuC3C,MAAvC,EACE;AAAA,W,YAAM;AAAA,YAAA4C,W,GAAY3C,MAAD,CACGX,WAAD,CACGqB,gBAAD,CAAoBP,SAApB,EAAiCvB,CAAH,GAAK8D,MAAnC,EAA2ClC,CAA3C,CADF,EAEGE,gBAAD,CAAoBP,SAApB,EAAiCvB,CAAH,GAAK8D,MAAnC,EAA2ClC,CAA3C,CAFF,EAGGE,gBAAD,CAAoBP,SAApB,EAA8BvB,CAA9B,EAAmC4B,CAAH,GAAKkC,MAArC,CAHF,EAIGhC,gBAAD,CAAoBP,SAApB,EAA8BvB,CAA9B,EAAmC4B,CAAH,GAAKkC,MAArC,CAJF,CADF,EAME3C,MANF,CAAX;AAAA,QAOJ,OAACa,YAAD,CAAgBT,SAAhB,EAA0BvB,CAA1B,EAA4B4B,CAA5B,EAA8BmC,WAA9B,EAPI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAWA,IAAME,SAAA,GAAAxE,OAAA,CAAAwE,SAAA,GAAN,SAAMA,SAAN,CAAkB1C,SAAlB,EAA4BuC,MAA5B,EAAmC3C,MAAnC,EACE;AAAA,W;uBAAiB2C,M;qBAAQxC,mBAAD,CAAsBC,SAAtB,C;wBAAoC,CAAH,GAAKuC,M;;;gBAAlDpB,G;;wBAAAA,G;;;4BAAEC,G;;oCAAAA,G;6CACZ;AAAA,2CAACkB,QAAD,CAAWtC,SAAX,EAAqBmB,GAArB,EAAuBC,GAAvB,EAAyBmB,MAAzB,EAAgC3C,MAAhC;AAAA,iC,CAAA,G;iDADYwB,G;;iCAAAA,G;;;qCAAFD,G;;qBAAAA,G;;;UAAZ,C,IAAA;AAAA,CADF,C;AAIA,IAAMwB,UAAA,GAAAzE,OAAA,CAAAyE,UAAA,GAAN,SAAMA,UAAN,CAAmB3C,SAAnB,EAA6BuC,MAA7B,EAAoC3C,MAApC,EACE;AAAA,W,YAAM;AAAA,YAAAgD,M,GAAM7C,mBAAD,CAAsBC,SAAtB,CAAL;AAAA,QACJ,O;4BAAe,C;0BAAE4C,M;6BAAKL,M;;;oBAAVnB,G;;4BAAAA,G;qCACV;AAAA,mC,YAAM;AAAA,oCAAAyB,O,GAAWxD,MAAD,CAAU+B,GAAH,GAAKmB,MAAZ,CAAJ,GAAyBA,MAAzB,GAAgC,CAAtC;AAAA,gCACJ,O;oDAAeM,O;kDAAMD,M;qDAAQ,CAAH,GAAKL,M;;;4CAAnBpB,G;;oDAAAA,G;6DACV;AAAA,2DAACsB,SAAD,CAAYzC,SAAZ,EAAsBmB,GAAtB,EAAwBC,GAAxB,EAA0BmB,MAA1B,EAAiC3C,MAAjC;AAAA,iD,CAAA,G;iEADUuB,G;;iDAAAA,G;;;sCAAZ,C,IAAA,EADI;AAAA,6B,KAAN,C,IAAA;AAAA,yB,CAAA,G;yCADUC,G;;yBAAAA,G;;;cAAZ,C,IAAA,EADI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAOA,IAAM0B,aAAA,GAAA5E,OAAA,CAAA4E,aAAA,GAAN,SAAMA,aAAN,CAAsB9C,SAAtB,EACE;AAAA,W,YAAM;AAAA,YAAA+C,e,GAAe,GAAf;AAAA,QACA,IAAAC,iB,GAAiB,GAAjB,CADA;AAAA,QAEA,IAAAjB,Q,GAAQ5B,oBAAD,CAAwBH,SAAxB,CAAP,CAFA;AAAA,QAGA,IAAA4C,M,GAAW5C,SAAA,CAAUC,KAAhB,CAAsB,CAAtB,CAAL,CAHA;AAAA,QAIHoC,aAAD,CAAiBrC,SAAjB,EAJI;AAAA,QAKJ,C;;YAAO,IAAAiD,Q,GAAOlB,QAAP,C;YACA,IAAAmB,Q,GAAOH,eAAP,C;;wBACKE,QAAJ,IAAW,CAAjB,G,aACE;AAAA,oBAACP,SAAD,CAAY1C,SAAZ,EAAsBiD,QAAtB,EAA6BC,QAA7B;AAAA,oBACCP,UAAD,CAAa3C,SAAb,EAAuBiD,QAAvB,EAA8BC,QAA9B,EADA;AAAA,oBAEA,O,UAAUD,QAAH,GAAU,CAAjB,E,UACUC,QAAH,GAAUF,iBADjB,E,IAAA,CAFA;AAAA,iB,CAAA,EADF,G;qBAFKC,Q,YACAC,Q;;cADP,C,IAAA,GALI;AAAA,QAYJ,OAACrC,SAAD,CAAWb,SAAX,EAZI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAgBA,IAAMmD,cAAA,GAAAjF,OAAA,CAAAiF,cAAA,GAAN,SAAMA,cAAN,CAAwBnD,SAAxB,EACE;AAAA,IAACqC,aAAD,CAAiBrC,SAAjB;AAAA,IACA,OAACa,SAAD,CAAWb,SAAX,EADA;AAAA,CADF,C;AAIA,IAAMoD,cAAA,GAAAlF,OAAA,CAAAkF,cAAA,GAAN,SAAMA,cAAN,CAAwBpD,SAAxB,EACE;AAAA,W,YAAM;AAAA,YAAA+C,e,GAAe,GAAf;AAAA,QACA,IAAAC,iB,GAAiB,GAAjB,CADA;AAAA,QAEA,IAAAjB,Q,GAAQ5B,oBAAD,CAAwBH,SAAxB,CAAP,CAFA;AAAA,QAGA,IAAA4C,M,GAAW5C,SAAA,CAAUC,KAAhB,CAAsB,CAAtB,CAAL,CAHA;AAAA,QAIHoC,aAAD,CAAiBrC,SAAjB,EAJI;AAAA,QAKH0C,SAAD,CAAY1C,SAAZ,EAAsB+B,QAAtB,EAA6BgB,eAA7B,EALI;AAAA,QAMJ,OAAClC,SAAD,CAAWb,SAAX,EANI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AASA,IAAMqD,cAAA,GAAAnF,OAAA,CAAAmF,cAAA,GAAN,SAAMA,cAAN,CAAwBrD,SAAxB,EACE;AAAA,W,YAAM;AAAA,YAAA+C,e,GAAe,GAAf;AAAA,QACA,IAAAC,iB,GAAiB,GAAjB,CADA;AAAA,QAEA,IAAAjB,Q,GAAQ5B,oBAAD,CAAwBH,SAAxB,CAAP,CAFA;AAAA,QAGA,IAAA4C,M,GAAW5C,SAAA,CAAUC,KAAhB,CAAsB,CAAtB,CAAL,CAHA;AAAA,QAIHoC,aAAD,CAAiBrC,SAAjB,EAJI;AAAA,QAKH0C,SAAD,CAAY1C,SAAZ,EAAsB+B,QAAtB,EAA6BgB,eAA7B,EALI;AAAA,QAMHJ,UAAD,CAAa3C,SAAb,EAAuB+B,QAAvB,EAA8BgB,eAA9B,EANI;AAAA,QAOHL,SAAD,CAAY1C,SAAZ,EAAyB+B,QAAH,GAAU,CAAhC,EAAsCiB,iBAAH,GAAoBD,eAAvD,EAPI;AAAA,QAQHJ,UAAD,CAAa3C,SAAb,EAA0B+B,QAAH,GAAU,CAAjC,EAAuCiB,iBAAH,GAAoBD,eAAxD,EARI;AAAA,QASJ,OAAClC,SAAD,CAAWb,SAAX,EATI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAcA,IAAMsD,oBAAA,GAAApF,OAAA,CAAAoF,oBAAA,GAAN,SAAMA,oBAAN,GACE;AAAA,W,YAAM;AAAA,YAAAC,O,GAAM,IAAKC,KAAA,CAAMC,gBAAX,CAA4B,QAA5B,EAAqC,CAArC,CAAN;AAAA,QACHF,OAAA,CAAMG,YAAP,CAAoB,GAApB,EAAwB,CAAxB,EAA0B,GAA1B,EADI;AAAA,QAEJ,OAAAH,OAAA,CAFI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAKA,IAAMI,UAAA,GAAAzF,OAAA,CAAAyF,UAAA,GAAN,SAAMA,UAAN,GACE;AAAA,W,YAAM;AAAA,YAAAC,Q,GAAO,IAAKJ,KAAA,CAAMK,iBAAX,CACK,EADL,EAEQ5F,KAAH,GAASE,MAFd,EAGK,GAHL,EAIK,IAJL,CAAP;AAAA,QAKHyF,QAAA,CAAOF,YAAR,CAAqB,CAArB,EAAuB,C,GAAvB,EAA4B,GAA5B,EALI;AAAA,QAMJ,OAAAE,QAAA,CANI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AASA,IAAME,YAAA,GAAA5F,OAAA,CAAA4F,YAAA,GAAN,SAAMA,YAAN,GACE;AAAA,W,YAAM;AAAA,YAAAC,U,GAAS,IAAKP,KAAA,CAAMQ,aAAX,CAAyB,E,kBAAA,EAAzB,CAAT;AAAA,QACHD,UAAA,CAASE,aAAV,CAAwB,QAAxB,EADI;AAAA,QAEHF,UAAA,CAASG,OAAV,CAAkBjG,KAAlB,EAAwBE,MAAxB,EAFI;AAAA,QAGH4F,UAAA,CAASI,aAAV,CAAwB,CAAxB,EAHI;AAAA,QAIJ,OAAAJ,UAAA,CAJI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAOA,IAAMK,YAAA,GAAAlG,OAAA,CAAAkG,YAAA,GAAN,SAAMA,YAAN,CAAqBpE,SAArB,EACE;AAAA,W,YAAM;AAAA,YAAAuB,Y,GAAiBvB,SAAA,CAAUC,KAAhB,CAAsB,CAAtB,CAAX;AAAA,QACA,IAAAoE,U,GAAS,IAAKb,KAAA,CAAMc,aAAX,CACK/F,WADL,EAEKA,WAFL,EAGQgD,YAAH,GAAc,CAHnB,EAIQA,YAAH,GAAc,CAJnB,CAAT,CADA;AAAA,QAMJ,OAAA8C,UAAA,CANI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AASA,IAAME,YAAA,GAAArG,OAAA,CAAAqG,YAAA,GAAN,SAAMA,YAAN,CAAqBC,MAArB,EAA4BC,QAA5B,EACE;AAAA,W,YAAM;AAAA,YAAAC,U,GAAS,IAAKlB,KAAA,CAAMmB,iBAAX,CAA6BH,MAA7B,EAAoCC,QAAA,CAASG,UAA7C,CAAT;AAAA,QACEF,UAAA,CAASG,WAAf,GAA2B,GAA3B,CADI;AAAA,QAEEH,UAAA,CAASI,SAAf,GAAyB,GAAzB,CAFI;AAAA,QAGEJ,UAAA,CAASK,YAAf,G,IAAA,CAHI;AAAA,QAIEL,UAAA,CAASM,oBAAf,GAAoC,GAApC,CAJI;AAAA,QAKJ,OAAAN,UAAA,CALI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAQA,IAAMO,SAAA,GAAA/G,OAAA,CAAA+G,SAAA,GAAN,SAAMA,SAAN,CAAkBC,QAAlB,EACE;AAAA,W,YAAM;AAAA,YAAAC,U,GAAS,IAAK3B,KAAA,CAAM4B,mBAAX,CACK;AAAA,Y,aAAYhH,SAAZ;AAAA,Y,sBACqBC,cADrB;AAAA,Y,SAEQ,KAFR;AAAA,SADL,CAAT;AAAA,QAIJ,WAAKmF,KAAA,CAAM6B,IAAX,CAAgBH,QAAhB,EAAyBC,UAAzB,EAJI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAQA,IAAMG,WAAA,GAAApH,OAAA,CAAAoH,WAAA,GAAN,SAAMA,WAAN,CAAqBb,QAArB,EAA8Bc,MAA9B,EAAsCC,SAAtC,EACE;AAAA,W,YAAM;AAAA,YAAAC,W,GAAWC,QAAA,CAASC,cAAV,CAAyBJ,MAAzB,CAAV;AAAA,QACA,IAAAK,U,GAAUF,QAAA,CAASG,aAAV,CAAwB,KAAxB,CAAT,CADA;AAAA,QAEA,IAAAC,e,GAAgBJ,QAAA,CAASG,aAAV,CAAwB,QAAxB,CAAf,CAFA;AAAA,QAGA,IAAAE,Y,GAAaL,QAAA,CAASM,cAAV,CAAyB,SAAzB,CAAZ,CAHA;AAAA,QAIA,IAAAC,c,GAAc,UAAKC,CAAL,EAAQ;AAAA,mBAAiBA,CAAhB,CAACC,cAAF;AAAA,SAAtB,CAJA;AAAA,QAKEL,eAAA,CAAeM,OAArB,GAA6BZ,SAA7B,CALI;AAAA,QAMEf,QAAA,CAAS4B,uBAAf,GAAuCJ,cAAvC,CANI;AAAA,QAOHxB,QAAA,CAAS6B,2BAAV,CAAsC,qBAAtC,EAA4DL,cAA5D,E,KAAA,EAPI;AAAA,QAQUH,eAAb,CAACS,WAAF,CAA6BR,YAA7B,EARI;AAAA,QASUN,WAAb,CAACc,WAAF,CAAwB9B,QAAA,CAASG,UAAjC,EATI;AAAA,QAUUa,WAAb,CAACc,WAAF,CAAwBX,UAAxB,EAVI;AAAA,QAWJ,OAAcA,UAAb,CAACW,WAAF,CAAuBT,eAAvB,EAXI;AAAA,K,KAAN,C,IAAA;AAAA,CADF,C;AAeA,IAAMU,cAAA,GAAAtI,OAAA,CAAAsI,cAAA,GAAN,SAAMA,cAAN,CAAuBtB,QAAvB,EAAgClF,SAAhC,EACE;AAAA,K;;QAAO,IAAAyG,G,GAAE,CAAF,C;;oBACEA,GAAH,GAAKvB,QAAA,CAASwB,eAAlB,G,aACM;AAAA,gBAAiBxB,QAAA,CAASyB,QAAf,CAAwBF,GAAxB,CAAL,CAAGG,CAAT,GACStI,aAAH,GAAgC0B,SAAR,CAAG6G,IAAT,CAAyBJ,GAAzB,CADxB;AAAA,gBAEF,O,UAAUA,GAAH,GAAK,CAAZ,E,IAAA,CAFE;AAAA,a,CAAA,EADN,G;iBADKA,G;;UAAP,C,IAAA;AAAA,IAKCvB,QAAA,CAAS4B,oBAAV,GALA;AAAA,IAMA,OAAA5B,QAAA,CANA;AAAA,CADF,C;AAWA,IAAM6B,QAAA,GAAA7I,OAAA,CAAA6I,QAAA,GAAN,SAAMA,QAAN,CAAiBC,SAAjB,EAA4BC,SAA5B,EAAsCC,IAAtC,E;IACE,IAAKC,KAAA,GAAM,IAAK3D,KAAA,CAAM4D,KAAX,EAAX,C;IACCD,KAAA,CAAME,GAAP,CAAW,IAAK7D,KAAA,CAAM8D,UAAX,CAAsB,GAAtB,CAAX,E;IAEA,IAAKC,KAAA,GAAM,IAAK/D,KAAA,CAAMgE,KAAX,EAAX,C;IACA,IAAKhD,MAAA,GAAQb,UAAD,EAAZ,C;IACA,IAAKc,QAAA,GAAUX,YAAD,EAAd,C;IAEA,IAAKoB,QAAA,G,MAAL,C;IACA,IAAKuC,KAAA,G,MAAL,C;IAECN,KAAA,CAAME,GAAP,CAAY/D,oBAAD,EAAX,E;IACC6D,KAAA,CAAME,GAAP,CAAW,IAAK7D,KAAA,CAAMkE,YAAX,CAAwB,QAAxB,EAAiC,IAAjC,CAAX,E;IAEA,IAAMC,OAAA,GAAN,SAAMA,OAAN,GACE;AAAA,e,YAAM;AAAA,gBAAAlG,W,GAAWJ,aAAD,CAAgB6F,IAAhB,CAAV;AAAA,Y,WACJ,CAAG,uBAAH,EADI;AAAA,YAEJ,C;;2CAAM;AAAA,2BAACD,SAAD,CAAWxF,WAAX;AAAA,iB,CAAA,E;;;;kBAAN,C,IAAA,GAFI;AAAA,Y,WAIJ,CAAG,wBAAH,EAJI;AAAA,YAKJ,C;;2CACE;AAAA,oBAAMyD,QAAN,GAAgBd,YAAD,CAAe3C,WAAf,CAAf;AAAA,oBACA,OAAC+E,cAAD,CAAiBtB,QAAjB,EAA0BzD,WAA1B,EADA;AAAA,iB,CAAA,E;;;;kBADF,C,IAAA,GALI;AAAA,Y,WASJ,CAAG,qBAAH,EATI;AAAA,YAUJ,O;;2CACE;AAAA,oBAAC0F,KAAA,CAAMS,MAAP,CAAcH,KAAd;AAAA,oBACMA,KAAN,GAAaxC,SAAD,CAAYC,QAAZ,CAAZ,CADA;AAAA,oBAEA,OAACiC,KAAA,CAAME,GAAP,CAAWI,KAAX,EAFA;AAAA,iB,CAAA,E;;;;kBADF,C,IAAA,EAVI;AAAA,S,KAAN,C,IAAA;AAAA,KADF,C;IAgBCnC,WAAD,CAAeb,QAAf,EAAwBuC,SAAxB,EAAmCW,OAAnC,E;IACA,IAAKE,QAAA,GAAUtD,YAAD,CAAeC,MAAf,EAAsBC,QAAtB,CAAd,C;IAEA,IAAMqD,MAAA,GAAN,SAAMA,MAAN,GACE;AAAA,e,YAAM;AAAA,gBAAAC,O,GAAOR,KAAA,CAAMS,QAAP,EAAN;AAAA,YACHC,qBAAD,CAAuBH,MAAvB,EADI;AAAA,YAEKD,QAAR,CAACK,MAAF,CAAkBH,OAAlB,EAFI;AAAA,YAGJ,OAACtD,QAAA,CAASqD,MAAV,CAAiBX,KAAjB,EAAuB3C,MAAvB,EAHI;AAAA,S,KAAN,C,IAAA;AAAA,KADF,C;IAMCmD,OAAD,G;IACCG,MAAD,G;;CAxCF,C;AA4CA,IAAMK,SAAA,GAAAjK,OAAA,CAAAiK,SAAA,GAAN,SAAMA,SAAN,CAAkBnB,SAAlB,E;IACE,IAAKG,KAAA,GAAM,IAAK3D,KAAA,CAAM4D,KAAX,EAAX,C;IACCD,KAAA,CAAME,GAAP,CAAW,IAAK7D,KAAA,CAAM8D,UAAX,CAAsB,GAAtB,CAAX,E;IAEA,IAAKC,KAAA,GAAM,IAAK/D,KAAA,CAAMgE,KAAX,EAAX,C;IACA,IAAKhD,MAAA,GAAQb,UAAD,EAAZ,C;IACA,IAAKc,QAAA,GAAUX,YAAD,EAAd,C;IAEA,IAAKoB,QAAA,G,MAAL,C;IACA,IAAKuC,KAAA,G,MAAL,C;IAECN,KAAA,CAAME,GAAP,CAAY/D,oBAAD,EAAX,E;IACC6D,KAAA,CAAME,GAAP,CAAW,IAAK7D,KAAA,CAAMkE,YAAX,CAAwB,QAAxB,EAAiC,IAAjC,CAAX,E;IAEA,IAAMC,OAAA,GAAN,SAAMA,OAAN,GACE;AAAA,e,YAAM;AAAA,gBAAAlG,W,GAAWJ,aAAD,CAAgB,CAAhB,CAAV;AAAA,Y,WACJ,CAAG,uBAAH,EADI;AAAA,YAEJ,C;;2CAAM;AAAA,2BAACyB,aAAD,CAAgBrB,WAAhB;AAAA,iB,CAAA,E;;;;kBAAN,C,IAAA,GAFI;AAAA,Y,WAIJ,CAAG,wBAAH,EAJI;AAAA,YAKJ,C;;2CACE;AAAA,oBAAMyD,QAAN,GAAgBd,YAAD,CAAe3C,WAAf,CAAf;AAAA,oBACA,OAAC+E,cAAD,CAAiBtB,QAAjB,EAA0BzD,WAA1B,EADA;AAAA,iB,CAAA,E;;;;kBADF,C,IAAA,GALI;AAAA,Y,WASJ,CAAG,qBAAH,EATI;AAAA,YAUJ,O;;2CACE;AAAA,oBAAC0F,KAAA,CAAMS,MAAP,CAAcH,KAAd;AAAA,oBACMA,KAAN,GAAaxC,SAAD,CAAYC,QAAZ,CAAZ,CADA;AAAA,oBAEA,OAACiC,KAAA,CAAME,GAAP,CAAWI,KAAX,EAFA;AAAA,iB,CAAA,E;;;;kBADF,C,IAAA,EAVI;AAAA,S,KAAN,C,IAAA;AAAA,KADF,C;IAgBCnC,WAAD,CAAeb,QAAf,EAAwBuC,SAAxB,EAAmCW,OAAnC,E;IACA,IAAKE,QAAA,GAAUtD,YAAD,CAAeC,MAAf,EAAsBC,QAAtB,CAAd,C;IAEA,IAAMqD,MAAA,GAAN,SAAMA,MAAN,GACE;AAAA,e,YAAM;AAAA,gBAAAC,O,GAAOR,KAAA,CAAMS,QAAP,EAAN;AAAA,YACHC,qBAAD,CAAuBH,MAAvB,EADI;AAAA,YAEKD,QAAR,CAACK,MAAF,CAAkBH,OAAlB,EAFI;AAAA,YAGJ,OAACtD,QAAA,CAASqD,MAAV,CAAiBX,KAAjB,EAAuB3C,MAAvB,EAHI;AAAA,S,KAAN,C,IAAA;AAAA,KADF,C;IAMCmD,OAAD,G;IACCG,MAAD,G;;CAxCF,C;AA4CA,IAAMM,GAAA,GAAAlK,OAAA,CAAAkK,GAAA,GAAN,SAAMA,GAAN,GACE;AAAA,IAACrB,QAAD,CAAW,QAAX,EAAoB5D,cAApB,EAAqC,CAArC;AAAA,IACC4D,QAAD,CAAW,QAAX,EAAoB3D,cAApB,EAAqC,CAArC,EADA;AAAA,IAEC2D,QAAD,CAAW,QAAX,EAAoB1D,cAApB,EAAqC,CAArC,EAFA;AAAA,IAGA,OAAC8E,SAAD,CAAY,YAAZ,EAHA;AAAA,CADF,C;AAMCE,CAAD,CAAGD,GAAH","sourcesContent":["(ns demo\n  (:require [ndarray]))\n\n\n; Constants -------------------------------------------------------------------\n(def width 610)\n(def height 400)\n(def wireframe true)\n(def wireframe-width 1.2)\n(def terrain-height 50)\n(def terrain-size 100)\n\n; General Utilities -----------------------------------------------------------\n(defmacro when [condition & body]\n  `(if ~condition\n     (do ~@body)))\n\n(defmacro when-not [condition & body]\n  `(when (not ~condition)\n     ~@body))\n\n(defmacro -> [& operations]\n  (reduce\n    (fn [form operation]\n      (cons (first operation)\n            (cons form (rest operation))))\n    (first operations)\n    (rest operations)))\n\n\n(defn inc [x]\n  (+ x 1))\n\n(defn dec [x]\n  (- x 1))\n\n\n(defmacro do-times [varname limit & body]\n  (let [end (gensym)]\n    `(let [~end ~limit]\n       (loop [~varname 0]\n         (when (< ~varname ~end)\n           ~@body\n           (recur (inc ~varname)))))))\n\n(defmacro do-stride [varnames start-form end-form stride-form & body]\n  (let [stride (gensym \"stride\")\n        start (gensym \"start\")\n        end (gensym \"end\")\n        build (fn build [vars]\n                (if (empty? vars)\n                  `(do ~@body)\n                  (let [varname (first vars)]\n                    `(loop [~varname ~start]\n                       (when (< ~varname ~end)\n                         ~(build (rest vars))\n                         (recur (+ ~varname ~stride)))))))]\n    ; Fix the numbers once outside the nested loops,\n    ; and then build the guts.\n    `(let [~start ~start-form\n           ~end ~end-form\n           ~stride ~stride-form]\n       ~(build varnames))))\n\n\n(defmacro do-ndarray [vars array-form & body]\n  (let [array-var (gensym \"array\")\n        build (fn build [vars n]\n                (if (empty? vars)\n                  `(do ~@body)\n                  `(do-times ~(first vars) (aget (.-shape ~array-var) ~n)\n                     ~(build (rest vars) (inc n)))))]\n    `(let [~array-var ~array-form]\n       ~(build vars 0))))\n\n(defmacro do-ndarray-el [element array-form & body]\n  (let [index (gensym \"index\")\n        array (gensym \"array\")]\n    `(let [~array ~array-form]\n       (do-times ~index (.-length (.-data ~array))\n         (let [~element (aget (.-data ~array) ~index)]\n           ~@body)))))\n\n\n(defmacro inc! [place]\n  `(set! ~place (inc ~place)))\n\n(defmacro add! [place amount]\n  `(set! ~place (+ ~place ~amount)))\n\n\n(defmacro l [& forms]\n  `(console.log ~@forms))\n\n(defmacro time [& body]\n  (let [start (gensym)\n        end (gensym)\n        result (gensym)]\n    `(let [~start (.getTime (new Date))\n           ~result (do ~@body)\n           ~end (.getTime (new Date))]\n       (l (+ \"Elapsed time: \" (- ~end ~start) \"ms.\"))\n       ~result)))\n\n\n(defn midpoint [a b]\n  (/ (+ a b) 2))\n\n(defn average2 [a b]\n  (/ (+ a b) 2))\n\n(defn average4 [a b c d]\n  (/ (+ a b c d) 4))\n\n(defn safe-average [a b c d]\n  (let [total 0 count 0]\n    (when a (add! total a) (inc! count))\n    (when b (add! total b) (inc! count))\n    (when c (add! total c) (inc! count))\n    (when d (add! total d) (inc! count))\n    (/ total count)))\n\n\n(defn even? [n]\n  (== 0 (mod n 2)))\n\n(defn odd? [n]\n  (== 1 (mod n 2)))\n\n\n; Randomness ------------------------------------------------------------------\n(defn rand []\n  (Math.random))\n\n(defn rand-around-zero [spread]\n  (- (* spread (rand) 2) spread))\n\n(defn jitter [value spread]\n  (+ value (rand-around-zero spread)))\n\n\n; Heightmap Helpers -----------------------------------------------------------\n(defn heightmap-resolution [heightmap]\n  (aget heightmap.shape 0))\n\n(defn heightmap-last-index [heightmap]\n  (dec (heightmap-resolution heightmap)))\n\n(defn heightmap-center-index [heightmap]\n  (midpoint 0 (heightmap-last-index heightmap)))\n\n\n(defn heightmap-get [heightmap x y]\n  (.get heightmap x y))\n\n(defn heightmap-get-safe [heightmap x y]\n  (let [last (heightmap-last-index heightmap)]\n    (when (and (<= 0 x last)\n               (<= 0 y last))\n      (heightmap-get heightmap x y))))\n\n(defn heightmap-set! [heightmap x y val]\n  (.set heightmap x y val))\n\n(defn heightmap-set-if-unset! [heightmap x y val]\n  (when (== 0 (heightmap-get heightmap x y))\n    (heightmap-set! heightmap x y val)))\n\n\n(defn normalize [heightmap]\n  (let [max (- Infinity)\n        min Infinity]\n    (do-ndarray-el el heightmap\n      (when (< max el) (set! max el))\n      (when (> min el) (set! min el)))\n    (let [span (- max min)]\n      (do-ndarray [x y] heightmap\n        (heightmap-set! heightmap x y\n                        (/ (- (heightmap-get heightmap x y) min)\n                           span))))))\n\n\n(defn make-heightmap [exponent]\n  (let [resolution (+ (Math.pow 2 exponent) 1)]\n    (let [heightmap (ndarray (new Float64Array (* resolution resolution))\n                             [resolution resolution])]\n      (set! heightmap.exponent exponent)\n      (set! heightmap.resolution resolution)\n      (set! heightmap.last (dec resolution))\n      heightmap)))\n\n\n(defn top-left-corner [heightmap]\n  (let [center (heightmap-center-index heightmap)]\n    (-> heightmap\n      (.lo 0 0)\n      (.hi (inc center) (inc center)))))\n\n(defn top-right-corner [heightmap]\n  (let [center (heightmap-center-index heightmap)]\n    (-> heightmap\n      (.lo center 0)\n      (.hi (inc center) (inc center)))))\n\n(defn bottom-left-corner [heightmap]\n  (let [center (heightmap-center-index heightmap)]\n    (-> heightmap\n      (.lo 0 center)\n      (.hi (inc center) (inc center)))))\n\n(defn bottom-right-corner [heightmap]\n  (let [center (heightmap-center-index heightmap)]\n    (-> heightmap\n      (.lo center center)\n      (.hi (inc center) (inc center)))))\n\n\n; Diamond-Square --------------------------------------------------------------\n(defn ds-init-corners [heightmap]\n  (let [last (heightmap-last-index heightmap)]\n    (heightmap-set! heightmap 0    0    (rand))\n    (heightmap-set! heightmap 0    last (rand))\n    (heightmap-set! heightmap last 0    (rand))\n    (heightmap-set! heightmap last last (rand))))\n\n(defn ds-square [heightmap x y radius spread]\n  (let [new-height (jitter\n                     (average4\n                       (heightmap-get heightmap (- x radius) (- y radius))\n                       (heightmap-get heightmap (- x radius) (+ y radius))\n                       (heightmap-get heightmap (+ x radius) (- y radius))\n                       (heightmap-get heightmap (+ x radius) (+ y radius)))\n                     spread)]\n    (heightmap-set! heightmap x y new-height)))\n\n(defn ds-diamond [heightmap x y radius spread]\n  (let [new-height (jitter\n                     (safe-average\n                       (heightmap-get-safe heightmap (- x radius) y)\n                       (heightmap-get-safe heightmap (+ x radius) y)\n                       (heightmap-get-safe heightmap x (- y radius))\n                       (heightmap-get-safe heightmap x (+ y radius)))\n                     spread)]\n    (heightmap-set! heightmap x y new-height)))\n\n\n(defn ds-squares [heightmap radius spread]\n  (do-stride [x y] radius (heightmap-resolution heightmap) (* 2 radius)\n    (ds-square heightmap x y radius spread)))\n\n(defn ds-diamonds [heightmap radius spread]\n  (let [size (heightmap-resolution heightmap)]\n    (do-stride [y] 0 size radius\n      (let [shift (if (even? (/ y radius)) radius 0)]\n        (do-stride [x] shift size (* 2 radius)\n          (ds-diamond heightmap x y radius spread))))))\n\n(defn diamond-square [heightmap]\n  (let [initial-spread 0.3\n        spread-reduction 0.5\n        center (heightmap-center-index heightmap)\n        size (aget heightmap.shape 0)]\n    (ds-init-corners heightmap)\n    (loop [radius center\n           spread initial-spread]\n      (when (>= radius 1)\n        (ds-squares heightmap radius spread)\n        (ds-diamonds heightmap radius spread)\n        (recur (/ radius 2)\n               (* spread spread-reduction))))\n    (normalize heightmap)))\n\n\n(defn diamond-square-1 [heightmap]\n  (ds-init-corners heightmap)\n  (normalize heightmap))\n\n(defn diamond-square-2 [heightmap]\n  (let [initial-spread 0.3\n        spread-reduction 0.5\n        center (heightmap-center-index heightmap)\n        size (aget heightmap.shape 0)]\n    (ds-init-corners heightmap)\n    (ds-squares heightmap center initial-spread)\n    (normalize heightmap)))\n\n(defn diamond-square-3 [heightmap]\n  (let [initial-spread 0.3\n        spread-reduction 0.5\n        center (heightmap-center-index heightmap)\n        size (aget heightmap.shape 0)]\n    (ds-init-corners heightmap)\n    (ds-squares heightmap center initial-spread)\n    (ds-diamonds heightmap center initial-spread)\n    (ds-squares heightmap (/ center 2) (* spread-reduction initial-spread))\n    (ds-diamonds heightmap (/ center 2) (* spread-reduction initial-spread))\n    (normalize heightmap)))\n\n\n; Three.js Helpers ------------------------------------------------------------\n(defn make-directional-light []\n  (let [light (new THREE.DirectionalLight 0xffffff 1)]\n    (light.position.set 100 0 150)\n    light))\n\n(defn make-camera []\n  (let [camera (new THREE.PerspectiveCamera\n                    55,\n                    (/ width height)\n                    0.1,\n                    1000)]\n    (camera.position.set 0 -100 150)\n    camera))\n\n(defn make-renderer []\n  (let [renderer (new THREE.WebGLRenderer {:antialias false})]\n    (renderer.setClearColor 0xffffff)\n    (renderer.setSize width height)\n    (renderer.setPixelRatio 2)\n    renderer))\n\n(defn make-geometry [heightmap]\n  (let [resolution (aget heightmap.shape 0)\n        geometry (new THREE.PlaneGeometry\n                      terrain-size\n                      terrain-size\n                      (- resolution 1)\n                      (- resolution 1))]\n    geometry))\n\n(defn make-controls [camera renderer]\n  (let [controls (new THREE.TrackballControls camera renderer.domElement)]\n    (set! controls.rotateSpeed 1.4)\n    (set! controls.zoomSpeed 0.5)\n    (set! controls.staticMoving true)\n    (set! controls.dynamicDampingFactor 0.3)\n    controls))\n\n(defn make-plane [geometry]\n  (let [material (new THREE.MeshLambertMaterial\n                      {:wireframe wireframe\n                       :wireframeLinewidth wireframe-width\n                       :color 0x00bb00})]\n    (new THREE.Mesh geometry material)))\n\n\n(defn attach-to-dom [renderer el-name refresh-fn]\n  (let [container (document.getElementById el-name)\n        settings (document.createElement \"div\")\n        refresh-button (document.createElement \"button\")\n        button-text (document.createTextNode \"Refresh\")\n        cancel-scroll (fn [e] (.preventDefault e))]\n    (set! refresh-button.onclick refresh-fn)\n    (set! renderer.domElement.onmousewheel cancel-scroll)\n    (renderer.domElement.addEventListener \"MozMousePixelScroll\" cancel-scroll false)\n    (.appendChild refresh-button button-text)\n    (.appendChild container renderer.domElement)\n    (.appendChild container settings)\n    (.appendChild settings refresh-button)))\n\n\n(defn update-geometry [geometry heightmap]\n  (loop [i 0]\n    (if (< i geometry.vertices.length)\n      (do (set! (.-z (aget geometry.vertices i))\n                (* terrain-height (aget (.-data heightmap) i)))\n        (recur (+ i 1)))))\n  (geometry.computeVertexNormals)\n  geometry)\n\n\n; Main ------------------------------------------------------------------------\n(defn make-demo [element-id algorithm size]\n  (def scene (new THREE.Scene))\n  (scene.add (new THREE.AxisHelper 100))\n\n  (def clock (new THREE.Clock))\n  (def camera (make-camera))\n  (def renderer (make-renderer))\n\n  (def geometry)\n  (def plane)\n\n  (scene.add (make-directional-light))\n  (scene.add (new THREE.AmbientLight 0xffffff 0.05))\n\n  (defn refresh []\n    (let [heightmap (make-heightmap size)]\n      (l \"Generating terrain...\")\n      (time (algorithm heightmap))\n\n      (l \"Rebuilding geometry...\")\n      (time\n        (set! geometry (make-geometry heightmap))\n        (update-geometry geometry heightmap))\n\n      (l \"Rebuilding plane...\")\n      (time\n        (scene.remove plane)\n        (set! plane (make-plane geometry))\n        (scene.add plane))))\n\n  (attach-to-dom renderer element-id refresh)\n  (def controls (make-controls camera renderer))\n\n  (defn render []\n    (let [delta (clock.getDelta)]\n      (requestAnimationFrame render)\n      (.update controls delta)\n      (renderer.render scene camera)))\n\n  (refresh)\n  (render)\n\n  nil)\n\n(defn make-final [element-id]\n  (def scene (new THREE.Scene))\n  (scene.add (new THREE.AxisHelper 100))\n\n  (def clock (new THREE.Clock))\n  (def camera (make-camera))\n  (def renderer (make-renderer))\n\n  (def geometry)\n  (def plane)\n\n  (scene.add (make-directional-light))\n  (scene.add (new THREE.AmbientLight 0xffffff 0.05))\n\n  (defn refresh []\n    (let [heightmap (make-heightmap 6)]\n      (l \"Generating terrain...\")\n      (time (diamond-square heightmap))\n\n      (l \"Rebuilding geometry...\")\n      (time\n        (set! geometry (make-geometry heightmap))\n        (update-geometry geometry heightmap))\n\n      (l \"Rebuilding plane...\")\n      (time\n        (scene.remove plane)\n        (set! plane (make-plane geometry))\n        (scene.add plane))))\n\n  (attach-to-dom renderer element-id refresh)\n  (def controls (make-controls camera renderer))\n\n  (defn render []\n    (let [delta (clock.getDelta)]\n      (requestAnimationFrame render)\n      (.update controls delta)\n      (renderer.render scene camera)))\n\n  (refresh)\n  (render)\n\n  nil)\n\n(defn run []\n  (make-demo \"demo-1\" diamond-square-1 2)\n  (make-demo \"demo-2\" diamond-square-2 4)\n  (make-demo \"demo-3\" diamond-square-3 4)\n  (make-final \"demo-final\"))\n\n($ run)\n\n\n; vim: lw+=do-times lw+=do-nested :\n"]}