1
0
mirror of https://github.com/veden/Rampant.git synced 2025-01-03 22:52:20 +02:00
Rampant/libs/SquadAttack.lua

278 lines
11 KiB
Lua
Raw Normal View History

2019-02-16 06:17:30 +02:00
if (squadAttackG) then
return squadAttackG
end
local squadAttack = {}
-- imports
local constants = require("Constants")
local mapUtils = require("MapUtils")
local unitGroupUtils = require("UnitGroupUtils")
local playerUtils = require("PlayerUtils")
local movementUtils = require("MovementUtils")
local mathUtils = require("MathUtils")
local chunkPropertyUtils = require("ChunkPropertyUtils")
-- constants
2017-05-14 00:32:16 +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
2018-02-13 09:10:17 +02:00
local RESOURCE_PHEROMONE = constants.RESOURCE_PHEROMONE
2019-02-16 20:45:42 +02:00
local ATTACK_SCORE = constants.ATTACK_SCORE
local ATTACK_SCORE_KAMIKAZE = constants.ATTACK_SCORE_KAMIKAZE
local SQUAD_BUILDING = constants.SQUAD_BUILDING
local SQUAD_RAIDING = constants.SQUAD_RAIDING
2018-02-13 09:10:17 +02:00
local SQUAD_SETTLING = constants.SQUAD_SETTLING
local SQUAD_GUARDING = constants.SQUAD_GUARDING
local AI_STATE_SIEGE = constants.AI_STATE_SIEGE
2017-06-11 02:59:06 +02:00
local PLAYER_PHEROMONE_MULTIPLER = constants.PLAYER_PHEROMONE_MULTIPLER
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
2017-11-21 09:27:03 +02:00
local SENTINEL_IMPASSABLE_CHUNK = constants.SENTINEL_IMPASSABLE_CHUNK
-- imported functions
local mRandom = math.random
local euclideanDistancePoints = mathUtils.euclideanDistancePoints
local findMovementPosition = movementUtils.findMovementPosition
local getNestCount = chunkPropertyUtils.getNestCount
2018-02-17 05:31:29 +02:00
2017-06-08 02:57:24 +02:00
local getNeighborChunks = mapUtils.getNeighborChunks
local addSquadToChunk = chunkPropertyUtils.addSquadToChunk
2018-01-14 07:48:21 +02:00
local getChunkByXY = mapUtils.getChunkByXY
local positionToChunkXY = mapUtils.positionToChunkXY
local addMovementPenalty = movementUtils.addMovementPenalty
local lookupMovementPenalty = movementUtils.lookupMovementPenalty
2017-01-20 07:58:36 +02:00
local calculateKamikazeThreshold = unitGroupUtils.calculateKamikazeThreshold
2016-10-31 05:24:14 +02:00
local positionFromDirectionAndChunk = mapUtils.positionFromDirectionAndChunk
local euclideanDistanceNamed = mathUtils.euclideanDistanceNamed
local playersWithinProximityToPosition = playerUtils.playersWithinProximityToPosition
local getPlayerBaseGenerator = chunkPropertyUtils.getPlayerBaseGenerator
local getResourceGenerator = chunkPropertyUtils.getResourceGenerator
2017-11-21 09:27:03 +02:00
local scoreNeighborsForAttack = movementUtils.scoreNeighborsForAttack
2018-09-24 06:56:45 +02:00
local scoreNeighborsForSettling = movementUtils.scoreNeighborsForSettling
-- module code
2018-02-13 09:10:17 +02:00
local function scoreResourceLocation(squad, neighborChunk)
2019-02-11 08:14:17 +02:00
local settle = neighborChunk[MOVEMENT_PHEROMONE] + neighborChunk[RESOURCE_PHEROMONE]
return settle - lookupMovementPenalty(squad, neighborChunk) - (neighborChunk[PLAYER_PHEROMONE] * PLAYER_PHEROMONE_MULTIPLER)
2018-02-13 09:10:17 +02:00
end
local function scoreSiegeLocation(squad, neighborChunk)
2019-02-11 08:14:17 +02:00
local settle = neighborChunk[MOVEMENT_PHEROMONE] + neighborChunk[BASE_PHEROMONE] + neighborChunk[RESOURCE_PHEROMONE] + (neighborChunk[PLAYER_PHEROMONE] * PLAYER_PHEROMONE_MULTIPLER)
return settle - lookupMovementPenalty(squad, neighborChunk)
end
2019-02-16 06:17:30 +02:00
local function scoreAttackLocation(natives, squad, neighborChunk)
2019-02-11 08:14:17 +02:00
local damage
if (neighborChunk[MOVEMENT_PHEROMONE] >= 0) then
damage = neighborChunk[MOVEMENT_PHEROMONE] + (neighborChunk[BASE_PHEROMONE] ) + (neighborChunk[PLAYER_PHEROMONE] * PLAYER_PHEROMONE_MULTIPLER)
else
damage = (neighborChunk[BASE_PHEROMONE] * (1 - (neighborChunk[MOVEMENT_PHEROMONE] / -natives.retreatThreshold))) + (neighborChunk[PLAYER_PHEROMONE] * PLAYER_PHEROMONE_MULTIPLER)
end
return damage - lookupMovementPenalty(squad, neighborChunk)
end
local function scoreAttackKamikazeLocation(natives, squad, neighborChunk)
local damage = neighborChunk[BASE_PHEROMONE] + (neighborChunk[PLAYER_PHEROMONE] * PLAYER_PHEROMONE_MULTIPLER)
2017-12-21 05:50:36 +02:00
return damage - lookupMovementPenalty(squad, neighborChunk)
end
2019-02-11 08:14:17 +02:00
local function settleMove(map, attackPosition, attackCmd, settleCmd, squad, group, natives, surface)
2018-02-13 09:10:17 +02:00
local groupState = group.state
2018-02-13 09:10:17 +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 x, y = positionToChunkXY(groupPosition)
local chunk = getChunkByXY(map, x, y)
2018-09-24 06:56:45 +02:00
local attackChunk, attackDirection = scoreNeighborsForSettling(map,
chunk,
getNeighborChunks(map, x, y),
((natives.state == AI_STATE_SIEGE) and scoreSiegeLocation) or
scoreResourceLocation,
squad)
2018-02-13 09:10:17 +02:00
if (chunk ~= SENTINEL_IMPASSABLE_CHUNK) then
addSquadToChunk(map, chunk, squad)
addMovementPenalty(natives, squad, chunk)
elseif (attackChunk ~= SENTINEL_IMPASSABLE_CHUNK) then
addSquadToChunk(map, attackChunk, squad)
addMovementPenalty(natives, squad, attackChunk)
end
if group.valid and (attackChunk ~= SENTINEL_IMPASSABLE_CHUNK) then
local resourceGenerator = getResourceGenerator(map, groupPosition)
local distance = euclideanDistancePoints(groupPosition.x, groupPosition.y, squad.originPosition.x, squad.originPosition.y)
if (distance >= squad.maxDistance) or ((resourceGenerator ~= 0) and (getNestCount(map, chunk) == 0)) then
local position = findMovementPosition(surface, groupPosition)
if position then
attackPosition.x = position.x
attackPosition.y = position.y
2019-02-11 08:14:17 +02:00
squad.status = SQUAD_BUILDING
2019-02-11 08:14:17 +02:00
group.set_command(settleCmd)
group.start_moving()
else
addMovementPenalty(natives, squad, attackChunk)
end
elseif (groupState == DEFINES_GROUP_FINISHED) or (groupState == DEFINES_GROUP_GATHERING) then
2018-02-13 09:10:17 +02:00
squad.cycles = ((#squad.group.members > 80) and 6) or 4
attackCmd.distraction = DEFINES_DISTRACTION_BY_ENEMY
2019-02-11 08:14:17 +02:00
2018-02-13 09:10:17 +02:00
local position = findMovementPosition(surface, positionFromDirectionAndChunk(attackDirection, groupPosition, attackPosition, 1.35))
if position then
attackPosition.x = position.x
attackPosition.y = position.y
group.set_command(attackCmd)
group.start_moving()
else
addMovementPenalty(natives, squad, attackChunk)
end
end
end
end
end
local function attackMove(map, attackPosition, attackCmd, squad, group, natives, surface)
local groupState = group.state
2019-02-11 08:14:17 +02:00
2018-02-13 09:10:17 +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 x, y = positionToChunkXY(groupPosition)
local chunk = getChunkByXY(map, x, y)
2019-02-16 20:45:42 +02:00
local attackScorer = scoreAttackLocation
if (squad.attackScoreFunction == ATTACK_SCORE_KAMIKAZE) then
attackScorer = scoreAttackKamikazeLocation
end
2018-09-24 06:56:45 +02:00
local attackChunk, attackDirection = scoreNeighborsForAttack(map,
2019-02-11 08:14:17 +02:00
natives,
2018-09-24 06:56:45 +02:00
chunk,
2018-02-13 09:10:17 +02:00
getNeighborChunks(map, x, y),
2019-02-16 20:45:42 +02:00
attackScorer,
2018-02-13 09:10:17 +02:00
squad)
if (chunk ~= SENTINEL_IMPASSABLE_CHUNK) then
addSquadToChunk(map, chunk, squad)
addMovementPenalty(natives, squad, chunk)
elseif (attackChunk ~= SENTINEL_IMPASSABLE_CHUNK) then
addSquadToChunk(map, attackChunk, squad)
addMovementPenalty(natives, squad, attackChunk)
end
if group.valid and (attackChunk ~= SENTINEL_IMPASSABLE_CHUNK) then
local playerBaseGenerator = getPlayerBaseGenerator(map, attackChunk)
if (playerBaseGenerator == 0) or ((groupState == DEFINES_GROUP_FINISHED) or (groupState == DEFINES_GROUP_GATHERING)) then
2019-02-11 08:14:17 +02:00
2018-02-13 09:10:17 +02:00
squad.cycles = ((#squad.group.members > 80) and 6) or 4
local moreFrenzy = not squad.rabid and squad.frenzy and (euclideanDistanceNamed(groupPosition, squad.frenzyPosition) < 100)
squad.frenzy = moreFrenzy
2019-02-11 08:14:17 +02:00
2018-02-13 09:10:17 +02:00
if squad.rabid or squad.frenzy then
attackCmd.distraction = DEFINES_DISTRACTION_BY_ANYTHING
else
attackCmd.distraction = DEFINES_DISTRACTION_BY_ENEMY
end
local position = findMovementPosition(surface, positionFromDirectionAndChunk(attackDirection, groupPosition, attackPosition, 1.35))
if position then
attackPosition.x = position.x
attackPosition.y = position.y
group.set_command(attackCmd)
group.start_moving()
else
addMovementPenalty(natives, squad, attackChunk)
end
elseif not squad.frenzy and not squad.rabid and
((groupState == DEFINES_GROUP_ATTACKING_DISTRACTION) or (groupState == DEFINES_GROUP_ATTACKING_TARGET) or
(playerBaseGenerator ~= 0)) then
squad.frenzy = true
squad.frenzyPosition.x = groupPosition.x
squad.frenzyPosition.y = groupPosition.y
end
end
end
end
function squadAttack.squadsDispatch(map, surface, natives)
2016-08-26 00:20:06 +02:00
local squads = natives.squads
2018-01-14 07:48:21 +02:00
local attackPosition = map.position
local attackCmd = map.attackAreaCommand
2018-02-13 09:10:17 +02:00
local settleCmd = map.settleCommand
2019-02-11 08:14:17 +02:00
2016-08-26 00:20:06 +02:00
for i=1,#squads do
local squad = squads[i]
local group = squad.group
2018-02-13 09:10:17 +02:00
if group and group.valid then
if (squad.status == SQUAD_RAIDING) then
attackMove(map, attackPosition, attackCmd, squad, group, natives, surface)
elseif (squad.status == SQUAD_SETTLING) then
settleMove(map, attackPosition, attackCmd, settleCmd, squad, group, natives, surface)
2016-11-04 01:51:35 +02:00
end
end
end
end
function squadAttack.squadsBeginAttack(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.settlers then
if (squad.status == SQUAD_GUARDING) and group and group.valid then
local kamikazeThreshold = calculateKamikazeThreshold(#squad.group.members, natives)
squad.kamikaze = mRandom() < kamikazeThreshold
squad.status = SQUAD_SETTLING
end
else
if (squad.status == SQUAD_GUARDING) and group and group.valid then
2019-02-11 08:14:17 +02:00
local groupPosition = group.position
2018-05-24 02:25:08 +02:00
if playersWithinProximityToPosition(players, groupPosition, 100, natives) then
squad.frenzy = true
squad.frenzyPosition.x = groupPosition.x
squad.frenzyPosition.y = groupPosition.y
end
2019-02-11 08:14:17 +02:00
local kamikazeThreshold = calculateKamikazeThreshold(#squad.group.members, natives)
if not squad.kamikaze then
squad.kamikaze = (mRandom() < kamikazeThreshold)
end
if squad.kamikaze and (mRandom() < (kamikazeThreshold * 0.75)) then
2019-02-16 20:45:42 +02:00
squad.attackScoreFunction = ATTACK_SCORE_KAMIKAZE
2019-02-11 08:14:17 +02:00
end
squad.status = SQUAD_RAIDING
end
2016-10-15 02:00:18 +02:00
end
end
end
2019-02-16 06:17:30 +02:00
squadAttackG = squadAttack
return squadAttack