201 lines
6.3 KiB
JavaScript

/**********************************************************************
* LOGGING & MONEY
**********************************************************************/
function logAction(text) {
const entry = document.createElement('div');
entry.className = 'log-entry';
entry.textContent = text;
logContainer.appendChild(entry);
logContainer.scrollTop = logContainer.scrollHeight;
}
function addMoney(amount, reason="") {
money += amount;
if(money < 0) money = 0;
updateMoneyDisplay();
if(reason) {
logAction(`Money ${amount >= 0 ? '+' : ''}$${amount} from ${reason}. Total=$${money}`);
}
}
function updateMoneyDisplay() {
moneyDisplay.textContent = `Money: $${money}`;
citizenCountDisplay.textContent = `Citizens: ${citizens.length}`;
}
/**********************************************************************
* RANDOM UTILS
**********************************************************************/
const firstNames = ["Al", "Bea", "Cal", "Dee", "Eve", "Fay", "Gil", "Hal", "Ian", "Joy",
"Kay", "Lee", "Max", "Ned", "Oda", "Pam", "Ray", "Sue", "Tim", "Ula",
"Vic", "Wyn", "Xan", "Yel", "Zed"];
const lastNames = ["Apple", "Berry", "Cherry", "Delta", "Echo", "Flint", "Gran", "Hills",
"Iris", "Jones", "Knight", "Lemon", "Myer", "Noble", "Olson", "Prime",
"Quartz", "Row", "Smith", "Turn", "Umbra", "Vale", "Wick", "Xeno", "Yolk", "Zoom"];
function randomName() {
const f = firstNames[Math.floor(Math.random() * firstNames.length)];
const l = lastNames[Math.floor(Math.random() * lastNames.length)];
return `${f} ${l}`;
}
function randInt(min, max) {
return Math.floor(Math.random() * (max - min)) + min;
}
/**********************************************************************
* MOVE & DISTANCE
**********************************************************************/
// Default implementation of isWater - will be overridden by terrain.js
function isWater(x, y) {
return false; // Default to no water
}
// Water movement penalty constant
const WATER_MOVEMENT_PENALTY = 0.5;
function moveToward(obj, tx, ty, speed=0.4) {
let dx = tx - obj.x;
let dy = ty - obj.y;
let dist = Math.sqrt(dx*dx + dy*dy);
// Apply water movement penalty if in water
let actualSpeed = speed;
if (isWater(obj.x, obj.y)) {
actualSpeed *= WATER_MOVEMENT_PENALTY;
}
if(dist > 1) {
obj.vx = (dx/dist) * actualSpeed;
obj.vy = (dy/dist) * actualSpeed;
} else {
obj.vx = 0;
obj.vy = 0;
}
}
function distance(x1, y1, x2, y2) {
let dx = x2 - x1;
let dy = y2 - y1;
return Math.sqrt(dx*dx + dy*dy);
}
function knockback(ent, tx, ty, dist) {
let dx = ent.x - tx;
let dy = ent.y - ty;
let length = Math.sqrt(dx*dx + dy*dy);
if(length > 0.1) {
ent.x += (dx/length) * dist;
ent.y += (dy/length) * dist;
}
}
/**********************************************************************
* PLACEMENT VALIDATION
**********************************************************************/
// Check if a position is valid for placement (not in water)
function isValidPlacement(x, y) {
return !isWater(x, y);
}
/**********************************************************************
* FINDERS
**********************************************************************/
function findNearestResourceOfType(ref, rtype) {
let best = null;
let bestD = Infinity;
resources.forEach((res) => {
if(res.type === rtype && res.amount > 0) {
let d = distance(ref.x, ref.y, res.x, res.y);
if(d < bestD) {
bestD = d;
best = res;
}
}
});
return best;
}
function findNearestAnimalOfType(me, targetType) {
let best = null;
let bestD = Infinity;
animals.forEach((a) => {
if(a.type === targetType && !a.dead && a !== me) {
let d = distance(me.x, me.y, a.x, a.y);
if(d < bestD) {
bestD = d;
best = a;
}
}
});
return best;
}
function findHouseWithFruit() {
return buildings.find(b => b.buildingType === "House" && b.completed && b.storedFruit > 0);
}
function findAnyCompletedHouse() {
return buildings.find(b => b.buildingType === "House" && b.completed);
}
function findHouseNeedingFruit() {
return buildings.find(b => b.buildingType === "House" && b.completed && b.storedFruit < b.maxFruit);
}
function findCompletedMarket() {
return buildings.find(b => b.buildingType === "Market" && b.completed);
}
function findCompletedHospital() {
return buildings.find(b => b.buildingType === "Hospital" && b.completed);
}
function findCompletedSchool() {
return buildings.find(b => b.buildingType === "School" && b.completed);
}
/**********************************************************************
* ROAD BUILDING WHEN HOUSE COMPLETES
**********************************************************************/
function maybeBuildRoad(newHouse) {
let otherHouses = buildings.filter(b => b.buildingType === "House" && b.completed && b !== newHouse);
if(otherHouses.length === 0) return;
let nearest = null;
let minD = Infinity;
otherHouses.forEach((oh) => {
let d = distance(newHouse.x, newHouse.y, oh.x, oh.y);
if(d < minD) {
minD = d;
nearest = oh;
}
});
if(!nearest) return;
let road = createRoadSite(newHouse.x, newHouse.y, nearest.x, nearest.y);
buildings.push(road);
logAction(`A Road site created between House@(${Math.floor(newHouse.x)},${Math.floor(newHouse.y)}) & House@(${Math.floor(nearest.x)},${Math.floor(nearest.y)}).`);
}
/**********************************************************************
* DEPOSIT WOOD LOGIC
**********************************************************************/
function depositWoodToStorage(cit) {
let buildingNeedingWood = buildings.find(b => !b.completed && b.deliveredWood < b.requiredWood);
if(cit.carryingWood > 0 && !buildingNeedingWood) {
// deposit wood to city center (0,0)
let d = distance(cit.x, cit.y, 0, 0);
if(d < 20) {
cityStorage.wood += cit.carryingWood;
logAction(`${cit.name} deposited ${cit.carryingWood} wood into city storage.`);
cit.carryingWood = 0;
return true;
} else {
moveToward(cit, 0, 0, 0.4);
return true;
}
}
return false;
}