mirror of
https://github.com/ComfyFactory/ComfyFactorio.git
synced 2025-01-16 02:47:48 +02:00
1068 lines
32 KiB
Lua
1068 lines
32 KiB
Lua
-- This file is part of thesixthroc's Pirate Ship softmod, licensed under GPLv3 and stored at https://github.com/ComfyFactory/ComfyFactorio and https://github.com/danielmartin0/ComfyFactorio-Pirates.
|
|
|
|
local Memory = require('maps.pirates.memory')
|
|
local Balance = require('maps.pirates.balance')
|
|
local Common = require('maps.pirates.common')
|
|
local CoreData = require('maps.pirates.coredata')
|
|
-- local Utils = require 'maps.pirates.utils_local'
|
|
local Math = require('maps.pirates.math')
|
|
local Raffle = require('utils.math.raffle')
|
|
local _inspect = require('utils.inspect').inspect
|
|
|
|
-- local Structures = require 'maps.pirates.structures.structures'
|
|
local Boats = require('maps.pirates.structures.boats.boats')
|
|
local Surfaces = require('maps.pirates.surfaces.surfaces')
|
|
-- local Islands = require 'maps.pirates.surfaces.islands.islands'
|
|
local IslandEnum = require('maps.pirates.surfaces.islands.island_enum')
|
|
-- local Sea = require 'maps.pirates.surfaces.sea.sea'
|
|
-- local Crew = require 'maps.pirates.crew'
|
|
-- local Quest = require 'maps.pirates.quest'
|
|
local SurfacesCommon = require('maps.pirates.surfaces.common')
|
|
local Utils = require('maps.pirates.utils_local')
|
|
|
|
local Public = {}
|
|
|
|
local function fake_boat_target()
|
|
local memory = Memory.get_crew_memory()
|
|
if memory.boat and memory.boat.position then
|
|
return {
|
|
valid = true,
|
|
position = { x = memory.boat.position.x - 60, y = memory.boat.position.y } or nil,
|
|
name = 'boatarea',
|
|
}
|
|
end
|
|
end
|
|
|
|
-- fff 283 discussed pollution mechanics: https://factorio.com/blog/post/fff-283
|
|
|
|
local side_attack_target_names = {
|
|
'character',
|
|
'pumpjack',
|
|
'radar',
|
|
'electric-mining-drill',
|
|
'assembling-machine-1',
|
|
'solar-panel',
|
|
'nuclear-reactor',
|
|
'oil-refinery',
|
|
'centrifuge',
|
|
}
|
|
|
|
--=== Tick Actions
|
|
|
|
function Public.Tick_actions(tick_interval)
|
|
local memory = Memory.get_crew_memory()
|
|
local destination = Common.current_destination()
|
|
|
|
if destination.type ~= Surfaces.enum.ISLAND then
|
|
return
|
|
end
|
|
if memory.boat.state ~= Boats.enum_state.LANDED and memory.boat.state ~= Boats.enum_state.RETREATING then
|
|
return
|
|
end
|
|
|
|
if
|
|
memory.game_lost
|
|
or (
|
|
destination.dynamic_data.timeratlandingtime
|
|
and destination.dynamic_data.timer
|
|
< destination.dynamic_data.timeratlandingtime + Balance.grace_period_on_arriving_at_island_seconds
|
|
)
|
|
then
|
|
return
|
|
end
|
|
|
|
if game.tick % (tick_interval * 2) == 0 and memory.boat.state == Boats.enum_state.LANDED then
|
|
local extra_evo = 2 * tick_interval / 60 * Balance.evolution_per_second()
|
|
Common.increment_evo(extra_evo)
|
|
destination.dynamic_data.evolution_accrued_time = destination.dynamic_data.evolution_accrued_time + extra_evo
|
|
end
|
|
|
|
-- if destination.subtype == IslandEnum.enum.RED_DESERT then return end -- This was a hack to stop biter boats causing attacks, but, it has the even worse effect of stopping all floating_pollution gathering.
|
|
|
|
local ai_cycle_seconds = { -- warning: use even seconds only
|
|
[2] = Public.tell_biters_near_silo_to_attack_it,
|
|
[4] = Public.poke_script_groups,
|
|
[6] = Public.try_main_attack,
|
|
|
|
[16] = Public.tell_biters_near_silo_to_attack_it,
|
|
[18] = Public.poke_script_groups,
|
|
[20] = Public.try_secondary_attack,
|
|
|
|
[42] = Public.tell_biters_near_silo_to_attack_it,
|
|
[44] = Public.poke_script_groups,
|
|
[46] = Public.try_rogue_attack,
|
|
|
|
[56] = Public.poke_inactive_scripted_biters,
|
|
[58] = Public.create_mail_delivery_biters,
|
|
|
|
[74] = Public.try_boat_biters_attack,
|
|
}
|
|
|
|
if ai_cycle_seconds[(game.tick / 60) % 80] then
|
|
ai_cycle_seconds[(game.tick / 60) % 80]()
|
|
end
|
|
end
|
|
|
|
function Public.eat_up_fraction_of_all_pollution_wrapped()
|
|
local surface = game.surfaces[Common.current_destination().surface_name]
|
|
Public.eat_up_fraction_of_all_pollution(surface, 0.05)
|
|
end
|
|
|
|
function Public.eat_up_fraction_of_all_pollution(surface, fraction_of_global_pollution)
|
|
local memory = Memory.get_crew_memory()
|
|
-- local enemy_force_name = memory.enemy_force_name
|
|
|
|
local pollution_available = memory.floating_pollution
|
|
|
|
local chunk_positions = {}
|
|
for i = 1, Math.ceil(surface.map_gen_settings.width / 32), 1 do
|
|
for j = 1, Math.ceil(surface.map_gen_settings.height / 32), 1 do
|
|
chunk_positions[#chunk_positions + 1] = {
|
|
x = 16 + i * 32 - surface.map_gen_settings.width / 2,
|
|
y = 16 + j * 32 - surface.map_gen_settings.height / 2,
|
|
}
|
|
end
|
|
end
|
|
|
|
for i = 1, #chunk_positions do
|
|
local p = chunk_positions[i]
|
|
local pollution = surface.get_pollution(p)
|
|
local pollution_to_eat = pollution * fraction_of_global_pollution
|
|
|
|
surface.pollute(p, -pollution_to_eat)
|
|
-- Radioactive world doesn't absorb map pollution:
|
|
if Common.current_destination().subtype ~= IslandEnum.enum.RADIOACTIVE then
|
|
pollution_available = pollution_available + pollution_to_eat
|
|
end
|
|
end
|
|
|
|
-- if _DEBUG then
|
|
-- game.print(string.format('ate %f pollution', pollution_available))
|
|
-- end
|
|
|
|
memory.floating_pollution = pollution_available
|
|
end
|
|
|
|
function Public.wave_size_rng() -- random variance in attack sizes
|
|
local memory = Memory.get_crew_memory()
|
|
local destination = Common.current_destination()
|
|
|
|
local rng_scale = Balance.crew_scale() ^ (1 / 4) -- slightly dampen wave variance for small crews as they can't handle it
|
|
|
|
-- prevent situation where when player reveals spawner, he immediately gets surrounded by massive amount of biters (especially late game)
|
|
if destination and destination.type == SurfacesCommon.enum.ISLAND then
|
|
if
|
|
destination.dynamic_data
|
|
and destination.dynamic_data.disabled_wave_timer
|
|
and destination.dynamic_data.disabled_wave_timer > 0
|
|
then
|
|
return 0
|
|
end
|
|
end
|
|
|
|
local wave_percentage_chance = Math.clamp(0, 50, 11 + 8 * memory.floating_pollution / 1500)
|
|
|
|
local wave_size_multiplier = 1
|
|
local rng1 = Math.random(100)
|
|
|
|
if rng1 > wave_percentage_chance then
|
|
wave_size_multiplier = 0
|
|
elseif memory.overworldx > 0 then
|
|
local rng2 = Math.random(1000)
|
|
if rng2 >= 110 * rng_scale then
|
|
wave_size_multiplier = 1
|
|
elseif rng2 >= 30 * rng_scale then
|
|
wave_size_multiplier = 1.5
|
|
elseif rng2 >= 15 * rng_scale then
|
|
wave_size_multiplier = 2
|
|
elseif rng2 >= 2 * rng_scale then
|
|
wave_size_multiplier = 3
|
|
else
|
|
wave_size_multiplier = 4
|
|
end
|
|
end
|
|
|
|
return wave_size_multiplier
|
|
end
|
|
|
|
function Public.try_main_attack()
|
|
Public.eat_up_fraction_of_all_pollution_wrapped()
|
|
|
|
local wave_size_multiplier = Public.wave_size_rng()
|
|
|
|
if wave_size_multiplier == 0 then
|
|
log('Attacks: ' .. 'Not attempted (by chance).')
|
|
return nil
|
|
else
|
|
local group = Public.spawn_group_of_scripted_biters(2 / 3, 6, 350, wave_size_multiplier)
|
|
local target = Public.generate_main_attack_target()
|
|
|
|
Public.group_set_commands(group, Public.attack_target(target))
|
|
|
|
-- if _DEBUG then game.print(game.tick .. string.format(": sending main attack of %s units from {%f,%f} to %s", #group.members, group.position.x, group.position.y, target.name)) end
|
|
end
|
|
end
|
|
|
|
function Public.try_secondary_attack()
|
|
Public.eat_up_fraction_of_all_pollution_wrapped()
|
|
|
|
local wave_size_multiplier = Public.wave_size_rng()
|
|
|
|
if wave_size_multiplier == 0 then
|
|
log('Attacks: ' .. 'Not attempted (by chance).')
|
|
return nil
|
|
else
|
|
local surface = game.surfaces[Common.current_destination().surface_name]
|
|
|
|
local group = Public.spawn_group_of_scripted_biters(2 / 3, 12, 275, wave_size_multiplier)
|
|
if not (group and group.valid) then
|
|
return
|
|
end
|
|
|
|
local target
|
|
if Math.random(2) == 1 then
|
|
target = Public.generate_main_attack_target()
|
|
else
|
|
target = Public.generate_side_attack_target(surface, group.position)
|
|
end
|
|
|
|
Public.group_set_commands(group, Public.attack_target(target))
|
|
|
|
-- if _DEBUG then game.print(game.tick .. string.format(": sending main attack of %s units from {%f,%f} to %s", #group.members, group.position.x, group.position.y, target.name)) end
|
|
end
|
|
end
|
|
|
|
function Public.try_rogue_attack()
|
|
Public.eat_up_fraction_of_all_pollution_wrapped()
|
|
|
|
local wave_size_multiplier = Public.wave_size_rng()
|
|
|
|
if wave_size_multiplier == 0 then
|
|
log('Attacks: ' .. 'Not attempted (by chance).')
|
|
return nil
|
|
else
|
|
local surface = game.surfaces[Common.current_destination().surface_name]
|
|
|
|
local group = Public.spawn_group_of_scripted_biters(1 / 2, 6, 200, wave_size_multiplier)
|
|
if not (group and group.valid) then
|
|
return
|
|
end
|
|
local target = Public.generate_side_attack_target(surface, group.position)
|
|
|
|
Public.group_set_commands(group, Public.attack_target(target))
|
|
|
|
-- if _DEBUG then game.print(game.tick .. string.format(": sending rogue attack of %s units from {%f,%f} to %s", #group.members, group.position.x, group.position.y, target.name)) end
|
|
end
|
|
end
|
|
|
|
function Public.tell_biters_near_silo_to_attack_it()
|
|
-- careful with this function, you don't want to pull biters onto the silo before any aggro has happened
|
|
local memory = Memory.get_crew_memory()
|
|
local destination = Common.current_destination()
|
|
local surface = game.surfaces[destination.surface_name]
|
|
local enemy_force_name = memory.enemy_force_name
|
|
|
|
-- don't do this too early
|
|
if
|
|
destination.dynamic_data.timeratlandingtime
|
|
and destination.dynamic_data.timer
|
|
< destination.dynamic_data.timeratlandingtime + Balance.grace_period_on_arriving_at_island_seconds * 2
|
|
then
|
|
return
|
|
end
|
|
if
|
|
not (
|
|
destination.dynamic_data.rocketsilos
|
|
and destination.dynamic_data.rocketsilos[1]
|
|
and destination.dynamic_data.rocketsilos[1].valid
|
|
and destination.dynamic_data.rocketsilos[1].destructible
|
|
)
|
|
then
|
|
return
|
|
end
|
|
|
|
local attackcommand = Public.attack_target_entity(destination.dynamic_data.rocketsilos[1])
|
|
|
|
if attackcommand then
|
|
surface.set_multi_command({
|
|
command = attackcommand,
|
|
unit_count = Math.random(1, Math.floor(1 + memory.evolution_factor * 100)),
|
|
force = enemy_force_name,
|
|
unit_search_distance = 10,
|
|
})
|
|
end
|
|
end
|
|
|
|
function Public.poke_script_groups()
|
|
local memory = Memory.get_crew_memory()
|
|
for index, group in pairs(memory.scripted_unit_groups) do
|
|
local groupref = group.ref
|
|
if
|
|
not groupref.valid
|
|
or groupref.surface.index ~= game.surfaces[Common.current_destination().surface_name].index
|
|
or #groupref.members < 1
|
|
then
|
|
memory.scripted_unit_groups[index] = nil
|
|
else
|
|
if groupref.state == defines.group_state.finished then
|
|
if Math.random(20) == 20 then
|
|
local command =
|
|
Public.attack_obstacles(groupref.surface, { x = groupref.position.x, y = groupref.position.y })
|
|
groupref.set_command(command)
|
|
else
|
|
groupref.set_autonomous() --means go home, really
|
|
end
|
|
elseif group.state == defines.group_state.gathering then
|
|
groupref.start_moving()
|
|
-- elseif group.state == defines.group_state.wander_in_group then
|
|
-- groupref.set_autonomous() --means go home, really
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
function Public.poke_inactive_scripted_biters()
|
|
local memory = Memory.get_crew_memory()
|
|
for unit_number, biter in pairs(memory.scripted_biters) do
|
|
if Public.is_biter_inactive(biter) then
|
|
memory.scripted_biters[unit_number] = nil
|
|
if biter.entity and biter.entity.valid then
|
|
local target = Public.nearest_target()
|
|
Public.group_set_commands(biter.entity, Public.attack_target(target))
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
function Public.create_mail_delivery_biters() --these travel cross-map between bases to add a little life and spookyness
|
|
local memory = Memory.get_crew_memory()
|
|
local surface = game.surfaces[Common.current_destination().surface_name]
|
|
local enemy_force_name = memory.enemy_force_name
|
|
|
|
local spawners = Common.get_valid_spawners(surface)
|
|
|
|
local try_how_many_groups = Math.clamp(0, 4, (#spawners - 8) / 100)
|
|
|
|
for i = 1, try_how_many_groups do
|
|
if Math.random(2) == 1 then
|
|
local s1 = spawners[Math.random(#spawners)]
|
|
|
|
local far_spawners = {}
|
|
for j = 1, #spawners do
|
|
local s2 = spawners[i]
|
|
if not (i == j or Math.distance(s1.position, s2.position) < 250) then
|
|
far_spawners[#far_spawners + 1] = s2
|
|
end
|
|
end
|
|
|
|
if #far_spawners > 0 then
|
|
local s2 = far_spawners[Math.random(#far_spawners)]
|
|
|
|
memory.floating_pollution = memory.floating_pollution + 64
|
|
local units = Public.try_spawner_spend_fraction_of_available_pollution_on_biters(
|
|
s1.position,
|
|
1 / 4,
|
|
4,
|
|
32,
|
|
1,
|
|
'small-biter'
|
|
)
|
|
memory.floating_pollution = memory.floating_pollution - 64
|
|
|
|
if (not units) or not #units or (#units == 0) then
|
|
return
|
|
end
|
|
|
|
local start_p = surface.find_non_colliding_position('rocket-silo', s1.position, 256, 2) or s1.position
|
|
|
|
local unit_group = surface.create_unit_group({ position = start_p, force = enemy_force_name })
|
|
if not (unit_group and unit_group.valid) then
|
|
return
|
|
end
|
|
|
|
for _, unit in pairs(units) do
|
|
unit_group.add_member(unit)
|
|
end
|
|
memory.scripted_unit_groups[unit_group.unique_id] = { ref = unit_group, script_type = 'mail-delivery' }
|
|
|
|
Public.group_set_commands(unit_group, {
|
|
Public.move_to(s2.position),
|
|
Public.wander_around(),
|
|
})
|
|
|
|
-- game.print(string.format('%f biters delivering mail from %f, %f to %f, %f', #units, s1.position.x, s1.position.y, s2.position.x, s2.position.y))
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
--=== Spawn scripted biters
|
|
|
|
function Public.spawn_group_of_scripted_biters(
|
|
fraction_of_floating_pollution,
|
|
minimum_avg_units,
|
|
maximum_units,
|
|
wave_size_multiplier
|
|
)
|
|
local memory = Memory.get_crew_memory()
|
|
local surface = game.surfaces[Common.current_destination().surface_name]
|
|
local enemy_force_name = memory.enemy_force_name
|
|
|
|
local spawner = Common.get_random_valid_spawner(surface)
|
|
if not spawner then
|
|
log('no spawner found')
|
|
return
|
|
end
|
|
|
|
local nearby_units_to_bring
|
|
if Public.get_scripted_biter_count() >= 9 / 10 * CoreData.total_max_biters then
|
|
-- pick up nearby units that might be idle:
|
|
nearby_units_to_bring = surface.find_units({
|
|
area = {
|
|
{ spawner.position.x - 8, spawner.position.y - 8 },
|
|
{ spawner.position.x + 8, spawner.position.y + 8 },
|
|
},
|
|
force = enemy_force_name,
|
|
condition = 'same',
|
|
})
|
|
else
|
|
nearby_units_to_bring = {}
|
|
end
|
|
|
|
local new_units = Public.try_spawner_spend_fraction_of_available_pollution_on_biters(
|
|
spawner.position,
|
|
fraction_of_floating_pollution,
|
|
minimum_avg_units,
|
|
maximum_units,
|
|
wave_size_multiplier
|
|
)
|
|
|
|
if new_units and nearby_units_to_bring and (#new_units + #nearby_units_to_bring) == 0 then
|
|
return
|
|
end
|
|
|
|
local position = surface.find_non_colliding_position('rocket-silo', spawner.position, 256, 2) or spawner.position
|
|
|
|
local unit_group = surface.create_unit_group({ position = position, force = enemy_force_name })
|
|
for _, unit in pairs(nearby_units_to_bring) do
|
|
unit_group.add_member(unit)
|
|
end
|
|
for _, unit in pairs(new_units) do
|
|
unit_group.add_member(unit)
|
|
end
|
|
memory.scripted_unit_groups[unit_group.unique_id] = { ref = unit_group, script_type = 'attacker' }
|
|
return unit_group
|
|
end
|
|
|
|
function Public.try_spawner_spend_fraction_of_available_pollution_on_biters(
|
|
spawnposition,
|
|
fraction_of_floating_pollution,
|
|
minimum_avg_units,
|
|
maximum_units,
|
|
wave_size_multiplier,
|
|
enforce_type
|
|
)
|
|
maximum_units = maximum_units or 400
|
|
|
|
-- log('ai spawning attempt params: ' .. (fraction_of_floating_pollution or '') .. ' ' .. (minimum_avg_units or '') .. ' ' .. (maximum_units or '') .. ' ' .. (unit_pollutioncost_multiplier or '') .. ' ' .. (enforce_type or ''))
|
|
|
|
local memory = Memory.get_crew_memory()
|
|
local surface = game.surfaces[Common.current_destination().surface_name]
|
|
-- local surface = spawner.surface
|
|
-- local spawnposition = spawner.position
|
|
-- local difficulty = memory.difficulty
|
|
local enemy_force_name = memory.enemy_force_name
|
|
local evolution = memory.evolution_factor
|
|
|
|
local units_created_count = 0
|
|
local units_created = {}
|
|
|
|
local temp_floating_pollution = memory.floating_pollution
|
|
local budget = fraction_of_floating_pollution * temp_floating_pollution
|
|
|
|
local initialpollution = memory.floating_pollution
|
|
-- local initialbudget = budget
|
|
|
|
local map_pollution_cost_multiplier = 1
|
|
local destination = Common.current_destination()
|
|
|
|
if destination.dynamic_data and destination.dynamic_data.initial_spawner_count then
|
|
local initial_spawner_count = destination.dynamic_data.initial_spawner_count
|
|
|
|
if initial_spawner_count > 0 then
|
|
local spawners_count = Common.spawner_count(surface)
|
|
if spawners_count > 0 then
|
|
map_pollution_cost_multiplier = Math.max(initial_spawner_count / spawners_count, 1)
|
|
else
|
|
map_pollution_cost_multiplier = 1000000
|
|
end
|
|
end
|
|
end
|
|
|
|
-- if memory.overworldx == 0 then
|
|
-- -- less biters:
|
|
-- base_pollution_cost_multiplier = base_pollution_cost_multiplier * 2.30 --tuned to teach players to defend, but then to feel relaxing once they do
|
|
-- end -- moved to scripted_biters_pollution_cost_multiplier
|
|
|
|
local base_scripted_biters_pollution_cost_multiplier = Balance.scripted_biters_pollution_cost_multiplier()
|
|
|
|
map_pollution_cost_multiplier = map_pollution_cost_multiplier * base_scripted_biters_pollution_cost_multiplier
|
|
|
|
if destination.subtype == IslandEnum.enum.SWAMP then
|
|
map_pollution_cost_multiplier = map_pollution_cost_multiplier * 0.95 --biters 5% more aggressive
|
|
end
|
|
|
|
-- if destination.subtype == IslandEnum.enum.MAZE then
|
|
-- base_pollution_cost_multiplier = base_pollution_cost_multiplier * 1.2 --biters 20% less aggressive
|
|
-- end
|
|
|
|
if budget >= minimum_avg_units * Common.averageUnitPollutionCost(evolution) * map_pollution_cost_multiplier then
|
|
local function spawn(name2)
|
|
units_created_count = units_created_count + 1
|
|
|
|
local unit_pollutioncost = CoreData.biterPollutionValues[name2]
|
|
* map_pollution_cost_multiplier
|
|
/ wave_size_multiplier
|
|
|
|
local p = surface.find_non_colliding_position(name2, spawnposition, 60, 1)
|
|
if not p then
|
|
p = spawnposition
|
|
log('no position found, using spawnposition')
|
|
end
|
|
|
|
local biter = surface.create_entity({ name = name2, force = enemy_force_name, position = p })
|
|
Common.try_make_biter_elite(biter)
|
|
|
|
units_created[#units_created + 1] = biter
|
|
memory.scripted_biters[biter.unit_number] = { entity = biter, created_at = game.tick }
|
|
|
|
temp_floating_pollution = temp_floating_pollution - unit_pollutioncost
|
|
budget = budget - unit_pollutioncost
|
|
-- flow statistics should reflect the number of biters generated. Therefore it should mismatch the actual pollution spent, because it should miss all the factors that can vary:
|
|
game.get_pollution_statistics(surface).on_flow(name2, -CoreData.biterPollutionValues[name2])
|
|
|
|
return biter.unit_number
|
|
end
|
|
|
|
local mixed = (Math.random(3) <= 2)
|
|
if mixed then
|
|
local whilesafety = 5000
|
|
local next_name = enforce_type or Common.get_random_unit_type(evolution)
|
|
|
|
while
|
|
units_created_count < maximum_units
|
|
and budget >= CoreData.biterPollutionValues[next_name] * map_pollution_cost_multiplier
|
|
and #memory.scripted_biters < CoreData.total_max_biters
|
|
and whilesafety > 0
|
|
do
|
|
whilesafety = whilesafety - 1
|
|
spawn(next_name)
|
|
next_name = enforce_type or Common.get_random_unit_type(evolution)
|
|
end
|
|
else
|
|
local name = enforce_type or Common.get_random_unit_type(evolution)
|
|
|
|
local whilesafety = 5000
|
|
while
|
|
units_created_count < maximum_units
|
|
and budget >= CoreData.biterPollutionValues[name] * map_pollution_cost_multiplier
|
|
and #memory.scripted_biters < CoreData.total_max_biters
|
|
and whilesafety > 0
|
|
do
|
|
whilesafety = whilesafety - 1
|
|
spawn(name)
|
|
end
|
|
end
|
|
|
|
memory.floating_pollution = temp_floating_pollution
|
|
else
|
|
log('Attacks: ' .. 'Insufficient pollution for wave.')
|
|
end
|
|
|
|
if units_created_count > 0 then
|
|
log(
|
|
'Attacks: '
|
|
.. 'Spent '
|
|
.. Math.floor(100 * (initialpollution - temp_floating_pollution) / initialpollution)
|
|
.. '% of '
|
|
.. Math.ceil(initialpollution)
|
|
.. ' pollution budget on biters, at '
|
|
.. Math.ceil(map_pollution_cost_multiplier / wave_size_multiplier * 100) / 100
|
|
.. 'x price.'
|
|
)
|
|
end
|
|
|
|
return units_created
|
|
end
|
|
|
|
--=== Misc Functions
|
|
|
|
function Public.generate_main_attack_target()
|
|
local memory = Memory.get_crew_memory()
|
|
local destination = Common.current_destination()
|
|
local target
|
|
local fractioncharged = 0
|
|
if
|
|
not destination.dynamic_data.rocket_launched
|
|
and destination.dynamic_data.rocketsilos
|
|
and destination.dynamic_data.rocketsilos[1]
|
|
and destination.dynamic_data.rocketsilos[1].valid
|
|
and destination.dynamic_data.rocketsilos[1].destructible
|
|
and destination.dynamic_data.rocket_silo_energy_consumed
|
|
and destination.dynamic_data.rocket_silo_energy_needed
|
|
and destination.dynamic_data.rocket_silo_energy_needed > 0
|
|
then
|
|
fractioncharged = destination.dynamic_data.rocket_silo_energy_consumed
|
|
/ destination.dynamic_data.rocket_silo_energy_needed
|
|
|
|
if memory.overworldx > 40 * 22 then --chance of biters going directly to silo
|
|
fractioncharged = fractioncharged + 0.03
|
|
end
|
|
end
|
|
|
|
local rng = Math.random()
|
|
if rng <= fractioncharged ^ (1 / 2) then
|
|
target = destination.dynamic_data.rocketsilos[1]
|
|
else
|
|
target = fake_boat_target()
|
|
end
|
|
return target
|
|
end
|
|
|
|
function Public.generate_side_attack_target(surface, position)
|
|
local entities = surface.find_entities_filtered({ name = side_attack_target_names })
|
|
if not entities then
|
|
return
|
|
end
|
|
if Math.random(20) >= #entities then
|
|
return
|
|
end
|
|
|
|
entities = Math.shuffle(entities)
|
|
entities = Math.shuffle_distancebiased(entities, position)
|
|
local weights = {}
|
|
for index, _ in pairs(entities) do
|
|
weights[#weights + 1] = 1 + Math.floor((#entities - index) / 2)
|
|
end
|
|
return Raffle.raffle(entities, weights)
|
|
end
|
|
|
|
function Public.nearest_target(surface, position)
|
|
local names = { 'rocket-silo' }
|
|
for _, name in pairs(side_attack_target_names) do
|
|
names[#names + 1] = name
|
|
end
|
|
local entities = surface.find_entities_filtered({ name = names })
|
|
local d = 9999
|
|
local nearest = nil
|
|
for i = 1, #entities do
|
|
local e = entities[i]
|
|
if e and e.valid and Math.distance(e.position, position) < d then
|
|
nearest = e
|
|
end
|
|
end
|
|
return nearest
|
|
end
|
|
|
|
function Public.is_biter_inactive(biter)
|
|
if (not biter.entity) or not biter.entity.valid then
|
|
return true
|
|
end
|
|
if game.tick - biter.created_at > 30 * 60 * 60 then
|
|
biter.entity.destroy()
|
|
return true
|
|
end
|
|
return false
|
|
end
|
|
|
|
function Public.get_scripted_biter_count()
|
|
local memory = Memory.get_crew_memory()
|
|
local count = 0
|
|
for k, biter in pairs(memory.scripted_biters) do
|
|
if biter.entity and biter.entity.valid then
|
|
count = count + 1
|
|
else
|
|
memory.scripted_biters[k] = nil
|
|
end
|
|
end
|
|
return count
|
|
end
|
|
|
|
-----------commands-----------
|
|
|
|
function Public.stop()
|
|
local command = {
|
|
type = defines.command.stop,
|
|
distraction = defines.distraction.none,
|
|
}
|
|
return command
|
|
end
|
|
|
|
function Public.move_to(position)
|
|
local command = {
|
|
type = defines.command.go_to_location,
|
|
destination = position,
|
|
distraction = defines.distraction.by_anything,
|
|
}
|
|
return command
|
|
end
|
|
|
|
function Public.attack_target_entity(target)
|
|
if not (target and target.valid) then
|
|
return
|
|
end
|
|
local command = {
|
|
type = defines.command.attack,
|
|
target = target,
|
|
distraction = defines.distraction.by_anything,
|
|
}
|
|
return command
|
|
end
|
|
|
|
function Public.attack_area(position, radius)
|
|
local command = {
|
|
type = defines.command.attack_area,
|
|
destination = position,
|
|
radius = radius or 25,
|
|
distraction = defines.distraction.by_anything,
|
|
}
|
|
return command
|
|
end
|
|
|
|
function Public.attack_obstacles(surface, position)
|
|
local commands = {}
|
|
local obstacles = surface.find_entities_filtered({
|
|
position = position,
|
|
radius = 25,
|
|
type = { 'simple-entity', 'tree', 'simple-entity-with-owner' },
|
|
limit = 100,
|
|
})
|
|
if obstacles then
|
|
Math.shuffle(obstacles)
|
|
Math.shuffle_distancebiased(obstacles, position)
|
|
for i = 1, #obstacles, 1 do
|
|
if obstacles[i].valid then
|
|
commands[#commands + 1] = {
|
|
type = defines.command.attack,
|
|
target = obstacles[i],
|
|
distraction = defines.distraction.by_anything,
|
|
}
|
|
end
|
|
end
|
|
end
|
|
commands[#commands + 1] = Public.move_to(position)
|
|
local command = {
|
|
type = defines.command.compound,
|
|
structure_type = defines.compound_command.return_last,
|
|
commands = commands,
|
|
}
|
|
return command
|
|
end
|
|
|
|
function Public.wander_around(ticks_to_wait) --wander individually inside group radius
|
|
local command = {
|
|
type = defines.command.wander,
|
|
distraction = defines.distraction.by_anything,
|
|
ticks_to_wait = ticks_to_wait,
|
|
}
|
|
return command
|
|
end
|
|
|
|
function Public.group_set_commands(group, commands)
|
|
if not (group and group.valid) then
|
|
return
|
|
end
|
|
|
|
if commands and #commands > 0 then
|
|
local command = {
|
|
type = defines.command.compound,
|
|
structure_type = defines.compound_command.return_last,
|
|
commands = commands,
|
|
}
|
|
group.set_command(command)
|
|
end
|
|
end
|
|
|
|
function Public.attack_target(target)
|
|
if not (target and target.valid) then
|
|
return
|
|
end
|
|
|
|
local commands
|
|
if target.name == 'boatarea' then
|
|
commands = {
|
|
Public.attack_area(target.position, 32),
|
|
Public.attack_area(target.position, 32),
|
|
}
|
|
else
|
|
commands = {
|
|
Public.attack_area(target.position, 8),
|
|
Public.attack_target_entity(target),
|
|
}
|
|
end
|
|
-- if Math.random(20) == 20 then
|
|
-- commands = {
|
|
-- Public.attack_obstacles(group.surface, {x = (group.position.x * 0.90 + target.position.x * 0.10), y = (group.position.y * 0.90 + target.position.y * 0.10)}),
|
|
-- attackcommand,
|
|
-- }
|
|
-- else
|
|
-- commands = {attackcommand}
|
|
-- end
|
|
return commands
|
|
end
|
|
|
|
--- small group of revenge biters ---
|
|
|
|
function Public.revenge_group(surface, p, target, type, bonus_evo, amount_multiplier)
|
|
amount_multiplier = amount_multiplier or 1
|
|
bonus_evo = bonus_evo or 0
|
|
type = type or 'biter'
|
|
local memory = Memory.get_crew_memory()
|
|
local enemy_force_name = memory.enemy_force_name
|
|
|
|
local name, count
|
|
if type == 'biter' then
|
|
name = Common.get_random_biter_type(memory.evolution_factor + bonus_evo)
|
|
|
|
if name == 'small-biter' then
|
|
count = 5
|
|
elseif name == 'medium-biter' then
|
|
count = 3
|
|
elseif name == 'big-biter' then
|
|
count = 2
|
|
elseif name == 'behemoth-biter' then
|
|
count = 1
|
|
end
|
|
elseif type == 'spitter' then
|
|
name = Common.get_random_spitter_type(memory.evolution_factor + bonus_evo)
|
|
|
|
if name == 'small-spitter' then
|
|
count = 7
|
|
elseif name == 'medium-spitter' then
|
|
count = 4
|
|
elseif name == 'big-spitter' then
|
|
count = 3
|
|
elseif name == 'behemoth-spitter' then
|
|
count = 2
|
|
end
|
|
end
|
|
|
|
if not (name and count and count > 0) then
|
|
return
|
|
end
|
|
|
|
local units = {}
|
|
for i = 1, Math.floor(count * amount_multiplier) do
|
|
local p2 = surface.find_non_colliding_position('wooden-chest', p, 5, 0.5)
|
|
if p2 then
|
|
local biter = surface.create_entity({ name = name, force = enemy_force_name, position = p })
|
|
Common.try_make_biter_elite(biter)
|
|
-- local biter = surface.create_entity({name = name, force = enemy_force_name, position = p2})
|
|
units[#units + 1] = biter
|
|
end
|
|
end
|
|
|
|
if #units > 0 then
|
|
local unit_group = surface.create_unit_group({ position = p, force = enemy_force_name })
|
|
if not (unit_group and unit_group.valid) then
|
|
return
|
|
end
|
|
|
|
for _, unit in pairs(units) do
|
|
unit_group.add_member(unit)
|
|
end
|
|
|
|
Public.group_set_commands(unit_group, Public.attack_target(target))
|
|
unit_group.set_autonomous()
|
|
end
|
|
end
|
|
|
|
----------- biter raiding parties -----------
|
|
|
|
function Public.spawn_boat_biters(boat, max_evo, count, width)
|
|
-- max_evolution_bonus = max_evolution_bonus or 0.3
|
|
local memory = Memory.get_crew_memory()
|
|
local surface = game.surfaces[boat.surface_name]
|
|
-- local difficulty = memory.difficulty
|
|
local enemy_force_name = boat.force_name
|
|
-- local evolution = memory.evolution_factor
|
|
|
|
local p = { boat.position.x - width / 2, boat.position.y }
|
|
|
|
local units = {}
|
|
for i = 1, count do
|
|
local name = Common.get_random_unit_type(max_evo - i * 0.04)
|
|
-- local name = Common.get_random_unit_type(evolution + i/15 * max_evolution_bonus)
|
|
-- local name = Common.get_random_unit_type(evolution + 3 * i/100)
|
|
|
|
local p2 = surface.find_non_colliding_position('wooden-chest', p, 8, 0.5) --needs to be wooden-chest for collisions to work properly
|
|
if p2 then
|
|
local biter = surface.create_entity({ name = name, force = enemy_force_name, position = p2 })
|
|
Common.try_make_biter_elite(biter)
|
|
|
|
memory.scripted_biters[biter.unit_number] = { entity = biter, created_at = game.tick }
|
|
|
|
units[#units + 1] = biter
|
|
end
|
|
end
|
|
|
|
if #units > 0 then
|
|
local unit_group = surface.create_unit_group({ position = p, force = enemy_force_name })
|
|
for _, unit in pairs(units) do
|
|
unit_group.add_member(unit)
|
|
end
|
|
boat.unit_group = { ref = unit_group, script_type = 'landing-party' }
|
|
end
|
|
end
|
|
|
|
function Public.update_landing_party_unit_groups(boat, step_distance)
|
|
local memory = Memory.get_crew_memory()
|
|
|
|
-- move unit groups:
|
|
local group = boat.unit_group
|
|
local surface = game.surfaces[boat.surface_name]
|
|
if not (group and surface and surface.valid) then
|
|
return
|
|
end
|
|
|
|
local groupref = group.ref
|
|
if not (groupref and groupref.valid) then
|
|
return
|
|
end
|
|
|
|
local p2 = groupref.position
|
|
if not p2 then
|
|
return
|
|
end
|
|
|
|
local enemy_force_name = memory.enemy_force_name
|
|
local m = groupref.members
|
|
groupref.destroy()
|
|
|
|
local new_group =
|
|
surface.create_unit_group({ position = { x = p2.x + step_distance, y = p2.y }, force = enemy_force_name })
|
|
|
|
boat.unit_group = { ref = new_group, script_type = 'landing-party' }
|
|
for i = 1, #m do
|
|
local b = m[i]
|
|
new_group.add_member(b)
|
|
end
|
|
|
|
-- if boat.spawner and boat.spawner.valid then
|
|
-- new_group.set_command(Public.move_to(boat.spawner.position))
|
|
-- end
|
|
end
|
|
|
|
function Public.send_boat_biters_for_attack(eboat)
|
|
local memory = Memory.get_crew_memory()
|
|
local enemy_force_name = memory.enemy_force_name
|
|
local boat = memory.boat
|
|
|
|
local units = game.surfaces[eboat.surface_name].find_units({
|
|
area = { { eboat.position.x - 12, eboat.position.y - 12 }, { eboat.position.x + 12, eboat.position.y + 12 } },
|
|
force = enemy_force_name,
|
|
condition = 'same',
|
|
})
|
|
|
|
if #units > 0 then
|
|
local unit_group =
|
|
game.surfaces[eboat.surface_name].create_unit_group({ position = eboat.position, force = enemy_force_name })
|
|
for _, unit in pairs(units) do
|
|
unit_group.add_member(unit)
|
|
end
|
|
boat.unit_group = { ref = unit_group, script_type = 'landing-party' }
|
|
|
|
boat.unit_group.ref.set_command({
|
|
type = defines.command.attack_area,
|
|
destination = ({ memory.boat.position.x - 32, memory.boat.position.y } or { 0, 0 }),
|
|
radius = 32,
|
|
distraction = defines.distraction.by_enemy,
|
|
})
|
|
end
|
|
end
|
|
|
|
function Public.try_boat_biters_attack()
|
|
local memory = Memory.get_crew_memory()
|
|
local destination = Common.current_destination()
|
|
|
|
if not destination.dynamic_data.enemyboats then
|
|
return
|
|
end
|
|
|
|
for i = 1, #destination.dynamic_data.enemyboats do
|
|
local eboat = destination.dynamic_data.enemyboats[i]
|
|
if
|
|
eboat
|
|
and eboat.surface_name
|
|
and game.surfaces[eboat.surface_name]
|
|
and game.surfaces[eboat.surface_name].valid
|
|
then
|
|
if eboat.state == Boats.enum_state.LANDED and memory.game_lost == false then
|
|
Public.send_boat_biters_for_attack(eboat)
|
|
end
|
|
else
|
|
Utils.fast_remove(destination.dynamic_data.enemyboats, i)
|
|
end
|
|
end
|
|
end
|
|
|
|
local function on_object_destroyed(event)
|
|
local registration_number = event.registration_number
|
|
|
|
local p
|
|
local biter_name
|
|
local surface_name
|
|
local memory
|
|
for i = 1, Common.starting_ships_count do
|
|
Memory.set_working_id(i)
|
|
memory = Memory.get_crew_memory()
|
|
if memory.elite_biters_stream_registrations then
|
|
for j, r in pairs(memory.elite_biters_stream_registrations) do
|
|
if r.number == registration_number then
|
|
p = r.position
|
|
biter_name = r.name
|
|
surface_name = r.surface_name
|
|
memory.elite_biters_stream_registrations[j] = nil
|
|
break
|
|
end
|
|
end
|
|
end
|
|
if p then
|
|
break
|
|
end
|
|
end
|
|
if p then
|
|
local surface = game.surfaces[surface_name]
|
|
if not (surface and surface.valid) then
|
|
return
|
|
end
|
|
|
|
local p2 = surface.find_non_colliding_position('medium-biter', p, 10, 0.2)
|
|
if not p2 then
|
|
return
|
|
end
|
|
|
|
surface.create_entity({ name = biter_name, position = p2, force = memory.enemy_force_name })
|
|
end
|
|
end
|
|
|
|
-- function Public.destroy_inactive_scripted_biters()
|
|
-- local memory = Memory.get_crew_memory()
|
|
-- local floating_pollution_accrued = 0
|
|
-- for unit_number, biter in pairs(memory.scripted_biters) do
|
|
-- if Public.is_biter_inactive(biter) then
|
|
-- memory.floating_pollution = memory.floating_pollution + CoreData.biterPollutionValues[biter.entity.name]
|
|
-- floating_pollution_accrued = floating_pollution_accrued + CoreData.biterPollutionValues[biter.entity.name]
|
|
-- memory.scripted_biters[unit_number] = nil
|
|
-- end
|
|
-- end
|
|
-- if _DEBUG and floating_pollution_accrued > 0 then game.print(game.tick .. string.format(":%f of spare pollution accrued", floating_pollution_accrued)) end
|
|
-- end
|
|
|
|
--=== Data
|
|
|
|
local event = require('utils.event')
|
|
event.add(defines.events.on_object_destroyed, on_object_destroyed)
|
|
|
|
return Public
|