-- imports local chunkPropertyUtils = require("libs/ChunkPropertyUtils") local unitUtils = require("libs/UnitUtils") local baseUtils = require("libs/BaseUtils") local mapUtils = require("libs/MapUtils") local mathUtils = require("libs/MathUtils") local unitGroupUtils = require("libs/UnitGroupUtils") local chunkProcessor = require("libs/ChunkProcessor") local mapProcessor = require("libs/MapProcessor") local constants = require("libs/Constants") local pheromoneUtils = require("libs/PheromoneUtils") local squadDefense = require("libs/SquadDefense") local squadAttack = require("libs/SquadAttack") local aiAttackWave = require("libs/AIAttackWave") local aiPlanning = require("libs/AIPlanning") local tests = require("tests") local chunkUtils = require("libs/ChunkUtils") local upgrade = require("Upgrade") local aiPredicates = require("libs/AIPredicates") local stringUtils = require("libs/StringUtils") -- constants local COMMAND_TIMEOUT = constants.COMMAND_TIMEOUT local AI_SQUAD_COST = constants.AI_SQUAD_COST local AI_SETTLER_COST = constants.AI_SETTLER_COST local RECOVER_NEST_COST = constants.RECOVER_NEST_COST local RECOVER_WORM_COST = constants.RECOVER_WORM_COST local RETREAT_GRAB_RADIUS = constants.RETREAT_GRAB_RADIUS local RETREAT_SPAWNER_GRAB_RADIUS = constants.RETREAT_SPAWNER_GRAB_RADIUS local DEFINES_WIRE_TYPE_RED = defines.wire_type.red local DEFINES_WIRE_TYPE_GREEN = defines.wire_type.green local ENERGY_THIEF_CONVERSION_TABLE = constants.ENERGY_THIEF_CONVERSION_TABLE local ENERGY_THIEF_LOOKUP = constants.ENERGY_THIEF_LOOKUP -- imported functions local distortPosition = mathUtils.distortPosition local prepMap = upgrade.prepMap local processMapAIs = aiPlanning.processMapAIs local registerEnemyBaseStructure = chunkUtils.registerEnemyBaseStructure local queueGeneratedChunk = mapUtils.queueGeneratedChunk local isRampantSetting = stringUtils.isRampantSetting local processPendingUpgrades = chunkProcessor.processPendingUpgrades local canMigrate = aiPredicates.canMigrate local convertTypeToDrainCrystal = unitUtils.convertTypeToDrainCrystal local squadDispatch = squadAttack.squadDispatch local cleanUpMapTables = mapProcessor.cleanUpMapTables local positionToChunkXY = mapUtils.positionToChunkXY local processVengence = mapProcessor.processVengence local processAttackWaves = mapProcessor.processAttackWaves local processStaticMap = mapProcessor.processStaticMap local disperseVictoryScent = pheromoneUtils.disperseVictoryScent local getChunkByPosition = mapUtils.getChunkByPosition local entityForPassScan = chunkUtils.entityForPassScan local processPendingChunks = chunkProcessor.processPendingChunks local processScanChunks = chunkProcessor.processScanChunks local processMap = mapProcessor.processMap local processPlayers = mapProcessor.processPlayers local scanEnemyMap = mapProcessor.scanEnemyMap local scanPlayerMap = mapProcessor.scanPlayerMap local scanResourceMap = mapProcessor.scanResourceMap local processNests = mapProcessor.processNests local rallyUnits = aiAttackWave.rallyUnits local recycleBases = baseUtils.recycleBases local deathScent = pheromoneUtils.deathScent local victoryScent = pheromoneUtils.victoryScent local createSquad = unitGroupUtils.createSquad local createBase = baseUtils.createBase local findNearbyBase = baseUtils.findNearbyBase local processActiveNests = mapProcessor.processActiveNests local getDeathGenerator = chunkPropertyUtils.getDeathGenerator local retreatUnits = squadDefense.retreatUnits local accountPlayerEntity = chunkUtils.accountPlayerEntity local unregisterEnemyBaseStructure = chunkUtils.unregisterEnemyBaseStructure local makeImmortalEntity = chunkUtils.makeImmortalEntity local registerResource = chunkUtils.registerResource local unregisterResource = chunkUtils.unregisterResource local cleanSquads = squadAttack.cleanSquads local upgradeEntity = baseUtils.upgradeEntity local rebuildNativeTables = baseUtils.rebuildNativeTables local tRemove = table.remove local sFind = string.find -- local references to global local universe -- manages the chunks that make up the game universe -- hook functions local function onIonCannonFired(event) --[[ event.force, event.surface, event.player_index, event.position, event.radius --]] local map = universe.maps[event.surface.index] if not map then return end map.ionCannonBlasts = map.ionCannonBlasts + 1 map.points = map.points + 4000 if universe.aiPointsPrintGainsToChat then game.print(map.surface.name .. ": Points: +" .. 4000 .. ". [Ion Cannon] Total: " .. string.format("%.2f", map.points)) end local chunk = getChunkByPosition(map, event.position) if (chunk ~= -1) then rallyUnits(chunk, map, event.tick) end end local function onAbandonedRuins(event) local entity = event.entity if entity.valid and (entity.force.name ~= "enemy") then local map = universe.maps[entity.surface.index] if not map then return end accountPlayerEntity(entity, map, true, false) end end local function hookEvents() if settings.startup["ion-cannon-radius"] ~= nil then script.on_event(remote.call("orbital_ion_cannon", "on_ion_cannon_fired"), onIonCannonFired) end if settings.global["AbandonedRuins-set"] ~= nil then script.on_event(remote.call("AbandonedRuins", "get_on_entity_force_changed_event"), onAbandonedRuins) end end local function onLoad() universe = global.universe hookEvents() end local function onChunkGenerated(event) -- queue generated chunk for delayed processing, queuing is required because -- some mods (RSO) mess with chunk as they are generated, which messes up the -- scoring. queueGeneratedChunk(universe, event) end local function onModSettingsChange(event) if not isRampantSetting(event.setting) then return end upgrade.compareTable(universe.safeEntities, "curved-rail", settings.global["rampant--safeBuildings-curvedRail"].value) upgrade.compareTable(universe.safeEntities, "straight-rail", settings.global["rampant--safeBuildings-straightRail"].value) upgrade.compareTable(universe.safeEntities, "rail-signal", settings.global["rampant--safeBuildings-railSignals"].value) upgrade.compareTable(universe.safeEntities, "rail-chain-signal", settings.global["rampant--safeBuildings-railChainSignals"].value) upgrade.compareTable(universe.safeEntities, "train-stop", settings.global["rampant--safeBuildings-trainStops"].value) upgrade.compareTable(universe.safeEntities, "lamp", settings.global["rampant--safeBuildings-lamps"].value) upgrade.compareTable(universe, "MAX_BASE_MUTATIONS", settings.global["rampant--max-base-mutations"].value) local changed, newValue = upgrade.compareTable(universe.safeEntities, "big-electric-pole", settings.global["rampant--safeBuildings-bigElectricPole"].value) if changed then universe.safeEntities["big-electric-pole"] = newValue universe.safeEntities["big-electric-pole-2"] = newValue universe.safeEntities["big-electric-pole-3"] = newValue universe.safeEntities["big-electric-pole-4"] = newValue universe.safeEntities["lighted-big-electric-pole-4"] = newValue universe.safeEntities["lighted-big-electric-pole-3"] = newValue universe.safeEntities["lighted-big-electric-pole-2"] = newValue universe.safeEntities["lighted-big-electric-pole"] = newValue end upgrade.compareTable(universe, "temperamentRateModifier", settings.global["rampant--temperamentRateModifier"].value) upgrade.compareTable(universe, "baseDistanceModifier", settings.global["rampant--baseDistanceModifier"].value) upgrade.compareTable(universe, "printBaseAdaptation", settings.global["rampant--printBaseAdaptation"].value) upgrade.compareTable(universe, "adaptationModifier", settings.global["rampant--adaptationModifier"].value) upgrade.compareTable(universe, "raidAIToggle", settings.global["rampant--raidAIToggle"].value) upgrade.compareTable(universe, "siegeAIToggle", settings.global["rampant--siegeAIToggle"].value) upgrade.compareTable(universe, "attackPlayerThreshold", settings.global["rampant--attackPlayerThreshold"].value) upgrade.compareTable(universe, "attackUsePlayer", settings.global["rampant--attackWaveGenerationUsePlayerProximity"].value) upgrade.compareTable(universe, "attackWaveMaxSize", settings.global["rampant--attackWaveMaxSize"].value) upgrade.compareTable(universe, "aiNocturnalMode", settings.global["rampant--permanentNocturnal"].value) upgrade.compareTable(universe, "aiPointsScaler", settings.global["rampant--aiPointsScaler"].value) universe.aiPointsPrintGainsToChat = settings.global["rampant--aiPointsPrintGainsToChat"].value universe.aiPointsPrintSpendingToChat = settings.global["rampant--aiPointsPrintSpendingToChat"].value universe.printBaseUpgrades = settings.global["rampant--printBaseUpgrades"].value 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 upgrade.compareTable(universe, "AI_MAX_SQUAD_COUNT", settings.global["rampant--maxNumberOfSquads"].value) upgrade.compareTable(universe, "AI_MAX_BUILDER_COUNT", settings.global["rampant--maxNumberOfBuilders"].value) return true end local function onConfigChanged() local version = upgrade.attempt(universe) if version then if not universe then universe = global.universe end end onModSettingsChange({setting="rampant--"}) upgrade.compareTable(universe, "ENEMY_SEED", settings.startup["rampant--enemySeed"].value) upgrade.compareTable(universe, "ENEMY_VARIATIONS", settings.startup["rampant--newEnemyVariations"].value) upgrade.compareTable(universe, "NEW_ENEMIES", settings.startup["rampant--newEnemies"].value) if universe.NEW_ENEMIES then rebuildNativeTables(universe, universe.random) else universe.buildingHiveTypeLookup = {} universe.buildingHiveTypeLookup["biter-spawner"] = "biter-spawner" universe.buildingHiveTypeLookup["spitter-spawner"] = "spitter-spawner" universe.buildingHiveTypeLookup["small-worm-turret"] = "turret" universe.buildingHiveTypeLookup["medium-worm-turret"] = "turret" universe.buildingHiveTypeLookup["big-worm-turret"] = "turret" universe.buildingHiveTypeLookup["behemoth-worm-turret"] = "turret" end -- not a completed implementation needs if checks to use all forces -- both in the data stage, commands, and if then logic local enemyForces = { ["enemy"]=true } local npcForces = { ["neutral"]=true } if game.active_mods["AbandonedRuins"] then npcForces["AbandonedRuins:enemy"] = true end universe.setCommandForces(npcForces, enemyForces) for _,surface in pairs(game.surfaces) do if not universe.maps then universe.maps = {} end if not universe.maps[surface.index] then prepMap(universe, surface) end end end local function onBuild(event) local entity = event.created_entity or event.entity if entity.valid then local map = universe.maps[entity.surface.index] if not map then return end if (entity.type == "resource") and (entity.force.name == "neutral") then registerResource(entity, map) else accountPlayerEntity(entity, map, true, false) if universe.safeEntities[entity.type] or universe.safeEntities[entity.name] then entity.destructible = false end end end end local function onMine(event) local entity = event.entity if entity.valid then local map = universe.maps[entity.surface.index] if not map then return end accountPlayerEntity(entity, map, false, false) end end local function onDeath(event) local entity = event.entity if not entity.valid then return end local surface = entity.surface local map = universe.maps[surface.index] if not map then return end if (entity.force.name == "neutral") then if (entity.name == "cliff") then entityForPassScan(map, entity) end return end local entityPosition = entity.position local chunk = getChunkByPosition(map, entityPosition) local cause = event.cause local tick = event.tick local entityType = entity.type if (entity.force.name == "enemy") then local artilleryBlast = (cause and ((cause.type == "artillery-wagon") or (cause.type == "artillery-turret"))) if artilleryBlast then map.artilleryBlasts = map.artilleryBlasts + 1 end if (entityType == "unit") then if (chunk ~= -1) and event.force and (event.force.name ~= "enemy") then local group = entity.unit_group if group then local damageType = event.damage_type local squad = universe.groupNumberToSquad[group.group_number] if damageType and squad then local base = squad.base if base then local damageTypeName = damageType.name base.damagedBy[damageTypeName] = (base.damagedBy[damageTypeName] or 0) + 0.01 base.deathEvents = base.deathEvents + 1 end end end -- drop death pheromone where unit died deathScent(map, chunk) if (-getDeathGenerator(map, chunk) < -universe.retreatThreshold) and cause and cause.valid then retreatUnits(chunk, cause, map, tick, (artilleryBlast and RETREAT_SPAWNER_GRAB_RADIUS) or RETREAT_GRAB_RADIUS) end map.lostEnemyUnits = map.lostEnemyUnits + 1 if (universe.random() < universe.rallyThreshold) and not surface.peaceful_mode then rallyUnits(chunk, map, tick) end end elseif map.drainPylons[entity.unit_number] then local entityUnitNumber = entity.unit_number local pair = map.drainPylons[entityUnitNumber] if pair then local target = pair[1] local pole = pair[2] if target.unit_number == entityUnitNumber then map.drainPylons[entityUnitNumber] = nil if pole.valid then map.drainPylons[pole.unit_number] = nil pole.die() end elseif (pole.unit_number == entityUnitNumber) then map.drainPylons[entityUnitNumber] = nil if target.valid then map.drainPylons[target.unit_number] = nil target.destroy() end end end else if event.force and (event.force.name ~= "enemy") then local rally = false if (entityType == "unit-spawner") then map.points = map.points + RECOVER_NEST_COST if universe.aiPointsPrintGainsToChat then game.print(map.surface.name .. ": Points: +" .. RECOVER_NEST_COST .. ". [Nest Lost] Total: " .. string.format("%.2f", map.points)) end rally = true elseif (entityType == "turret") then map.points = map.points + RECOVER_WORM_COST if universe.aiPointsPrintGainsToChat then game.print(map.surface.name .. ": Points: +" .. RECOVER_WORM_COST .. ". [Worm Lost] Total: " .. string.format("%.2f", map.points)) end rally = true end if (chunk ~= -1) and rally then rallyUnits(chunk, map, tick) if cause and cause.valid then retreatUnits(chunk, cause, map, tick, RETREAT_SPAWNER_GRAB_RADIUS) end end end if universe.buildingHiveTypeLookup[entity.name] or (entityType == "unit-spawner") or (entityType == "turret") then unregisterEnemyBaseStructure(map, entity, event.damage_type) end end else local creditNatives = false if (event.force ~= nil) and (event.force.name == "enemy") then creditNatives = true local drained if (chunk ~= -1) then victoryScent(map, chunk, entityType) drained = (entityType == "electric-turret") and map.chunkToDrained[chunk.id] end if cause or (drained and (drained - tick) > 0) then if ((cause and ENERGY_THIEF_LOOKUP[cause.name]) or (not cause)) then local conversion = ENERGY_THIEF_CONVERSION_TABLE[entityType] if conversion then local newEntity = surface.create_entity({ position=entity.position, name=convertTypeToDrainCrystal(entity.force.evolution_factor, conversion), direction=entity.direction }) if (conversion == "pole") then local targetEntity = surface.create_entity({ position=entity.position, name="pylon-target-rampant", direction=entity.direction }) targetEntity.backer_name = "" local pair = {targetEntity, newEntity} map.drainPylons[targetEntity.unit_number] = pair map.drainPylons[newEntity.unit_number] = pair local wires = entity.neighbours if wires then for _,v in pairs(wires.copper) do if (v.valid) then newEntity.connect_neighbour(v); end end for _,v in pairs(wires.red) do if (v.valid) then newEntity.connect_neighbour({ wire = DEFINES_WIRE_TYPE_RED, target_entity = v }); end end for _,v in pairs(wires.green) do if (v.valid) then newEntity.connect_neighbour({ wire = DEFINES_WIRE_TYPE_GREEN, target_entity = v }); end end end elseif newEntity.backer_name then newEntity.backer_name = "" end end end end end if creditNatives and (universe.safeEntities[entityType] or universe.safeEntities[entity.name]) then makeImmortalEntity(surface, entity) else accountPlayerEntity(entity, map, false, creditNatives) end end end local function onEnemyBaseBuild(event) local entity = event.entity if entity.valid then local map = universe.maps[entity.surface.index] if not map then return end local chunk = getChunkByPosition(map, entity.position) if (chunk ~= -1) then local base if universe.NEW_ENEMIES then base = findNearbyBase(map, chunk) if not base then base = createBase(map, chunk, event.tick) end registerEnemyBaseStructure(map, entity, event.tick, base) upgradeEntity(entity, base, map, nil, true, true) end else local x,y = positionToChunkXY(entity.position) onChunkGenerated({ surface = entity.surface, tick = event.tick, area = { left_top = { x = x, y = y } } }) end end end local function onSurfaceTileChange(event) local surfaceIndex = event.surface_index or (event.robot and event.robot.surface and event.robot.surface.index) local map = universe.maps[surfaceIndex] if not map then return end local surface = map.surface local chunks = {} local tiles = event.tiles if event.tile then if ((event.tile.name == "landfill") or sFind(event.tile.name, "water")) then for i=1,#tiles do local position = tiles[i].position local chunk = getChunkByPosition(map, position) if (chunk ~= -1) then map.chunkToPassScan[chunk.id] = chunk else local x,y = positionToChunkXY(position) local addMe = true for ci=1,#chunks do local c = chunks[ci] if (c.x == x) and (c.y == y) then addMe = false break end end if addMe then local chunkXY = {x=x,y=y} chunks[#chunks+1] = chunkXY onChunkGenerated({area = { left_top = chunkXY }, tick = event.tick, surface = surface}) end end end end else for i=1,#tiles do local tile = tiles[i] if (tile.name == "landfill") or sFind(tile.name, "water") then local position = tile.position local chunk = getChunkByPosition(map, position) if (chunk ~= -1) then map.chunkToPassScan[chunk.id] = chunk else local x,y = positionToChunkXY(position) local addMe = true for ci=1,#chunks do local c = chunks[ci] if (c.x == x) and (c.y == y) then addMe = false break end end if addMe then local chunkXY = {x=x,y=y} chunks[#chunks+1] = chunkXY onChunkGenerated({area = { left_top = chunkXY }, tick = event.tick, surface = surface}) end end end end end end local function onResourceDepleted(event) local entity = event.entity if entity.valid then local map = universe.maps[entity.surface.index] if not map then return end unregisterResource(entity, map) end end local function onRobotCliff(event) local entity = event.robot if entity.valid then local map = universe.maps[entity.surface.index] if not map then return end if (event.item.name == "cliff-explosives") then entityForPassScan(map, event.cliff) end end end local function onUsedCapsule(event) local surface = game.players[event.player_index].surface local map = universe.maps[surface.index] if not map then return end if (event.item.name == "cliff-explosives") then local position2Top = universe.position2Top local position2Bottom = universe.position2Bottom position2Top.x = event.position.x-0.75 position2Top.y = event.position.y-0.75 position2Bottom.x = event.position.x+0.75 position2Bottom.y = event.position.y+0.75 local cliffs = surface.find_entities_filtered(universe.cliffQuery) for i=1,#cliffs do entityForPassScan(map, cliffs[i]) end end end local function onRocketLaunch(event) local entity = event.rocket_silo or event.rocket if entity.valid then local map = universe.maps[entity.surface.index] if not map then return end map.rocketLaunched = map.rocketLaunched + 1 map.points = map.points + 5000 if universe.aiPointsPrintGainsToChat then game.print(map.surface.name .. ": Points: +" .. 5000 .. ". [Rocket Launch] Total: " .. string.format("%.2f", map.points)) end end end local function onTriggerEntityCreated(event) if (event.effect_id == "rampant-drain-trigger") then local entity = event.target_entity if (entity.valid) then local map = universe.maps[event.surface_index] if not map then return end local chunk = getChunkByPosition(map, entity.position) if (chunk ~= -1) then map.chunkToDrained[chunk.id] = event.tick + 60 end end end end local function onInit() global.universe = {} universe = global.universe hookEvents() onConfigChanged() end local function onEntitySpawned(event) local entity = event.mine if universe.NEW_ENEMIES and entity.valid then local map = universe.maps[entity.surface.index] if not map then return end if universe.buildingHiveTypeLookup[entity.name] then local disPos = distortPosition(universe.random, entity.position, 8) local chunk = getChunkByPosition(map, disPos) if (chunk ~= -1) then local base = findNearbyBase(map, chunk) if not base then base = createBase(map, chunk, event.tick) end registerEnemyBaseStructure(map, entity, event.tick, base) upgradeEntity(entity, base, map, disPos, true, true) else local x,y = positionToChunkXY(entity.position) onChunkGenerated({ surface = entity.surface, tick = event.tick, area = { left_top = { x = x, y = y } } }) entity.destroy() end end end end local function onUnitGroupCreated(event) local group = event.group if (group.force.name == "enemy") then local surface = group.surface local squad if not group.is_script_driven then local map = universe.maps[surface.index] if not map then return end if not universe.aiNocturnalMode then local settler = universe.random() < 0.25 and canMigrate(map) and (universe.builderCount < universe.AI_MAX_BUILDER_COUNT) if not settler and universe.squadCount > universe.AI_MAX_SQUAD_COUNT then group.destroy() map.points = map.points + AI_SQUAD_COST if universe.aiPointsPrintGainsToChat then game.print(map.surface.name .. ": Points: +" .. AI_SQUAD_COST .. ". [Squad Refund] Total: " .. string.format("%.2f", map.points)) end return end squad = createSquad(nil, map, group, settler) universe.groupNumberToSquad[group.group_number] = squad if universe.NEW_ENEMIES then local chunk = getChunkByPosition(map, group.position) if (chunk ~= -1) then squad.base = findNearbyBase(map, chunk) end end if settler then universe.builderCount = universe.builderCount + 1 else universe.squadCount = universe.squadCount + 1 end else if not (surface.darkness > 0.65) then group.destroy() return end local settler = universe.random() < 0.25 and canMigrate(map) and (universe.builderCount < universe.AI_MAX_BUILDER_COUNT) if not settler and universe.squadCount > universe.AI_MAX_SQUAD_COUNT then group.destroy() map.points = map.points + AI_SQUAD_COST if universe.aiPointsPrintGainsToChat then game.print(map.surface.name .. ": Points: +" .. AI_SQUAD_COST .. ". [Squad Refund] Total: " .. string.format("%.2f", map.points)) end return end squad = createSquad(nil, map, group, settler) universe.groupNumberToSquad[group.group_number] = squad if universe.NEW_ENEMIES then local chunk = getChunkByPosition(group.position) if (chunk ~= -1) then squad.base = findNearbyBase(map, chunk) end end if settler then universe.builderCount = universe.builderCount + 1 else universe.squadCount = universe.squadCount + 1 end end end end end local function onGroupFinishedGathering(event) local group = event.group if group.valid and (group.force.name == "enemy") then local map = universe.maps[group.surface.index] if not map then return end local squad = universe.groupNumberToSquad[group.group_number] if squad then if squad.settler then if (universe.builderCount < universe.AI_MAX_BUILDER_COUNT) then squadDispatch(map, squad, event.tick) else group.destroy() map.points = map.points + AI_SETTLER_COST if universe.aiPointsPrintGainsToChat then game.print(map.surface.name .. ": Points: +" .. AI_SETTLER_COST .. ". [Settler Refund] Total: " .. string.format("%.2f", map.points)) end end else if (universe.squadCount < universe.AI_MAX_SQUAD_COUNT) then squadDispatch(map, squad, event.tick) else group.destroy() map.points = map.points + AI_SQUAD_COST if universe.aiPointsPrintGainsToChat then game.print(map.surface.name .. ": Points: +" .. AI_SQUAD_COST .. ". [Squad Refund] Total: " .. string.format("%.2f", map.points)) end end end else local settler = universe.random() < 0.25 and canMigrate(map) and (universe.builderCount < universe.AI_MAX_BUILDER_COUNT) if not settler and universe.squadCount > universe.AI_MAX_SQUAD_COUNT then group.destroy() map.points = map.points + AI_SQUAD_COST if universe.aiPointsPrintGainsToChat then game.print(map.surface.name .. ": Points: +" .. AI_SQUAD_COST .. ". [Squad Refund] Total: " .. string.format("%.2f", map.points)) end return end squad = createSquad(nil, map, group, settler) universe.groupNumberToSquad[group.group_number] = squad if settler then universe.builderCount = universe.builderCount + 1 else universe.squadCount = universe.squadCount + 1 end squadDispatch(map, squad, event.tick) end end end local function onForceCreated(event) universe.playerForces[#universe.playerForces+1] = event.force.name end local function onForceMerged(event) for i=#universe.playerForces,1,-1 do if (universe.playerForces[i] == event.source_name) then tRemove(universe.playerForces, i) break end end end local function onSurfaceCreated(event) prepMap(universe, game.surfaces[event.surface_index]) end local function onSurfaceDeleted(event) local surfaceIndex = event.surface_index if (universe.mapIterator == surfaceIndex) then universe.mapIterator, universe.activeMap = next(universe.maps, universe.mapIterator) end if (universe.processMapAIIterator == surfaceIndex) then universe.processMapAIIterator = nil end universe.maps[surfaceIndex] = nil end local function onBuilderArrived(event) local builder = event.group if not (builder and builder.valid) then builder = event.unit if not (builder and builder.valid and builder.force.name == "enemy") then return end elseif (builder.force.name ~= "enemy") then return end local targetPosition = universe.position targetPosition.x = builder.position.x targetPosition.y = builder.position.y local map = universe.maps[builder.surface.index] if not map then return end local squad = universe.groupNumberToSquad[builder.group_number] squad.commandTick = event.tick + COMMAND_TIMEOUT * 10 if universe.aiPointsPrintSpendingToChat then game.print("Settled: [gps=" .. targetPosition.x .. "," .. targetPosition.y .."]") end map.surface.create_entity(universe.createBuildCloudQuery) end -- hooks script.on_event(defines.events.on_tick, function () local gameRef = game local tick = gameRef.tick local pick = tick % 8 -- local profiler = game.create_profiler() local map = universe.activeMap if (not map) or (universe.processedChunks > (#map.processQueue * 0.05)) then universe.mapIterator, map = next(universe.maps, universe.mapIterator) if not map then universe.mapIterator, map = next(universe.maps, universe.mapIterator) end universe.processedChunks = 0 universe.activeMap = map end if (pick == 0) then processPendingChunks(universe, tick) processMapAIs(universe, gameRef.forces.enemy.evolution_factor, tick) if universe.NEW_ENEMIES then recycleBases(map) end cleanUpMapTables(map, tick) elseif (pick == 1) then processPlayers(gameRef.connected_players, universe, tick) elseif (pick == 2) then processMap(map, tick) elseif (pick == 3) then processStaticMap(map) disperseVictoryScent(universe) processVengence(universe) elseif (pick == 4) then scanResourceMap(map, tick) scanEnemyMap(map, tick) elseif (pick == 5) then processAttackWaves(map) scanEnemyMap(map, tick) elseif (pick == 6) then scanPlayerMap(map, tick) processNests(map, tick) elseif (pick == 7) then processPendingChunks(universe, tick) processScanChunks(map) end processActiveNests(universe, tick) processPendingUpgrades(universe, tick) processPendingUpgrades(universe, tick) cleanSquads(universe, tick) -- game.print({"", "--dispatch4 ", profiler, ", ", pick, ", ", game.tick, " ", universe.random()}) end) script.on_event(defines.events.on_surface_deleted, onSurfaceDeleted) script.on_event(defines.events.on_surface_cleared, onSurfaceCreated) script.on_event(defines.events.on_surface_created, onSurfaceCreated) script.on_init(onInit) script.on_load(onLoad) script.on_event(defines.events.on_runtime_mod_setting_changed, onModSettingsChange) script.on_configuration_changed(onConfigChanged) script.on_event(defines.events.on_resource_depleted, onResourceDepleted) script.on_event({defines.events.on_player_built_tile, defines.events.on_robot_built_tile, defines.events.script_raised_set_tiles}, onSurfaceTileChange) script.on_event(defines.events.on_player_used_capsule, onUsedCapsule) script.on_event(defines.events.on_script_trigger_effect, onTriggerEntityCreated) script.on_event(defines.events.on_pre_robot_exploded_cliff, onRobotCliff) script.on_event(defines.events.on_biter_base_built, onEnemyBaseBuild) script.on_event({defines.events.on_player_mined_entity, defines.events.on_robot_mined_entity}, onMine) script.on_event({defines.events.on_built_entity, defines.events.on_robot_built_entity, defines.events.script_raised_built, defines.events.script_raised_revive}, onBuild) script.on_event(defines.events.on_land_mine_armed, onEntitySpawned) script.on_event(defines.events.on_rocket_launched, onRocketLaunch) script.on_event({defines.events.on_entity_died, defines.events.script_raised_destroy}, onDeath) script.on_event(defines.events.on_chunk_generated, onChunkGenerated) script.on_event(defines.events.on_unit_group_created, onUnitGroupCreated) script.on_event(defines.events.on_force_created, onForceCreated) script.on_event(defines.events.on_forces_merged, onForceMerged) script.on_event(defines.events.on_unit_group_finished_gathering, onGroupFinishedGathering) script.on_event(defines.events.on_build_base_arrived, onBuilderArrived) -- testing remote.add_interface("rampantTests", { pheromoneLevels = tests.pheromoneLevels, activeSquads = tests.activeSquads, entitiesOnPlayerChunk = tests.entitiesOnPlayerChunk, findNearestPlayerEnemy = tests.findNearestPlayerEnemy, morePoints = tests.morePoints, aiStats = tests.aiStats, dumpEnvironment = tests.dumpEnvironment, fillableDirtTest = tests.fillableDirtTest, tunnelTest = tests.tunnelTest, dumpNatives = tests.dumpatives, createEnemy = tests.createEnemy, attackOrigin = tests.attackOrigin, cheatMode = tests.cheatMode, gaussianRandomTest = tests.gaussianRandomTest, reveal = tests.reveal, showMovementGrid = tests.showMovementGrid, showBaseGrid = tests.showBaseGrid, baseStats = tests.baseStats, mergeBases = tests.mergeBases, clearBases = tests.clearBases, getOffsetChunk = tests.getOffsetChunk, registeredNest = tests.registeredNest, colorResourcePoints = tests.colorResourcePoints, entityStats = tests.entityStats, stepAdvanceTendrils = tests.stepAdvanceTendrils, unitGroupBuild = tests.unitGroupBuild, exportAiState = tests.exportAiState(nil), createEnergyTest = tests.createEnergyTest, killActiveSquads = tests.killActiveSquads, scanChunkPaths = tests.scanChunkPaths, scanEnemy = tests.scanEnemy, getEnemyStructureCount = tests.getEnemyStructureCount } ) local function rampantSetAIState(event) local surfaceIndex = game.players[event.player_index].surface.index local map = universe.maps[surfaceIndex] if not map then return end game.print(map.surface.name .. " is in " .. constants.stateEnglish[map.state]) if event.parameter then local target = tonumber(event.parameter) if (target == nil) then game.print("invalid param") return end if (target ~= constants.AI_STATE_PEACEFUL and target ~= constants.AI_STATE_AGGRESSIVE and target ~= constants.AI_STATE_RAIDING and target ~= constants.AI_STATE_MIGRATING and target ~= constants.AI_STATE_SIEGE and target ~= constants.AI_STATE_ONSLAUGHT) then game.print(target .. " is not a valid state") return else map.state = target game.print(map.surface.name .. " is now in " .. constants.stateEnglish[map.state]) end end end commands.add_command('rampantSetAIState', "", rampantSetAIState)