1
0
mirror of https://github.com/veden/Rampant.git synced 2025-01-07 23:01:39 +02:00
Rampant/libs/BaseUtils.lua

520 lines
16 KiB
Lua
Executable File

local baseUtils = {}
-- imports
local stringUtils = require("StringUtils")
local mathUtils = require("MathUtils")
local constants = require("Constants")
local mapUtils = require("MapUtils")
local chunkPropertyUtils = require("ChunkPropertyUtils")
-- constants
local NEUTRAL_WORM_TIERS = constants.NEUTRAL_WORM_TIERS
local NEUTRAL_WORM_VARIATIONS = constants.NEUTRAL_WORM_VARIATIONS
local NEUTRAL_NEST_TIERS = constants.NEUTRAL_NEST_TIERS
local NEUTRAL_NEST_VARIATIONS = constants.NEUTRAL_NEST_VARIATIONS
local PHYSICAL_WORM_TIERS = constants.PHYSICAL_WORM_TIERS
local PHYSICAL_WORM_VARIATIONS = constants.PHYSICAL_WORM_VARIATIONS
local PHYSICAL_NEST_TIERS = constants.PHYSICAL_NEST_TIERS
local PHYSICAL_NEST_VARIATIONS = constants.PHYSICAL_NEST_VARIATIONS
local ELECTRIC_WORM_TIERS = constants.ELECTRIC_WORM_TIERS
local ELECTRIC_WORM_VARIATIONS = constants.ELECTRIC_WORM_VARIATIONS
local ELECTRIC_NEST_TIERS = constants.ELECTRIC_NEST_TIERS
local ELECTRIC_NEST_VARIATIONS = constants.ELECTRIC_NEST_VARIATIONS
local ACID_WORM_TIERS = constants.ACID_WORM_TIERS
local ACID_WORM_VARIATIONS = constants.ACID_WORM_VARIATIONS
local ACID_NEST_TIERS = constants.ACID_NEST_TIERS
local ACID_NEST_VARIATIONS = constants.ACID_NEST_VARIATIONS
local SUICIDE_WORM_TIERS = constants.SUICIDE_WORM_TIERS
local SUICIDE_WORM_VARIATIONS = constants.SUICIDE_WORM_VARIATIONS
local SUICIDE_NEST_TIERS = constants.SUICIDE_NEST_TIERS
local SUICIDE_NEST_VARIATIONS = constants.SUICIDE_NEST_VARIATIONS
local NUCLEAR_WORM_TIERS = constants.NUCLEAR_WORM_TIERS
local NUCLEAR_WORM_VARIATIONS = constants.NUCLEAR_WORM_VARIATIONS
local NUCLEAR_NEST_TIERS = constants.NUCLEAR_NEST_TIERS
local NUCLEAR_NEST_VARIATIONS = constants.NUCLEAR_NEST_VARIATIONS
local FIRE_WORM_TIERS = constants.FIRE_WORM_TIERS
local FIRE_WORM_VARIATIONS = constants.FIRE_WORM_VARIATIONS
local FIRE_NEST_TIERS = constants.FIRE_NEST_TIERS
local FIRE_NEST_VARIATIONS = constants.FIRE_NEST_VARIATIONS
local INFERNO_WORM_TIERS = constants.INFERNO_WORM_TIERS
local INFERNO_WORM_VARIATIONS = constants.INFERNO_WORM_VARIATIONS
local INFERNO_NEST_TIERS = constants.INFERNO_NEST_TIERS
local INFERNO_NEST_VARIATIONS = constants.INFERNO_NEST_VARIATIONS
local TROLL_WORM_TIERS = constants.TROLL_WORM_TIERS
local TROLL_WORM_VARIATIONS = constants.TROLL_WORM_VARIATIONS
local TROLL_NEST_TIERS = constants.TROLL_NEST_TIERS
local TROLL_NEST_VARIATIONS = constants.TROLL_NEST_VARIATIONS
local FAST_WORM_TIERS = constants.FAST_WORM_TIERS
local FAST_WORM_VARIATIONS = constants.FAST_WORM_VARIATIONS
local FAST_NEST_TIERS = constants.FAST_NEST_TIERS
local FAST_NEST_VARIATIONS = constants.FAST_NEST_VARIATIONS
local BASE_ALIGNMENT_NEUTRAL = constants.BASE_ALIGNMENT_NEUTRAL
local BASE_ALIGNMENT_ACID = constants.BASE_ALIGNMENT_ACID
local BASE_ALIGNMENT_ELECTRIC = constants.BASE_ALIGNMENT_ELECTRIC
local BASE_ALIGNMENT_PHYSICAL = constants.BASE_ALIGNMENT_PHYSICAL
local BASE_ALIGNMENT_SUICIDE = constants.BASE_ALIGNMENT_SUICIDE
local BASE_ALIGNMENT_NUCLEAR = constants.BASE_ALIGNMENT_NUCLEAR
local BASE_ALIGNMENT_INFERNO = constants.BASE_ALIGNMENT_INFERNO
local BASE_ALIGNMENT_FIRE = constants.BASE_ALIGNMENT_FIRE
local BASE_ALIGNMENT_FAST = constants.BASE_ALIGNMENT_FAST
local BASE_ALIGNMENT_TROLL = constants.BASE_ALIGNMENT_TROLL
local BASE_WORM_UPGRADE = constants.BASE_WORM_UPGRADE
local BASE_SPAWNER_UPGRADE = constants.BASE_SPAWNER_UPGRADE
local BASE_UPGRADE = constants.BASE_UPGRADE
local BASE_DISTANCE_THRESHOLD = constants.BASE_DISTANCE_THRESHOLD
local BASE_DISTANCE_LEVEL_BONUS = constants.BASE_DISTANCE_LEVEL_BONUS
local BASE_DISTANCE_TO_EVO_INDEX = constants.BASE_DISTANCE_TO_EVO_INDEX
local CHUNK_SIZE = constants.CHUNK_SIZE
local BASE_ALIGNMENT_PATHS = constants.BASE_ALIGNMENT_PATHS
local EVOLUTION_INCREMENTS = constants.EVOLUTION_INCREMENTS
local MAGIC_MAXIMUM_NUMBER = constants.MAGIC_MAXIMUM_NUMBER
local SENTINEL_IMPASSABLE_CHUNK = constants.SENTINEL_IMPASSABLE_CHUNK
-- imported functions
local euclideanDistancePoints = mathUtils.euclideanDistancePoints
local roundToFloor = mathUtils.roundToFloor
local gaussianRandomRange = mathUtils.gaussianRandomRange
local mFloor = math.floor
local mMin = math.min
local mMax = math.max
local getChunkByPosition = mapUtils.getChunkByPosition
local getChunkByXY = mapUtils.getChunkByXY
local getChunkBase = chunkPropertyUtils.getChunkBase
local setChunkBase = chunkPropertyUtils.setChunkBase
local mRandom = math.random
-- module code
function baseUtils.findNearbyBase(map, chunk, chunkRadius)
if (chunk == SENTINEL_IMPASSABLE_CHUNK) then
return nil
end
local tested = {}
local x = chunk.x
local y = chunk.y
local foundBase = getChunkBase(map, chunk)
if foundBase then
return foundBase
end
local closest = MAGIC_MAXIMUM_NUMBER
for xi = x-chunkRadius, x+chunkRadius, CHUNK_SIZE do
for yi = y-chunkRadius, y+chunkRadius, CHUNK_SIZE do
if (xi ~= x) and (yi ~= y) then
local base = getChunkBase(map, getChunkByXY(map, xi, yi))
if base then
if not tested[base] then
tested[base] = true
local distance = euclideanDistancePoints(base.x, base.y, x, y)
if (distance <= (BASE_DISTANCE_THRESHOLD + (base.level * BASE_DISTANCE_LEVEL_BONUS))) and (distance < closest) then
closest = distance
foundBase = base
end
end
end
end
end
end
return foundBase
end
local function findUpgrade(base, evoIndex, natives, evolutionTable)
-- local used = { }
-- local alignments = base.alignments
-- local alignmentPick = base.alignments[mRandom(#base.alignments)]
local alignmentPick = base.alignment
-- while alignmentPick do
-- used[alignmentPick] = true
if not evolutionTable then
print(alignmentPick)
end
for evo=evoIndex, 0, -EVOLUTION_INCREMENTS do
local entitySet = evolutionTable[alignmentPick][evo]
if entitySet and (#entitySet > 0) then
return entitySet[mRandom(#entitySet)]
end
end
-- alignmentPick = nil
-- for x = #alignments, 1, -1 do
-- if not used[alignments[x]] then
-- alignmentPick = alignments[x]
-- break
-- end
-- end
-- end
return nil
end
function baseUtils.upgradeEntity(map, entity, surface, natives, evolutionFactor, tick)
local position = entity.position
local entityType = entity.type
entity.destroy()
local chunk = getChunkByPosition(map, position)
local base = getChunkBase(map, chunk)
if not base then
base = baseUtils.createBase(map, natives, evolutionFactor, chunk, surface, tick)
end
local distance = roundToFloor(mMin(1,
euclideanDistancePoints(position.x, position.y, 0, 0) * BASE_DISTANCE_TO_EVO_INDEX),
EVOLUTION_INCREMENTS)
local evoIndex = mMax(distance, roundToFloor(evolutionFactor, EVOLUTION_INCREMENTS))
local spawnerName = findUpgrade(base, evoIndex, natives, ((entityType == "unit-spawner") and natives.evolutionTableUnitSpawner) or natives.evolutionTableWorm)
if spawnerName then
local newPosition = surface.find_non_colliding_position(spawnerName, position, CHUNK_SIZE, 4)
if newPosition then
entity = surface.create_entity({name = spawnerName, position = newPosition})
end
end
return entity
end
function baseUtils.upgradeBase(base)
local paths = BASE_ALIGNMENT_PATHS[base.alignment]
if paths then
base.alignment = paths[mRandom(#paths)]
return true
end
return false
end
function baseUtils.processBase(map, surface, natives, tick, base, evolutionFactor)
local areaTop = map.position2Top
local areaBottom = map.position2Bottom
areaTop[1] = base.x
areaTop[2] = base.y
areaBottom[1] = base.x + CHUNK_SIZE
areaBottom[2] = base.y + CHUNK_SIZE
local entity
local cost
local choice = mRandom()
if (choice <= 0.3) then
if (base.points >= BASE_SPAWNER_UPGRADE) then
entity = surface.find_entities_filtered(map.filteredEntitiesSpawnerQueryLimited)
cost = BASE_SPAWNER_UPGRADE
end
elseif (choice <= 0.6) then
if (base.points >= BASE_WORM_UPGRADE) then
entity = surface.find_entities_filtered(map.filteredEntitiesWormQueryLimited)
cost = BASE_WORM_UPGRADE
end
elseif (choice >= 0.995) then
if (base.points >= BASE_UPGRADE) then
if baseUtils.upgradeBase(base) then
base.points = base.points - BASE_UPGRADE
end
end
end
if entity and (#entity > 0) then
baseUtils.upgradeEntity(map, entity[mRandom(#entity)], surface, natives, evolutionFactor, tick)
base.points = base.points - cost
end
base.points = base.points + natives.baseIncrement
base.tick = tick
end
function baseUtils.createBase(map, natives, evolutionFactor, chunk, surface, tick)
local x = chunk.x
local y = chunk.y
local distance = euclideanDistancePoints(x, y, 0, 0)
local meanLevel = mFloor(distance / 200)
local distanceIndex = roundToFloor(mMin(1, distance * BASE_DISTANCE_TO_EVO_INDEX), EVOLUTION_INCREMENTS)
local evoIndex = mMax(distanceIndex, roundToFloor(evolutionFactor, EVOLUTION_INCREMENTS))
--local alignments = { BASE_ALIGNMENT_NEUTRAL }
local alignment = BASE_ALIGNMENT_NEUTRAL
local base = {
x = x,
y = y,
tick = tick,
alignment = alignment,
points = 0,
level = gaussianRandomRange(meanLevel, meanLevel * 0.3, meanLevel * 0.50, meanLevel * 1.50)
}
setChunkBase(map, chunk, base)
-- if not buildHive(map, base, surface) then
-- return nil
-- end
natives.bases[#natives.bases+1] = base
return base
end
local function fileEntity(baseAlignment, entity, evolutionTable)
local evoRequirement = mFloor(entity.prototype.build_base_evolution_requirement/EVOLUTION_INCREMENTS) * EVOLUTION_INCREMENTS
local eTable = evolutionTable[baseAlignment]
if not eTable then
eTable = {}
evolutionTable[baseAlignment] = eTable
end
local aTable = eTable[evoRequirement]
if not aTable then
aTable = {}
eTable[evoRequirement] = aTable
end
aTable[#aTable+1] = entity.name
end
local function processUnitClass(biterVariation, biterTier, spitterVariation, spitterTier, wormVariation, wormTier, surface, natives, baseAlignment, baseAlignmentString)
local position = { x = 0, y = 0 }
for v=1,biterVariation do
for t=1,biterTier do
local entity = surface.create_entity({
name= baseAlignmentString .. "-biter-nest-v" .. v .. "-t" .. t .. "-rampant",
position = position
})
fileEntity(baseAlignment, entity, natives.evolutionTableUnitSpawner)
entity.destroy()
end
end
for v=1,spitterVariation do
for t=1,spitterTier do
local entity = surface.create_entity({
name=baseAlignmentString .. "-spitter-nest-v" .. v .. "-t" .. t .. "-rampant",
position = position
})
fileEntity(baseAlignment, entity, natives.evolutionTableUnitSpawner)
entity.destroy()
end
end
for v=1,wormVariation do
for t=1,wormTier do
local entity = surface.create_entity({
name=baseAlignmentString .. "-worm-v" .. v .. "-t" .. t .. "-rampant",
position = position
})
fileEntity(baseAlignment, entity, natives.evolutionTableWorm)
entity.destroy()
end
end
end
function baseUtils.rebuildNativeTables(natives, surface)
natives.evolutionTableUnitSpawner = {}
natives.evolutionTableWorm = {}
natives.evolutionTableAlignment = {}
-- todo fill out alignment evolution levels
natives.evolutionTableAlignment =
processUnitClass(NEUTRAL_NEST_VARIATIONS,
NEUTRAL_NEST_TIERS,
NEUTRAL_NEST_VARIATIONS,
NEUTRAL_NEST_TIERS,
NEUTRAL_WORM_VARIATIONS,
NEUTRAL_WORM_TIERS,
surface,
natives,
BASE_ALIGNMENT_NEUTRAL,
"neutral")
processUnitClass(ACID_NEST_VARIATIONS,
ACID_NEST_TIERS,
ACID_NEST_VARIATIONS,
ACID_NEST_TIERS,
ACID_WORM_VARIATIONS,
ACID_WORM_TIERS,
surface,
natives,
BASE_ALIGNMENT_ACID,
"acid")
processUnitClass(PHYSICAL_NEST_VARIATIONS,
PHYSICAL_NEST_TIERS,
0,
0,
PHYSICAL_WORM_VARIATIONS,
PHYSICAL_WORM_TIERS,
surface,
natives,
BASE_ALIGNMENT_PHYSICAL,
"physical")
processUnitClass(FIRE_NEST_VARIATIONS,
FIRE_NEST_TIERS,
FIRE_NEST_VARIATIONS,
FIRE_NEST_TIERS,
FIRE_WORM_VARIATIONS,
FIRE_WORM_TIERS,
surface,
natives,
BASE_ALIGNMENT_FIRE,
"fire")
processUnitClass(ELECTRIC_NEST_VARIATIONS,
ELECTRIC_NEST_TIERS,
0,
0,
ELECTRIC_WORM_VARIATIONS,
ELECTRIC_WORM_TIERS,
surface,
natives,
BASE_ALIGNMENT_ELECTRIC,
"electric")
processUnitClass(SUICIDE_NEST_VARIATIONS,
SUICIDE_NEST_TIERS,
0,
0,
SUICIDE_WORM_VARIATIONS,
SUICIDE_WORM_TIERS,
surface,
natives,
BASE_ALIGNMENT_SUICIDE,
"suicide")
processUnitClass(NUCLEAR_NEST_VARIATIONS,
NUCLEAR_NEST_TIERS,
0,
0,
NUCLEAR_WORM_VARIATIONS,
NUCLEAR_WORM_TIERS,
surface,
natives,
BASE_ALIGNMENT_NUCLEAR,
"nuclear")
processUnitClass(TROLL_NEST_VARIATIONS,
TROLL_NEST_TIERS,
TROLL_NEST_VARIATIONS,
TROLL_NEST_TIERS,
TROLL_WORM_VARIATIONS,
TROLL_WORM_TIERS,
surface,
natives,
BASE_ALIGNMENT_TROLL,
"troll")
processUnitClass(0,
0,
INFERNO_NEST_VARIATIONS,
INFERNO_NEST_TIERS,
INFERNO_WORM_VARIATIONS,
INFERNO_WORM_TIERS,
surface,
natives,
BASE_ALIGNMENT_INFERNO,
"inferno")
processUnitClass(FAST_NEST_VARIATIONS,
FAST_NEST_TIERS,
FAST_NEST_VARIATIONS,
FAST_NEST_TIERS,
FAST_WORM_VARIATIONS,
FAST_WORM_TIERS,
surface,
natives,
BASE_ALIGNMENT_FAST,
"fast")
-- processUnitClass(DECAYING_NEST_VARIATIONS,
-- DECAYING_NEST_TIERS,
-- DECAYING_NEST_VARIATIONS,
-- DECAYING_NEST_TIERS,
-- DECAYING_WORM_VARIATIONS,
-- DECAYING_WORM_TIERS,
-- surface,
-- natives,
-- BASE_ALIGNMENT_DECAYING,
-- "decaying")
-- processUnitClass(UNDYING_NEST_VARIATIONS,
-- UNDYING_NEST_TIERS,
-- UNDYING_NEST_VARIATIONS,
-- UNDYING_NEST_TIERS,
-- UNDYING_WORM_VARIATIONS,
-- UNDYING_WORM_TIERS,
-- surface,
-- natives,
-- BASE_ALIGNMENT_UNDYING,
-- "undying")
-- processUnitClass(POSION_NEST_VARIATIONS,
-- POSION_NEST_TIERS,
-- POSION_NEST_VARIATIONS,
-- POSION_NEST_TIERS,
-- POSION_WORM_VARIATIONS,
-- POSION_WORM_TIERS,
-- surface,
-- natives,
-- BASE_ALIGNMENT_POSION,
-- "posion")
-- processUnitClass(LASER_NEST_VARIATIONS,
-- LASER_NEST_TIERS,
-- LASER_NEST_VARIATIONS,
-- LASER_NEST_TIERS,
-- LASER_WORM_VARIATIONS,
-- LASER_WORM_TIERS,
-- surface,
-- natives,
-- BASE_ALIGNMENT_LASER,
-- "laser")
-- processUnitClass(WASP_NEST_VARIATIONS,
-- WASP_NEST_TIERS,
-- WASP_NEST_VARIATIONS,
-- WASP_NEST_TIERS,
-- WASP_WORM_VARIATIONS,
-- WASP_WORM_TIERS,
-- surface,
-- natives,
-- BASE_ALIGNMENT_WASP,
-- "wasp")
end
return baseUtils