worldbox-minigame/terrain.js

147 lines
4.1 KiB
JavaScript

/**********************************************************************
* TERRAIN GENERATION
**********************************************************************/
// Perlin noise implementation
// Based on https://github.com/josephg/noisejs
class Perlin {
constructor() {
this.grad3 = [[1,1,0],[-1,1,0],[1,-1,0],[-1,-1,0],
[1,0,1],[-1,0,1],[1,0,-1],[-1,0,-1],
[0,1,1],[0,-1,1],[0,1,-1],[0,-1,-1]];
this.p = [];
for (let i=0; i<256; i++) {
this.p[i] = Math.floor(Math.random()*256);
}
// To remove the need for index wrapping, double the permutation table length
this.perm = new Array(512);
this.gradP = new Array(512);
// Skipping the seed function for simplicity
this.seed(0);
}
seed(seed) {
if(seed > 0 && seed < 1) {
// Scale the seed out
seed *= 65536;
}
seed = Math.floor(seed);
if(seed < 256) {
seed |= seed << 8;
}
for(let i = 0; i < 256; i++) {
let v;
if (i & 1) {
v = this.p[i] ^ (seed & 255);
} else {
v = this.p[i] ^ ((seed>>8) & 255);
}
this.perm[i] = this.perm[i + 256] = v;
this.gradP[i] = this.gradP[i + 256] = this.grad3[v % 12];
}
}
// 2D Perlin Noise
noise(x, y) {
// Find unit grid cell containing point
let X = Math.floor(x), Y = Math.floor(y);
// Get relative xy coordinates of point within that cell
x = x - X; y = y - Y;
// Wrap the integer cells at 255 (smaller integer period can be introduced here)
X = X & 255; Y = Y & 255;
// Calculate noise contributions from each of the four corners
let n00 = this.dot2(this.gradP[X+this.perm[Y]], x, y);
let n01 = this.dot2(this.gradP[X+this.perm[Y+1]], x, y-1);
let n10 = this.dot2(this.gradP[X+1+this.perm[Y]], x-1, y);
let n11 = this.dot2(this.gradP[X+1+this.perm[Y+1]], x-1, y-1);
// Compute the fade curve value for x
let u = this.fade(x);
// Interpolate the four results
return this.lerp(
this.lerp(n00, n10, u),
this.lerp(n01, n11, u),
this.fade(y));
}
// Fading function
fade(t) {
return t*t*t*(t*(t*6-15)+10);
}
// Linear interpolation
lerp(a, b, t) {
return (1-t)*a + t*b;
}
// Dot product
dot2(g, x, y) {
return g[0]*x + g[1]*y;
}
}
// Terrain types
const TERRAIN_WATER = 0;
const TERRAIN_GRASS = 1;
// Generate terrain map
function generateTerrain(width, height, scale) {
const terrain = new Array(width);
for (let x = 0; x < width; x++) {
terrain[x] = new Array(height);
for (let y = 0; y < height; y++) {
// Generate noise value
let nx = x / scale;
let ny = y / scale;
let value = perlin.noise(nx, ny);
// Adjust the value to get more interesting terrain
// Add multiple octaves of noise for more natural look
value += 0.5 * perlin.noise(nx * 2, ny * 2);
value += 0.25 * perlin.noise(nx * 4, ny * 4);
value /= 1.75; // Normalize
// Determine terrain type based on noise value
// Lower water threshold to create more land areas
if (value < -0.3) { // Changed from 0.0 to -0.3 to reduce water
terrain[x][y] = TERRAIN_WATER;
} else {
terrain[x][y] = TERRAIN_GRASS;
}
}
}
return terrain;
}
// Check if a position is in water
function isWater(x, y) {
// Convert world coordinates to terrain grid coordinates
const gridX = Math.floor((x + 2000) / 10);
const gridY = Math.floor((y + 2000) / 10);
// Check bounds
if (gridX < 0 || gridX >= terrainWidth || gridY < 0 || gridY >= terrainHeight) {
return false; // Default to land if out of bounds
}
return terrainMap[gridX][gridY] === TERRAIN_WATER;
}
// Terrain dimensions
const terrainWidth = 400;
const terrainHeight = 400;
const terrainScale = 50; // Smaller scale = more detailed terrain
// Initialize Perlin noise
const perlin = new Perlin();
// Generate the terrain map
let terrainMap = generateTerrain(terrainWidth, terrainHeight, terrainScale);