2019-02-15 20:17:30 -08:00
|
|
|
if aiDefenseG then
|
|
|
|
return aiDefenseG
|
|
|
|
end
|
2016-08-17 22:55:08 -07:00
|
|
|
local aiDefense = {}
|
|
|
|
|
2016-08-19 19:52:27 -07:00
|
|
|
-- imports
|
|
|
|
|
2016-08-17 22:55:08 -07:00
|
|
|
local constants = require("Constants")
|
|
|
|
local mapUtils = require("MapUtils")
|
|
|
|
local unitGroupUtils = require("UnitGroupUtils")
|
2017-06-15 18:30:26 -07:00
|
|
|
local movementUtils = require("MovementUtils")
|
2018-01-14 23:41:55 -08:00
|
|
|
local chunkPropetyUtils = require("ChunkPropertyUtils")
|
2016-08-17 22:55:08 -07:00
|
|
|
|
2016-08-19 19:52:27 -07:00
|
|
|
-- constants
|
|
|
|
|
2018-01-13 21:48:21 -08:00
|
|
|
-- local RETREAT_SPAWNER_GRAB_RADIUS = constants.RETREAT_SPAWNER_GRAB_RADIUS
|
2016-10-14 17:00:18 -07:00
|
|
|
local MOVEMENT_PHEROMONE = constants.MOVEMENT_PHEROMONE
|
2016-08-19 19:52:27 -07:00
|
|
|
local PLAYER_PHEROMONE = constants.PLAYER_PHEROMONE
|
2016-10-14 17:00:18 -07:00
|
|
|
local BASE_PHEROMONE = constants.BASE_PHEROMONE
|
2016-08-19 19:52:27 -07:00
|
|
|
|
2019-02-10 22:14:17 -08:00
|
|
|
local PLAYER_PHEROMONE_MULTIPLER = constants.PLAYER_PHEROMONE_MULTIPLER
|
|
|
|
|
2016-08-19 19:52:27 -07:00
|
|
|
local SQUAD_RETREATING = constants.SQUAD_RETREATING
|
|
|
|
|
2018-02-11 19:21:28 -08:00
|
|
|
local INTERVAL_RETREAT = constants.INTERVAL_RETREAT
|
2017-04-21 16:14:04 -07:00
|
|
|
|
2017-11-20 23:27:03 -08:00
|
|
|
local SENTINEL_IMPASSABLE_CHUNK = constants.SENTINEL_IMPASSABLE_CHUNK
|
2017-05-11 21:50:06 -07:00
|
|
|
|
2016-08-19 19:52:27 -07:00
|
|
|
-- imported functions
|
|
|
|
|
2018-01-14 16:14:38 -08:00
|
|
|
local mRandom = math.random
|
|
|
|
|
2018-01-14 23:41:55 -08:00
|
|
|
local addSquadToChunk = chunkPropetyUtils.addSquadToChunk
|
2018-01-13 21:48:21 -08:00
|
|
|
|
2018-01-14 16:14:38 -08:00
|
|
|
local calculateKamikazeThreshold = unitGroupUtils.calculateKamikazeThreshold
|
|
|
|
|
2017-06-07 17:57:24 -07:00
|
|
|
local positionFromDirectionAndChunk = mapUtils.positionFromDirectionAndChunk
|
|
|
|
local getNeighborChunks = mapUtils.getNeighborChunks
|
2018-01-13 21:48:21 -08:00
|
|
|
local findNearbySquadFiltered = unitGroupUtils.findNearbySquadFiltered
|
2017-06-15 18:30:26 -07:00
|
|
|
local addMovementPenalty = movementUtils.addMovementPenalty
|
2016-08-19 19:52:27 -07:00
|
|
|
local createSquad = unitGroupUtils.createSquad
|
|
|
|
local membersToSquad = unitGroupUtils.membersToSquad
|
2017-11-20 23:27:03 -08:00
|
|
|
local scoreNeighborsForRetreat = movementUtils.scoreNeighborsForRetreat
|
2017-06-15 18:30:26 -07:00
|
|
|
local findMovementPosition = movementUtils.findMovementPosition
|
2016-08-19 19:52:27 -07:00
|
|
|
|
2018-01-14 23:41:55 -08:00
|
|
|
local getRetreatTick = chunkPropetyUtils.getRetreatTick
|
|
|
|
local getPlayerBaseGenerator = chunkPropetyUtils.getPlayerBaseGenerator
|
|
|
|
local setRetreatTick = chunkPropetyUtils.setRetreatTick
|
|
|
|
local getEnemyStructureCount = chunkPropetyUtils.getEnemyStructureCount
|
2017-11-20 23:27:03 -08:00
|
|
|
|
2016-08-19 19:52:27 -07:00
|
|
|
-- module code
|
2016-08-17 22:55:08 -07:00
|
|
|
|
2018-01-13 21:48:21 -08:00
|
|
|
local function scoreRetreatLocation(map, neighborChunk)
|
2019-03-07 19:40:55 -08:00
|
|
|
return -(neighborChunk[BASE_PHEROMONE] +
|
|
|
|
neighborChunk[MOVEMENT_PHEROMONE] +
|
|
|
|
-(neighborChunk[PLAYER_PHEROMONE] * PLAYER_PHEROMONE_MULTIPLER) +
|
|
|
|
-(getPlayerBaseGenerator(map, neighborChunk)))
|
2016-08-26 23:44:17 -07:00
|
|
|
end
|
|
|
|
|
2018-02-11 19:21:28 -08:00
|
|
|
function aiDefense.retreatUnits(chunk, position, squad, map, surface, natives, tick, radius, artilleryBlast, force)
|
|
|
|
if (tick - getRetreatTick(map, chunk) > INTERVAL_RETREAT) and ((getEnemyStructureCount(map, chunk) == 0) or artilleryBlast or force) then
|
2016-11-03 16:51:35 -07:00
|
|
|
local performRetreat = false
|
2017-05-27 21:50:37 -07:00
|
|
|
local enemiesToSquad = nil
|
2019-02-06 22:28:02 -08:00
|
|
|
|
2017-05-27 21:50:37 -07:00
|
|
|
if not squad then
|
2018-01-01 15:31:14 -08:00
|
|
|
enemiesToSquad = surface.find_enemy_units(position, radius)
|
2017-05-23 23:46:23 -07:00
|
|
|
performRetreat = #enemiesToSquad > 0
|
2018-01-14 16:14:38 -08:00
|
|
|
if (mRandom() < calculateKamikazeThreshold(#enemiesToSquad, natives)) then
|
2018-02-11 19:21:28 -08:00
|
|
|
setRetreatTick(map, chunk, tick)
|
2019-03-07 19:40:55 -08:00
|
|
|
return
|
2018-01-14 16:14:38 -08:00
|
|
|
end
|
2018-02-06 23:57:41 -08:00
|
|
|
elseif squad.group and squad.group.valid and (squad.status ~= SQUAD_RETREATING) and not squad.kamikaze then
|
2019-03-07 19:40:55 -08:00
|
|
|
performRetreat = #squad.group.members > 3
|
2016-11-03 16:51:35 -07:00
|
|
|
end
|
2019-02-06 22:28:02 -08:00
|
|
|
|
2016-11-03 16:51:35 -07:00
|
|
|
if performRetreat then
|
2018-01-13 21:48:21 -08:00
|
|
|
setRetreatTick(map, chunk, tick)
|
2017-06-10 17:59:06 -07:00
|
|
|
local exitPath,exitDirection = scoreNeighborsForRetreat(chunk,
|
2018-01-13 21:48:21 -08:00
|
|
|
getNeighborChunks(map, chunk.x, chunk.y),
|
2017-11-20 23:27:03 -08:00
|
|
|
scoreRetreatLocation,
|
2018-01-13 21:48:21 -08:00
|
|
|
map)
|
2017-11-20 23:27:03 -08:00
|
|
|
if (exitPath ~= SENTINEL_IMPASSABLE_CHUNK) then
|
2017-06-15 18:30:26 -07:00
|
|
|
local retreatPosition = findMovementPosition(surface,
|
2019-03-07 19:40:55 -08:00
|
|
|
positionFromDirectionAndChunk(exitDirection, position, map.position, 0.98))
|
2019-02-06 22:28:02 -08:00
|
|
|
|
2017-06-15 18:30:26 -07:00
|
|
|
if not retreatPosition then
|
|
|
|
return
|
2017-06-10 01:38:20 -07:00
|
|
|
end
|
2018-01-01 15:31:14 -08:00
|
|
|
|
2016-11-03 16:51:35 -07:00
|
|
|
-- in order for units in a group attacking to retreat, we have to create a new group and give the command to join
|
|
|
|
-- to each unit, this is the only way I have found to have snappy mid battle retreats even after 0.14.4
|
2019-02-06 22:28:02 -08:00
|
|
|
|
2019-03-07 19:40:55 -08:00
|
|
|
local newSquad = findNearbySquadFiltered(map, exitPath)
|
2017-08-08 01:19:51 -07:00
|
|
|
|
2018-01-01 15:31:14 -08:00
|
|
|
if not newSquad then
|
2019-03-06 22:12:39 -08:00
|
|
|
newSquad = createSquad(retreatPosition, surface)
|
|
|
|
natives.squads[#natives.squads+1] = newSquad
|
2016-11-03 16:51:35 -07:00
|
|
|
end
|
2017-08-08 01:19:51 -07:00
|
|
|
|
|
|
|
if newSquad then
|
2019-03-06 22:12:39 -08:00
|
|
|
newSquad.status = SQUAD_RETREATING
|
2019-03-07 19:40:55 -08:00
|
|
|
newSquad.cycles = 4
|
2019-03-06 22:12:39 -08:00
|
|
|
|
2018-01-13 21:48:21 -08:00
|
|
|
local cmd = map.retreatCommand
|
|
|
|
cmd.group = newSquad.group
|
2017-08-08 01:19:51 -07:00
|
|
|
if enemiesToSquad then
|
2018-02-11 19:21:28 -08:00
|
|
|
membersToSquad(cmd, enemiesToSquad, artilleryBlast)
|
2017-08-08 01:19:51 -07:00
|
|
|
else
|
2018-01-13 21:48:21 -08:00
|
|
|
membersToSquad(cmd, squad.group.members, true)
|
2017-08-08 01:19:51 -07:00
|
|
|
newSquad.penalties = squad.penalties
|
|
|
|
if squad.rabid then
|
|
|
|
newSquad.rabid = true
|
|
|
|
end
|
2016-09-14 10:12:29 -07:00
|
|
|
end
|
2019-03-07 19:40:55 -08:00
|
|
|
|
|
|
|
if not newSquad.rapid then
|
|
|
|
newSquad.frenzy = true
|
|
|
|
local squadPosition = newSquad.group.position
|
|
|
|
newSquad.frenzyPosition.x = squadPosition.x
|
|
|
|
newSquad.frenzyPosition.y = squadPosition.y
|
|
|
|
end
|
2018-01-13 21:48:21 -08:00
|
|
|
addSquadToChunk(map, chunk, newSquad)
|
2017-12-20 19:50:36 -08:00
|
|
|
addMovementPenalty(natives, newSquad, chunk)
|
2016-11-03 16:51:35 -07:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-08-17 22:55:08 -07:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-02-15 20:17:30 -08:00
|
|
|
aiDefenseG = aiDefense
|
2016-08-29 21:08:22 -07:00
|
|
|
return aiDefense
|