mirror of
https://github.com/ComfyFactory/ComfyFactorio.git
synced 2025-01-24 03:47:58 +02:00
210 lines
5.9 KiB
Lua
210 lines
5.9 KiB
Lua
|
local Public = {}
|
||
|
local math_random = math.random
|
||
|
local math_floor = math.floor
|
||
|
local math_sqrt = math.sqrt
|
||
|
local math_round = math.round
|
||
|
local table_size = table.size
|
||
|
local table_insert = table.insert
|
||
|
local table_remove = table.remove
|
||
|
local table_shuffle = table.shuffle_table
|
||
|
|
||
|
local Global = require 'utils.global'
|
||
|
|
||
|
local tick_schedule = {}
|
||
|
Global.register(
|
||
|
tick_schedule,
|
||
|
function(t)
|
||
|
tick_schedule = t
|
||
|
end
|
||
|
)
|
||
|
|
||
|
local Table = require 'modules.scrap_towny_ffa.table'
|
||
|
local Evolution = require "modules.scrap_towny_ffa.evolution"
|
||
|
|
||
|
local function get_commmands(target, group)
|
||
|
local commands = {}
|
||
|
local group_position = { x = group.position.x, y = group.position.y }
|
||
|
local step_length = 128
|
||
|
|
||
|
local target_position = target.position
|
||
|
local distance_to_target = math_floor(math_sqrt((target_position.x - group_position.x) ^ 2 + (target_position.y - group_position.y) ^ 2))
|
||
|
local steps = math_floor(distance_to_target / step_length) + 1
|
||
|
local vector = { math_round((target_position.x - group_position.x) / steps, 3), math_round((target_position.y - group_position.y) / steps, 3) }
|
||
|
|
||
|
for _ = 1, steps, 1 do
|
||
|
group_position.x = group_position.x + vector[1]
|
||
|
group_position.y = group_position.y + vector[2]
|
||
|
local position = group.surface.find_non_colliding_position("small-biter", group_position, step_length, 2)
|
||
|
if position then
|
||
|
commands[#commands + 1] = {
|
||
|
type = defines.command.attack_area,
|
||
|
destination = { x = position.x, y = position.y },
|
||
|
radius = 16,
|
||
|
distraction = defines.distraction.by_damage
|
||
|
}
|
||
|
end
|
||
|
end
|
||
|
|
||
|
commands[#commands + 1] = {
|
||
|
type = defines.command.attack_area,
|
||
|
destination = target.position,
|
||
|
radius = 12,
|
||
|
distraction = defines.distraction.by_enemy,
|
||
|
}
|
||
|
commands[#commands + 1] = {
|
||
|
type = defines.command.attack,
|
||
|
target = target,
|
||
|
distraction = defines.distraction.by_anything,
|
||
|
}
|
||
|
|
||
|
return commands
|
||
|
end
|
||
|
|
||
|
local function roll_market()
|
||
|
local ffatable = Table.get_table()
|
||
|
local town_centers = ffatable.town_centers
|
||
|
if town_centers == nil or table_size(town_centers) == 0 then return end
|
||
|
local keyset = {}
|
||
|
for town_name, _ in pairs(town_centers) do
|
||
|
table_insert(keyset, town_name)
|
||
|
end
|
||
|
local tc = math_random(1, #keyset)
|
||
|
return town_centers[keyset[tc]]
|
||
|
end
|
||
|
|
||
|
local function get_random_close_spawner(surface, market, radius)
|
||
|
local units = surface.find_enemy_units(market.position, radius, market.force)
|
||
|
if units ~= nil and #units > 0 then
|
||
|
-- found units, shuffle the list
|
||
|
table_shuffle(units)
|
||
|
while units[1] do
|
||
|
local unit = units[1]
|
||
|
if unit.spawner then return unit.spawner end
|
||
|
table_remove(units, 1)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
local function is_swarm_valid(swarm)
|
||
|
local group = swarm.group
|
||
|
if not group then return end
|
||
|
if not group.valid then return end
|
||
|
if game.tick >= swarm.timeout then
|
||
|
group.destroy()
|
||
|
return
|
||
|
end
|
||
|
return true
|
||
|
end
|
||
|
|
||
|
function Public.validate_swarms()
|
||
|
local ffatable = Table.get_table()
|
||
|
for k, swarm in pairs(ffatable.swarms) do
|
||
|
if not is_swarm_valid(swarm) then
|
||
|
table_remove(ffatable.swarms, k)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
function Public.unit_groups_start_moving()
|
||
|
local ffatable = Table.get_table()
|
||
|
for _, swarm in pairs(ffatable.swarms) do
|
||
|
if swarm.group then
|
||
|
if swarm.group.valid then
|
||
|
swarm.group.start_moving()
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
function Public.swarm(town_center, radius)
|
||
|
if town_center == nil then return end
|
||
|
local ffatable = Table.get_table()
|
||
|
local r = radius or 32
|
||
|
local tc = town_center or roll_market()
|
||
|
if not tc or r > 512 then return end
|
||
|
|
||
|
-- skip if town evolution < 0.25
|
||
|
if town_center.get_biter_evolution < 0.25 then return end
|
||
|
|
||
|
-- skip if we have to many swarms already
|
||
|
local count = table_size(ffatable.swarms)
|
||
|
local towns = table_size(ffatable.town_centers)
|
||
|
if count > 3 * towns then return end
|
||
|
|
||
|
local market = tc.market
|
||
|
local surface = market.surface
|
||
|
|
||
|
-- find a spawner
|
||
|
local spawner = get_random_close_spawner(surface, market, r)
|
||
|
if not spawner then
|
||
|
r = r + 16
|
||
|
local future = game.tick + 1
|
||
|
-- schedule to run this method again with a higher radius on next tick
|
||
|
if not tick_schedule[future] then tick_schedule[future] = {} end
|
||
|
tick_schedule[future][#tick_schedule[future] + 1] = {
|
||
|
callback = 'swarm',
|
||
|
params = { tc, r }
|
||
|
}
|
||
|
return
|
||
|
end
|
||
|
|
||
|
-- get our evolution
|
||
|
local evolution = 0
|
||
|
if spawner.name == "spitter-spawner" then
|
||
|
evolution = Evolution.get_biter_evolution(spawner)
|
||
|
else
|
||
|
evolution = Evolution.get_spitter_evolution(spawner)
|
||
|
end
|
||
|
|
||
|
-- get our target amount of enemies
|
||
|
local count2 = (evolution * 124) + 4
|
||
|
|
||
|
local units = spawner.surface.find_enemy_units(spawner.position, 16, market.force)
|
||
|
if #units < count2 then
|
||
|
units = spawner.surface.find_enemy_units(spawner.position, 32, market.force)
|
||
|
end
|
||
|
if #units < count2 then
|
||
|
units = spawner.surface.find_enemy_units(spawner.position, 64, market.force)
|
||
|
end
|
||
|
if #units < count2 then
|
||
|
units = spawner.surface.find_enemy_units(spawner.position, 128, market.force)
|
||
|
end
|
||
|
if not units[1] then return end
|
||
|
|
||
|
local unit_group_position = surface.find_non_colliding_position("biter-spawner", units[1].position, 256, 1)
|
||
|
if not unit_group_position then return end
|
||
|
local unit_group = surface.create_unit_group({ position = unit_group_position, force = units[1].force })
|
||
|
|
||
|
for key, unit in pairs(units) do
|
||
|
if key > count2 then break end
|
||
|
unit_group.add_member(unit)
|
||
|
end
|
||
|
|
||
|
unit_group.set_command({
|
||
|
type = defines.command.compound,
|
||
|
structure_type = defines.compound_command.return_last,
|
||
|
commands = get_commmands(market, unit_group)
|
||
|
})
|
||
|
table_insert(ffatable.swarms, { group = unit_group, timeout = game.tick + 36000 })
|
||
|
end
|
||
|
|
||
|
local function on_tick()
|
||
|
if not tick_schedule[game.tick] then return end
|
||
|
for _, token in pairs(tick_schedule[game.tick]) do
|
||
|
local callback = token.callback
|
||
|
local params = token.params
|
||
|
if callback == 'swarm' then Public.swarm(params[1], params[2]) end
|
||
|
end
|
||
|
tick_schedule[game.tick] = nil
|
||
|
end
|
||
|
|
||
|
local on_init = function ()
|
||
|
local ffatable = Table.get_table()
|
||
|
ffatable.swarms = {}
|
||
|
end
|
||
|
|
||
|
local Event = require 'utils.event'
|
||
|
Event.on_init(on_init)
|
||
|
Event.add(defines.events.on_tick, on_tick)
|
||
|
|
||
|
return Public
|