-- 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