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

339 lines
15 KiB
Lua
Raw Normal View History

2019-02-15 20:17:30 -08:00
if (aiAttackWaveG) then
return aiAttackWaveG
end
local aiAttackWave = {}
-- imports
2016-08-18 19:02:13 -07:00
local constants = require("Constants")
local mapUtils = require("MapUtils")
2018-09-23 21:56:45 -07:00
local chunkPropertyUtils = require("ChunkPropertyUtils")
2016-08-18 19:02:13 -07:00
local unitGroupUtils = require("UnitGroupUtils")
2017-11-20 23:27:03 -08:00
local movementUtils = require("MovementUtils")
local mathUtils = require("MathUtils")
2019-02-27 18:53:59 -08:00
local config = require("__Rampant__/config")
-- constants
2016-10-14 17:00:18 -07:00
local BASE_PHEROMONE = constants.BASE_PHEROMONE
local PLAYER_PHEROMONE = constants.PLAYER_PHEROMONE
2016-10-14 17:00:18 -07:00
local MOVEMENT_PHEROMONE = constants.MOVEMENT_PHEROMONE
local RESOURCE_PHEROMONE = constants.RESOURCE_PHEROMONE
2019-04-24 23:13:22 -07:00
local AGGRESSIVE_CAN_ATTACK_WAIT_MAX_DURATION = constants.AGGRESSIVE_CAN_ATTACK_WAIT_MAX_DURATION
local AGGRESSIVE_CAN_ATTACK_WAIT_MIN_DURATION = constants.AGGRESSIVE_CAN_ATTACK_WAIT_MIN_DURATION
2016-10-14 17:00:18 -07:00
local AI_SQUAD_COST = constants.AI_SQUAD_COST
2019-02-19 22:16:43 -08:00
local AI_SETTLER_COST = constants.AI_SETTLER_COST
2018-09-23 21:56:45 -07:00
local AI_MAX_SQUAD_COUNT = constants.AI_MAX_SQUAD_COUNT
2016-11-04 00:26:19 -07:00
local AI_VENGENCE_SQUAD_COST = constants.AI_VENGENCE_SQUAD_COST
2019-04-24 23:13:22 -07:00
local AI_STATE_AGGRESSIVE = constants.AI_STATE_AGGRESSIVE
2016-10-14 17:00:18 -07:00
2018-02-11 19:21:28 -08:00
local INTERVAL_RALLY = constants.INTERVAL_RALLY
2017-05-13 15:32:16 -07:00
2017-06-10 01:38:20 -07:00
local CHUNK_ALL_DIRECTIONS = constants.CHUNK_ALL_DIRECTIONS
2017-11-20 23:27:03 -08:00
local CHUNK_SIZE = constants.CHUNK_SIZE
2017-05-26 17:58:33 -07:00
local RALLY_CRY_DISTANCE = constants.RALLY_CRY_DISTANCE
2018-02-16 19:31:29 -08:00
local SETTLER_DISTANCE = constants.SETTLER_DISTANCE
2017-05-26 17:58:33 -07:00
local RESOURCE_MINIMUM_FORMATION_DELTA = constants.RESOURCE_MINIMUM_FORMATION_DELTA
2019-02-05 22:25:43 -08:00
local AI_STATE_SIEGE = constants.AI_STATE_SIEGE
local AI_STATE_RAIDING = constants.AI_STATE_RAIDING
2017-11-20 23:27:03 -08:00
local SENTINEL_IMPASSABLE_CHUNK = constants.SENTINEL_IMPASSABLE_CHUNK
-- imported functions
2019-04-24 23:13:22 -07:00
local randomTickEvent = mathUtils.randomTickEvent
local mRandom = math.random
2017-06-07 17:57:24 -07:00
local positionFromDirectionAndChunk = mapUtils.positionFromDirectionAndChunk
2018-09-23 21:56:45 -07:00
local getPassable = chunkPropertyUtils.getPassable
local getNestCount = chunkPropertyUtils.getNestCount
local getChunkSettlerTick = chunkPropertyUtils.getChunkSettlerTick
2019-02-19 22:16:43 -08:00
local getRaidNestActiveness = chunkPropertyUtils.getRaidNestActiveness
local getNestActiveness = chunkPropertyUtils.getNestActiveness
2018-09-23 21:56:45 -07:00
local setChunkSettlerTick = chunkPropertyUtils.setChunkSettlerTick
local getRallyTick = chunkPropertyUtils.getRallyTick
local setRallyTick = chunkPropertyUtils.setRallyTick
2017-11-20 23:27:03 -08:00
local gaussianRandomRange = mathUtils.gaussianRandomRange
local getNeighborChunks = mapUtils.getNeighborChunks
local getChunkByXY = mapUtils.getChunkByXY
2017-11-20 23:27:03 -08:00
local scoreNeighborsForFormation = movementUtils.scoreNeighborsForFormation
local scoreNeighborsForResource = movementUtils.scoreNeighborsForResource
local createSquad = unitGroupUtils.createSquad
local attackWaveScaling = config.attackWaveScaling
local settlerWaveScaling = config.settlerWaveScaling
-- module code
2016-08-18 19:02:13 -07:00
2019-02-19 22:16:43 -08:00
local function attackWaveValidCandidate(chunk, natives, map)
local isValid = getNestActiveness(map, chunk)
if natives.state == AI_STATE_RAIDING then
isValid = isValid + getRaidNestActiveness(map, chunk)
2017-06-10 01:38:20 -07:00
end
2019-02-19 22:16:43 -08:00
return (isValid > 0)
end
local function scoreSettlerLocation(neighborChunk)
return neighborChunk[RESOURCE_PHEROMONE] +
neighborChunk[MOVEMENT_PHEROMONE] +
-neighborChunk[PLAYER_PHEROMONE]
end
2019-02-05 22:25:43 -08:00
local function scoreSiegeSettlerLocation(neighborChunk)
return neighborChunk[RESOURCE_PHEROMONE] +
neighborChunk[BASE_PHEROMONE] +
neighborChunk[MOVEMENT_PHEROMONE] +
-neighborChunk[PLAYER_PHEROMONE]
2019-02-05 22:25:43 -08:00
end
local function scoreUnitGroupLocation(neighborChunk)
return neighborChunk[PLAYER_PHEROMONE] +
neighborChunk[MOVEMENT_PHEROMONE] +
neighborChunk[BASE_PHEROMONE]
2019-02-05 22:25:43 -08:00
end
local function validSiegeSettlerLocation(map, neighborChunk)
return (getPassable(map, neighborChunk) == CHUNK_ALL_DIRECTIONS) and
(getNestCount(map, neighborChunk) == 0)
end
2019-02-05 22:25:43 -08:00
local function validSettlerLocation(map, chunk, neighborChunk)
local chunkResource = chunk[RESOURCE_PHEROMONE]
return (getPassable(map, neighborChunk) == CHUNK_ALL_DIRECTIONS) and
(getNestCount(map, neighborChunk) == 0) and
(neighborChunk[RESOURCE_PHEROMONE] >= (chunkResource * RESOURCE_MINIMUM_FORMATION_DELTA))
end
2018-01-13 21:48:21 -08:00
local function validUnitGroupLocation(map, neighborChunk)
return getPassable(map, neighborChunk) == CHUNK_ALL_DIRECTIONS and
(getNestCount(map, neighborChunk) == 0)
end
2019-11-29 16:49:22 -08:00
function aiAttackWave.rallyUnits(chunk, map, surface, tick)
if ((tick - getRallyTick(map, chunk) > INTERVAL_RALLY) and (map.natives.points >= AI_VENGENCE_SQUAD_COST)) then
setRallyTick(map, chunk, tick)
local cX = chunk.x
local cY = chunk.y
for x=cX - RALLY_CRY_DISTANCE, cX + RALLY_CRY_DISTANCE, 32 do
for y=cY - RALLY_CRY_DISTANCE, cY + RALLY_CRY_DISTANCE, 32 do
if (x ~= cX) and (y ~= cY) then
local rallyChunk = getChunkByXY(map, x, y)
if (rallyChunk ~= SENTINEL_IMPASSABLE_CHUNK) and (getNestCount(map, rallyChunk) > 0) then
2019-11-29 16:49:22 -08:00
if not aiAttackWave.formVengenceSquad(map, surface, rallyChunk) then
2019-12-06 21:57:20 -08:00
return false
end
end
end
end
end
2019-12-06 21:57:20 -08:00
return true
end
2017-01-19 21:58:36 -08:00
end
2019-12-06 21:57:20 -08:00
function aiAttackWave.formAttackWave(chunk, map, surface, tick)
if (map.natives.points >= AI_SQUAD_COST) then
setRallyTick(map, chunk, tick)
local cX = chunk.x
local cY = chunk.y
for x=cX - RALLY_CRY_DISTANCE, cX + RALLY_CRY_DISTANCE, 32 do
for y=cY - RALLY_CRY_DISTANCE, cY + RALLY_CRY_DISTANCE, 32 do
if (x ~= cX) and (y ~= cY) then
local rallyChunk = getChunkByXY(map, x, y)
if (rallyChunk ~= SENTINEL_IMPASSABLE_CHUNK) and (getNestCount(map, rallyChunk) > 0) then
if not aiAttackWave.formSquads(map, surface, rallyChunk, tick) then
return false
end
end
end
end
end
return true
end
return false
end
2018-02-16 19:31:29 -08:00
local function noNearbySettlers(map, chunk, tick)
local cX = chunk.x
local cY = chunk.y
for x=cX - SETTLER_DISTANCE, cX + SETTLER_DISTANCE, 32 do
for y=cY - SETTLER_DISTANCE, cY + SETTLER_DISTANCE, 32 do
if (x ~= cX) and (y ~= cY) then
local c = getChunkByXY(map, x, y)
if (c ~= SENTINEL_IMPASSABLE_CHUNK) and ((tick - getChunkSettlerTick(map, c)) < 0) then
return false
end
end
end
2018-02-16 19:31:29 -08:00
end
return true
end
2019-11-29 16:49:22 -08:00
function aiAttackWave.formSettlers(map, surface, chunk, tick)
local natives = map.natives
2019-10-19 12:13:48 -07:00
if (mRandom() < natives.formSquadThreshold) and ((natives.squads.len + #natives.building) < AI_MAX_SQUAD_COUNT) then
2019-02-05 22:25:43 -08:00
local squadPath, squadDirection
if (natives.state == AI_STATE_SIEGE) then
squadPath, squadDirection = scoreNeighborsForFormation(getNeighborChunks(map, chunk.x, chunk.y),
validSiegeSettlerLocation,
scoreSiegeSettlerLocation,
map)
else
squadPath, squadDirection = scoreNeighborsForResource(chunk,
getNeighborChunks(map, chunk.x, chunk.y),
validSettlerLocation,
scoreSettlerLocation,
map)
end
if (squadPath ~= SENTINEL_IMPASSABLE_CHUNK) and noNearbySettlers(map, chunk, tick) then
local squadPosition = surface.find_non_colliding_position("chunk-scanner-squad-rampant",
positionFromDirectionAndChunk(squadDirection,
chunk,
map.position,
0.98),
CHUNK_SIZE,
4,
2019-03-05 22:18:03 -08:00
true)
if squadPosition then
local squad = createSquad(squadPosition, surface, nil, true)
squad.maxDistance = gaussianRandomRange(natives.expansionMaxDistance * 0.5,
natives.expansionMaxDistanceDerivation,
10,
natives.expansionMaxDistance)
2019-02-05 22:25:43 -08:00
local scaledWaveSize = settlerWaveScaling(natives)
map.formGroupCommand.group = squad.group
map.formCommand.unit_count = scaledWaveSize
local foundUnits = surface.set_multi_command(map.formCommand)
if (foundUnits > 0) then
setChunkSettlerTick(map, squadPath, tick + natives.settlerCooldown)
2019-10-19 12:13:48 -07:00
local pending = natives.pendingAttack
pending.len = pending.len + 1
2019-10-20 13:45:43 -07:00
squad.cycles = 30
2019-10-19 12:13:48 -07:00
pending[pending.len] = squad
natives.points = natives.points - AI_SETTLER_COST
2019-03-06 22:12:39 -08:00
else
if (squad.group.valid) then
squad.group.destroy()
end
end
end
end
end
2019-02-05 22:25:43 -08:00
2019-02-19 22:16:43 -08:00
return (natives.points - AI_SETTLER_COST) > 0
end
2019-11-29 16:49:22 -08:00
function aiAttackWave.formVengenceSquad(map, surface, chunk)
local natives = map.natives
2019-10-19 12:13:48 -07:00
if (mRandom() < natives.formSquadThreshold) and (natives.squads.len < AI_MAX_SQUAD_COUNT)
2019-02-19 22:16:43 -08:00
then
local squadPath, squadDirection = scoreNeighborsForFormation(getNeighborChunks(map, chunk.x, chunk.y),
validUnitGroupLocation,
scoreUnitGroupLocation,
map)
if (squadPath ~= SENTINEL_IMPASSABLE_CHUNK) then
local squadPosition = surface.find_non_colliding_position("chunk-scanner-squad-rampant",
positionFromDirectionAndChunk(squadDirection,
chunk,
map.position,
0.98),
CHUNK_SIZE,
4,
2019-03-05 22:18:03 -08:00
true)
if squadPosition then
local squad = createSquad(squadPosition, surface)
2019-02-05 22:25:43 -08:00
squad.rabid = mRandom() < 0.03
2019-02-19 22:16:43 -08:00
local scaledWaveSize = attackWaveScaling(natives)
map.formGroupCommand.group = squad.group
map.formCommand.unit_count = scaledWaveSize
local foundUnits = surface.set_multi_command(map.formCommand)
if (foundUnits > 0) then
2019-10-19 12:13:48 -07:00
local pending = natives.pendingAttack
pending.len = pending.len + 1
2019-10-20 13:45:43 -07:00
squad.cycles = 13
2019-10-19 12:13:48 -07:00
pending[pending.len] = squad
natives.points = natives.points - AI_VENGENCE_SQUAD_COST
2019-03-06 22:12:39 -08:00
else
if (squad.group.valid) then
squad.group.destroy()
end
end
end
end
2019-02-19 22:16:43 -08:00
end
return (natives.points - AI_VENGENCE_SQUAD_COST) > 0
end
2019-11-29 16:49:22 -08:00
function aiAttackWave.formSquads(map, surface, chunk, tick)
local natives = map.natives
2019-02-19 22:16:43 -08:00
if attackWaveValidCandidate(chunk, natives, map) and
(mRandom() < natives.formSquadThreshold) and
2019-10-19 12:13:48 -07:00
(natives.squads.len < AI_MAX_SQUAD_COUNT)
2019-02-19 22:16:43 -08:00
then
local squadPath, squadDirection = scoreNeighborsForFormation(getNeighborChunks(map, chunk.x, chunk.y),
validUnitGroupLocation,
scoreUnitGroupLocation,
map)
if (squadPath ~= SENTINEL_IMPASSABLE_CHUNK) then
local squadPosition = surface.find_non_colliding_position("chunk-scanner-squad-rampant",
positionFromDirectionAndChunk(squadDirection,
chunk,
map.position,
0.98),
CHUNK_SIZE,
4,
2019-03-05 22:18:03 -08:00
true)
if squadPosition then
local squad = createSquad(squadPosition, surface)
2019-02-05 22:25:43 -08:00
squad.rabid = mRandom() < 0.03
2017-06-08 22:18:59 -07:00
local scaledWaveSize = attackWaveScaling(natives)
map.formGroupCommand.group = squad.group
map.formCommand.unit_count = scaledWaveSize
local foundUnits = surface.set_multi_command(map.formCommand)
if (foundUnits > 0) then
2019-10-19 12:13:48 -07:00
local pending = natives.pendingAttack
pending.len = pending.len + 1
2019-10-20 13:45:43 -07:00
squad.cycles = 30
2019-10-19 12:13:48 -07:00
pending[pending.len] = squad
natives.points = natives.points - AI_SQUAD_COST
2019-04-24 23:13:22 -07:00
if tick and (natives.state == AI_STATE_AGGRESSIVE) then
natives.canAttackTick = randomTickEvent(tick,
AGGRESSIVE_CAN_ATTACK_WAIT_MIN_DURATION,
AGGRESSIVE_CAN_ATTACK_WAIT_MAX_DURATION)
return false
end
2019-03-06 22:12:39 -08:00
else
if (squad.group.valid) then
squad.group.destroy()
end
end
2019-03-06 22:12:39 -08:00
end
end
2016-08-18 19:02:13 -07:00
end
2019-10-20 13:45:43 -07:00
2019-02-19 22:16:43 -08:00
return (natives.points - AI_SQUAD_COST) > 0
2016-08-18 19:02:13 -07:00
end
2019-02-19 22:16:43 -08:00
2019-02-15 20:17:30 -08:00
aiAttackWaveG = aiAttackWave
return aiAttackWave