From 22adaeb7002b5f301f85856567135839f829b112 Mon Sep 17 00:00:00 2001 From: "Kacper Kostka (aider)" Date: Wed, 2 Apr 2025 22:39:45 +0200 Subject: [PATCH] feat: Modify planner to place random buildings in random locations --- ai.js | 201 +++++++++++++++++----------------------------------------- 1 file changed, 57 insertions(+), 144 deletions(-) diff --git a/ai.js b/ai.js index 61f358b..16487f9 100644 --- a/ai.js +++ b/ai.js @@ -389,59 +389,34 @@ function soldierTasks(cit) { } function plannerTasks(cit) { - // Check if it's time to plan a new building + // Randomly place buildings with no planning if(Math.random() < 0.05) { cit.task = "planBuilding"; cit.target = null; return; } - // Help with other tasks when not planning - if(Math.random() < 0.5) { - builderTasks(cit); - } else { - // Explore the map to find good building locations - cit.task = "explore"; - cit.target = { - x: randInt(-500, 500), - y: randInt(-500, 500) - }; - } + // Just wander around when not placing buildings + cit.task = null; + cit.target = null; } /********************************************************************** * NEW TASK HANDLERS **********************************************************************/ function planBuildingTask(cit) { - // Find a good spot to place a new building - let buildingType = null; + // Choose a completely random building type + const buildingTypes = ["House", "Road", "Market", "Hospital", "School"]; + const buildingType = buildingTypes[Math.floor(Math.random() * buildingTypes.length)]; + + // Set cost based on building type let buildingCost = 0; - - // Decide what type of building to place based on current needs - const houseCount = buildings.filter(b => b.buildingType === "House" && b.completed).length; - const marketCount = buildings.filter(b => b.buildingType === "Market" && b.completed).length; - const hospitalCount = buildings.filter(b => b.buildingType === "Hospital" && b.completed).length; - const schoolCount = buildings.filter(b => b.buildingType === "School" && b.completed).length; - - // Prioritize based on what the city needs - if (houseCount < 3 || Math.random() < 0.5) { - buildingType = "House"; - buildingCost = COST_HOUSE; - } else if (marketCount < 1 && money >= COST_MARKET) { - buildingType = "Market"; - buildingCost = COST_MARKET; - } else if (hospitalCount < 1 && money >= COST_HOSPITAL) { - buildingType = "Hospital"; - buildingCost = COST_HOSPITAL; - } else if (schoolCount < 1 && money >= COST_SCHOOL) { - buildingType = "School"; - buildingCost = COST_SCHOOL; - } else if (Math.random() < 0.3) { - buildingType = "Road"; - buildingCost = COST_ROAD; - } else { - buildingType = "House"; - buildingCost = COST_HOUSE; + switch (buildingType) { + case "House": buildingCost = COST_HOUSE; break; + case "Road": buildingCost = COST_ROAD; break; + case "Market": buildingCost = COST_MARKET; break; + case "Hospital": buildingCost = COST_HOSPITAL; break; + case "School": buildingCost = COST_SCHOOL; break; } // Check if we can afford it @@ -450,118 +425,56 @@ function planBuildingTask(cit) { return; } - // Find a good location - away from other buildings but not too far from city center - let bestX = 0, bestY = 0; - let bestScore = -Infinity; + // Choose a completely random location + let randomX, randomY; + let attempts = 0; - for (let attempt = 0; attempt < 10; attempt++) { - // Try to find locations expanding outward from existing buildings - let existingBuilding = buildings[Math.floor(Math.random() * buildings.length)]; - if (!existingBuilding) { - // If no buildings yet, start near center - existingBuilding = { x: 0, y: 0 }; - } - - const direction = Math.random() * Math.PI * 2; - const distance = randInt(100, 300); - const testX = existingBuilding.x + Math.cos(direction) * distance; - const testY = existingBuilding.y + Math.sin(direction) * distance; - - // Skip if not valid placement - if (!isValidPlacement(testX, testY)) continue; - - // Calculate score based on: - // - Distance from other buildings (not too close) - // - Distance from center (not too far) - // - Terrain type - - let minDistToBuilding = Infinity; - buildings.forEach(b => { - const dist = distance(testX, testY, b.x, b.y); - if (dist < minDistToBuilding) minDistToBuilding = dist; - }); - - const distToCenter = distance(testX, testY, 0, 0); - - // Score calculation - higher is better - let score = 0; - - // Prefer some distance from other buildings - if (minDistToBuilding < 50) score -= 100; - else if (minDistToBuilding < 100) score += minDistToBuilding - 50; - else score += 50; - - // Prefer not too far from center - score -= distToCenter * 0.1; - - if (score > bestScore) { - bestScore = score; - bestX = testX; - bestY = testY; + // Just try random locations until we find one that's not water + do { + randomX = randInt(-1500, 1500); + randomY = randInt(-1500, 1500); + attempts++; + if (attempts > 20) { + // Give up after too many attempts + cit.task = null; + return; } + } while (!isValidPlacement(randomX, randomY)); + + // Place the building at the random location + addMoney(-buildingCost, `Planner: ${buildingType}`); + + let newBuilding; + switch (buildingType) { + case "House": + newBuilding = createHouseSite(randomX, randomY); + break; + case "Road": + // Just make a random road segment + newBuilding = createRoadSite( + randomX, + randomY, + randomX + randInt(-100, 100), + randomY + randInt(-100, 100) + ); + break; + case "Market": + newBuilding = createMarketSite(randomX, randomY); + break; + case "Hospital": + newBuilding = createHospitalSite(randomX, randomY); + break; + case "School": + newBuilding = createSchoolSite(randomX, randomY); + break; } - // If we found a good spot, place the building - if (bestScore > -Infinity) { - addMoney(-buildingCost, `Planner: ${buildingType}`); - - let newBuilding; - switch (buildingType) { - case "House": - newBuilding = createHouseSite(bestX, bestY); - break; - case "Road": - // Find nearest building to connect to - let nearest = null; - let minDist = Infinity; - buildings.forEach(b => { - if (b.buildingType !== "Road") { - const dist = distance(bestX, bestY, b.x, b.y); - if (dist < minDist) { - minDist = dist; - nearest = b; - } - } - }); - - if (nearest) { - newBuilding = createRoadSite(bestX, bestY, nearest.x, nearest.y); - } else { - newBuilding = createRoadSite(bestX - 50, bestY, bestX + 50, bestY); - } - break; - case "Market": - newBuilding = createMarketSite(bestX, bestY); - break; - case "Hospital": - newBuilding = createHospitalSite(bestX, bestY); - break; - case "School": - newBuilding = createSchoolSite(bestX, bestY); - break; - } - - buildings.push(newBuilding); - logAction(`${cit.name} [Planner] placed a new ${buildingType} site at (${Math.floor(bestX)}, ${Math.floor(bestY)})`); - } + buildings.push(newBuilding); + logAction(`${cit.name} [Planner] randomly placed a ${buildingType} at (${Math.floor(randomX)}, ${Math.floor(randomY)})`); cit.task = null; } -function exploreTask(cit) { - if (!cit.target) { - cit.task = null; - return; - } - - moveToward(cit, cit.target.x, cit.target.y, 0.5); - - // When reached exploration point, find a new task - if (distance(cit.x, cit.y, cit.target.x, cit.target.y) < 10) { - cit.task = null; - cit.target = null; - } -} function huntWolfTask(cit) { let wolf = cit.target;