2025-04-02 22:33:18 +02:00

310 lines
9.8 KiB
JavaScript

/**********************************************************************
* GAME CORE
**********************************************************************/
// AI functions are defined in ai.js and loaded via script tag
let frameCount = 0;
let money = 5000; // Start with 5000
let purchaseMode = null;
// Shared city storage (wood, etc.)
const cityStorage = {
wood: 0,
food: 0,
medicine: 0,
knowledge: 0
};
// Arrays
let resources = []; // Trees, FruitTrees
let buildings = []; // House, Road, Market, Hospital, School
let citizens = [];
let animals = [];
/**********************************************************************
* COSTS & EARNINGS
**********************************************************************/
const COST_HOUSE = 300;
const COST_ROAD = 150;
const COST_BUILDER = 100;
const COST_FARMER = 100;
const COST_MERCHANT = 150;
const COST_DOCTOR = 200;
const COST_TEACHER = 180;
const COST_MARKET = 400;
const COST_HOSPITAL = 500;
const COST_SCHOOL = 450;
const COST_SPAWNER = 500;
const COST_TREE = 50;
const COST_SOLDIER = 250;
const COST_PLANNER = 1000;
// Terrain costs
const COST_WATER = 20;
const COST_GRASS = 10;
const COST_SAND = 15;
const COST_DIRT = 5;
const COST_STONE = 25;
// Earn money
const REWARD_DELIVER_WOOD = 5;
const REWARD_BUILD_COMPLETE = 20;
const REWARD_MARKET_INCOME = 15;
const REWARD_EDUCATION = 10;
/**********************************************************************
* STATS & CONSTANTS
**********************************************************************/
// Hunger & energy
const HUNGER_MAX = 100;
const ENERGY_MAX = 100;
const HEALTH_MAX = 100;
const EDUCATION_MAX = 100;
const HUNGER_INCREMENT = 0.005;
const ENERGY_DECREMENT_WORK = 0.02;
const ENERGY_INCREMENT_REST = 0.05;
const HUNGER_THRESHOLD = 50;
const ENERGY_THRESHOLD = 30;
const HEALTH_THRESHOLD = 40;
// Trees
const FRUIT_TREE_START_AMOUNT = 20;
const FRUIT_GATHER_RATE = 1;
const FRUIT_PLANT_COST = 1;
// House & Road
const HOUSE_WOOD_REQUIRED = 50;
const HOUSE_BUILD_RATE = 0.2;
const HOUSE_MAX_FRUIT = 30;
const ROAD_WOOD_REQUIRED = 10;
const ROAD_BUILD_RATE = 0.3;
// Market, Hospital, School
const MARKET_WOOD_REQUIRED = 60;
const MARKET_BUILD_RATE = 0.15;
const HOSPITAL_WOOD_REQUIRED = 70;
const HOSPITAL_BUILD_RATE = 0.1;
const SCHOOL_WOOD_REQUIRED = 65;
const SCHOOL_BUILD_RATE = 0.12;
// Professions
const PROFESSIONS = ["Farmer", "Builder", "Merchant", "Doctor", "Teacher", "Soldier", "Planner"];
// Animals
const STARTING_RABBITS = 10;
const STARTING_WOLVES = 5;
const RABBIT_HUNGER_INCREMENT = 0.003;
const RABBIT_REPRO_COOLDOWN = 5000;
const RABBIT_REPRO_CHANCE = 0.0002;
const WOLF_SPEED = 0.7; // Faster than rabbits
/**********************************************************************
* INIT WORLD
**********************************************************************/
// Function to spawn wolves at the corners of the map
function spawnWolvesAtCorners() {
const corners = [
{ x: -1800, y: -1800 },
{ x: -1800, y: 1800 },
{ x: 1800, y: -1800 },
{ x: 1800, y: 1800 }
];
corners.forEach(corner => {
// Try to find valid placement near the corner
let x, y;
let attempts = 0;
do {
x = corner.x + randInt(-100, 100);
y = corner.y + randInt(-100, 100);
attempts++;
// Give up after too many attempts
if (attempts > 20) break;
} while (!isValidPlacement(x, y));
// If we found a valid spot, spawn 3 wolves there
if (attempts <= 20) {
for (let i = 0; i < 3; i++) {
const wolf = createAnimal("Wolf", x + randInt(-50, 50), y + randInt(-50, 50));
animals.push(wolf);
}
logAction(`A pack of wolves has appeared near (${Math.floor(x)}, ${Math.floor(y)})!`);
}
});
}
function initWorld() {
// Normal trees - only on land
for(let i=0; i<15; i++) {
let x, y;
do {
x = randInt(-1000,1000);
y = randInt(-1000,1000);
} while (!isValidPlacement(x, y));
resources.push(createResource("Tree", x, y, 100));
}
// Fruit trees - only on land
for(let i=0; i<10; i++) {
let x, y;
do {
x = randInt(-1000,1000);
y = randInt(-1000,1000);
} while (!isValidPlacement(x, y));
resources.push(createResource("FruitTree", x, y, FRUIT_TREE_START_AMOUNT));
}
// Start with 1 citizen => always a "Builder" - only on land
let cx, cy;
do {
cx = randInt(-200,200);
cy = randInt(-200,200);
} while (!isValidPlacement(cx, cy));
let c = createCitizen(randomName(), cx, cy, "Builder");
citizens.push(c);
logAction(`Initial Citizen joined: ${c.name} [Builder]`);
// Spawn some animals - only on land
for(let i=0; i<STARTING_RABBITS; i++) {
let x, y;
do {
x = randInt(-1000,1000);
y = randInt(-1000,1000);
} while (!isValidPlacement(x, y));
animals.push(createAnimal("Rabbit", x, y));
}
for(let i=0; i<STARTING_WOLVES; i++) {
let x, y;
do {
x = randInt(-1000,1000);
y = randInt(-1000,1000);
} while (!isValidPlacement(x, y));
animals.push(createAnimal("Wolf", x, y));
}
// Spawn wolves at the corners of the map
spawnWolvesAtCorners();
// Spawn additional wolves in random locations
for(let i=0; i<5; i++) {
let x, y;
do {
x = randInt(-1500,1500);
y = randInt(-1500,1500);
} while (!isValidPlacement(x, y));
animals.push(createAnimal("Wolf", x, y));
}
requestAnimationFrame(update);
}
/**********************************************************************
* UPDATE LOOP
**********************************************************************/
function update() {
frameCount++;
// Citizens
citizens.forEach((cit) => {
if(typeof updateCitizen === 'function') {
updateCitizen(cit);
} else {
console.error("updateCitizen function is not defined!");
}
});
// Animals
animals.forEach((ani) => {
if(!ani.dead) updateAnimal(ani);
});
animals = animals.filter(a => !a.dead);
// Automatic new citizen every 3000 frames
if(frameCount % 3000 === 0) {
let baby = createCitizen(randomName(), randInt(-200,200), randInt(-200,200));
citizens.push(baby);
logAction(`A new citizen is born: ${baby.name} [${baby.profession}]`);
}
// Buildings
buildings.forEach((b) => {
if(!b.completed && b.deliveredWood >= b.requiredWood) {
let buildRate;
switch(b.buildingType) {
case "Road": buildRate = ROAD_BUILD_RATE; break;
case "House": buildRate = HOUSE_BUILD_RATE; break;
case "Market": buildRate = MARKET_BUILD_RATE; break;
case "Hospital": buildRate = HOSPITAL_BUILD_RATE; break;
case "School": buildRate = SCHOOL_BUILD_RATE; break;
default: buildRate = HOUSE_BUILD_RATE;
}
b.buildProgress += buildRate;
if(b.buildProgress >= 100) {
b.completed = true;
addMoney(REWARD_BUILD_COMPLETE, `Complete ${b.buildingType}`);
if(b.buildingType === "House") {
logAction(`A new House completed @(${Math.floor(b.x)},${Math.floor(b.y)})!`);
maybeBuildRoad(b);
} else if(b.buildingType === "Road") {
logAction(`A Road has been completed!`);
} else if(b.buildingType === "Market") {
logAction(`A Market has been completed! Will generate income.`);
} else if(b.buildingType === "Hospital") {
logAction(`A Hospital has been completed! Citizens can heal here.`);
} else if(b.buildingType === "School") {
logAction(`A School has been completed! Citizens can learn here.`);
}
}
}
// Special building functions
if(b.completed) {
if(b.buildingType === "Market" && frameCount % 500 === 0) {
addMoney(REWARD_MARKET_INCOME, "Market income");
}
if(b.buildingType === "School" && frameCount % 800 === 0) {
addMoney(REWARD_EDUCATION, "Education benefits");
}
}
});
drawWorld();
updateHUD();
requestAnimationFrame(update);
}
function updateHUD() {
// Update money and citizen count
moneyDisplay.textContent = `Money: $${money}`;
citizenCountDisplay.textContent = `Citizens: ${citizens.length}`;
// Update resource counts
woodDisplay.textContent = `Wood: ${cityStorage.wood}`;
foodDisplay.textContent = `Food: ${cityStorage.food}`;
medicineDisplay.textContent = `Medicine: ${cityStorage.medicine}`;
knowledgeDisplay.textContent = `Knowledge: ${cityStorage.knowledge}`;
// Update building counts
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;
buildingCountsDisplay.textContent = `Buildings: 🏠${houseCount} 🏪${marketCount} 🏥${hospitalCount} 🏫${schoolCount}`;
// Update profession counts
const builderCount = citizens.filter(c => c.profession === "Builder").length;
const farmerCount = citizens.filter(c => c.profession === "Farmer").length;
const merchantCount = citizens.filter(c => c.profession === "Merchant").length;
const doctorCount = citizens.filter(c => c.profession === "Doctor").length;
const teacherCount = citizens.filter(c => c.profession === "Teacher").length;
const soldierCount = citizens.filter(c => c.profession === "Soldier").length;
professionCountsDisplay.textContent = `Citizens: 👷${builderCount} 🌾${farmerCount} 💰${merchantCount} 💉${doctorCount} 📚${teacherCount} ⚔️${soldierCount}`;
}