2016-08-18 07:55:08 +02:00
|
|
|
local aiAttack = {}
|
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- imports
|
2016-08-18 07:55:08 +02:00
|
|
|
|
|
|
|
local constants = require("Constants")
|
|
|
|
local mapUtils = require("MapUtils")
|
|
|
|
local unitGroupUtils = require("UnitGroupUtils")
|
2016-08-27 08:44:17 +02:00
|
|
|
local playerUtils = require("PlayerUtils")
|
|
|
|
local neighborUtils = require("NeighborUtils")
|
2016-08-18 07:55:08 +02:00
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- constants
|
2017-05-14 00:32:16 +02:00
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
local PLAYER_PHEROMONE = constants.PLAYER_PHEROMONE
|
2016-10-15 02:00:18 +02:00
|
|
|
local MOVEMENT_PHEROMONE = constants.MOVEMENT_PHEROMONE
|
|
|
|
local BASE_PHEROMONE = constants.BASE_PHEROMONE
|
2016-08-20 04:52:27 +02:00
|
|
|
|
|
|
|
local SQUAD_RAIDING = constants.SQUAD_RAIDING
|
|
|
|
local SQUAD_GUARDING = constants.SQUAD_GUARDING
|
|
|
|
|
2016-10-15 02:00:18 +02:00
|
|
|
local PLAYER_BASE_GENERATOR = constants.PLAYER_BASE_GENERATOR
|
2016-08-20 04:52:27 +02:00
|
|
|
|
2017-05-28 06:50:37 +02:00
|
|
|
local NEST_COUNT = constants.NEST_COUNT
|
|
|
|
|
|
|
|
local DEFINES_COMMAND_ATTACK_AREA = defines.command.attack_area
|
2017-05-27 02:58:33 +02:00
|
|
|
local DEFINES_GROUP_FINISHED = defines.group_state.finished
|
|
|
|
local DEFINES_GROUP_GATHERING = defines.group_state.gathering
|
|
|
|
local DEFINES_GROUP_MOVING = defines.group_state.moving
|
|
|
|
local DEFINES_GROUP_ATTACKING_DISTRACTION = defines.group_state.attacking_distraction
|
|
|
|
local DEFINES_GROUP_ATTACKING_TARGET = defines.group_state.attacking_target
|
|
|
|
local DEFINES_DISTRACTION_BY_ENEMY = defines.distraction.by_enemy
|
|
|
|
local DEFINES_DISTRACTION_BY_ANYTHING = defines.distraction.by_anything
|
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- imported functions
|
|
|
|
|
2017-06-08 02:57:24 +02:00
|
|
|
local getNeighborChunks = mapUtils.getNeighborChunks
|
2016-08-28 02:57:20 +02:00
|
|
|
local getChunkByPosition = mapUtils.getChunkByPosition
|
2016-10-31 05:24:14 +02:00
|
|
|
local canMoveChunkDirection = mapUtils.canMoveChunkDirection
|
2016-08-20 04:52:27 +02:00
|
|
|
local addSquadMovementPenalty = unitGroupUtils.addSquadMovementPenalty
|
|
|
|
local lookupSquadMovementPenalty = unitGroupUtils.lookupSquadMovementPenalty
|
2017-01-20 07:58:36 +02:00
|
|
|
local calculateKamikazeThreshold = unitGroupUtils.calculateKamikazeThreshold
|
2016-10-31 05:24:14 +02:00
|
|
|
local positionFromDirectionAndChunk = mapUtils.positionFromDirectionAndChunk
|
2016-08-27 08:44:17 +02:00
|
|
|
|
2016-09-14 19:12:29 +02:00
|
|
|
local euclideanDistanceNamed = mapUtils.euclideanDistanceNamed
|
|
|
|
|
2016-08-27 08:44:17 +02:00
|
|
|
local playersWithinProximityToPosition = playerUtils.playersWithinProximityToPosition
|
|
|
|
|
|
|
|
local scoreNeighborsWithDirection = neighborUtils.scoreNeighborsWithDirection
|
2016-08-18 07:55:08 +02:00
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- module code
|
|
|
|
|
2016-08-27 08:44:17 +02:00
|
|
|
local function validLocation(x, chunk, neighborChunk)
|
2016-10-31 05:24:14 +02:00
|
|
|
return canMoveChunkDirection(x, chunk, neighborChunk)
|
2016-08-27 08:44:17 +02:00
|
|
|
end
|
|
|
|
|
2017-05-28 06:50:37 +02:00
|
|
|
local function scoreAttackLocation(squad, neighborChunk, surface)
|
2016-08-27 08:44:17 +02:00
|
|
|
local squadMovementPenalty = lookupSquadMovementPenalty(squad, neighborChunk.cX, neighborChunk.cY)
|
2017-06-09 07:18:59 +02:00
|
|
|
local r = -- (surface.get_pollution(neighborChunk) * 0.01) +
|
|
|
|
neighborChunk[MOVEMENT_PHEROMONE] + neighborChunk[BASE_PHEROMONE] + (neighborChunk[PLAYER_PHEROMONE] * 50)
|
|
|
|
return r - squadMovementPenalty -- - (neighborChunk[NEST_COUNT] * 30)
|
2016-08-27 08:44:17 +02:00
|
|
|
end
|
|
|
|
|
2016-09-14 19:12:29 +02:00
|
|
|
function aiAttack.squadAttack(regionMap, surface, natives)
|
2016-08-26 00:20:06 +02:00
|
|
|
local squads = natives.squads
|
2016-09-14 19:12:29 +02:00
|
|
|
local attackPosition
|
|
|
|
local attackCmd
|
2017-06-01 03:46:53 +02:00
|
|
|
local tempNeighbors
|
2017-05-27 02:58:33 +02:00
|
|
|
|
2016-09-14 19:12:29 +02:00
|
|
|
if (#squads > 0) then
|
2017-06-08 02:57:24 +02:00
|
|
|
tempNeighbors = {nil, nil, nil, nil, nil, nil, nil, nil}
|
2016-09-14 19:12:29 +02:00
|
|
|
attackPosition = {x=0, y=0}
|
2017-05-28 06:50:37 +02:00
|
|
|
attackCmd = { type = DEFINES_COMMAND_ATTACK_AREA,
|
2016-09-14 19:12:29 +02:00
|
|
|
destination = attackPosition,
|
2017-04-16 08:04:22 +02:00
|
|
|
radius = 28,
|
2017-05-27 02:58:33 +02:00
|
|
|
distraction = DEFINES_DISTRACTION_BY_ENEMY }
|
2016-09-14 19:12:29 +02:00
|
|
|
end
|
2017-06-01 09:03:07 +02:00
|
|
|
|
2016-08-26 00:20:06 +02:00
|
|
|
for i=1,#squads do
|
|
|
|
local squad = squads[i]
|
2016-08-18 07:55:08 +02:00
|
|
|
local group = squad.group
|
2017-05-24 08:46:23 +02:00
|
|
|
if group.valid and (squad.status == SQUAD_RAIDING) then
|
|
|
|
local groupState = group.state
|
2017-05-27 02:58:33 +02:00
|
|
|
if (groupState == DEFINES_GROUP_FINISHED) or (groupState == DEFINES_GROUP_GATHERING) or ((groupState == DEFINES_GROUP_MOVING) and (squad.cycles == 0)) then
|
|
|
|
local groupPosition = group.position
|
|
|
|
local chunk = getChunkByPosition(regionMap, groupPosition.x, groupPosition.y)
|
|
|
|
if chunk then
|
2016-11-04 01:51:35 +02:00
|
|
|
local attackChunk, attackDirection = scoreNeighborsWithDirection(chunk,
|
2017-06-08 02:57:24 +02:00
|
|
|
getNeighborChunks(regionMap,
|
|
|
|
chunk.cX,
|
|
|
|
chunk.cY,
|
|
|
|
tempNeighbors),
|
2016-11-04 01:51:35 +02:00
|
|
|
validLocation,
|
|
|
|
scoreAttackLocation,
|
|
|
|
squad,
|
|
|
|
surface,
|
|
|
|
true)
|
2016-10-31 05:24:14 +02:00
|
|
|
addSquadMovementPenalty(squad, chunk.cX, chunk.cY)
|
2017-05-24 08:46:23 +02:00
|
|
|
if attackChunk then
|
2016-11-04 01:51:35 +02:00
|
|
|
if (attackChunk[PLAYER_BASE_GENERATOR] == 0) or
|
2017-05-27 02:58:33 +02:00
|
|
|
((groupState == DEFINES_GROUP_FINISHED) or (groupState == DEFINES_GROUP_GATHERING)) then
|
2016-08-27 08:44:17 +02:00
|
|
|
|
2017-06-08 02:57:24 +02:00
|
|
|
positionFromDirectionAndChunk(attackDirection, groupPosition, attackPosition, 1.25)
|
2016-09-14 19:12:29 +02:00
|
|
|
|
2017-04-16 08:04:22 +02:00
|
|
|
if (#squad.group.members > 80) then
|
|
|
|
squad.cycles = 6
|
|
|
|
else
|
|
|
|
squad.cycles = 4
|
|
|
|
end
|
2016-09-14 19:12:29 +02:00
|
|
|
|
2017-05-28 06:50:37 +02:00
|
|
|
local moreFrenzy = not squad.rabid and squad.frenzy and (euclideanDistanceNamed(groupPosition, squad.frenzyPosition) < 100)
|
|
|
|
squad.frenzy = moreFrenzy
|
2016-09-14 19:12:29 +02:00
|
|
|
|
|
|
|
if squad.rabid or squad.frenzy then
|
2017-05-27 02:58:33 +02:00
|
|
|
attackCmd.distraction = DEFINES_DISTRACTION_BY_ANYTHING
|
2016-09-14 19:12:29 +02:00
|
|
|
else
|
2017-05-27 02:58:33 +02:00
|
|
|
attackCmd.distraction = DEFINES_DISTRACTION_BY_ENEMY
|
2016-09-14 19:12:29 +02:00
|
|
|
end
|
|
|
|
|
2016-10-15 02:00:18 +02:00
|
|
|
group.set_command(attackCmd)
|
|
|
|
group.start_moving()
|
2016-09-14 19:12:29 +02:00
|
|
|
elseif not squad.frenzy and not squad.rabid and
|
2017-05-27 02:58:33 +02:00
|
|
|
((groupState == DEFINES_GROUP_ATTACKING_DISTRACTION) or (groupState == DEFINES_GROUP_ATTACKING_TARGET) or
|
2016-10-15 02:00:18 +02:00
|
|
|
(attackChunk[PLAYER_BASE_GENERATOR] ~= 0)) then
|
|
|
|
squad.frenzy = true
|
2017-05-27 02:58:33 +02:00
|
|
|
squad.frenzyPosition.x = groupPosition.x
|
|
|
|
squad.frenzyPosition.y = groupPosition.y
|
2016-09-14 19:12:29 +02:00
|
|
|
end
|
2016-11-04 01:51:35 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-08-18 07:55:08 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
function aiAttack.squadBeginAttack(natives, players)
|
2016-08-26 00:20:06 +02:00
|
|
|
local squads = natives.squads
|
|
|
|
for i=1,#squads do
|
|
|
|
local squad = squads[i]
|
2017-06-01 03:46:53 +02:00
|
|
|
local group = squad.group
|
|
|
|
if (squad.status == SQUAD_GUARDING) and group.valid then
|
|
|
|
local groupPosition = group.position
|
|
|
|
local kamikazeThreshold = calculateKamikazeThreshold(squad, natives)
|
2016-10-15 02:00:18 +02:00
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
local playerNearby = playersWithinProximityToPosition(players, groupPosition, 100)
|
2016-09-14 19:12:29 +02:00
|
|
|
if playerNearby then
|
|
|
|
squad.frenzy = true
|
2017-06-01 03:46:53 +02:00
|
|
|
squad.frenzyPosition.x = groupPosition.x
|
|
|
|
squad.frenzyPosition.y = groupPosition.y
|
2016-09-14 19:12:29 +02:00
|
|
|
end
|
2017-05-14 00:32:16 +02:00
|
|
|
|
2016-10-15 02:00:18 +02:00
|
|
|
if (math.random() < 0.70) then
|
2017-06-01 03:46:53 +02:00
|
|
|
squad.kamikaze = math.random() < kamikazeThreshold
|
2016-11-04 01:51:35 +02:00
|
|
|
squad.status = SQUAD_RAIDING
|
2016-10-15 02:00:18 +02:00
|
|
|
end
|
|
|
|
end
|
2016-08-18 07:55:08 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-08-30 06:08:22 +02:00
|
|
|
return aiAttack
|