mirror of
https://github.com/Azgaar/Fantasy-Map-Generator.git
synced 2025-12-17 17:51:24 +01:00
refactor: expand states
This commit is contained in:
parent
ff974a4fd0
commit
aa6cefb683
6 changed files with 220 additions and 108 deletions
|
|
@ -1,90 +1,195 @@
|
||||||
import {TIME} from "config/logging";
|
|
||||||
import FlatQueue from "flatqueue";
|
import FlatQueue from "flatqueue";
|
||||||
|
|
||||||
|
import {TIME} from "config/logging";
|
||||||
|
import {getInputNumber} from "utils/nodeUtils";
|
||||||
import {minmax} from "utils/numberUtils";
|
import {minmax} from "utils/numberUtils";
|
||||||
|
import type {createCapitals} from "./createCapitals";
|
||||||
|
import type {createStates} from "./createStates";
|
||||||
|
import {ELEVATION, FOREST_BIOMES, MIN_LAND_HEIGHT, DISTANCE_FIELD} from "config/generation";
|
||||||
|
|
||||||
|
type TCapitals = ReturnType<typeof createCapitals>;
|
||||||
|
type TStates = ReturnType<typeof createStates>;
|
||||||
|
|
||||||
// growth algorithm to assign cells to states
|
// growth algorithm to assign cells to states
|
||||||
export function expandStates() {
|
export function expandStates(
|
||||||
|
capitals: TCapitals,
|
||||||
|
states: TStates,
|
||||||
|
features: TPackFeatures,
|
||||||
|
cells: Pick<IPack["cells"], "c" | "h" | "f" | "t" | "r" | "fl" | "s" | "biome" | "culture">
|
||||||
|
) {
|
||||||
TIME && console.time("expandStates");
|
TIME && console.time("expandStates");
|
||||||
const {cells, states, cultures, burgs} = pack;
|
|
||||||
|
|
||||||
cells.state = new Uint16Array(cells.i.length);
|
const cellsNumber = cells.s.length;
|
||||||
const queue = new FlatQueue();
|
const stateIds = new Uint16Array(cellsNumber);
|
||||||
const cost = [];
|
|
||||||
const neutral = (cells.i.length / 5000) * 2500 * neutralInput.value * statesNeutral; // limit cost for state growth
|
|
||||||
|
|
||||||
states
|
const queue = new FlatQueue<{cellId: number; stateId: number}>();
|
||||||
.filter(s => s.i && !s.removed)
|
const cost: number[] = [];
|
||||||
.forEach(state => {
|
|
||||||
const capitalCell = burgs[state.capital].cell;
|
const neutralInput = getInputNumber("neutralInput");
|
||||||
cells.state[capitalCell] = state.i;
|
const maxExpansionCost = (cellsNumber / 2) * neutralInput * statesNeutral;
|
||||||
const cultureCenter = cultures[state.culture].center;
|
|
||||||
const biome = cells.biome[cultureCenter]; // state native biome
|
for (const {i: stateId, cell: cellId} of capitals) {
|
||||||
queue.push({cellId: state.center, stateId: state.i, b: biome}, 0);
|
stateIds[cellId] = stateId;
|
||||||
cost[state.center] = 1;
|
cost[cellId] = 1;
|
||||||
});
|
queue.push({cellId, stateId}, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
// expansion costs (less is better)
|
||||||
|
const SAME_CULTURE_BONUS = -9;
|
||||||
|
const DIFFERENT_CULTURES_FEE = 100;
|
||||||
|
|
||||||
|
const MAX_SUITABILITY_COST = 20;
|
||||||
|
const UNINHABITED_LAND_FEE = 5000;
|
||||||
|
|
||||||
|
const NATIVE_BIOME_FIXED_COST = 10;
|
||||||
|
const HUNTERS_NON_NATIVE_BIOME_FEE_MULTIPLIER = 2;
|
||||||
|
const NOMADS_FOREST_BIOMES_FEE_MULTIPLIER = 3;
|
||||||
|
const GENERIC_NON_NATIVE_BIOME_FEE_MULTIPLIER = 1;
|
||||||
|
|
||||||
|
const GENERIC_DEEP_WATER_FEE_MULTIPLIER = 2;
|
||||||
|
const GENERIC_WATER_CROSSING_FEE = 1000;
|
||||||
|
const NOMADS_WATER_CROSSING_FEE = 10000;
|
||||||
|
const NAVAL_WATER_CROSSING_FEE = 300;
|
||||||
|
const LAKE_STATES_LAKE_CROSSING_FEE = 10;
|
||||||
|
const GENERIC_MOUNTAINS_CROSSING_FEE = 2200;
|
||||||
|
const GENERIC_HILLS_CROSSING_FEE = 300;
|
||||||
|
const HIGHLAND_STATE_LOWLANDS_FEE = 1100;
|
||||||
|
const HIGHLAND_STATE_HIGHTLAND_COST = 0;
|
||||||
|
|
||||||
|
const RIVER_STATE_RIVER_CROSSING_COST = 0;
|
||||||
|
const RIVER_STATE_NO_RIVER_COST = 100;
|
||||||
|
const RIVER_CROSSING_MIN_COST = 20;
|
||||||
|
const RIVER_CROSSING_MAX_COST = 100;
|
||||||
|
|
||||||
|
const GENERIC_LAND_COAST_FEE = 20;
|
||||||
|
const MARITIME_LAND_COAST_FEE = 0;
|
||||||
|
const NOMADS_LAND_COAST_FEE = 60;
|
||||||
|
const GENERIC_LANDLOCKED_FEE = 0;
|
||||||
|
const NAVAL_LANDLOCKED_FEE = 30;
|
||||||
|
|
||||||
while (queue.length) {
|
while (queue.length) {
|
||||||
const priority = queue.peekValue();
|
const priority = queue.peekValue()!;
|
||||||
const {cellId, stateId, biome} = queue.pop();
|
const {cellId, stateId} = queue.pop()!;
|
||||||
const {type, culture} = states[stateId];
|
|
||||||
|
const {type, culture, center, expansionism} = getState(stateId);
|
||||||
|
const capitalBiome = cells.biome[center];
|
||||||
|
|
||||||
cells.c[cellId].forEach(neibCellId => {
|
cells.c[cellId].forEach(neibCellId => {
|
||||||
if (cells.state[neibCellId] && neibCellId === states[cells.state[neibCellId]].center) return; // do not overwrite capital cells
|
if (neibCellId === center && stateIds[neibCellId]) return; // do not overwrite capital cells
|
||||||
|
|
||||||
|
const cultureCost = getCultureCost(culture, neibCellId);
|
||||||
|
const populationCost = getPopulationCost(neibCellId);
|
||||||
|
const biomeCost = getBiomeCost(neibCellId, capitalBiome, type);
|
||||||
|
const heightCost = getHeightCost(neibCellId, type);
|
||||||
|
const riverCost = getRiverCost(neibCellId, type);
|
||||||
|
const typeCost = getTypeCost(neibCellId, type);
|
||||||
|
|
||||||
const cultureCost = culture === cells.culture[neibCellId] ? -9 : 100;
|
|
||||||
const populationCost =
|
|
||||||
cells.h[neibCellId] < 20 ? 0 : cells.s[neibCellId] ? Math.max(20 - cells.s[neibCellId], 0) : 5000;
|
|
||||||
const biomeCost = getBiomeCost(biome, cells.biome[neibCellId], type);
|
|
||||||
const heightCost = getHeightCost(pack.features[cells.f[neibCellId]], cells.h[neibCellId], type);
|
|
||||||
const riverCost = getRiverCost(cells.r[neibCellId], neibCellId, type);
|
|
||||||
const typeCost = getTypeCost(cells.t[neibCellId], type);
|
|
||||||
const cellCost = Math.max(cultureCost + populationCost + biomeCost + heightCost + riverCost + typeCost, 0);
|
const cellCost = Math.max(cultureCost + populationCost + biomeCost + heightCost + riverCost + typeCost, 0);
|
||||||
const totalCost = priority + 10 + cellCost / states[stateId].expansionism;
|
const totalCost = priority + 10 + cellCost / expansionism;
|
||||||
|
if (totalCost > maxExpansionCost) return;
|
||||||
if (totalCost > neutral) return;
|
|
||||||
|
|
||||||
if (!cost[neibCellId] || totalCost < cost[neibCellId]) {
|
if (!cost[neibCellId] || totalCost < cost[neibCellId]) {
|
||||||
if (cells.h[neibCellId] >= 20) cells.state[neibCellId] = stateId; // assign state to cell
|
if (cells.h[neibCellId] >= MIN_LAND_HEIGHT) stateIds[neibCellId] = stateId; // assign state to cell
|
||||||
cost[neibCellId] = totalCost;
|
cost[neibCellId] = totalCost;
|
||||||
|
|
||||||
queue.push({cellId: neibCellId, stateId, biome}, totalCost);
|
queue.push({cellId: neibCellId, stateId}, totalCost);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
burgs.filter(b => b.i && !b.removed).forEach(b => (b.state = cells.state[b.cell])); // assign state to burgs
|
|
||||||
|
|
||||||
function getBiomeCost(b, biome, type) {
|
|
||||||
if (b === biome) return 10; // tiny penalty for native biome
|
|
||||||
if (type === "Hunting") return biomesData.cost[biome] * 2; // non-native biome penalty for hunters
|
|
||||||
if (type === "Nomadic" && biome > 4 && biome < 10) return biomesData.cost[biome] * 3; // forest biome penalty for nomads
|
|
||||||
return biomesData.cost[biome]; // general non-native biome penalty
|
|
||||||
}
|
|
||||||
|
|
||||||
function getHeightCost(f, h, type) {
|
|
||||||
if (type === "Lake" && f.type === "lake") return 10; // low lake crossing penalty for Lake cultures
|
|
||||||
if (type === "Naval" && h < 20) return 300; // low sea crossing penalty for Navals
|
|
||||||
if (type === "Nomadic" && h < 20) return 10000; // giant sea crossing penalty for Nomads
|
|
||||||
if (h < 20) return 1000; // general sea crossing penalty
|
|
||||||
if (type === "Highland" && h < 62) return 1100; // penalty for highlanders on lowlands
|
|
||||||
if (type === "Highland") return 0; // no penalty for highlanders on highlands
|
|
||||||
if (h >= 67) return 2200; // general mountains crossing penalty
|
|
||||||
if (h >= 44) return 300; // general hills crossing penalty
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
function getRiverCost(r, i, type) {
|
|
||||||
if (type === "River") return r ? 0 : 100; // penalty for river cultures
|
|
||||||
if (!r) return 0; // no penalty for others if there is no river
|
|
||||||
return minmax(cells.fl[i] / 10, 20, 100); // river penalty from 20 to 100 based on flux
|
|
||||||
}
|
|
||||||
|
|
||||||
function getTypeCost(t, type) {
|
|
||||||
if (t === 1) return type === "Naval" || type === "Lake" ? 0 : type === "Nomadic" ? 60 : 20; // penalty for coastline
|
|
||||||
if (t === 2) return type === "Naval" || type === "Nomadic" ? 30 : 0; // low penalty for land level 2 for Navals and nomads
|
|
||||||
if (t !== -1) return type === "Naval" || type === "Lake" ? 100 : 0; // penalty for mainland for navals
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
TIME && console.timeEnd("expandStates");
|
TIME && console.timeEnd("expandStates");
|
||||||
|
return stateIds;
|
||||||
|
|
||||||
|
function isNeutrals(state: Entry<TStates>): state is TNeutrals {
|
||||||
|
return state.i === 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getState(stateId: number) {
|
||||||
|
const state = states[stateId];
|
||||||
|
if (isNeutrals(state)) throw new Error("Neutrals cannot expand");
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCultureCost(cellId: number, stateCulture: number) {
|
||||||
|
return cells.culture[cellId] === stateCulture ? SAME_CULTURE_BONUS : DIFFERENT_CULTURES_FEE;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getPopulationCost(cellId: number) {
|
||||||
|
const isWater = cells.h[cellId] < MIN_LAND_HEIGHT;
|
||||||
|
if (isWater) return 0;
|
||||||
|
|
||||||
|
const suitability = cells.s[cellId];
|
||||||
|
if (suitability) return Math.max(MAX_SUITABILITY_COST - suitability, 0);
|
||||||
|
|
||||||
|
return UNINHABITED_LAND_FEE;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getBiomeCost(cellId: number, capitalBiome: number, type: TCultureType) {
|
||||||
|
const biome = cells.biome[cellId];
|
||||||
|
if (biome === capitalBiome) return NATIVE_BIOME_FIXED_COST;
|
||||||
|
|
||||||
|
const defaultCost = biomesData.cost[biome];
|
||||||
|
if (type === "Hunting") return defaultCost * HUNTERS_NON_NATIVE_BIOME_FEE_MULTIPLIER;
|
||||||
|
if (type === "Nomadic" && FOREST_BIOMES.includes(biome)) return defaultCost * NOMADS_FOREST_BIOMES_FEE_MULTIPLIER;
|
||||||
|
return defaultCost * GENERIC_NON_NATIVE_BIOME_FEE_MULTIPLIER;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getHeightCost(cellId: number, type: TCultureType) {
|
||||||
|
const height = cells.h[cellId];
|
||||||
|
const isWater = height < MIN_LAND_HEIGHT;
|
||||||
|
|
||||||
|
if (isWater) {
|
||||||
|
const feature = features[cells.f[cellId]];
|
||||||
|
if (feature === 0) throw new Error(`No feature for cell ${cellId}`);
|
||||||
|
const isDeepWater = cells.t[cellId] > DISTANCE_FIELD.WATER_COAST;
|
||||||
|
const multiplier = isDeepWater ? GENERIC_DEEP_WATER_FEE_MULTIPLIER : 1;
|
||||||
|
|
||||||
|
if (type === "Lake" && feature.type === "lake") return LAKE_STATES_LAKE_CROSSING_FEE * multiplier;
|
||||||
|
if (type === "Naval") return NAVAL_WATER_CROSSING_FEE * multiplier;
|
||||||
|
if (type === "Nomadic") return NOMADS_WATER_CROSSING_FEE * multiplier;
|
||||||
|
return GENERIC_WATER_CROSSING_FEE * multiplier;
|
||||||
|
}
|
||||||
|
|
||||||
|
const isLowlands = height <= ELEVATION.FOOTHILLS;
|
||||||
|
const isHills = height >= ELEVATION.HILLS;
|
||||||
|
const isMountains = height >= ELEVATION.MOUNTAINS;
|
||||||
|
|
||||||
|
if (type === "Highland") {
|
||||||
|
if (isLowlands) return HIGHLAND_STATE_LOWLANDS_FEE;
|
||||||
|
return HIGHLAND_STATE_HIGHTLAND_COST;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isMountains) return GENERIC_MOUNTAINS_CROSSING_FEE;
|
||||||
|
if (isHills) return GENERIC_HILLS_CROSSING_FEE;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getRiverCost(cellId: number, type: TCultureType) {
|
||||||
|
const isRiver = cells.r[cellId] !== 0;
|
||||||
|
if (type === "River") return isRiver ? RIVER_STATE_RIVER_CROSSING_COST : RIVER_STATE_NO_RIVER_COST;
|
||||||
|
if (!isRiver) return 0;
|
||||||
|
|
||||||
|
const flux = cells.fl[cellId];
|
||||||
|
return minmax(flux / 10, RIVER_CROSSING_MIN_COST, RIVER_CROSSING_MAX_COST);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTypeCost(cellId: number, type: TCultureType) {
|
||||||
|
const isMaritime = type === "Naval" || type === "Lake";
|
||||||
|
const t = cells.t[cellId];
|
||||||
|
|
||||||
|
const isLandCoast = t === DISTANCE_FIELD.LAND_COAST;
|
||||||
|
if (isLandCoast) {
|
||||||
|
if (isMaritime) return MARITIME_LAND_COAST_FEE;
|
||||||
|
if (type === "Nomadic") return NOMADS_LAND_COAST_FEE;
|
||||||
|
return GENERIC_LAND_COAST_FEE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const isLandlocked = t === DISTANCE_FIELD.LANDLOCKED;
|
||||||
|
if (isLandlocked) {
|
||||||
|
if (type === "Naval") return NAVAL_LANDLOCKED_FEE;
|
||||||
|
return GENERIC_LANDLOCKED_FEE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -9,15 +9,12 @@ import {expandStates} from "./expandStates";
|
||||||
import {specifyBurgs} from "./specifyBurgs";
|
import {specifyBurgs} from "./specifyBurgs";
|
||||||
|
|
||||||
export function generateBurgsAndStates(
|
export function generateBurgsAndStates(
|
||||||
cells: Pick<
|
|
||||||
IPack["cells"],
|
|
||||||
"v" | "p" | "i" | "g" | "h" | "f" | "haven" | "harbor" | "r" | "fl" | "biome" | "s" | "culture"
|
|
||||||
>,
|
|
||||||
vertices: IGraphVertices,
|
|
||||||
cultures: TCultures,
|
cultures: TCultures,
|
||||||
features: TPackFeatures,
|
features: TPackFeatures,
|
||||||
temp: Int8Array,
|
cells: Pick<
|
||||||
rivers: Omit<IRiver, "name" | "basin" | "type">[]
|
IPack["cells"],
|
||||||
|
"v" | "c" | "p" | "i" | "g" | "h" | "f" | "t" | "haven" | "harbor" | "r" | "fl" | "biome" | "s" | "culture"
|
||||||
|
>
|
||||||
) {
|
) {
|
||||||
const cellsNumber = cells.i.length;
|
const cellsNumber = cells.i.length;
|
||||||
const burgIds = new Uint16Array(cellsNumber);
|
const burgIds = new Uint16Array(cellsNumber);
|
||||||
|
|
@ -30,13 +27,19 @@ export function generateBurgsAndStates(
|
||||||
const states = createStates(capitals, cultures);
|
const states = createStates(capitals, cultures);
|
||||||
const towns = createTowns(burgIds, cultures, pick(cells, "p", "i", "f", "s", "culture"));
|
const towns = createTowns(burgIds, cultures, pick(cells, "p", "i", "f", "s", "culture"));
|
||||||
|
|
||||||
expandStates();
|
const stateIds = expandStates(
|
||||||
|
capitals,
|
||||||
|
states,
|
||||||
|
features,
|
||||||
|
pick(cells, "c", "h", "f", "t", "r", "fl", "s", "biome", "culture")
|
||||||
|
);
|
||||||
// normalizeStates();
|
// normalizeStates();
|
||||||
|
// burgs.filter(b => b.i && !b.removed).forEach(b => (b.state = stateIds[b.cell])); // assign state to burgs
|
||||||
const roadScores = new Uint16Array(cellsNumber); // TODO: define roads
|
const roadScores = new Uint16Array(cellsNumber); // TODO: define roads
|
||||||
|
|
||||||
const burgs = specifyBurgs(capitals, towns, roadScores);
|
const burgs = specifyBurgs(capitals, towns, roadScores);
|
||||||
|
|
||||||
return {burgIds, states, burgs};
|
return {burgIds, stateIds, burgs, states};
|
||||||
|
|
||||||
function getScoredCellIds() {
|
function getScoredCellIds() {
|
||||||
// cell score for capitals placement
|
// cell score for capitals placement
|
||||||
|
|
|
||||||
|
|
@ -282,35 +282,33 @@ export const expandCultures = function (
|
||||||
TIME && console.time("expandCultures");
|
TIME && console.time("expandCultures");
|
||||||
|
|
||||||
const cultureIds = new Uint16Array(cells.h.length); // cell cultures
|
const cultureIds = new Uint16Array(cells.h.length); // cell cultures
|
||||||
const isWilderness = (culture: ICulture | TWilderness): culture is TWilderness => culture.i === 0;
|
|
||||||
|
|
||||||
const queue = new FlatQueue<{cellId: number; cultureId: number}>();
|
const queue = new FlatQueue<{cellId: number; cultureId: number}>();
|
||||||
|
|
||||||
|
const isWilderness = (culture: ICulture | TWilderness): culture is TWilderness => culture.i === 0;
|
||||||
cultures.forEach(culture => {
|
cultures.forEach(culture => {
|
||||||
if (isWilderness(culture) || culture.removed) return;
|
if (isWilderness(culture) || culture.removed) return;
|
||||||
queue.push({cellId: culture.center, cultureId: culture.i}, 0);
|
queue.push({cellId: culture.center, cultureId: culture.i}, 0);
|
||||||
});
|
});
|
||||||
|
|
||||||
const cellsNumberFactor = cells.h.length / 1.6;
|
const cellsNumberFactor = cells.h.length / 1.6;
|
||||||
const neutral = cellsNumberFactor * getInputNumber("neutralInput"); // limit cost for culture growth
|
const maxExpansionCost = cellsNumberFactor * getInputNumber("neutralInput"); // limit cost for culture growth
|
||||||
const cost: number[] = [];
|
const cost: number[] = [];
|
||||||
|
|
||||||
while (queue.length) {
|
while (queue.length) {
|
||||||
const priority = queue.peekValue()!;
|
const priority = queue.peekValue()!;
|
||||||
const {cellId, cultureId} = queue.pop()!;
|
const {cellId, cultureId} = queue.pop()!;
|
||||||
|
|
||||||
if (cultureId === 0) throw new Error("Wilderness culture should not expand");
|
const {type, expansionism, center} = getCulture(cultureId);
|
||||||
const {type, expansionism} = cultures[cultureId] as ICulture;
|
const cultureBiome = cells.biome[center];
|
||||||
const cultureBiome = cells.biome[cellId];
|
|
||||||
|
|
||||||
cells.c[cellId].forEach(neibCellId => {
|
cells.c[cellId].forEach(neibCellId => {
|
||||||
const biome = cells.biome[neibCellId];
|
const biomeCost = getBiomeCost(neibCellId, cultureBiome, type);
|
||||||
const biomeCost = getBiomeCost(biome, cultureBiome, type);
|
|
||||||
const heightCost = getHeightCost(neibCellId, cells.h[neibCellId], type);
|
const heightCost = getHeightCost(neibCellId, cells.h[neibCellId], type);
|
||||||
const riverCost = getRiverCost(cells.r[neibCellId], neibCellId, type);
|
const riverCost = getRiverCost(cells.r[neibCellId], neibCellId, type);
|
||||||
const typeCost = getTypeCost(cells.t[neibCellId], type);
|
const typeCost = getTypeCost(cells.t[neibCellId], type);
|
||||||
const totalCost = priority + (biomeCost + heightCost + riverCost + typeCost) / expansionism;
|
|
||||||
|
|
||||||
if (totalCost > neutral) return;
|
const totalCost = priority + (biomeCost + heightCost + riverCost + typeCost) / expansionism;
|
||||||
|
if (totalCost > maxExpansionCost) return;
|
||||||
|
|
||||||
if (!cost[neibCellId] || totalCost < cost[neibCellId]) {
|
if (!cost[neibCellId] || totalCost < cost[neibCellId]) {
|
||||||
if (cells.pop[neibCellId] > 0) cultureIds[neibCellId] = cultureId; // assign culture to populated cell
|
if (cells.pop[neibCellId] > 0) cultureIds[neibCellId] = cultureId; // assign culture to populated cell
|
||||||
|
|
@ -323,7 +321,14 @@ export const expandCultures = function (
|
||||||
TIME && console.timeEnd("expandCultures");
|
TIME && console.timeEnd("expandCultures");
|
||||||
return cultureIds;
|
return cultureIds;
|
||||||
|
|
||||||
function getBiomeCost(biome: number, cultureBiome: number, type: TCultureType) {
|
function getCulture(cultureId: number) {
|
||||||
|
const culture = cultures[cultureId];
|
||||||
|
if (isWilderness(culture)) throw new Error("Wilderness culture cannot expand");
|
||||||
|
return culture;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getBiomeCost(cellId: number, cultureBiome: number, type: TCultureType) {
|
||||||
|
const biome = cells.biome[cellId];
|
||||||
if (cultureBiome === biome) return 10; // tiny penalty for native biome
|
if (cultureBiome === biome) return 10; // tiny penalty for native biome
|
||||||
if (type === "Hunting") return biomesData.cost[biome] * 5; // non-native biome penalty for hunters
|
if (type === "Hunting") return biomesData.cost[biome] * 5; // non-native biome penalty for hunters
|
||||||
if (type === "Nomadic" && FOREST_BIOMES.includes(biome)) return biomesData.cost[biome] * 10; // forest biome penalty for nomads
|
if (type === "Nomadic" && FOREST_BIOMES.includes(biome)) return biomesData.cost[biome] * 10; // forest biome penalty for nomads
|
||||||
|
|
|
||||||
|
|
@ -95,25 +95,19 @@ export function createPack(grid: IGrid): IPack {
|
||||||
pop: population
|
pop: population
|
||||||
});
|
});
|
||||||
|
|
||||||
const {burgIds, states, burgs} = generateBurgsAndStates(
|
const {burgIds, stateIds, burgs, states} = generateBurgsAndStates(cultures, mergedFeatures, {
|
||||||
{
|
...pick(cells, "v", "c", "p", "i", "g"),
|
||||||
...pick(cells, "v", "p", "i", "g"),
|
h: heights,
|
||||||
h: heights,
|
f: featureIds,
|
||||||
f: featureIds,
|
t: distanceField,
|
||||||
haven,
|
haven,
|
||||||
harbor,
|
harbor,
|
||||||
r: riverIds,
|
r: riverIds,
|
||||||
fl: flux,
|
fl: flux,
|
||||||
biome,
|
biome,
|
||||||
s: suitability,
|
s: suitability,
|
||||||
culture: cultureIds
|
culture: cultureIds
|
||||||
},
|
});
|
||||||
vertices,
|
|
||||||
cultures,
|
|
||||||
mergedFeatures,
|
|
||||||
temp,
|
|
||||||
rawRivers
|
|
||||||
);
|
|
||||||
|
|
||||||
// Religions.generate();
|
// Religions.generate();
|
||||||
// BurgsAndStates.defineStateForms();
|
// BurgsAndStates.defineStateForms();
|
||||||
|
|
@ -152,8 +146,9 @@ export function createPack(grid: IGrid): IPack {
|
||||||
s: suitability,
|
s: suitability,
|
||||||
pop: population,
|
pop: population,
|
||||||
culture: cultureIds,
|
culture: cultureIds,
|
||||||
burg: burgIds
|
burg: burgIds,
|
||||||
// state, religion, province
|
state: stateIds
|
||||||
|
// religion, province
|
||||||
},
|
},
|
||||||
features: mergedFeatures,
|
features: mergedFeatures,
|
||||||
rivers: rawRivers, // "name" | "basin" | "type"
|
rivers: rawRivers, // "name" | "basin" | "type"
|
||||||
|
|
|
||||||
3
src/types/common.d.ts
vendored
3
src/types/common.d.ts
vendored
|
|
@ -2,6 +2,9 @@ type Logical = number & (1 | 0); // data type for logical numbers
|
||||||
|
|
||||||
type UnknownObject = {[key: string]: unknown};
|
type UnknownObject = {[key: string]: unknown};
|
||||||
|
|
||||||
|
// extract element from array
|
||||||
|
type Entry<T> = T[number];
|
||||||
|
|
||||||
type noop = () => void;
|
type noop = () => void;
|
||||||
|
|
||||||
interface Dict<T> {
|
interface Dict<T> {
|
||||||
|
|
|
||||||
1
src/types/pack/states.d.ts
vendored
1
src/types/pack/states.d.ts
vendored
|
|
@ -2,6 +2,7 @@ interface IState {
|
||||||
i: number;
|
i: number;
|
||||||
name: string;
|
name: string;
|
||||||
culture: number;
|
culture: number;
|
||||||
|
type: TCultureType;
|
||||||
fullName: string;
|
fullName: string;
|
||||||
removed?: boolean;
|
removed?: boolean;
|
||||||
coa: ICoa | string;
|
coa: ICoa | string;
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue