2016-08-05 06:47:51 +02:00
|
|
|
local unitGroupUtils = {}
|
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- imports
|
|
|
|
|
|
|
|
local mapUtils = require("MapUtils")
|
2016-08-07 05:38:47 +02:00
|
|
|
local constants = require("Constants")
|
2016-08-05 06:47:51 +02:00
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- constants
|
|
|
|
|
2016-10-15 02:00:18 +02:00
|
|
|
local MOVEMENT_PHEROMONE_GENERATOR_AMOUNT = constants.MOVEMENT_PHEROMONE_GENERATOR_AMOUNT
|
2016-08-20 04:52:27 +02:00
|
|
|
|
2017-06-01 04:48:59 +02:00
|
|
|
local SQUAD_QUEUE_SIZE = constants.SQUAD_QUEUE_SIZE
|
|
|
|
|
2017-05-27 02:58:33 +02:00
|
|
|
local DEFINES_GROUP_STATE_FINISHED = defines.group_state.finished
|
|
|
|
local DEFINES_GROUP_STATE_ATTACKING_TARGET = defines.group_state.attacking_target
|
|
|
|
local DEFINES_GROUP_STATE_ATTACKING_DISTRACTION = defines.group_state.attacking_distraction
|
2017-05-28 06:50:37 +02:00
|
|
|
local DEFINES_COMMAND_GROUP = defines.command.group
|
|
|
|
local DEFINES_DISTRACTION_NONE = defines.distraction.none
|
2016-08-18 07:55:08 +02:00
|
|
|
|
2017-06-10 10:38:20 +02:00
|
|
|
local MAX_PENALTY_BEFORE_PURGE = constants.MAX_PENALTY_BEFORE_PURGE
|
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
local SQUAD_RETREATING = constants.SQUAD_RETREATING
|
|
|
|
local SQUAD_GUARDING = constants.SQUAD_GUARDING
|
2017-01-20 07:58:36 +02:00
|
|
|
local GROUP_MERGE_DISTANCE = constants.GROUP_MERGE_DISTANCE
|
|
|
|
|
|
|
|
local NO_RETREAT_SQUAD_SIZE_BONUS_MAX = constants.NO_RETREAT_SQUAD_SIZE_BONUS_MAX
|
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
local AI_MAX_OVERFLOW_POINTS = constants.AI_MAX_OVERFLOW_POINTS
|
2017-05-24 08:46:23 +02:00
|
|
|
|
2017-04-16 08:04:22 +02:00
|
|
|
local AI_MAX_BITER_GROUP_SIZE = constants.AI_MAX_BITER_GROUP_SIZE
|
2017-06-01 09:03:07 +02:00
|
|
|
local AI_SQUAD_MERGE_THRESHOLD = constants.AI_SQUAD_MERGE_THRESHOLD
|
2017-04-16 08:04:22 +02:00
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
-- imported functions
|
|
|
|
|
2017-01-20 07:58:36 +02:00
|
|
|
local mLog = math.log10
|
|
|
|
|
2017-06-01 04:48:59 +02:00
|
|
|
local mMin = math.min
|
|
|
|
|
2016-08-20 04:52:27 +02:00
|
|
|
local tableRemove = table.remove
|
|
|
|
local tableInsert = table.insert
|
|
|
|
local euclideanDistanceNamed = mapUtils.euclideanDistanceNamed
|
|
|
|
|
|
|
|
-- module code
|
2017-01-20 07:58:36 +02:00
|
|
|
|
2016-08-08 03:35:36 +02:00
|
|
|
function unitGroupUtils.findNearBySquad(natives, position, distance, filter)
|
2016-08-05 06:47:51 +02:00
|
|
|
local squads = natives.squads
|
2017-06-01 03:46:53 +02:00
|
|
|
|
2016-08-26 00:20:06 +02:00
|
|
|
for i=1,#squads do
|
2016-08-05 06:47:51 +02:00
|
|
|
local squad = squads[i]
|
2016-08-07 05:38:47 +02:00
|
|
|
local unitGroup = squad.group
|
2017-05-28 06:50:37 +02:00
|
|
|
if unitGroup.valid and (not filter or (filter and filter[squad.status])) then
|
2016-08-20 04:52:27 +02:00
|
|
|
if (euclideanDistanceNamed(unitGroup.position, position) <= distance) then
|
2016-08-05 06:47:51 +02:00
|
|
|
return squad
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-08-07 05:38:47 +02:00
|
|
|
function unitGroupUtils.createSquad(position, surface, natives)
|
|
|
|
local unitGroup = surface.create_unit_group({position=position})
|
|
|
|
|
|
|
|
local squad = { group = unitGroup,
|
2016-08-20 04:52:27 +02:00
|
|
|
status = SQUAD_GUARDING,
|
|
|
|
penalties = {},
|
2016-09-14 19:12:29 +02:00
|
|
|
rabid = false,
|
|
|
|
frenzy = false,
|
2016-11-04 01:51:35 +02:00
|
|
|
kamikaze = false,
|
2016-09-14 19:12:29 +02:00
|
|
|
frenzyPosition = {x = 0,
|
|
|
|
y = 0},
|
2016-08-18 07:55:08 +02:00
|
|
|
cycles = 0 }
|
2016-08-07 05:38:47 +02:00
|
|
|
natives.squads[#natives.squads+1] = squad
|
|
|
|
return squad
|
|
|
|
end
|
|
|
|
|
2016-09-13 01:23:26 +02:00
|
|
|
function unitGroupUtils.membersToSquad(squad, members, overwriteGroup)
|
2016-08-08 03:35:36 +02:00
|
|
|
if (members ~= nil) then
|
2017-05-28 06:50:37 +02:00
|
|
|
local cmd = { type = DEFINES_COMMAND_GROUP,
|
2016-09-14 19:12:29 +02:00
|
|
|
group = squad.group,
|
2017-05-28 06:50:37 +02:00
|
|
|
distraction = DEFINES_DISTRACTION_NONE }
|
2016-09-13 01:23:26 +02:00
|
|
|
for i=1,#members do
|
2016-08-07 05:38:47 +02:00
|
|
|
local member = members[i]
|
2017-05-28 06:50:37 +02:00
|
|
|
if member.valid and (overwriteGroup or (not overwriteGroup and not member.unit_group)) then
|
2016-09-14 19:12:29 +02:00
|
|
|
member.set_command(cmd)
|
2016-08-07 05:38:47 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
function unitGroupUtils.convertUnitGroupToSquad(natives, unitGroup)
|
2017-06-01 03:46:53 +02:00
|
|
|
if not unitGroup then
|
|
|
|
return nil
|
2016-08-07 05:38:47 +02:00
|
|
|
end
|
2017-06-01 03:46:53 +02:00
|
|
|
local squads = natives.squads
|
|
|
|
for i=1,#squads do
|
|
|
|
local squad = squads[i]
|
|
|
|
if (squad.group == unitGroup) then
|
|
|
|
return squad
|
|
|
|
end
|
|
|
|
end
|
|
|
|
local returnSquad = { group = unitGroup,
|
|
|
|
status = SQUAD_GUARDING,
|
|
|
|
penalties = {},
|
|
|
|
rabid = false,
|
|
|
|
frenzy = false,
|
|
|
|
kamikaze = false,
|
|
|
|
frenzyPosition = {x = 0,
|
|
|
|
y = 0},
|
|
|
|
cycles = 0 }
|
|
|
|
squads[#squads+1] = returnSquad
|
2016-08-07 05:38:47 +02:00
|
|
|
return returnSquad
|
|
|
|
end
|
|
|
|
|
2017-06-10 10:38:20 +02:00
|
|
|
function unitGroupUtils.addSquadMovementPenalty(natives, squad, chunkX, chunkY)
|
2016-08-20 04:52:27 +02:00
|
|
|
local penalties = squad.penalties
|
2016-08-26 00:20:06 +02:00
|
|
|
for i=1,#penalties do
|
|
|
|
local penalty = penalties[i]
|
2016-08-20 04:52:27 +02:00
|
|
|
if (penalty.x == chunkX) and (penalty.y == chunkY) then
|
2016-10-15 02:00:18 +02:00
|
|
|
penalty.v = penalty.v + MOVEMENT_PHEROMONE_GENERATOR_AMOUNT
|
2017-06-10 10:38:20 +02:00
|
|
|
if (penalty.v > MAX_PENALTY_BEFORE_PURGE) then
|
|
|
|
local group = squad.group
|
|
|
|
local members = group.members
|
|
|
|
for x=1,#members do
|
|
|
|
members[x].destroy()
|
|
|
|
end
|
|
|
|
natives.points = natives.points + (#members * natives.unitRefundAmount)
|
|
|
|
|
|
|
|
if (natives.points > AI_MAX_OVERFLOW_POINTS) then
|
|
|
|
natives.points = AI_MAX_OVERFLOW_POINTS
|
|
|
|
end
|
|
|
|
|
|
|
|
group.destroy()
|
|
|
|
end
|
2016-08-20 04:52:27 +02:00
|
|
|
return
|
|
|
|
end
|
|
|
|
end
|
2017-06-10 10:38:20 +02:00
|
|
|
if (#penalties == 7) then
|
|
|
|
tableRemove(penalties, 7)
|
2016-08-20 04:52:27 +02:00
|
|
|
end
|
2016-10-15 02:00:18 +02:00
|
|
|
tableInsert(penalties, 1, { v = MOVEMENT_PHEROMONE_GENERATOR_AMOUNT,
|
2016-08-20 04:52:27 +02:00
|
|
|
x = chunkX,
|
|
|
|
y = chunkY })
|
|
|
|
end
|
|
|
|
|
|
|
|
function unitGroupUtils.lookupSquadMovementPenalty(squad, chunkX, chunkY)
|
|
|
|
local penalties = squad.penalties
|
2016-08-26 00:20:06 +02:00
|
|
|
for i=1,#penalties do
|
|
|
|
local penalty = penalties[i]
|
2016-08-20 04:52:27 +02:00
|
|
|
if (penalty.x == chunkX) and (penalty.y == chunkY) then
|
|
|
|
return penalty.v
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return 0
|
|
|
|
end
|
2016-08-07 05:38:47 +02:00
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
function unitGroupUtils.calculateKamikazeThreshold(squad, natives)
|
|
|
|
local squadSizeBonus = mLog((#squad.group.members / natives.attackWaveMaxSize) + 0.1) + 1
|
|
|
|
return natives.kamikazeThreshold + (NO_RETREAT_SQUAD_SIZE_BONUS_MAX * squadSizeBonus)
|
2017-01-20 07:58:36 +02:00
|
|
|
end
|
|
|
|
|
2017-04-22 01:14:04 +02:00
|
|
|
local function isAttacking(group)
|
|
|
|
local state = group.state
|
2017-05-27 02:58:33 +02:00
|
|
|
return (state == DEFINES_GROUP_STATE_ATTACKING_TARGET) or (state == DEFINES_GROUP_STATE_ATTACKING_DISTRACTION)
|
2017-01-20 07:58:36 +02:00
|
|
|
end
|
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
function unitGroupUtils.cleanSquads(natives)
|
2016-08-09 04:18:19 +02:00
|
|
|
local squads = natives.squads
|
2017-04-22 01:14:04 +02:00
|
|
|
local squadCount = #squads
|
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
local weight = natives.unitRefundAmount
|
2017-05-24 08:46:23 +02:00
|
|
|
|
|
|
|
local cleanSquads = {}
|
2017-06-01 09:03:07 +02:00
|
|
|
|
|
|
|
for i=1, squadCount do
|
2017-04-22 01:14:04 +02:00
|
|
|
local squad = squads[i]
|
|
|
|
local group = squad.group
|
|
|
|
if group.valid then
|
|
|
|
local memberCount = #group.members
|
2017-05-24 08:46:23 +02:00
|
|
|
if (memberCount == 0) then
|
|
|
|
group.destroy()
|
|
|
|
elseif (memberCount > AI_MAX_BITER_GROUP_SIZE) then
|
|
|
|
local members = group.members
|
|
|
|
for x=1,memberCount do
|
|
|
|
members[x].destroy()
|
|
|
|
end
|
2017-06-01 03:46:53 +02:00
|
|
|
natives.points = natives.points + (memberCount * weight)
|
2017-05-24 08:46:23 +02:00
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
if (natives.points > AI_MAX_OVERFLOW_POINTS) then
|
|
|
|
natives.points = AI_MAX_OVERFLOW_POINTS
|
2017-05-24 08:46:23 +02:00
|
|
|
end
|
2017-04-22 01:14:04 +02:00
|
|
|
group.destroy()
|
2017-05-24 08:46:23 +02:00
|
|
|
else
|
|
|
|
local status = squad.status
|
|
|
|
local cycles = squad.cycles
|
|
|
|
if (status == SQUAD_RETREATING) and (cycles == 0) then
|
|
|
|
squad.status = SQUAD_GUARDING
|
|
|
|
squad.frenzy = true
|
2017-06-01 04:48:59 +02:00
|
|
|
local squadPosition = group.position
|
2017-05-24 08:46:23 +02:00
|
|
|
squad.frenzyPosition.x = squadPosition.x
|
|
|
|
squad.frenzyPosition.y = squadPosition.y
|
2017-05-27 02:58:33 +02:00
|
|
|
elseif (group.state == DEFINES_GROUP_STATE_FINISHED) then
|
2017-05-24 08:46:23 +02:00
|
|
|
squad.status = SQUAD_GUARDING
|
|
|
|
elseif (cycles > 0) then
|
|
|
|
squad.cycles = cycles - 1
|
|
|
|
end
|
|
|
|
|
|
|
|
cleanSquads[#cleanSquads+1] = squad
|
2017-01-20 07:58:36 +02:00
|
|
|
end
|
|
|
|
end
|
2016-08-09 04:18:19 +02:00
|
|
|
end
|
2017-05-24 08:46:23 +02:00
|
|
|
|
|
|
|
natives.squads = cleanSquads
|
|
|
|
end
|
|
|
|
|
2017-01-20 07:58:36 +02:00
|
|
|
|
2017-06-01 03:46:53 +02:00
|
|
|
function unitGroupUtils.regroupSquads(natives)
|
2017-06-01 09:03:07 +02:00
|
|
|
local groupThreshold = AI_SQUAD_MERGE_THRESHOLD
|
2017-05-24 08:46:23 +02:00
|
|
|
|
|
|
|
local squads = natives.squads
|
|
|
|
local squadCount = #squads
|
2017-06-01 04:48:59 +02:00
|
|
|
|
|
|
|
local startIndex = natives.regroupIndex
|
2017-04-22 01:14:04 +02:00
|
|
|
|
2017-06-01 04:48:59 +02:00
|
|
|
local maxSquadIndex = mMin(startIndex + SQUAD_QUEUE_SIZE, squadCount)
|
|
|
|
for i=startIndex,maxSquadIndex do
|
2017-01-20 07:58:36 +02:00
|
|
|
local squad = squads[i]
|
2017-04-22 01:14:04 +02:00
|
|
|
local group = squad.group
|
2017-05-24 08:46:23 +02:00
|
|
|
if group.valid and not isAttacking(group) then
|
2017-04-22 01:14:04 +02:00
|
|
|
local status = squad.status
|
2017-06-01 04:48:59 +02:00
|
|
|
local memberCount = #group.members
|
2017-05-24 08:46:23 +02:00
|
|
|
if (memberCount < groupThreshold) then
|
|
|
|
local squadPosition = group.position
|
2017-04-22 01:14:04 +02:00
|
|
|
local mergedSquads = false
|
|
|
|
for x=i+1,squadCount do
|
2017-05-24 08:46:23 +02:00
|
|
|
local mergeSquad = squads[x]
|
|
|
|
local mergeGroup = mergeSquad.group
|
2017-06-01 04:48:59 +02:00
|
|
|
if mergeGroup.valid and (euclideanDistanceNamed(squadPosition, mergeGroup.position) < GROUP_MERGE_DISTANCE) and (mergeSquad.status == status) and not isAttacking(mergeGroup) then
|
2017-05-24 08:46:23 +02:00
|
|
|
local mergeMembers = mergeGroup.members
|
|
|
|
local mergeCount = #mergeMembers
|
|
|
|
if ((mergeCount + memberCount) < AI_MAX_BITER_GROUP_SIZE) then
|
2017-04-22 01:14:04 +02:00
|
|
|
for memberIndex=1, mergeCount do
|
|
|
|
group.add_member(mergeMembers[memberIndex])
|
|
|
|
end
|
|
|
|
if mergeSquad.kamikaze then
|
|
|
|
squad.kamikaze = true
|
|
|
|
end
|
|
|
|
mergedSquads = true
|
|
|
|
mergeGroup.destroy()
|
2017-05-24 08:46:23 +02:00
|
|
|
end
|
|
|
|
memberCount = memberCount + mergeCount
|
|
|
|
if (memberCount > groupThreshold) then
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
2017-04-22 01:14:04 +02:00
|
|
|
end
|
|
|
|
if mergedSquads and not squad.kamikaze then
|
2017-06-01 03:46:53 +02:00
|
|
|
local kamikazeThreshold = unitGroupUtils.calculateKamikazeThreshold(squad, natives)
|
2017-05-24 08:46:23 +02:00
|
|
|
if (math.random() < kamikazeThreshold) then
|
|
|
|
squad.kamikaze = true
|
|
|
|
end
|
2017-04-22 01:14:04 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2017-06-01 04:48:59 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
if (maxSquadIndex == squadCount) then
|
|
|
|
natives.regroupIndex = 1
|
|
|
|
else
|
|
|
|
natives.regroupIndex = maxSquadIndex
|
|
|
|
end
|
2016-08-05 06:47:51 +02:00
|
|
|
end
|
|
|
|
|
2016-08-30 06:08:22 +02:00
|
|
|
return unitGroupUtils
|