const smoothstep = x => x*x*(3 - 2*x); function interpolate(a, b, x, f = smoothstep) { const val = a + f(x) * (b - a); return val; } const sigmoid = (a, b, f = smoothstep) => x => { if (x < a) return 0; if (x > b) return 1; return f((x - a) / (b - a)); }; // super ghetto random function xorshift(x) { x ^= x << 13; x ^= x >> 7; x ^= x << 17; return x; } export function random(seed, z, x) { return xorshift(1337 * z + seed + 80085 * x); } function ghettoPerlinNoise(seed, x, y, gridSize = 16) { const dot = (vx, vy) => vx[0] * vy[0] + vx[1] * vy[1]; const randGrad = (x0, y0) => { const rand = random(seed, x0, y0); return [Math.sin(rand), Math.cos(rand)]; }; const x0 = Math.floor(x / gridSize); const y0 = Math.floor(y / gridSize); const sx = x / gridSize - x0; const sy = y / gridSize - y0; const n0 = dot(randGrad(x0, y0), [sx, sy]); const n1 = dot(randGrad(x0 + 1, y0), [sx - 1, sy]); const n2 = dot(randGrad(x0, y0 + 1), [sx, sy - 1]); const n3 = dot(randGrad(x0 + 1, y0 + 1), [sx - 1, sy - 1]); return 1.5 * interpolate(interpolate(n0, n1, sx), interpolate(n2, n3, sx), sy); } function cliffPerlin(seed, x, y) { const noise1 = ghettoPerlinNoise(seed, x, y); const noise2 = ghettoPerlinNoise(seed+1, x, y); const softerEdge = x => { if (x < 0.1) return 0.5 - 5 * x; if (x > 0.8) return 1.0 - 2.5 * (x - 0.8); return (x - 0.1) / 0.8; }; return interpolate(-1, 1, 0.5 * (1 + Math.atan2(noise1, noise2) / Math.PI), softerEdge); } export function makeTerrain(seed, x, y) { const lacunarity = 2.1; const persistence = 0.35; const noiseMap = (x, y) => cliffPerlin(seed, x / 2, y / 2); const fractalNoise = (x, y) => { let value = 0; let power = 0.6; let scale = 0.1; const noises = [ghettoPerlinNoise, ghettoPerlinNoise, cliffPerlin, cliffPerlin]; for (const noiseFun of noises) { const noise = noiseFun(seed, scale * x, scale * y); value += noise * power; power *= persistence; scale *= lacunarity; } return interpolate(-0.2, 0.6, value);//, x=>x*x); // between 0 and 1 } const scaledNoise = (x, y) => noiseMap(x / 1, y / 1); const outputNoise = fractalNoise; // const outputNoise = scaledNoise; const terrain = new Uint8Array(16 * 16); for (let i = 0; i < 16; i++) { for (let j = 0; j < 16; j++) { terrain[i * 16 + j] = Math.floor(64 + 64 * outputNoise(x + i, y + j)); } } return terrain; } function colorInterp(x) { const ocean = [0.0, 0.0, 0.5]; const grass = [0.2, 0.7, 0.2]; const mountain = [0.4, 0.3, 0.1]; const snow = [0.9, 0.9, 0.9]; const grassHeight = 75 / 255; const mountainHeight = 100 / 255; const interp = dim => { return x => { if (x < grassHeight) { return interpolate(ocean[dim], grass[dim], x / grassHeight, sigmoid(0.6, 0.95)); } else if (x < mountainHeight) { return interpolate(grass[dim], mountain[dim], (x - grassHeight) / (mountainHeight - grassHeight)); } else { return interpolate(mountain[dim], snow[dim], (x - mountainHeight) / (1.0 - mountainHeight), sigmoid(0, 0.2)); } }; }; return { r: 255 * interp(0)(x/255), g: 255 * interp(1)(x/255), b: 255 * interp(2)(x/255), }; } function main() { const canvas = document.querySelector('#canvas'); const ctx = canvas.getContext('2d'); const seed = 1337; ctx.fillStyle = 'gray'; ctx.fill(); for (let i = 0; i < canvas.width; i += 16) { for (let j = 0; j < canvas.height; j += 16) { const imgdat = ctx.createImageData(16, 16); const dat = imgdat.data; const chunk = makeTerrain(seed, i - canvas.width / 2, j - canvas.height / 2); for (let k = 0; k < 16; k++) { for (let l = 0; l < 16; l++) { const offset = 4 * (16 * (15 - l) + k); const val = chunk[16 * k + l]; dat[offset + 0] = colorInterp(val).r; dat[offset + 1] = colorInterp(val).g; dat[offset + 2] = colorInterp(val).b; dat[offset + 3] = 255; } } ctx.putImageData(imgdat, i, canvas.height - 16 - j); } } } //window.onload = main;