2019-12-19 18:26:02 +01:00
|
|
|
local Public = {}
|
|
|
|
local math_random = math.random
|
2019-12-20 23:10:03 +01:00
|
|
|
local math_floor = math.floor
|
|
|
|
local math_sqrt = math.sqrt
|
|
|
|
local math_round = math.round
|
2019-12-24 16:35:40 +01:00
|
|
|
local table_insert = table.insert
|
|
|
|
local table_remove = table.remove
|
2019-12-20 23:10:03 +01:00
|
|
|
|
|
|
|
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 i = 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_anything
|
|
|
|
}
|
|
|
|
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_enemy,
|
|
|
|
}
|
|
|
|
|
|
|
|
return commands
|
|
|
|
end
|
2019-12-19 18:26:02 +01:00
|
|
|
|
|
|
|
local function roll_market()
|
|
|
|
local r_max = 0
|
|
|
|
local town_centers = global.towny.town_centers
|
2019-12-28 14:56:04 +01:00
|
|
|
|
|
|
|
--Skip Towns that are too low in reserach for the current biter evolution.
|
|
|
|
local research_threshold = game.forces.enemy.evolution_factor * #game.technology_prototypes * 0.175
|
2019-12-27 18:59:50 -05:00
|
|
|
|
2019-12-19 18:26:02 +01:00
|
|
|
for k, town_center in pairs(town_centers) do
|
2019-12-27 18:59:50 -05:00
|
|
|
if town_center.research_counter >= research_threshold then
|
|
|
|
r_max = r_max + town_center.research_counter
|
|
|
|
end
|
2019-12-19 18:26:02 +01:00
|
|
|
end
|
|
|
|
if r_max == 0 then return end
|
|
|
|
local r = math_random(0, r_max)
|
|
|
|
|
|
|
|
local chance = 0
|
|
|
|
for k, town_center in pairs(town_centers) do
|
2019-12-27 18:59:50 -05:00
|
|
|
if town_center.research_counter >= research_threshold then
|
|
|
|
chance = chance + town_center.research_counter
|
|
|
|
if r <= chance then return town_center end
|
|
|
|
end
|
2019-12-19 18:26:02 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function get_random_close_spawner(surface, market)
|
|
|
|
local spawners = surface.find_entities_filtered({type = "unit-spawner"})
|
|
|
|
if not spawners[1] then return false end
|
|
|
|
local size_of_spawners = #spawners
|
|
|
|
local center = market.position
|
|
|
|
local spawner = spawners[math_random(1, size_of_spawners)]
|
2019-12-24 12:43:47 +01:00
|
|
|
for i = 1, 64, 1 do
|
2019-12-19 18:26:02 +01:00
|
|
|
local spawner_2 = spawners[math_random(1, size_of_spawners)]
|
2019-12-20 23:10:03 +01:00
|
|
|
if (center.x - spawner_2.position.x) ^ 2 + (center.y - spawner_2.position.y) ^ 2 < (center.x - spawner.position.x) ^ 2 + (center.y - spawner.position.y) ^ 2 then
|
|
|
|
spawner = spawner_2
|
|
|
|
end
|
2019-12-19 18:26:02 +01:00
|
|
|
end
|
|
|
|
return spawner
|
|
|
|
end
|
|
|
|
|
2019-12-24 16:35:40 +01:00
|
|
|
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()
|
|
|
|
for k, swarm in pairs(global.towny.swarms) do
|
|
|
|
if not is_swarm_valid(swarm) then
|
|
|
|
table_remove(global.towny.swarms, k)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-12-28 14:56:04 +01:00
|
|
|
--Destroy biters that are out of the current evolution range.
|
|
|
|
function Public.wipe_units_out_of_evo_range()
|
|
|
|
local units_to_wipe = {}
|
|
|
|
local evo = game.forces.enemy.evolution_factor
|
|
|
|
if evo > 0.80 then return end
|
|
|
|
units_to_wipe[#units_to_wipe + 1] = "behemoth-biter"
|
|
|
|
units_to_wipe[#units_to_wipe + 1] = "behemoth-spitter"
|
|
|
|
if evo < 0.40 then
|
|
|
|
units_to_wipe[#units_to_wipe + 1] = "big-biter"
|
|
|
|
units_to_wipe[#units_to_wipe + 1] = "big-spitter"
|
|
|
|
end
|
|
|
|
if evo < 0.10 then
|
|
|
|
units_to_wipe[#units_to_wipe + 1] = "medium-biter"
|
|
|
|
units_to_wipe[#units_to_wipe + 1] = "medium-spitter"
|
|
|
|
end
|
|
|
|
for k, surface in pairs(game.surfaces) do
|
|
|
|
for k2, unit in pairs(surface.find_entities_filtered({name = units_to_wipe, force = "enemy"})) do
|
|
|
|
unit.destroy()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
function Public.clear_spawn_for_player(player)
|
|
|
|
local area = {{player.position.x - 64, player.position.y - 64}, {player.position.x + 64, player.position.y + 64}}
|
|
|
|
for _, e in pairs(player.surface.find_entities_filtered({force = "enemy", type = {"unit-spawner", "unit", "turret"}, area = area})) do
|
|
|
|
e.destroy()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-12-24 16:35:40 +01:00
|
|
|
function Public.unit_groups_start_moving()
|
|
|
|
for k, swarm in pairs(global.towny.swarms) do
|
|
|
|
if swarm.group then
|
|
|
|
if swarm.group.valid then
|
|
|
|
swarm.group.start_moving()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-12-19 18:26:02 +01:00
|
|
|
function Public.swarm()
|
2019-12-24 16:35:40 +01:00
|
|
|
local count = 0
|
|
|
|
for k, swarm in pairs(global.towny.swarms) do
|
|
|
|
count = count + 1
|
|
|
|
end
|
|
|
|
if count > 6 then return end
|
|
|
|
|
2019-12-19 18:26:02 +01:00
|
|
|
local town_center = roll_market()
|
|
|
|
if not town_center then return end
|
|
|
|
local market = town_center.market
|
|
|
|
local surface = market.surface
|
|
|
|
local spawner = get_random_close_spawner(surface, market)
|
|
|
|
if not spawner then return end
|
2019-12-24 12:43:47 +01:00
|
|
|
local units = spawner.surface.find_enemy_units(spawner.position, 160, market.force)
|
2019-12-19 18:26:02 +01:00
|
|
|
if not units[1] then return end
|
2019-12-19 19:58:37 +01:00
|
|
|
local unit_group_position = surface.find_non_colliding_position("market", 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})
|
2019-12-23 14:23:41 +01:00
|
|
|
local count = (town_center.research_counter * 1.5) + 4
|
2019-12-19 18:26:02 +01:00
|
|
|
for key, unit in pairs(units) do
|
|
|
|
if key > count then break end
|
|
|
|
unit_group.add_member(unit)
|
|
|
|
end
|
|
|
|
unit_group.set_command({
|
|
|
|
type = defines.command.compound,
|
|
|
|
structure_type = defines.compound_command.return_last,
|
2019-12-20 23:10:03 +01:00
|
|
|
commands = get_commmands(market, unit_group)
|
2019-12-19 18:26:02 +01:00
|
|
|
})
|
2019-12-24 16:35:40 +01:00
|
|
|
table_insert(global.towny.swarms, {group = unit_group, timeout = game.tick + 36000})
|
2019-12-19 18:26:02 +01:00
|
|
|
end
|
|
|
|
|
2019-12-19 19:00:42 +01:00
|
|
|
function Public.set_evolution()
|
|
|
|
local town_center_count = global.towny.size_of_town_centers
|
|
|
|
if town_center_count == 0 then
|
|
|
|
game.forces.enemy.evolution_factor = 0
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
2019-12-24 16:35:40 +01:00
|
|
|
local max_research_count = math_floor(#game.technology_prototypes * 0.30)
|
2019-12-19 19:00:42 +01:00
|
|
|
|
|
|
|
local evo = 0
|
|
|
|
for _, town_center in pairs(global.towny.town_centers) do
|
|
|
|
evo = evo + town_center.research_counter
|
|
|
|
end
|
|
|
|
evo = evo / town_center_count
|
|
|
|
evo = evo / max_research_count
|
|
|
|
if evo > 1 then evo = 1 end
|
|
|
|
|
|
|
|
game.forces.enemy.evolution_factor = evo
|
|
|
|
end
|
|
|
|
|
2019-12-19 18:26:02 +01:00
|
|
|
return Public
|