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

739 lines
22 KiB
Lua
Raw Normal View History

2022-01-15 00:08:58 +02:00
-- Copyright (C) 2022 veden
-- This program is free software: you can redistribute it and/or modify
-- it under the terms of the GNU General Public License as published by
-- the Free Software Foundation, either version 3 of the License, or
-- (at your option) any later version.
-- This program is distributed in the hope that it will be useful,
-- but WITHOUT ANY WARRANTY; without even the implied warranty of
-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-- GNU General Public License for more details.
-- You should have received a copy of the GNU General Public License
-- along with this program. If not, see <https://www.gnu.org/licenses/>.
local Upgrade = {}
-- imports
local Constants = require("libs/Constants")
local ChunkProcessor = require("libs/ChunkProcessor")
local ChunkPropertyUtils = require("libs/ChunkPropertyUtils")
local MapUtils = require("libs/MapUtils")
--
local Universe
-- Constants
local MINIMUM_EXPANSION_DISTANCE = Constants.MINIMUM_EXPANSION_DISTANCE
2021-02-20 07:41:30 +02:00
local DEFINES_COMMAND_GROUP = defines.command.group
local DEFINES_COMMAND_WANDER = defines.command.wander
local DEFINES_COMMAND_BUILD_BASE = defines.command.build_base
local DEFINES_COMMAND_ATTACK_AREA = defines.command.attack_area
local DEFINES_COMMAND_GO_TO_LOCATION = defines.command.go_to_location
local DEFINES_COMMMAD_COMPOUND = defines.command.compound
local DEFINES_COMMAND_FLEE = defines.command.flee
local DEFINES_COMMAND_STOP = defines.command.stop
local DEFINES_COMPOUND_COMMAND_RETURN_LAST = defines.compound_command.return_last
local DEFINES_DISTRACTION_NONE = defines.distraction.none
local DEFINES_DISTRACTION_BY_ENEMY = defines.distraction.by_enemy
local DEFINES_DISTRACTION_BY_ANYTHING = defines.distraction.by_anything
local CHUNK_SIZE = Constants.CHUNK_SIZE
local TRIPLE_CHUNK_SIZE = Constants.TRIPLE_CHUNK_SIZE
local ENEMY_PHEROMONE = Constants.ENEMY_PHEROMONE
local CHUNK_TICK = Constants.CHUNK_TICK
local TICKS_A_MINUTE = Constants.TICKS_A_MINUTE
-- imported functions
local addBaseResourceChunk = ChunkPropertyUtils.addBaseResourceChunk
local getChunkBase = ChunkPropertyUtils.getChunkBase
local getResourceGenerator = ChunkPropertyUtils.getResourceGenerator
local sFind = string.find
local queueGeneratedChunk = MapUtils.queueGeneratedChunk
local processPendingChunks = ChunkProcessor.processPendingChunks
-- module code
function Upgrade.isExcludedSurface(surfaceName)
return
(surfaceName == "aai-signals") or
(surfaceName == "RTStasisRealm") or
(surfaceName == "minime_dummy_dungeon") or
(surfaceName == "minime-preview-character") or
(surfaceName == "pipelayer") or
(surfaceName == "beltlayer") or
sFind(surfaceName, "Factory floor") or
sFind(surfaceName, " Orbit") or
sFind(surfaceName, "clonespace") or
sFind(surfaceName, "BPL_TheLabplayer") or
sFind(surfaceName, "starmap%-") or
sFind(surfaceName, "NiceFill") or
sFind(surfaceName, "Asteroid Belt") or
sFind(surfaceName, "Vault ") or
sFind(surfaceName, "spaceship") or
sFind(surfaceName, "bpsb%-lab%-")
end
2021-02-20 07:41:30 +02:00
local function addCommandSet(queriesAndCommands)
-- preallocating memory to be used in code, making it fast by reducing garbage generated.
queriesAndCommands.neighbors = {
-1,
-1,
-1,
-1,
-1,
-1,
-1,
-1
}
queriesAndCommands.cardinalNeighbors = {
-1,
-1,
-1,
-1
}
2021-12-07 07:32:44 +02:00
2021-02-20 07:41:30 +02:00
queriesAndCommands.chunkOverlapArray = {
-1,
-1,
-1,
-1
}
queriesAndCommands.playerForces = {}
queriesAndCommands.enemyForces = {}
queriesAndCommands.npcForces = {}
queriesAndCommands.nonPlayerForces = {}
-- pb
queriesAndCommands.pbFilteredEntitiesPointQueryLimited = {
position = {0, 0},
2021-02-20 07:41:30 +02:00
radius = 10,
limit = 1,
2021-12-07 07:32:44 +02:00
force = queriesAndCommands.enemyForces,
2021-02-20 07:41:30 +02:00
type = {
"unit-spawner",
"turret"
}
}
-- msec
queriesAndCommands.msecFilteredEntitiesEnemyStructureQuery = {
area={
{0,0},
{0,0}
},
force=queriesAndCommands.enemyForces,
type={
"turret",
"unit-spawner"
}
}
-- oba
queriesAndCommands.obaCreateBuildCloudQuery = {
2021-02-20 07:41:30 +02:00
name = "build-clear-cloud-rampant",
position = {0,0}
2021-02-20 07:41:30 +02:00
}
-- sp
local spbSharedChunkArea = {
{0,0},
{0,0}
}
queriesAndCommands.spbHasPlayerStructuresQuery = {
area=spbSharedChunkArea,
force=queriesAndCommands.nonPlayerForces,
invert=true,
limit=1
}
queriesAndCommands.spbFilteredEntitiesPlayerQueryLowest = {
area=spbSharedChunkArea,
2021-12-07 07:32:44 +02:00
force=queriesAndCommands.playerForces,
2021-02-20 07:41:30 +02:00
collision_mask = "player-layer",
type={
"wall",
"transport-belt"
}
}
queriesAndCommands.spbFilteredEntitiesPlayerQueryLow = {
area=spbSharedChunkArea,
2021-12-07 07:32:44 +02:00
force=queriesAndCommands.playerForces,
2021-02-20 07:41:30 +02:00
collision_mask = "player-layer",
type={
"splitter",
"pump",
"offshore-pump",
"lamp",
"solar-panel",
"programmable-speaker",
"accumulator",
"assembling-machine",
"turret",
"ammo-turret"
}
}
queriesAndCommands.spbFilteredEntitiesPlayerQueryHigh = {
area=spbSharedChunkArea,
2021-12-07 07:32:44 +02:00
force=queriesAndCommands.playerForces,
2021-02-20 07:41:30 +02:00
collision_mask = "player-layer",
type={
"furnace",
"lab",
"roboport",
"beacon",
"radar",
"electric-turret",
"boiler",
"generator",
"fluid-turret",
"mining-drill"
}
}
queriesAndCommands.spbFilteredEntitiesPlayerQueryHighest = {
area=spbSharedChunkArea,
2021-12-07 07:32:44 +02:00
force=queriesAndCommands.playerForces,
2021-02-20 07:41:30 +02:00
collision_mask = "player-layer",
type={
"artillery-turret",
"reactor",
"rocket-silo"
}
}
-- is
local isSharedChunkArea = {
{0,0},
{0,0}
}
queriesAndCommands.isFilteredTilesQuery = {
collision_mask="water-tile",
area=isSharedChunkArea
}
queriesAndCommands.isFilteredEntitiesChunkNeutral = {
area=isSharedChunkArea,
collision_mask = "player-layer",
type={
"tree",
"simple-entity"
}
}
queriesAndCommands.isFilteredEntitiesEnemyStructureQuery = {
area=isSharedChunkArea,
force=queriesAndCommands.enemyForces,
type={
"turret",
"unit-spawner"
}
}
queriesAndCommands.isCountResourcesQuery = {
area=isSharedChunkArea,
type="resource"
}
queriesAndCommands.isFilteredEntitiesUnitQuery = {
area=isSharedChunkArea,
force=queriesAndCommands.enemyForces,
type="unit"
}
-- cps
local cpsSharedChunkArea = {
{0,0},
{0,0}
}
queriesAndCommands.cpsFilteredTilesQuery = {
collision_mask="water-tile",
area=cpsSharedChunkArea
}
queriesAndCommands.cpsFilteredEntitiesChunkNeutral = {
area=cpsSharedChunkArea,
collision_mask = "player-layer",
type={
"tree",
"simple-entity"
}
}
queriesAndCommands.cpsFilteredEnemyAnyFound = {
area=cpsSharedChunkArea,
force=queriesAndCommands.enemyForces,
type={
"turret",
"unit-spawner"
},
limit = 1
}
-- msrc
local msrcSharedChunkArea = {
{0,0},
{0,0}
}
queriesAndCommands.msrcFilteredTilesQuery = {
collision_mask="water-tile",
area=msrcSharedChunkArea
}
queriesAndCommands.msrcFilteredEntitiesChunkNeutral = {
area=msrcSharedChunkArea,
2021-02-20 07:41:30 +02:00
collision_mask = "player-layer",
type={
"tree",
"simple-entity"
}
}
queriesAndCommands.msrcCountResourcesQuery = {
area=msrcSharedChunkArea,
type="resource"
}
2021-02-20 07:41:30 +02:00
-- sp
local spSharedAreaChunk = {
2021-02-20 07:41:30 +02:00
{0,0},
{0,0}
}
queriesAndCommands.spFilteredEntitiesCliffQuery = {
area=spSharedAreaChunk,
2021-02-20 07:41:30 +02:00
type="cliff",
limit = 1
}
queriesAndCommands.spFilteredTilesPathQuery = {
area=spSharedAreaChunk,
2021-02-20 07:41:30 +02:00
collision_mask="water-tile",
limit = 1
}
-- ouc
queriesAndCommands.oucCliffQuery = {
area={
{0,0},
{0,0}
},
2021-02-20 07:41:30 +02:00
type="cliff"
}
-- ppu
queriesAndCommands.ppuUpgradeEntityQuery = {
2021-02-20 07:41:30 +02:00
name = "",
position = {0,0}
2021-02-20 07:41:30 +02:00
}
queriesAndCommands.attackCommand = {
type = DEFINES_COMMAND_ATTACK_AREA,
destination = {0,0},
2021-02-20 07:41:30 +02:00
radius = CHUNK_SIZE * 1.5,
distraction = DEFINES_DISTRACTION_BY_ANYTHING
}
queriesAndCommands.moveCommand = {
type = DEFINES_COMMAND_GO_TO_LOCATION,
destination = {0,0},
2021-02-20 07:41:30 +02:00
pathfind_flags = { cache = true },
distraction = DEFINES_DISTRACTION_BY_ENEMY
}
queriesAndCommands.settleCommand = {
type = DEFINES_COMMAND_BUILD_BASE,
destination = {0,0},
2021-02-20 07:41:30 +02:00
distraction = DEFINES_DISTRACTION_BY_ENEMY,
ignore_planner = true
}
2021-11-29 02:35:31 +02:00
queriesAndCommands.wanderCommand = {
2021-02-20 07:41:30 +02:00
type = DEFINES_COMMAND_WANDER,
wander_in_group = false,
radius = TRIPLE_CHUNK_SIZE*2,
2021-11-29 02:28:52 +02:00
ticks_to_wait = 20 * 60
2021-02-20 07:41:30 +02:00
}
queriesAndCommands.wander2Command = {
type = DEFINES_COMMAND_WANDER,
wander_in_group = true,
radius = TRIPLE_CHUNK_SIZE*2,
ticks_to_wait = 2 * 60
}
2021-02-20 07:41:30 +02:00
queriesAndCommands.stopCommand = {
type = DEFINES_COMMAND_STOP
}
queriesAndCommands.compoundSettleCommand = {
type = DEFINES_COMMMAD_COMPOUND,
structure_type = DEFINES_COMPOUND_COMMAND_RETURN_LAST,
commands = {
queriesAndCommands.wonder2Command,
queriesAndCommands.settleCommand
}
}
queriesAndCommands.retreatCommand = {
type = DEFINES_COMMAND_GROUP,
group = nil,
distraction = DEFINES_DISTRACTION_BY_ANYTHING,
use_group_distraction = true
}
queriesAndCommands.fleeCommand = {
type = DEFINES_COMMAND_FLEE,
from = nil,
distraction = DEFINES_DISTRACTION_NONE
}
queriesAndCommands.compoundRetreatGroupCommand = {
type = DEFINES_COMMMAD_COMPOUND,
structure_type = DEFINES_COMPOUND_COMMAND_RETURN_LAST,
commands = {
queriesAndCommands.stopCommand,
queriesAndCommands.fleeCommand,
queriesAndCommands.retreatCommand
}
}
queriesAndCommands.formGroupCommand = {
type = DEFINES_COMMAND_GROUP,
group = nil,
distraction = DEFINES_DISTRACTION_BY_ANYTHING,
use_group_distraction = false
}
queriesAndCommands.formCommand = {
command = queriesAndCommands.formGroupCommand,
unit_count = 0,
unit_search_distance = TRIPLE_CHUNK_SIZE
}
queriesAndCommands.formRetreatCommand = {
command = queriesAndCommands.compoundRetreatGroupCommand,
unit_count = 1,
unit_search_distance = CHUNK_SIZE
}
end
2021-12-07 07:32:44 +02:00
function Upgrade.excludeSurface()
for mapId,map in pairs(Universe.maps) do
local toBeRemoved = not map.surface.valid or Upgrade.isExcludedSurface(map.surface.name) or Universe.excludedSurfaces[map.surface.name]
if toBeRemoved then
if Universe.mapIterator == mapId then
Universe.mapIterator, Universe.activeMap = next(Universe.maps, Universe.mapIterator)
end
if Universe.processMapAIIterator == mapId then
Universe.processMapAIIterator = nil
end
Universe.maps[mapId] = nil
end
end
end
function Upgrade.setCommandForces(npcForces, enemyForces)
for force in pairs(Universe.playerForces) do
Universe.playerForces[force] = nil
end
for force in pairs(Universe.npcForces) do
Universe.npcForces[force] = nil
end
for force in pairs(Universe.enemyForces) do
Universe.enemyForces[force] = nil
end
for force in pairs(Universe.nonPlayerForces) do
Universe.nonPlayerForces[force] = nil
end
for _,force in pairs(game.forces) do
if not npcForces[force.name] and not enemyForces[force.name] then
Universe.playerForces[#Universe.playerForces+1] = force.name
2021-12-07 07:32:44 +02:00
end
end
for force in pairs(enemyForces) do
Universe.enemyForces[#Universe.enemyForces+1] = force
Universe.nonPlayerForces[#Universe.nonPlayerForces+1] = force
end
for force in pairs(npcForces) do
Universe.npcForces[#Universe.npcForces+1] = force
Universe.nonPlayerForces[#Universe.nonPlayerForces+1] = force
end
2021-02-20 07:41:30 +02:00
end
function Upgrade.addUniverseProperties()
if not global.universePropertyVersion then
for key in pairs(global) do
if key ~= "universe" then
global[key] = nil
end
end
for key in pairs(Universe) do
Universe[key] = nil
end
global.universePropertyVersion = 0
end
if global.universePropertyVersion < 1 then
global.universePropertyVersion = 1
Universe.safeEntities = {}
Universe.aiPointsScaler = settings.global["rampant--aiPointsScaler"].value
Universe.aiPointsPrintGainsToChat = settings.global["rampant--aiPointsPrintGainsToChat"].value
Universe.aiPointsPrintSpendingToChat = settings.global["rampant--aiPointsPrintSpendingToChat"].value
Universe.aiNocturnalMode = settings.global["rampant--permanentNocturnal"].value
Universe.retreatThreshold = 0
Universe.rallyThreshold = 0
Universe.formSquadThreshold = 0
Universe.attackWaveSize = 0
Universe.attackWaveDeviation = 0
Universe.attackWaveUpperBound = 0
Universe.unitRefundAmount = 0
Universe.regroupIndex = 1
Universe.kamikazeThreshold = 0
Universe.attackWaveLowerBound = 1
Universe.settlerCooldown = 0
Universe.settlerWaveDeviation = 0
Universe.settlerWaveSize = 0
Universe.enabledMigration = Universe.expansion and settings.global["rampant--enableMigration"].value
Universe.peacefulAIToggle = settings.global["rampant--peacefulAIToggle"].value
Universe.printAIStateChanges = settings.global["rampant--printAIStateChanges"].value
Universe.debugTemperament = settings.global["rampant--debugTemperament"].value
Universe.eventId = 0
Universe.chunkId = 0
Universe.maps = {}
Universe.chunkIdToChunk = {}
Universe.groupNumberToSquad = {}
Universe.chunkToVictory = {}
Universe.pendingChunks = {}
Universe.processActiveNest = {}
Universe.processActiveNestIterator = nil
Universe.deployVengenceIterator = nil
Universe.pendingUpgradeIterator = nil
Universe.victoryScentIterator = nil
Universe.squadIterator = nil
Universe.processMapAIIterator = nil
Universe.processNestIterator = nil
Universe.vengenceQueue = {}
Universe.activeMap = nil
Universe.mapIterator = nil
Universe.builderCount = 0
Universe.squadCount = 0
Universe.chunkToNests = {}
Universe.processActiveSpawnerIterator = nil
Universe.processActiveRaidSpawnerIterator = nil
Universe.processMigrationIterator = nil
Universe.chunkToDrainedIterator = nil
Universe.chunkToActiveNest = {}
Universe.chunkToActiveRaidNest = {}
Universe.chunkToDrained = {}
Universe.chunkToRetreatIterator = nil
Universe.chunkToRetreats = {}
Universe.chunkToRallyIterator = nil
Universe.chunkToRallys = {}
Universe.chunkToPassScan = {}
Universe.chunkToPassScanIterator = nil
Universe.baseId = 0
Universe.awake = false
Universe.recycleBaseIterator = nil
Universe.maxPoints = 0
Universe.maxOverflowPoints = 0
addCommandSet(Universe)
Universe.bases = {}
Universe.processBaseAIIterator = nil
for _,map in pairs(Universe.maps) do
local processQueue = map.processQueue
for i=1,#processQueue do
local chunk = processQueue[i]
chunk[CHUNK_TICK] = chunk[ENEMY_PHEROMONE]
chunk[ENEMY_PHEROMONE] = 0
end
end
Universe.excludedSurfaces = {}
2021-05-01 19:26:05 +02:00
Universe.pendingUpgrades = {}
Universe.settlePurpleCloud = {}
Universe.settlePurpleCloud.len = 0
end
end
2021-05-01 19:26:05 +02:00
function Upgrade.attempt()
if not global.gameVersion then
global.gameVersion = 1
2019-02-03 08:01:28 +02:00
game.forces.enemy.kill_all_units()
2021-02-20 09:31:36 +02:00
game.map_settings.path_finder.min_steps_to_check_path_find_termination =
Constants.PATH_FINDER_MIN_STEPS_TO_CHECK_PATH
game.map_settings.unit_group.min_group_radius = Constants.UNIT_GROUP_MAX_RADIUS * 0.5
game.map_settings.unit_group.max_group_radius = Constants.UNIT_GROUP_MAX_RADIUS
game.map_settings.unit_group.max_member_speedup_when_behind = Constants.UNIT_GROUP_MAX_SPEED_UP
game.map_settings.unit_group.max_member_slowdown_when_ahead = Constants.UNIT_GROUP_MAX_SLOWDOWN
game.map_settings.unit_group.max_group_slowdown_factor = Constants.UNIT_GROUP_SLOWDOWN_FACTOR
2019-10-20 22:45:43 +02:00
game.map_settings.max_failed_behavior_count = 3
game.map_settings.unit_group.member_disown_distance = 10
game.map_settings.unit_group.tick_tolerance_when_member_arrives = 60
game.forces.enemy.ai_controllable = true
Universe.evolutionLevel = game.forces.enemy.evolution_factor
2021-11-25 19:04:52 +02:00
for _,map in pairs(Universe.maps) do
if (map.surface.valid) then
local entities = map.surface.find_entities_filtered({type="land-mine"})
for i=1,#entities do
local entity = entities[i]
if entity.valid and string.find(entity.name, "entity-proxy-") then
entity.destroy()
end
end
end
end
Universe.random = game.create_random_generator(Constants.ENEMY_SEED)
Upgrade.excludeSurface()
local minDiffuse = game.map_settings.pollution.min_to_diffuse
Universe.pollutionDiffuseMinimum = minDiffuse * 0.75
Universe.expansion = game.map_settings.enemy_expansion.enabled
Universe.expansionMaxDistance = game.map_settings.enemy_expansion.max_expansion_distance * CHUNK_SIZE
Universe.expansionMinTime = game.map_settings.enemy_expansion.min_expansion_cooldown / TICKS_A_MINUTE
Universe.expansionMaxTime = game.map_settings.enemy_expansion.max_expansion_cooldown / TICKS_A_MINUTE
Universe.expansionMinSize = game.map_settings.enemy_expansion.settler_group_min_size
Universe.expansionMaxSize = game.map_settings.enemy_expansion.settler_group_max_size
Universe.expansionLowTargetDistance = (Universe.expansionMaxDistance + MINIMUM_EXPANSION_DISTANCE) * 0.33
Universe.expansionMediumTargetDistance = (Universe.expansionMaxDistance + MINIMUM_EXPANSION_DISTANCE) * 0.50
Universe.expansionHighTargetDistance = (Universe.expansionMaxDistance + MINIMUM_EXPANSION_DISTANCE) * 0.75
Universe.expansionDistanceDeviation = Universe.expansionMediumTargetDistance * 0.33
for _,map in pairs(Universe.maps) do
if (map.surface.valid) then
for _, chunk in pairs(map.processQueue) do
if getResourceGenerator(map, chunk) > 0 then
local base = getChunkBase(map, chunk)
if base then
addBaseResourceChunk(base, chunk)
end
end
end
map.chunkToPlayerGenerator = {}
map.chunkToPermanentDeathGenerator = {}
end
end
2021-11-25 19:04:52 +02:00
end
end
function Upgrade.prepMap(surface)
local surfaceName = surface.name
if Upgrade.isExcludedSurface(surfaceName) then
return
end
game.print("Rampant - Indexing surface:" .. surfaceName .. ", index:" .. tostring(surface.index) .. ", please wait.")
local surfaceIndex = surface.index
if not Universe.maps then
Universe.maps = {}
end
local map = {}
Universe.maps[surfaceIndex] = map
2021-12-08 08:44:22 +02:00
map.activatedMap = false
map.processedChunks = 0
map.processQueue = {}
map.processIndex = 1
map.scanPlayerIndex = 1
map.scanResourceIndex = 1
map.scanEnemyIndex = 1
map.processStaticIndex = 1
map.outgoingScanWave = true
map.outgoingStaticScanWave = true
map.chunkToBase = {}
map.chunkToTurrets = {}
map.chunkToTraps = {}
map.chunkToUtilities = {}
map.chunkToHives = {}
map.chunkToNestIds = {}
map.chunkToHiveIds = {}
map.chunkToTrapIds = {}
map.chunkToTurretIds = {}
map.chunkToUtilityIds = {}
map.drainPylons = {}
map.chunkToPlayerBase = {}
map.chunkToResource = {}
map.chunkToPlayerCount = {}
map.playerToChunk = {}
map.chunkToSquad = {}
map.chunkToPassable = {}
map.chunkToPathRating = {}
map.chunkToDeathGenerator = {}
map.chunkToPermanentDeathGenerator = {}
map.chunkToPlayerGenerator = {}
map.chunkScanCounts = {}
map.emptySquadsOnChunk = {}
map.surface = surface
map.bases = {}
map.squads = nil
map.pendingAttack = nil
map.building = nil
-- queue all current chunks that wont be generated during play
local tick = game.tick
for chunk in surface.get_chunks() do
if surface.is_chunk_generated(chunk) then
queueGeneratedChunk({
surface = surface,
tick = tick,
area = {
left_top = {
x = chunk.x * 32,
y = chunk.y * 32
}
}
}
)
end
end
processPendingChunks(tick, true)
end
function Upgrade.init(universe)
Universe = universe
end
return Upgrade