mindcraft/test/test_generate_blueprint.js
2025-01-30 10:30:17 -08:00

413 lines
16 KiB
JavaScript

/** Main big funciton: Builds the foundation of the house. breaks down into smaller functions like generating the rooms, generating embellishments, etc...
*
* @param position coordinate that specify where the house should be built. come in as [x,y,z]
* @param windows an int that can be 0,1,2 for increasing frequencies of windows
* @param doors a boolean that has doors for room or no doors (if ur crazy ig)
*/
function buildHouse(position, windows, doors){
// randomly initalize a space for a 3D matrix (make sure its big enough)
const minSize = 30; // Minimum size for width, length, height
const randomness = 20; // Maximum randomness to add
const width = Math.floor(Math.random() * (randomness + 1)) + minSize;
const length = Math.floor(Math.random() * (randomness + 1)) + minSize;
const height = Math.floor(Math.random() * (randomness + 1)) + minSize;
// slice up the space ensuring each compartment has at least 4x4x2 space.
const resultMatrix = generateAbstractRooms(width, length, height, 3);
printMatrix(resultMatrix)
// todo: then, internally do things like windows / stairs / doors/ etc...
}
/**
*
* @param m - length (x-axis)
* @param n - width (y-axis)
* @param p - depth (z-axis, how many layers)
* @param rooms
* @returns {any[][][]}
*/
function generateAbstractRooms(m, n, p, rooms = 5) {
const matrix = Array.from({ length: p }, () =>
Array.from({ length: m }, () =>
Array(n).fill('air')
)
);
// Mark entire outer border with 'stone'
for (let z = 0; z < p; z++) {
for (let x = 0; x < m; x++) {
for (let y = 0; y < n; y++) {
if (
z === 0 || z === p - 1 || // Top and bottom faces
x === 0 || x === m - 1 || // Front and back faces
y === 0 || y === n - 1 // Left and right faces
) {
matrix[z][x][y] = 'stone';
}
}
}
}
const usedSpaces = new Set();
// Loop that places rooms
for (let roomCount = 0; roomCount < rooms; roomCount++) {
const length = Math.max(4, Math.floor(Math.random() * 6) + 4);
const width = Math.max(4, Math.floor(Math.random() * 6) + 4);
const depth = Math.max(3, Math.floor(Math.random() * 6) + 4);
let roomPlaced = false;
for (let attempt = 0; attempt < 50; attempt++) {
const x = Math.floor(Math.random() * (m - length - 1)) + 1;
const y = Math.floor(Math.random() * (n - width - 1)) + 1;
const z = Math.floor(Math.random() * (p - depth - 1)) + 1;
// Check space availability, excluding room's own edges (so that walls/ceilings can be shared)
console.log(`Attempting room: ${length}x${width}x${depth}`);
const spaceAvailable = !Array.from({ length: depth }).some((_, di) =>
Array.from({ length: length }).some((_, dj) =>
Array.from({ length: width }).some((_, dk) =>
// Exclude room's own edges from check
(di !== 0 && di !== depth - 1 &&
dj !== 0 && dj !== length - 1 &&
dk !== 0 && dk !== width - 1) &&
usedSpaces.has(`${x + dj},${y + dk},${z + di}`)
)
)
);
if (spaceAvailable) {
for (let di = 0; di < depth; di++) {
for (let dj = 0; dj < length; dj++) {
for (let dk = 0; dk < width; dk++) {
const spaceKey = `${x + dj},${y + dk},${z + di}`;
usedSpaces.add(spaceKey);
if (
z + di >= 0 && z + di < p &&
x + dj >= 0 && x + dj < m &&
y + dk >= 0 && y + dk < n
) {
// Mark only the outer edges of the room
if (di === 0 || di === depth - 1 ||
dj === 0 || dj === length - 1 ||
dk === 0 || dk === width - 1) {
matrix[z + di][x + dj][y + dk] = 'stone';
}
}
}
}
}
roomPlaced = true;
break;
}
}
if (!roomPlaced) {
console.warn(`Could not place room ${roomCount}`);
}
}
// TODO: Convert layers matrix into the right format
return matrix;
}
/**
* Systematically builds the houses by placing them next to the already existing rooms. Still uses randomness.
* @param m Width of the 3D space
* @param n Height of the 3D space
* @param p Depth of the 3D space
* @param rooms Number of rooms to generate
*/
function generateSequentialRooms(m, n, p, rooms) {
// Build 3D space
const matrix = Array.from({length: p}, () =>
Array.from({length: m}, () =>
Array(n).fill('air')
)
);
// Mark entire outer border with 'stone'
for (let z = 0; z < p; z++) {
for (let x = 0; x < m; x++) {
for (let y = 0; y < n; y++) {
if (
z === 0 || z === p - 1 || // Top and bottom faces
x === 0 || x === m - 1 || // Front and back faces
y === 0 || y === n - 1 // Left and right faces
) {
matrix[z][x][y] = 'stone';
}
}
}
}
let placedRooms = 0;
let lastRoom = null;
// Direction probabilities (e.g., 'above': 40%, 'left': 15%, etc.)
const directionChances = [
{direction: 'above', chance: 0.15},
{direction: 'left', chance: 0.15},
{direction: 'right', chance: 0.15},
{direction: 'forward', chance: 0.15},
{direction: 'backward', chance: 0.15},
];
// Function to pick a random direction based on percentages
function getRandomDirection() {
const rand = Math.random();
let cumulative = 0;
for (const {direction, chance} of directionChances) {
cumulative += chance;
if (rand <= cumulative) return direction;
}
return directionChances[1].direction; // Fallback to the first direction
}
// Ensures no rooms overlap except at edges
// Ensures no rooms overlap except at edges
function isSpaceValid(newX, newY, newZ, newLength, newWidth, newDepth) {
for (let di = 0; di < newDepth; di++) {
for (let dj = 0; dj < newLength; dj++) {
for (let dk = 0; dk < newWidth; dk++) {
const x = newX + dj;
const y = newY + dk;
const z = newZ + di;
// Skip checking the outermost borders of the new room (these can overlap with stone)
if (dj === 0 || dj === newLength - 1 ||
dk === 0 || dk === newWidth - 1 ||
di === 0 || di === newDepth - 1) {
continue;
}
// For non-border spaces, ensure they're air
if (matrix[z][x][y] !== 'air') {
return false;
}
}
}
}
return true;
}
function validateAndBuildBorder(matrix, newX, newY, newZ, newLength, newWidth, newDepth, m, n, p) {
// Allow rooms to use the matrix edges (note the <= instead of <)
if (
newX >= 0 && newX + newLength <= m &&
newY >= 0 && newY + newWidth <= n &&
newZ >= 0 && newZ + newDepth <= p &&
isSpaceValid(newX, newY, newZ, newLength, newWidth, newDepth)
) {
console.log(`Placing room at (${newX}, ${newY}, ${newZ}) with dimensions (${newLength}x${newWidth}x${newDepth})`);
for (let di = 0; di < newDepth; di++) {
for (let dj = 0; dj < newLength; dj++) {
for (let dk = 0; dk < newWidth; dk++) {
const x = newX + dj;
const y = newY + dk;
const z = newZ + di;
// If this is at a matrix border, don't modify it
// todo: change to be window
if (x === 0 || x === m - 1 ||
y === 0 || y === n - 1 ||
z === 0 || z === p - 1) {
continue;
}
// For non-border spaces, check if this is a floor that should be shared
if (di === 0 && matrix[z-1][x][y] === 'stone') {
// Skip creating floor if there's a ceiling below
matrix[z][x][y] = 'air';
} else if (di === 0 || di === newDepth - 1 ||
dj === 0 || dj === newLength - 1 ||
dk === 0 || dk === newWidth - 1) {
matrix[z][x][y] = 'stone';
} else {
matrix[z][x][y] = 'air';
}
}
}
}
return true;
}
return false;
}
function addDoor(matrix, x, y, z) {
matrix[z][x][y] = 'door';
}
function addStairs(matrix, x, y, z) {
matrix[z][x][y] = 'stair';
}
// Places rooms until we can't, or we place all
// attempts random configurations of rooms in random directions.
while (placedRooms < rooms) {
let roomPlaced = false;
for (let attempt = 0; attempt < 150; attempt++) {
const newLength = Math.max(6, Math.floor(Math.random() * 6) + 4);
const newWidth = Math.max(6, Math.floor(Math.random() * 6) + 4);
const newDepth = Math.max(5, Math.floor(Math.random() * 5) + 2);
let newX, newY, newZ;
// first room is special
if (placedRooms === 0) {
// First room placement
newX = Math.floor(Math.random() * (m - newLength - 1)) + 1;
newY = Math.floor(Math.random() * (n - newWidth - 1)) + 1;
newZ = 0; // Ground floor
if (validateAndBuildBorder(matrix, newX, newY, newZ, newLength, newWidth, newDepth, m, n, p)) {
lastRoom = { x: newX, y: newY, z: newZ, length: newLength, width: newWidth, depth: newDepth };
roomPlaced = true;
placedRooms++;
break;
}
// Todo: add doors to room on all sides
break;
}
else {
const direction = getRandomDirection();
switch (direction) {
case 'above':
// todo: the ceiling / floor are not the same when they should be
newX = lastRoom.x;
newY = lastRoom.y;
newZ = lastRoom.z + lastRoom.depth - 1;
if (validateAndBuildBorder(matrix, newX, newY, newZ, newLength, newWidth, newDepth, m, n, p)) {
addStairs(matrix, lastRoom.x + Math.floor(lastRoom.length / 2),
lastRoom.y + Math.floor(lastRoom.width / 2),
lastRoom.z);
lastRoom = { x: newX, y: newY, z: newZ, length: newLength, width: newWidth, depth: newDepth };
roomPlaced = true;
placedRooms++;
break;
}
break;
case 'left':
newX = lastRoom.x - newLength + 1;
newY = lastRoom.y;
newZ = lastRoom.z;
if (validateAndBuildBorder(matrix, newX, newY, newZ, newLength, newWidth, newDepth, m, n, p)) {
addDoor(matrix, lastRoom.x, lastRoom.y + Math.floor(lastRoom.width / 2), lastRoom.z);
lastRoom = { x: newX, y: newY, z: newZ, length: newLength, width: newWidth, depth: newDepth };
roomPlaced = true;
placedRooms++;
break;
}
break;
case 'right':
newX = lastRoom.x + lastRoom.length - 1;
newY = lastRoom.y;
newZ = lastRoom.z;
if (validateAndBuildBorder(matrix, newX, newY, newZ, newLength, newWidth, newDepth, m, n, p)) {
addDoor(matrix, lastRoom.x + lastRoom.length - 1,
lastRoom.y + Math.floor(lastRoom.width / 2),
lastRoom.z);
lastRoom = { x: newX, y: newY, z: newZ, length: newLength, width: newWidth, depth: newDepth };
roomPlaced = true;
placedRooms++;
break;
}
break;
case 'forward':
newX = lastRoom.x;
newY = lastRoom.y + lastRoom.width - 1;
newZ = lastRoom.z;
if (validateAndBuildBorder(matrix, newX, newY, newZ, newLength, newWidth, newDepth, m, n, p)) {
addDoor(matrix, lastRoom.x + Math.floor(lastRoom.length / 2),
lastRoom.y + lastRoom.width - 1,
lastRoom.z);
lastRoom = { x: newX, y: newY, z: newZ, length: newLength, width: newWidth, depth: newDepth };
roomPlaced = true;
placedRooms++;
break;
}
break;
case 'backward':
newX = lastRoom.x;
newY = lastRoom.y - newWidth + 1;
newZ = lastRoom.z;
if (validateAndBuildBorder(matrix, newX, newY, newZ, newLength, newWidth, newDepth, m, n, p)) {
addDoor(matrix, lastRoom.x + Math.floor(lastRoom.length / 2),
lastRoom.y,
lastRoom.z);
lastRoom = { x: newX, y: newY, z: newZ, length: newLength, width: newWidth, depth: newDepth };
roomPlaced = true;
placedRooms++;
break;
}
break;
}
if (roomPlaced) {
break;
}
}
}
if (!roomPlaced) {
console.warn(`Could not place room ${placedRooms + 1}`);
break;
}
}
return matrix
}
/**
* todo: Given a matrix, turn it into a blueprint
*/
function printMatrix(matrix) {
matrix.forEach((layer, layerIndex) => {
console.log(`Layer ${layerIndex}:`);
layer.forEach(row => {
console.log(
row.map(cell => {
switch (cell) {
case 'stone': return '█'; // Wall
case 'air': return '.'; // Open space
case 'door': return 'D'; // Door
case 'stair': return 'S'; // Stairs
default: return ' '; // Unknown or unmarked space
}
}).join(' ')
);
});
console.log('---');
});
}
const resultMatrix = generateSequentialRooms(20, 20, 20, 6);
printMatrix(resultMatrix)