201 lines
6.3 KiB
JavaScript
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;
|
|
}
|