2016-08-05 06:47:51 +02:00
|
|
|
local mapProcessor = {}
|
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- imports
|
|
|
|
|
2017-06-11 02:59:06 +02:00
|
|
|
local unitGroupUtils = require("UnitGroupUtils")
|
2016-08-21 23:48:55 +02:00
|
|
|
local pheromoneUtils = require("PheromoneUtils")
|
2017-06-13 05:16:43 +02:00
|
|
|
local aiAttackWave = require("AIAttackWave")
|
2017-05-19 09:47:24 +02:00
|
|
|
local aiPredicates = require("AIPredicates")
|
2016-08-29 02:05:28 +02:00
|
|
|
local constants = require("Constants")
|
2016-10-15 02:00:18 +02:00
|
|
|
local mapUtils = require("MapUtils")
|
2017-05-19 09:47:24 +02:00
|
|
|
local playerUtils = require("PlayerUtils")
|
2017-11-21 09:27:03 +02:00
|
|
|
local chunkUtils = require("ChunkUtils")
|
2017-06-16 03:30:26 +02:00
|
|
|
local mathUtils = require("MathUtils")
|
|
|
|
|
2016-09-14 13:16:33 +02:00
|
|
|
-- constants
|
|
|
|
|
|
|
|
local PROCESS_QUEUE_SIZE = constants.PROCESS_QUEUE_SIZE
|
2016-11-04 09:26:19 +02:00
|
|
|
|
2016-09-14 13:16:33 +02:00
|
|
|
local SCAN_QUEUE_SIZE = constants.SCAN_QUEUE_SIZE
|
|
|
|
|
|
|
|
local CHUNK_SIZE = constants.CHUNK_SIZE
|
2017-06-01 03:46:53 +02:00
|
|
|
local DOUBLE_CHUNK_SIZE = constants.DOUBLE_CHUNK_SIZE
|
|
|
|
local TRIPLE_CHUNK_SIZE = constants.TRIPLE_CHUNK_SIZE
|
2016-10-15 02:00:18 +02:00
|
|
|
|
|
|
|
local PROCESS_PLAYER_BOUND = constants.PROCESS_PLAYER_BOUND
|
|
|
|
local CHUNK_TICK = constants.CHUNK_TICK
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
local SENTINEL_IMPASSABLE_CHUNK = constants.SENTINEL_IMPASSABLE_CHUNK
|
2017-05-12 06:50:06 +02:00
|
|
|
|
2016-11-04 09:26:19 +02:00
|
|
|
local AI_SQUAD_COST = constants.AI_SQUAD_COST
|
|
|
|
local AI_VENGENCE_SQUAD_COST = constants.AI_VENGENCE_SQUAD_COST
|
|
|
|
|
|
|
|
local MOVEMENT_PHEROMONE = constants.MOVEMENT_PHEROMONE
|
2017-04-17 04:46:36 +02:00
|
|
|
local BUILDING_PHEROMONES = constants.BUILDING_PHEROMONES
|
2016-11-04 09:26:19 +02:00
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- imported functions
|
|
|
|
|
2016-08-21 23:48:55 +02:00
|
|
|
local scents = pheromoneUtils.scents
|
|
|
|
local processPheromone = pheromoneUtils.processPheromone
|
2017-11-21 09:27:03 +02:00
|
|
|
local playerScent = pheromoneUtils.playerScent
|
2016-08-20 04:52:27 +02:00
|
|
|
|
2017-06-13 05:16:43 +02:00
|
|
|
local formSquads = aiAttackWave.formSquads
|
2016-08-20 04:52:27 +02:00
|
|
|
|
2016-10-15 02:00:18 +02:00
|
|
|
local getChunkByPosition = mapUtils.getChunkByPosition
|
2017-11-21 09:27:03 +02:00
|
|
|
local positionToChunkXY = mapUtils.positionToChunkXY
|
2016-10-15 02:00:18 +02:00
|
|
|
|
2017-06-11 02:59:06 +02:00
|
|
|
local recycleBiters = unitGroupUtils.recycleBiters
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
local validPlayer = playerUtils.validPlayer
|
|
|
|
|
|
|
|
local setResourceGenerator = chunkUtils.setResourceGenerator
|
|
|
|
local setPlayerBaseGenerator = chunkUtils.setPlayerBaseGenerator
|
|
|
|
local getNestCount = chunkUtils.getNestCount
|
|
|
|
local getWormCount = chunkUtils.getWormCount
|
2016-08-20 04:52:27 +02:00
|
|
|
|
2017-05-19 09:47:24 +02:00
|
|
|
local canAttack = aiPredicates.canAttack
|
2017-05-14 00:32:16 +02:00
|
|
|
|
2017-06-16 03:30:26 +02:00
|
|
|
local euclideanDistanceNamed = mathUtils.euclideanDistanceNamed
|
2017-05-28 06:50:37 +02:00
|
|
|
|
2016-08-29 02:05:28 +02:00
|
|
|
local mMin = math.min
|
|
|
|
|
2017-07-01 06:36:23 +02:00
|
|
|
local mRandom = math.random
|
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- module code
|
|
|
|
|
2016-10-15 02:00:18 +02:00
|
|
|
local function nonRepeatingRandom(players)
|
|
|
|
local ordering = {}
|
|
|
|
for _,player in pairs(players) do
|
|
|
|
ordering[#ordering+1] = player.index
|
|
|
|
end
|
|
|
|
for i=#ordering,1,-1 do
|
2017-07-01 06:36:23 +02:00
|
|
|
local s = mRandom(i)
|
2016-10-15 02:00:18 +02:00
|
|
|
local t = ordering[i]
|
|
|
|
ordering[i] = ordering[s]
|
|
|
|
ordering[s] = t
|
|
|
|
end
|
|
|
|
return ordering
|
|
|
|
end
|
|
|
|
|
|
|
|
--[[
|
|
|
|
processing is not consistant as it depends on the number of chunks that have been generated
|
|
|
|
so if we process 400 chunks an iteration and 200 chunks have been generated than these are
|
|
|
|
processed 3 times a second and 1200 generated chunks would be processed once a second
|
|
|
|
In theory, this might be fine as smaller bases have less surface to attack and need to have
|
|
|
|
pheromone dissipate at a faster rate.
|
|
|
|
--]]
|
2017-06-01 03:46:53 +02:00
|
|
|
function mapProcessor.processMap(regionMap, surface, natives, tick)
|
2016-08-29 02:05:28 +02:00
|
|
|
local roll = regionMap.processRoll
|
2017-07-23 03:07:44 +02:00
|
|
|
local index = regionMap.processIndex
|
2016-08-25 01:30:45 +02:00
|
|
|
|
2016-08-29 02:05:28 +02:00
|
|
|
if (index == 1) then
|
2017-07-01 06:36:23 +02:00
|
|
|
roll = mRandom()
|
2016-08-29 02:05:28 +02:00
|
|
|
regionMap.processRoll = roll
|
2016-08-26 00:20:06 +02:00
|
|
|
end
|
2017-05-14 00:32:16 +02:00
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
local squads = canAttack(natives, surface) and (0.11 <= roll) and (roll <= 0.35) and (natives.points >= AI_SQUAD_COST)
|
2016-08-20 04:52:27 +02:00
|
|
|
|
2016-08-29 02:05:28 +02:00
|
|
|
local processQueue = regionMap.processQueue
|
2016-09-14 13:16:33 +02:00
|
|
|
local endIndex = mMin(index + PROCESS_QUEUE_SIZE, #processQueue)
|
2016-08-29 02:05:28 +02:00
|
|
|
for x=index,endIndex do
|
|
|
|
local chunk = processQueue[x]
|
2017-04-22 01:14:04 +02:00
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
if (chunk[CHUNK_TICK] ~= tick) then
|
|
|
|
chunk[CHUNK_TICK] = tick
|
|
|
|
|
2017-06-10 10:38:20 +02:00
|
|
|
processPheromone(regionMap, chunk)
|
2017-04-22 01:14:04 +02:00
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
if squads and (getNestCount(regionMap, chunk) > 0) then
|
2017-06-10 10:38:20 +02:00
|
|
|
formSquads(regionMap, surface, natives, chunk, AI_SQUAD_COST)
|
2017-08-08 10:19:51 +02:00
|
|
|
squads = (natives.points >= AI_SQUAD_COST) and (#natives.squads < natives.maxSquads)
|
2017-06-01 03:46:53 +02:00
|
|
|
end
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
scents(regionMap, chunk)
|
2017-06-01 03:46:53 +02:00
|
|
|
end
|
2016-08-29 02:05:28 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
if (endIndex == #processQueue) then
|
2017-07-23 03:07:44 +02:00
|
|
|
regionMap.processIndex = 1
|
2016-08-29 02:05:28 +02:00
|
|
|
else
|
2017-07-23 03:07:44 +02:00
|
|
|
regionMap.processIndex = endIndex + 1
|
2016-08-29 02:05:28 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-10-15 02:00:18 +02:00
|
|
|
--[[
|
|
|
|
Localized player radius were processing takes place in realtime, doesn't store state
|
|
|
|
between calls.
|
|
|
|
vs
|
|
|
|
the slower passive version processing the entire map in multiple passes.
|
|
|
|
--]]
|
2017-06-01 03:46:53 +02:00
|
|
|
function mapProcessor.processPlayers(players, regionMap, surface, natives, tick)
|
2016-10-15 02:00:18 +02:00
|
|
|
-- put down player pheromone for player hunters
|
|
|
|
-- randomize player order to ensure a single player isn't singled out
|
|
|
|
local playerOrdering = nonRepeatingRandom(players)
|
2017-03-25 23:46:30 +02:00
|
|
|
|
2017-07-01 06:36:23 +02:00
|
|
|
local roll = mRandom()
|
2017-06-01 03:46:53 +02:00
|
|
|
|
|
|
|
local allowingAttacks = canAttack(natives, surface)
|
2016-10-15 02:00:18 +02:00
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
local squads = allowingAttacks and (0.11 <= roll) and (roll <= 0.20) and (natives.points >= AI_SQUAD_COST)
|
|
|
|
|
2016-10-15 02:00:18 +02:00
|
|
|
for i=1,#playerOrdering do
|
|
|
|
local player = players[playerOrdering[i]]
|
2017-05-19 09:47:24 +02:00
|
|
|
if validPlayer(player) then
|
2017-11-21 09:27:03 +02:00
|
|
|
local chunkX, chunkY = positionToChunkXY(player.character.position)
|
|
|
|
local playerChunk = getChunkByPosition(regionMap, chunkX, chunkY)
|
2016-10-15 02:00:18 +02:00
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
if (playerChunk ~= SENTINEL_IMPASSABLE_CHUNK) then
|
2016-10-15 02:00:18 +02:00
|
|
|
playerScent(playerChunk)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
for i=1,#playerOrdering do
|
|
|
|
local player = players[playerOrdering[i]]
|
2017-05-19 09:47:24 +02:00
|
|
|
if validPlayer(player) then
|
2017-11-21 09:27:03 +02:00
|
|
|
local chunkX, chunkY = positionToChunkXY(player.character.position)
|
|
|
|
local playerChunk = getChunkByPosition(regionMap, chunkX, chunkY)
|
2016-10-15 02:00:18 +02:00
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
if (playerChunk ~= SENTINEL_IMPASSABLE_CHUNK) then
|
|
|
|
local vengence = (allowingAttacks and
|
|
|
|
(natives.points >= AI_VENGENCE_SQUAD_COST) and
|
|
|
|
((getWormCount(regionMap, playerChunk) > 0) or (getNestCount(regionMap, playerChunk) > 0) or (playerChunk[MOVEMENT_PHEROMONE] < natives.retreatThreshold)))
|
2017-06-01 03:46:53 +02:00
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
for x=playerChunk.x - PROCESS_PLAYER_BOUND, playerChunk.x + PROCESS_PLAYER_BOUND, 32 do
|
|
|
|
for y=playerChunk.y - PROCESS_PLAYER_BOUND, playerChunk.y + PROCESS_PLAYER_BOUND, 32 do
|
|
|
|
local chunk = getChunkByPosition(regionMap, x, y)
|
|
|
|
|
|
|
|
if (chunk ~= SENTINEL_IMPASSABLE_CHUNK) and (chunk[CHUNK_TICK] ~= tick) then
|
2016-10-15 02:00:18 +02:00
|
|
|
chunk[CHUNK_TICK] = tick
|
2017-04-22 01:14:04 +02:00
|
|
|
|
2017-06-10 10:38:20 +02:00
|
|
|
processPheromone(regionMap, chunk)
|
2017-04-22 01:14:04 +02:00
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
if (getNestCount(regionMap, chunk) > 0) then
|
2017-06-11 02:59:06 +02:00
|
|
|
if squads then
|
|
|
|
formSquads(regionMap, surface, natives, chunk, AI_SQUAD_COST)
|
2017-08-08 10:19:51 +02:00
|
|
|
squads = (natives.points >= AI_SQUAD_COST) and (#natives.squads < natives.maxSquads)
|
2017-06-11 02:59:06 +02:00
|
|
|
end
|
|
|
|
if vengence then
|
|
|
|
formSquads(regionMap, surface, natives, chunk, AI_VENGENCE_SQUAD_COST)
|
2017-08-08 10:19:51 +02:00
|
|
|
vengence = (natives.points >= AI_VENGENCE_SQUAD_COST) and (#natives.squads < natives.maxSquads)
|
2017-06-11 02:59:06 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
scents(regionMap, chunk)
|
2016-10-15 02:00:18 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-11-04 01:51:35 +02:00
|
|
|
--[[
|
2017-04-17 04:46:36 +02:00
|
|
|
Passive scan to find entities that have been generated outside the factorio event system
|
2016-11-04 01:51:35 +02:00
|
|
|
--]]
|
2017-06-01 03:46:53 +02:00
|
|
|
function mapProcessor.scanMap(regionMap, surface, natives)
|
2017-07-23 03:07:44 +02:00
|
|
|
local index = regionMap.scanIndex
|
2017-06-01 03:46:53 +02:00
|
|
|
|
|
|
|
local offset = {0, 0}
|
|
|
|
local chunkBox = {false, offset}
|
2017-05-24 08:46:23 +02:00
|
|
|
local playerQuery = {area = chunkBox,
|
|
|
|
force = "player"}
|
2017-06-08 02:57:24 +02:00
|
|
|
local resourceQuery = {area = chunkBox,
|
|
|
|
type = "resource"}
|
2017-06-01 03:46:53 +02:00
|
|
|
local unitCountQuery = { area = chunkBox,
|
|
|
|
type = "unit",
|
|
|
|
force = "enemy",
|
|
|
|
limit = 301 }
|
2017-05-24 08:46:23 +02:00
|
|
|
|
2016-08-29 02:05:28 +02:00
|
|
|
local processQueue = regionMap.processQueue
|
2016-09-14 13:16:33 +02:00
|
|
|
local endIndex = mMin(index + SCAN_QUEUE_SIZE, #processQueue)
|
2017-06-01 03:46:53 +02:00
|
|
|
|
2016-08-29 02:05:28 +02:00
|
|
|
for x=index,endIndex do
|
2016-10-15 02:00:18 +02:00
|
|
|
local chunk = processQueue[x]
|
2017-04-17 04:46:36 +02:00
|
|
|
|
2017-05-28 06:50:37 +02:00
|
|
|
chunkBox[1] = chunk
|
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
offset[1] = chunk.x + CHUNK_SIZE
|
|
|
|
offset[2] = chunk.y + CHUNK_SIZE
|
2017-05-24 08:46:23 +02:00
|
|
|
|
|
|
|
local unitCount = surface.count_entities_filtered(unitCountQuery)
|
|
|
|
|
|
|
|
if (unitCount > 300) then
|
2017-06-01 03:46:53 +02:00
|
|
|
local closeBy = false
|
|
|
|
local squads = natives.squads
|
|
|
|
for i=1, #squads do
|
|
|
|
local squadGroup = squads[i].group
|
|
|
|
if squadGroup.valid and (euclideanDistanceNamed(squadGroup.position, chunk) < DOUBLE_CHUNK_SIZE) then
|
2017-05-24 08:46:23 +02:00
|
|
|
closeBy = true
|
2017-06-01 03:46:53 +02:00
|
|
|
break
|
2017-05-24 08:46:23 +02:00
|
|
|
end
|
|
|
|
end
|
2017-06-01 03:46:53 +02:00
|
|
|
|
|
|
|
if not closeBy then
|
2017-06-11 02:59:06 +02:00
|
|
|
recycleBiters(natives, surface.find_enemy_units(chunk, TRIPLE_CHUNK_SIZE))
|
2017-04-22 01:14:04 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
setResourceGenerator(regionMap, chunk, surface.count_entities_filtered(resourceQuery) * 0.001)
|
2017-06-08 02:57:24 +02:00
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
local entities = surface.find_entities_filtered(playerQuery)
|
2017-04-17 04:46:36 +02:00
|
|
|
local playerBaseGenerator = 0
|
2017-05-24 08:46:23 +02:00
|
|
|
local safeBuildings = natives.safeBuildings
|
2017-04-17 04:46:36 +02:00
|
|
|
for i=1,#entities do
|
|
|
|
local entity = entities[i]
|
|
|
|
local value = BUILDING_PHEROMONES[entity.type]
|
2017-05-24 08:46:23 +02:00
|
|
|
if safeBuildings then
|
|
|
|
if natives.safeEntities[entity.type] or natives.safeEntityName[entity.name] then
|
|
|
|
entity.destructible = false
|
|
|
|
end
|
|
|
|
end
|
2017-06-11 02:59:06 +02:00
|
|
|
if value then
|
2017-04-17 04:46:36 +02:00
|
|
|
playerBaseGenerator = playerBaseGenerator + value
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-11-21 09:27:03 +02:00
|
|
|
setPlayerBaseGenerator(regionMap, chunk, playerBaseGenerator)
|
2016-08-05 06:47:51 +02:00
|
|
|
end
|
2017-04-22 01:14:04 +02:00
|
|
|
|
2016-08-29 02:05:28 +02:00
|
|
|
if (endIndex == #processQueue) then
|
2017-07-23 03:07:44 +02:00
|
|
|
regionMap.scanIndex = 1
|
2016-08-29 02:05:28 +02:00
|
|
|
else
|
2017-07-23 03:07:44 +02:00
|
|
|
regionMap.scanIndex = endIndex + 1
|
2016-08-20 04:52:27 +02:00
|
|
|
end
|
2016-08-05 06:47:51 +02:00
|
|
|
end
|
|
|
|
|
2016-09-08 22:02:23 +02:00
|
|
|
return mapProcessor
|