feat: Modify planner to place random buildings in random locations
This commit is contained in:
parent
18ac996f65
commit
22adaeb700
201
ai.js
201
ai.js
@ -389,59 +389,34 @@ function soldierTasks(cit) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function plannerTasks(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) {
|
if(Math.random() < 0.05) {
|
||||||
cit.task = "planBuilding";
|
cit.task = "planBuilding";
|
||||||
cit.target = null;
|
cit.target = null;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Help with other tasks when not planning
|
// Just wander around when not placing buildings
|
||||||
if(Math.random() < 0.5) {
|
cit.task = null;
|
||||||
builderTasks(cit);
|
cit.target = null;
|
||||||
} else {
|
|
||||||
// Explore the map to find good building locations
|
|
||||||
cit.task = "explore";
|
|
||||||
cit.target = {
|
|
||||||
x: randInt(-500, 500),
|
|
||||||
y: randInt(-500, 500)
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**********************************************************************
|
/**********************************************************************
|
||||||
* NEW TASK HANDLERS
|
* NEW TASK HANDLERS
|
||||||
**********************************************************************/
|
**********************************************************************/
|
||||||
function planBuildingTask(cit) {
|
function planBuildingTask(cit) {
|
||||||
// Find a good spot to place a new building
|
// Choose a completely random building type
|
||||||
let buildingType = null;
|
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;
|
let buildingCost = 0;
|
||||||
|
switch (buildingType) {
|
||||||
// Decide what type of building to place based on current needs
|
case "House": buildingCost = COST_HOUSE; break;
|
||||||
const houseCount = buildings.filter(b => b.buildingType === "House" && b.completed).length;
|
case "Road": buildingCost = COST_ROAD; break;
|
||||||
const marketCount = buildings.filter(b => b.buildingType === "Market" && b.completed).length;
|
case "Market": buildingCost = COST_MARKET; break;
|
||||||
const hospitalCount = buildings.filter(b => b.buildingType === "Hospital" && b.completed).length;
|
case "Hospital": buildingCost = COST_HOSPITAL; break;
|
||||||
const schoolCount = buildings.filter(b => b.buildingType === "School" && b.completed).length;
|
case "School": buildingCost = COST_SCHOOL; break;
|
||||||
|
|
||||||
// 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;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if we can afford it
|
// Check if we can afford it
|
||||||
@ -450,118 +425,56 @@ function planBuildingTask(cit) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find a good location - away from other buildings but not too far from city center
|
// Choose a completely random location
|
||||||
let bestX = 0, bestY = 0;
|
let randomX, randomY;
|
||||||
let bestScore = -Infinity;
|
let attempts = 0;
|
||||||
|
|
||||||
for (let attempt = 0; attempt < 10; attempt++) {
|
// Just try random locations until we find one that's not water
|
||||||
// Try to find locations expanding outward from existing buildings
|
do {
|
||||||
let existingBuilding = buildings[Math.floor(Math.random() * buildings.length)];
|
randomX = randInt(-1500, 1500);
|
||||||
if (!existingBuilding) {
|
randomY = randInt(-1500, 1500);
|
||||||
// If no buildings yet, start near center
|
attempts++;
|
||||||
existingBuilding = { x: 0, y: 0 };
|
if (attempts > 20) {
|
||||||
}
|
// Give up after too many attempts
|
||||||
|
cit.task = null;
|
||||||
const direction = Math.random() * Math.PI * 2;
|
return;
|
||||||
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;
|
|
||||||
}
|
}
|
||||||
|
} 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
|
buildings.push(newBuilding);
|
||||||
if (bestScore > -Infinity) {
|
logAction(`${cit.name} [Planner] randomly placed a ${buildingType} at (${Math.floor(randomX)}, ${Math.floor(randomY)})`);
|
||||||
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)})`);
|
|
||||||
}
|
|
||||||
|
|
||||||
cit.task = null;
|
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) {
|
function huntWolfTask(cit) {
|
||||||
let wolf = cit.target;
|
let wolf = cit.target;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user