2020-12-05 19:10:53 +02:00
local Event = require ' utils.event '
2021-02-10 22:21:29 +02:00
local BiterHealthBooster = require ' modules.biter_health_booster_v2 '
2021-05-25 22:19:20 +02:00
local Difficulty = require ' modules.difficulty_vote_by_amount '
2020-08-09 20:23:45 +02:00
local BiterRolls = require ' modules.wave_defense.biter_rolls '
local SideTargets = require ' modules.wave_defense.side_targets '
local ThreatEvent = require ' modules.wave_defense.threat_events '
local update_gui = require ' modules.wave_defense.gui '
local threat_values = require ' modules.wave_defense.threat_values '
local WD = require ' modules.wave_defense.table '
2020-06-07 13:33:24 +02:00
local Alert = require ' utils.alert '
2020-10-30 23:05:05 +02:00
local Public = { }
2019-11-09 14:18:30 +02:00
local math_random = math.random
local math_floor = math.floor
local table_insert = table.insert
local math_sqrt = math.sqrt
local math_round = math.round
2019-10-28 18:38:36 +02:00
2019-11-09 14:18:30 +02:00
local group_size_modifier_raffle = { }
2020-08-09 20:23:45 +02:00
local group_size_chances = {
{ 4 , 0.4 } ,
{ 5 , 0.5 } ,
{ 6 , 0.6 } ,
{ 7 , 0.7 } ,
{ 8 , 0.8 } ,
{ 9 , 0.9 } ,
{ 10 , 1 } ,
{ 9 , 1.1 } ,
{ 8 , 1.2 } ,
{ 7 , 1.3 } ,
{ 6 , 1.4 } ,
{ 5 , 1.5 } ,
{ 4 , 1.6 } ,
{ 3 , 1.7 } ,
{ 2 , 1.8 }
}
2020-11-23 23:10:32 +02:00
2019-11-09 14:18:30 +02:00
for _ , v in pairs ( group_size_chances ) do
2020-10-30 23:05:05 +02:00
for _ = 1 , v [ 1 ] , 1 do
2020-08-09 20:23:45 +02:00
table_insert ( group_size_modifier_raffle , v [ 2 ] )
end
2019-11-09 14:18:30 +02:00
end
local group_size_modifier_raffle_size = # group_size_modifier_raffle
2019-10-08 01:17:00 +02:00
local function debug_print ( msg )
2020-08-17 20:18:06 +02:00
local debug = WD.get ( ' debug ' )
if not debug then
2020-08-09 20:23:45 +02:00
return
end
print ( ' WaveDefense: ' .. msg )
2019-10-08 01:17:00 +02:00
end
2020-11-23 23:10:32 +02:00
local function valid ( userdata )
if not ( userdata and userdata.valid ) then
return false
end
return true
end
2020-11-18 16:24:22 +02:00
2021-11-11 02:56:03 +02:00
local function shuffle ( tbl )
local size = # tbl
for i = size , 1 , - 1 do
local rand = math.random ( size )
tbl [ i ] , tbl [ rand ] = tbl [ rand ] , tbl [ i ]
end
return tbl
end
2020-11-23 23:10:32 +02:00
local function find_initial_spot ( surface , position )
local spot = WD.get ( ' spot ' )
if not spot then
2021-11-11 02:56:03 +02:00
local pos = surface.find_non_colliding_position ( ' stone-furnace ' , position , 128 , 1 )
2020-12-05 19:10:53 +02:00
if not pos then
2021-11-11 02:56:03 +02:00
pos = surface.find_non_colliding_position ( ' stone-furnace ' , position , 148 , 1 )
2020-12-05 19:10:53 +02:00
end
if not pos then
2021-11-11 02:56:03 +02:00
pos = surface.find_non_colliding_position ( ' stone-furnace ' , position , 164 , 1 )
end
if not pos then
pos = position
end
if math_random ( 1 , 2 ) == 1 then
local random_pos = { { x = pos.x - 10 , y = pos.y - 5 } , { x = pos.x + 10 , y = pos.y + 5 } , { x = pos.x - 10 , y = pos.y + 5 } , { x = pos.x + 10 , y = pos.y - 5 } }
local actual_pos = shuffle ( random_pos )
pos = { x = actual_pos [ 1 ] . x , y = actual_pos [ 1 ] . y }
2020-12-05 19:10:53 +02:00
end
2021-11-11 02:56:03 +02:00
2020-12-05 19:10:53 +02:00
if not pos then
pos = position
end
2020-11-23 23:10:32 +02:00
WD.set ( ' spot ' , pos )
return pos
else
spot = WD.get ( ' spot ' )
return spot
end
2020-10-22 13:32:18 +02:00
end
2020-11-18 16:24:22 +02:00
local function is_closer ( pos1 , pos2 , pos )
return ( ( pos1.x - pos.x ) ^ 2 + ( pos1.y - pos.y ) ^ 2 ) < ( ( pos2.x - pos.x ) ^ 2 + ( pos2.y - pos.y ) ^ 2 )
2020-11-17 16:03:14 +02:00
end
2020-10-22 13:32:18 +02:00
local function shuffle_distance ( tbl , position )
local size = # tbl
for i = size , 1 , - 1 do
local rand = math_random ( size )
if is_closer ( tbl [ i ] . position , tbl [ rand ] . position , position ) and i > rand then
tbl [ i ] , tbl [ rand ] = tbl [ rand ] , tbl [ i ]
end
end
return tbl
end
2020-11-23 23:10:32 +02:00
local function is_position_near ( pos_to_check , check_against )
local function inside ( pos )
return pos.x >= pos_to_check.x and pos.y >= pos_to_check.y and pos.x <= pos_to_check.x and pos.y <= pos_to_check.y
end
if inside ( check_against ) then
return true
end
return false
end
2020-08-04 12:10:15 +02:00
local function remove_trees ( entity )
2020-11-23 23:10:32 +02:00
if not valid ( entity ) then
return
end
2020-08-09 20:23:45 +02:00
local surface = entity.surface
2020-11-18 16:24:22 +02:00
local radius = 10
2020-08-09 20:23:45 +02:00
local pos = entity.position
local area = { { pos.x - radius , pos.y - radius } , { pos.x + radius , pos.y + radius } }
local trees = surface.find_entities_filtered { area = area , type = ' tree ' }
if # trees > 0 then
for i , tree in pairs ( trees ) do
if tree and tree.valid then
tree.destroy ( )
end
end
end
2020-08-04 12:10:15 +02:00
end
local function remove_rocks ( entity )
2020-11-23 23:10:32 +02:00
if not valid ( entity ) then
return
end
2020-08-09 20:23:45 +02:00
local surface = entity.surface
2020-11-18 16:24:22 +02:00
local radius = 10
2020-08-09 20:23:45 +02:00
local pos = entity.position
local area = { { pos.x - radius , pos.y - radius } , { pos.x + radius , pos.y + radius } }
local rocks = surface.find_entities_filtered { area = area , type = ' simple-entity ' }
if # rocks > 0 then
for i , rock in pairs ( rocks ) do
if rock and rock.valid then
rock.destroy ( )
end
end
end
end
2020-10-30 23:05:05 +02:00
2020-11-23 23:10:32 +02:00
local function fill_tiles ( entity , size )
if not valid ( entity ) then
return
end
2020-11-18 16:24:22 +02:00
local surface = entity.surface
2020-11-23 23:10:32 +02:00
local radius = size or 10
2020-11-18 16:24:22 +02:00
local pos = entity.position
local t = {
' water ' ,
' water-green ' ,
' water-mud ' ,
' water-shallow ' ,
' deepwater ' ,
' deepwater-green '
}
local area = { { pos.x - radius , pos.y - radius } , { pos.x + radius , pos.y + radius } }
local tiles = surface.find_tiles_filtered { area = area , name = t }
if # tiles > 0 then
for _ , tile in pairs ( tiles ) do
surface.set_tiles ( { { name = ' sand-1 ' , position = tile.position } } , true )
end
end
2020-11-23 23:10:32 +02:00
debug_print ( ' fill_tiles - filled tiles cause we found non-placable tiles. ' )
end
local function get_spawn_pos ( )
local surface_index = WD.get ( ' surface_index ' )
local surface = game.surfaces [ surface_index ]
if not surface then
return debug_print ( ' get_spawn_pos - surface was not valid? ' )
end
local c = 0
:: retry ::
2021-03-01 22:49:30 +02:00
local initial_position = WD.get ( ' spawn_position ' )
2020-11-23 23:10:32 +02:00
2021-03-01 22:49:30 +02:00
local located_position = find_initial_spot ( surface , initial_position )
2021-11-11 02:56:03 +02:00
local valid_position = surface.find_non_colliding_position ( ' steel-chest ' , located_position , 32 , 1 )
2021-03-01 22:49:30 +02:00
local debug = WD.get ( ' debug ' )
if debug then
if valid_position then
local x = valid_position.x
local y = valid_position.y
game.print ( ' [gps= ' .. x .. ' , ' .. y .. ' , ' .. surface.name .. ' ] ' )
end
end
if not valid_position then
2020-11-23 23:10:32 +02:00
local remove_entities = WD.get ( ' remove_entities ' )
if remove_entities then
c = c + 1
2021-03-01 22:49:30 +02:00
valid_position = WD.get ( ' spawn_position ' )
debug_print ( serpent.block ( ' valid_position - x: ' .. valid_position.x .. ' y: ' .. valid_position.y ) )
remove_trees ( { surface = surface , position = valid_position , valid = true } )
remove_rocks ( { surface = surface , position = valid_position , valid = true } )
fill_tiles ( { surface = surface , position = valid_position , valid = true } )
2020-11-23 23:10:32 +02:00
WD.set ( ' spot ' , ' nil ' )
if c == 5 then
return debug_print ( ' get_spawn_pos - we could not find a spawning pos? ' )
end
goto retry
else
return debug_print ( ' get_spawn_pos - we could not find a spawning pos? ' )
end
end
2021-03-01 22:49:30 +02:00
debug_print ( serpent.block ( ' valid_position - x: ' .. valid_position.x .. ' y: ' .. valid_position.y ) )
return valid_position
2020-11-18 16:24:22 +02:00
end
2019-10-15 04:20:40 +02:00
local function is_unit_valid ( biter )
2020-11-17 13:45:47 +02:00
local max_biter_age = WD.get ( ' max_biter_age ' )
2020-08-09 20:23:45 +02:00
if not biter.entity then
debug_print ( ' is_unit_valid - unit destroyed - does no longer exist ' )
return false
end
if not biter.entity . valid then
debug_print ( ' is_unit_valid - unit destroyed - invalid ' )
return false
end
if not biter.entity . unit_group then
debug_print ( ' is_unit_valid - unit destroyed - no unitgroup ' )
return false
end
2020-11-17 13:45:47 +02:00
if biter.spawn_tick + max_biter_age < game.tick then
2020-08-09 20:23:45 +02:00
debug_print ( ' is_unit_valid - unit destroyed - timed out ' )
return false
end
return true
2019-10-07 22:40:05 +02:00
end
2019-10-25 08:09:39 +02:00
local function refresh_active_unit_threat ( )
2020-11-17 13:45:47 +02:00
local active_biter_threat = WD.get ( ' active_biter_threat ' )
local active_biters = WD.get ( ' active_biters ' )
debug_print ( ' refresh_active_unit_threat - current value ' .. active_biter_threat )
local biter_threat = 0
for k , biter in pairs ( active_biters ) do
2020-11-23 23:10:32 +02:00
if valid ( biter.entity ) then
biter_threat = biter_threat + threat_values [ biter.entity . name ]
else
active_biters [ k ] = nil
2020-08-09 20:23:45 +02:00
end
end
2021-02-10 22:21:29 +02:00
local biter_health_boost = BiterHealthBooster.get ( ' biter_health_boost ' )
WD.set ( ' active_biter_threat ' , math_round ( biter_threat * biter_health_boost , 2 ) )
2020-11-17 13:45:47 +02:00
debug_print ( ' refresh_active_unit_threat - new value ' .. active_biter_threat )
2019-10-25 08:09:39 +02:00
end
2019-10-14 06:52:17 +02:00
local function time_out_biters ( )
2020-11-17 13:45:47 +02:00
local active_biters = WD.get ( ' active_biters ' )
local active_biter_count = WD.get ( ' active_biter_count ' )
local active_biter_threat = WD.get ( ' active_biter_threat ' )
2020-11-23 23:10:32 +02:00
if active_biter_count >= 100 and # active_biters <= 10 then
WD.set ( ' active_biter_count ' , 50 )
end
2021-02-10 22:21:29 +02:00
local biter_health_boost = BiterHealthBooster.get ( ' biter_health_boost ' )
2020-11-17 13:45:47 +02:00
for k , biter in pairs ( active_biters ) do
2020-08-09 20:23:45 +02:00
if not is_unit_valid ( biter ) then
2020-11-18 16:24:22 +02:00
WD.set ( ' active_biter_count ' , active_biter_count - 1 )
2020-11-23 23:10:32 +02:00
if biter.entity then
if biter.entity . valid then
2021-02-10 22:21:29 +02:00
WD.set ( ' active_biter_threat ' , active_biter_threat - math_round ( threat_values [ biter.entity . name ] * biter_health_boost , 2 ) )
2020-11-23 23:10:32 +02:00
if biter.entity . force.index == 2 then
biter.entity . destroy ( )
end
debug_print ( ' time_out_biters: ' .. k .. ' got deleted. ' )
2020-08-09 20:23:45 +02:00
end
end
2020-11-17 13:45:47 +02:00
active_biters [ k ] = nil
2020-08-09 20:23:45 +02:00
end
end
2019-10-07 04:37:23 +02:00
end
2020-10-22 12:44:14 +02:00
local function get_random_close_spawner ( )
2020-11-17 13:45:47 +02:00
local nests = WD.get ( ' nests ' )
local target = WD.get ( ' target ' )
local get_random_close_spawner_attempts = WD.get ( ' get_random_close_spawner_attempts ' )
local center = target.position
2020-10-22 12:44:14 +02:00
local spawner
2020-11-18 16:24:22 +02:00
local retries = 0
2020-11-17 13:45:47 +02:00
for i = 1 , get_random_close_spawner_attempts , 1 do
2020-10-22 12:44:14 +02:00
:: retry ::
2020-11-17 13:45:47 +02:00
if # nests < 1 then
2020-10-22 12:44:14 +02:00
return false
end
2020-11-17 13:45:47 +02:00
local k = math_random ( 1 , # nests )
local spawner_2 = nests [ k ]
2020-10-22 17:39:23 +02:00
if not spawner_2 or not spawner_2.valid then
2020-11-17 13:45:47 +02:00
nests [ k ] = nil
2020-11-18 16:24:22 +02:00
retries = retries + 1
if retries == 5 then
break
end
2020-10-22 12:44:14 +02:00
goto retry
end
2020-12-05 19:10:53 +02:00
if not spawner or ( 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
2020-08-09 20:23:45 +02:00
spawner = spawner_2
2021-11-12 19:09:06 +02:00
if spawner and spawner.position then
debug_print ( ' get_random_close_spawner - Found at x ' .. spawner.position . x .. ' y ' .. spawner.position . y )
end
2020-08-09 20:23:45 +02:00
end
end
return spawner
2019-10-07 01:46:26 +02:00
end
2020-11-17 13:45:47 +02:00
local function get_random_character ( )
2020-08-09 20:23:45 +02:00
local characters = { }
2020-11-17 13:45:47 +02:00
local surface_index = WD.get ( ' surface_index ' )
2020-10-22 12:44:14 +02:00
local p = game.connected_players
for _ , player in pairs ( p ) do
2020-08-09 20:23:45 +02:00
if player.character then
if player.character . valid then
2020-11-17 13:45:47 +02:00
if player.character . surface.index == surface_index then
2020-08-09 20:23:45 +02:00
characters [ # characters + 1 ] = player.character
end
end
end
end
if not characters [ 1 ] then
return
end
return characters [ math_random ( 1 , # characters ) ]
2020-01-01 15:05:49 +02:00
end
2019-10-25 08:09:39 +02:00
local function set_main_target ( )
2020-11-17 13:45:47 +02:00
local target = WD.get ( ' target ' )
if target then
if target.valid then
2020-08-09 20:23:45 +02:00
return
end
end
2021-11-11 02:56:03 +02:00
local unit_groups_size = WD.get ( ' unit_groups_size ' )
if unit_groups_size < 0 then
unit_groups_size = 0
end
WD.set ( ' unit_groups_size ' , unit_groups_size )
2020-11-17 13:45:47 +02:00
local sec_target = SideTargets.get_side_target ( )
if not sec_target then
sec_target = get_random_character ( )
2020-08-09 20:23:45 +02:00
end
2020-11-17 13:45:47 +02:00
if not sec_target then
2020-08-09 20:23:45 +02:00
return
end
2020-11-17 13:45:47 +02:00
WD.set ( ' target ' , sec_target )
2020-12-05 19:10:53 +02:00
debug_print ( ' set_main_target -- New main target ' .. sec_target.name .. ' at position x ' .. sec_target.position . x .. ' y ' .. sec_target.position . y .. ' selected. ' )
2019-10-25 08:09:39 +02:00
end
2019-10-12 04:06:48 +02:00
local function set_group_spawn_position ( surface )
2020-10-22 12:44:14 +02:00
local spawner = get_random_close_spawner ( )
2020-08-09 20:23:45 +02:00
if not spawner then
return
end
2020-11-23 23:10:32 +02:00
local position = surface.find_non_colliding_position ( ' behemoth-biter ' , spawner.position , 128 , 1 )
2020-08-09 20:23:45 +02:00
if not position then
return
end
2020-11-17 13:45:47 +02:00
WD.set ( ' spawn_position ' , { x = position.x , y = position.y } )
2020-11-18 16:24:22 +02:00
local spawn_position = get_spawn_pos ( )
2020-11-17 13:45:47 +02:00
debug_print ( ' set_group_spawn_position -- Changed position to x ' .. spawn_position.x .. ' y ' .. spawn_position.y .. ' . ' )
2019-10-07 01:46:26 +02:00
end
local function set_enemy_evolution ( )
2020-11-17 13:45:47 +02:00
local wave_number = WD.get ( ' wave_number ' )
local biter_health_boost = WD.get ( ' biter_health_boost ' )
local threat = WD.get ( ' threat ' )
local evolution_factor = wave_number * 0.001
local biter_h_boost = 1
2021-05-23 17:03:52 +02:00
local enemy = game.forces . enemy
2020-08-09 20:23:45 +02:00
if evolution_factor > 1 then
evolution_factor = 1
end
2020-11-17 13:45:47 +02:00
if biter_health_boost then
2020-11-23 23:10:32 +02:00
biter_h_boost = math_round ( biter_health_boost + ( threat - 5000 ) * 0.000044 , 3 )
2020-10-25 18:02:37 +02:00
else
2020-11-23 23:10:32 +02:00
biter_h_boost = math_round ( biter_h_boost + ( threat - 5000 ) * 0.000044 , 3 )
2020-10-25 18:02:37 +02:00
end
2020-11-17 13:45:47 +02:00
if biter_h_boost <= 1 then
biter_h_boost = 1
2020-10-30 23:05:05 +02:00
end
2020-08-09 20:23:45 +02:00
2021-02-10 22:21:29 +02:00
BiterHealthBooster.set ( ' biter_health_boost ' , biter_h_boost )
2021-05-23 17:03:52 +02:00
if enemy.evolution_factor == 1 and evolution_factor == 1 then
return
end
enemy.evolution_factor = evolution_factor
2019-10-25 08:09:39 +02:00
end
local function can_units_spawn ( )
2020-11-17 13:45:47 +02:00
local threat = WD.get ( ' threat ' )
if threat <= 0 then
2020-08-09 20:23:45 +02:00
debug_print ( ' can_units_spawn - threat too low ' )
2020-11-23 23:10:32 +02:00
time_out_biters ( )
2020-08-09 20:23:45 +02:00
return false
end
2020-11-17 13:45:47 +02:00
local active_biter_count = WD.get ( ' active_biter_count ' )
local max_active_biters = WD.get ( ' max_active_biters ' )
if active_biter_count >= max_active_biters then
2020-08-09 20:23:45 +02:00
debug_print ( ' can_units_spawn - active biter count too high ' )
2020-11-23 23:10:32 +02:00
time_out_biters ( )
2020-08-09 20:23:45 +02:00
return false
end
2020-11-17 13:45:47 +02:00
local active_biter_threat = WD.get ( ' active_biter_threat ' )
if active_biter_threat >= threat then
debug_print ( ' can_units_spawn - active biter threat too high ( ' .. active_biter_threat .. ' ) ' )
2020-11-23 23:10:32 +02:00
time_out_biters ( )
2020-08-09 20:23:45 +02:00
return false
end
return true
2019-10-25 08:09:39 +02:00
end
local function get_active_unit_groups_count ( )
2020-11-17 13:45:47 +02:00
local unit_groups = WD.get ( ' unit_groups ' )
2020-08-09 20:23:45 +02:00
local count = 0
2020-11-23 23:10:32 +02:00
for k , g in pairs ( unit_groups ) do
2020-08-09 20:23:45 +02:00
if g.valid then
if # g.members > 0 then
count = count + 1
else
g.destroy ( )
2021-11-11 02:56:03 +02:00
unit_groups [ k ] = nil
local unit_groups_size = WD.get ( ' unit_groups_size ' )
WD.set ( ' unit_groups_size ' , unit_groups_size - 1 )
2020-08-09 20:23:45 +02:00
end
2020-11-23 23:10:32 +02:00
else
unit_groups [ k ] = nil
2021-11-11 02:56:03 +02:00
local unit_groups_size = WD.get ( ' unit_groups_size ' )
WD.set ( ' unit_groups_size ' , unit_groups_size - 1 )
2020-11-23 23:10:32 +02:00
local unit_group_last_command = WD.get ( ' unit_group_last_command ' )
if unit_group_last_command [ k ] then
unit_group_last_command [ k ] = nil
end
local unit_group_pos = WD.get ( ' unit_group_pos ' )
local positions = unit_group_pos.positions
if positions [ k ] then
positions [ k ] = nil
end
2020-08-09 20:23:45 +02:00
end
end
debug_print ( ' Active unit group count: ' .. count )
return count
2019-10-07 01:46:26 +02:00
end
2021-11-11 02:56:03 +02:00
local function spawn_biter ( surface , position , forceSpawn , is_boss_biter )
if not forceSpawn then
if not is_boss_biter then
if not can_units_spawn ( ) then
return
end
2020-08-09 20:23:45 +02:00
end
end
2021-02-10 22:21:29 +02:00
local boosted_health = BiterHealthBooster.get ( ' biter_health_boost ' )
2020-11-17 13:45:47 +02:00
2020-08-09 20:23:45 +02:00
local name
if math_random ( 1 , 100 ) > 73 then
name = BiterRolls.wave_defense_roll_spitter_name ( )
else
name = BiterRolls.wave_defense_roll_biter_name ( )
end
2021-11-11 02:56:03 +02:00
local old_position = position
position = surface.find_non_colliding_position ( ' steel-chest ' , position , 10 , 1 )
if not position then
position = old_position
end
2020-11-17 13:45:47 +02:00
2020-11-18 16:24:22 +02:00
local biter = surface.create_entity ( { name = name , position = position , force = ' enemy ' } )
2020-08-09 20:23:45 +02:00
biter.ai_settings . allow_destroy_when_commands_fail = true
2021-11-11 02:56:03 +02:00
biter.ai_settings . allow_try_return_to_spawner = false
2020-08-21 13:56:01 +02:00
biter.ai_settings . do_separation = true
2020-11-17 13:45:47 +02:00
2021-02-10 22:21:29 +02:00
local increase_health_per_wave = WD.get ( ' increase_health_per_wave ' )
if increase_health_per_wave and not is_boss_biter then
local modified_unit_health = WD.get ( ' modified_unit_health ' )
2021-02-15 23:37:15 +02:00
BiterHealthBooster.add_unit ( biter , modified_unit_health.current_value )
2021-02-10 22:21:29 +02:00
end
2020-08-09 20:23:45 +02:00
if is_boss_biter then
2021-02-10 22:21:29 +02:00
local increase_boss_health_per_wave = WD.get ( ' increase_boss_health_per_wave ' )
if increase_boss_health_per_wave then
local modified_boss_unit_health = WD.get ( ' modified_boss_unit_health ' )
BiterHealthBooster.add_boss_unit ( biter , modified_boss_unit_health , 0.55 )
2020-11-17 13:45:47 +02:00
else
local sum = boosted_health * 5
debug_print ( ' Boss Health Boosted: ' .. sum )
BiterHealthBooster.add_boss_unit ( biter , sum , 0.55 )
end
2020-08-09 20:23:45 +02:00
end
2021-02-10 22:21:29 +02:00
2020-11-17 13:45:47 +02:00
WD.set ( ' active_biters ' ) [ biter.unit_number ] = { entity = biter , spawn_tick = game.tick }
local active_biter_count = WD.get ( ' active_biter_count ' )
WD.set ( ' active_biter_count ' , active_biter_count + 1 )
local active_biter_threat = WD.get ( ' active_biter_threat ' )
WD.set ( ' active_biter_threat ' , active_biter_threat + math_round ( threat_values [ name ] * boosted_health , 2 ) )
2020-08-09 20:23:45 +02:00
return biter
2019-10-07 01:46:26 +02:00
end
2020-12-05 19:10:53 +02:00
local function increase_biter_damage ( )
2021-02-04 21:18:31 +02:00
local increase_damage_per_wave = WD.get ( ' increase_damage_per_wave ' )
if not increase_damage_per_wave then
return
end
2020-12-05 19:10:53 +02:00
local e = game.forces . enemy
2021-02-14 22:43:31 +02:00
local new = Difficulty.get ( ) . difficulty_vote_value * 0.04
local melee = new
local bio = new - 0.02
2020-12-05 19:10:53 +02:00
local e_old_melee = e.get_ammo_damage_modifier ( ' melee ' )
local e_old_biological = e.get_ammo_damage_modifier ( ' biological ' )
2021-02-14 22:43:31 +02:00
debug_print ( ' Melee: ' .. melee + e_old_melee )
debug_print ( ' Biological: ' .. bio + e_old_biological )
e.set_ammo_damage_modifier ( ' melee ' , melee + e_old_melee )
e.set_ammo_damage_modifier ( ' biological ' , bio + e_old_biological )
2020-12-05 19:10:53 +02:00
end
2021-02-10 22:21:29 +02:00
local function increase_biters_health ( )
local increase_health_per_wave = WD.get ( ' increase_health_per_wave ' )
if not increase_health_per_wave then
return
end
local boosted_health = BiterHealthBooster.get ( ' biter_health_boost ' )
local wave_number = WD.get ( ' wave_number ' )
-- this sets normal units health
local modified_unit_health = WD.get ( ' modified_unit_health ' )
2021-02-15 23:37:15 +02:00
if modified_unit_health.current_value > modified_unit_health.limit_value then
modified_unit_health.current_value = modified_unit_health.limit_value
2021-02-10 22:21:29 +02:00
end
2021-02-15 23:37:15 +02:00
debug_print ( ' [HEALTHBOOSTER] > Normal Units Health Boosted: ' .. modified_unit_health.current_value )
WD.set ( ' modified_unit_health ' ) . current_value = modified_unit_health.current_value + modified_unit_health.health_increase_per_boss_wave
2021-02-10 22:21:29 +02:00
-- this sets boss units health
if boosted_health == 1 then
2021-03-01 22:49:30 +02:00
boosted_health = 1.25
2021-02-10 22:21:29 +02:00
end
2021-03-01 22:49:30 +02:00
boosted_health = boosted_health * ( wave_number * 0.04 )
local sum = boosted_health * 5
2021-02-10 22:21:29 +02:00
debug_print ( ' [HEALTHBOOSTER] > Boss Health Boosted: ' .. sum )
2021-02-25 22:33:37 +02:00
if sum >= 300 then
sum = 300
2021-02-10 22:21:29 +02:00
end
WD.set ( ' modified_boss_unit_health ' , sum )
end
2019-10-07 22:40:05 +02:00
local function set_next_wave ( )
2020-11-17 13:45:47 +02:00
local wave_number = WD.get ( ' wave_number ' )
WD.set ( ' wave_number ' , wave_number + 1 )
wave_number = WD.get ( ' wave_number ' )
local threat_gain_multiplier = WD.get ( ' threat_gain_multiplier ' )
local threat_gain = wave_number * threat_gain_multiplier
if wave_number > 1000 then
threat_gain = threat_gain * ( wave_number * 0.001 )
end
if wave_number % 25 == 0 then
2020-12-05 19:10:53 +02:00
increase_biter_damage ( )
2021-02-10 22:21:29 +02:00
increase_biters_health ( )
2020-11-17 13:45:47 +02:00
WD.set ( ' boss_wave ' , true )
WD.set ( ' boss_wave_warning ' , true )
local alert_boss_wave = WD.get ( ' alert_boss_wave ' )
2020-11-18 16:24:22 +02:00
local spawn_position = get_spawn_pos ( )
2020-11-17 13:45:47 +02:00
if alert_boss_wave then
local msg = ' Boss Wave: ' .. wave_number
2020-08-09 20:23:45 +02:00
local pos = {
2020-11-17 13:45:47 +02:00
position = spawn_position
2020-08-09 20:23:45 +02:00
}
Alert.alert_all_players_location ( pos , msg , { r = 0.8 , g = 0.1 , b = 0.1 } )
end
threat_gain = threat_gain * 2
else
2020-11-17 13:45:47 +02:00
local boss_wave_warning = WD.get ( ' boss_wave_warning ' )
if boss_wave_warning then
WD.set ( ' boss_wave_warning ' , false )
2020-08-09 20:23:45 +02:00
end
end
2020-11-17 13:45:47 +02:00
local threat = WD.get ( ' threat ' )
WD.set ( ' threat ' , threat + math_floor ( threat_gain ) )
local wave_enforced = WD.get ( ' wave_enforced ' )
local next_wave = WD.get ( ' next_wave ' )
local wave_interval = WD.get ( ' wave_interval ' )
if not wave_enforced then
WD.set ( ' last_wave ' , next_wave )
WD.set ( ' next_wave ' , game.tick + wave_interval )
2020-10-24 14:46:14 +02:00
end
2020-11-17 13:45:47 +02:00
local clear_corpses = WD.get ( ' clear_corpses ' )
if clear_corpses then
local surface_index = WD.get ( ' surface_index ' )
local surface = game.surfaces [ surface_index ]
2020-08-09 20:23:45 +02:00
for _ , entity in pairs ( surface.find_entities_filtered { type = ' corpse ' } ) do
if math_random ( 1 , 2 ) == 1 then
entity.destroy ( )
end
end
end
2019-10-07 22:40:05 +02:00
end
2020-04-27 15:50:18 +02:00
local function reform_group ( group )
2020-11-17 13:45:47 +02:00
local unit_group_command_step_length = WD.get ( ' unit_group_command_step_length ' )
2020-08-09 20:23:45 +02:00
local group_position = { x = group.position . x , y = group.position . y }
2020-11-17 13:45:47 +02:00
local step_length = unit_group_command_step_length
2020-08-09 20:23:45 +02:00
local position = group.surface . find_non_colliding_position ( ' biter-spawner ' , group_position , step_length , 4 )
if position then
local new_group = group.surface . create_unit_group { position = position , force = group.force }
for key , biter in pairs ( group.members ) do
new_group.add_member ( biter )
end
debug_print ( ' Creating new unit group, because old one was stuck. ' )
2020-11-17 13:45:47 +02:00
local unit_groups = WD.get ( ' unit_groups ' )
unit_groups [ new_group.group_number ] = new_group
2021-11-11 02:56:03 +02:00
local unit_groups_size = WD.get ( ' unit_groups_size ' )
WD.set ( ' unit_groups_size ' , unit_groups_size + 1 )
2020-10-22 12:44:14 +02:00
2020-08-09 20:23:45 +02:00
return new_group
else
debug_print ( ' Destroying stuck group. ' )
2020-11-17 13:45:47 +02:00
local unit_groups = WD.get ( ' unit_groups ' )
if unit_groups [ group.group_number ] then
2020-11-23 23:10:32 +02:00
local unit_group_last_command = WD.get ( ' unit_group_last_command ' )
if unit_group_last_command [ group.group_number ] then
unit_group_last_command [ group.group_number ] = nil
2020-10-22 12:44:14 +02:00
end
2020-11-23 23:10:32 +02:00
local unit_group_pos = WD.get ( ' unit_group_pos ' )
local positions = unit_group_pos.positions
if positions [ group.group_number ] then
positions [ group.group_number ] = nil
end
table.remove ( unit_groups , group.group_number )
2021-11-11 02:56:03 +02:00
local unit_groups_size = WD.get ( ' unit_groups_size ' )
WD.set ( ' unit_groups_size ' , unit_groups_size - 1 )
2020-10-22 12:44:14 +02:00
end
2020-08-09 20:23:45 +02:00
group.destroy ( )
end
return nil
2020-04-27 15:50:18 +02:00
end
2020-11-23 23:10:32 +02:00
local function get_side_targets ( group )
2020-11-17 13:45:47 +02:00
local unit_group_command_step_length = WD.get ( ' unit_group_command_step_length ' )
2020-11-23 23:10:32 +02:00
2020-08-09 20:23:45 +02:00
local commands = { }
local group_position = { x = group.position . x , y = group.position . y }
2020-11-17 13:45:47 +02:00
local step_length = unit_group_command_step_length
2020-08-09 20:23:45 +02:00
2020-11-23 23:10:32 +02:00
local side_target = SideTargets.get_side_target ( )
local target_position = side_target.position
2020-12-05 19:10:53 +02:00
local distance_to_target = math_floor ( math_sqrt ( ( target_position.x - group_position.x ) ^ 2 + ( target_position.y - group_position.y ) ^ 2 ) )
2020-11-23 23:10:32 +02:00
local steps = math_floor ( distance_to_target / step_length ) + 1
2020-11-17 13:45:47 +02:00
2020-11-23 23:10:32 +02:00
for i = 1 , steps , 1 do
local old_position = group_position
local obstacles =
group.surface . find_entities_filtered {
position = old_position ,
radius = step_length * 2 ,
type = { ' simple-entity ' , ' tree ' } ,
limit = 100
}
if obstacles then
for v = 1 , # obstacles , 1 do
if obstacles [ v ] . valid then
2020-10-30 23:05:05 +02:00
commands [ # commands + 1 ] = {
2020-11-23 23:10:32 +02:00
type = defines.command . attack ,
destination = obstacles [ v ] . position ,
2020-10-30 23:05:05 +02:00
distraction = defines.distraction . by_anything
}
end
end
2020-08-09 20:23:45 +02:00
end
2020-11-23 23:10:32 +02:00
commands [ # commands + 1 ] = {
type = defines.command . attack ,
target = side_target ,
distraction = defines.distraction . by_anything
}
2020-08-09 20:23:45 +02:00
end
2020-11-23 23:10:32 +02:00
return commands
end
local function get_main_command ( group )
local unit_group_command_step_length = WD.get ( ' unit_group_command_step_length ' )
local commands = { }
local group_position = { x = group.position . x , y = group.position . y }
local step_length = unit_group_command_step_length
2020-11-17 13:45:47 +02:00
local target = WD.get ( ' target ' )
2020-11-23 23:10:32 +02:00
if not valid ( target ) then
2020-11-17 13:45:47 +02:00
return
end
2020-11-23 23:10:32 +02:00
debug_print ( ' get_main_command - starting ' )
2020-11-17 13:45:47 +02:00
local target_position = target.position
2020-12-05 19:10:53 +02:00
local distance_to_target = math_floor ( math_sqrt ( ( target_position.x - group_position.x ) ^ 2 + ( target_position.y - group_position.y ) ^ 2 ) )
2020-11-23 23:10:32 +02:00
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 )
}
debug_print ( ' get_commmands - to main target x ' .. target_position.x .. ' y ' .. target_position.y )
debug_print ( ' get_commmands - distance_to_target: ' .. distance_to_target .. ' steps: ' .. steps )
debug_print ( ' get_commmands - vector ' .. vector [ 1 ] .. ' _ ' .. vector [ 2 ] )
2020-08-09 20:23:45 +02:00
2020-11-23 23:10:32 +02:00
for i = 1 , steps , 1 do
local old_position = group_position
group_position.x = group_position.x + vector [ 1 ]
group_position.y = group_position.y + vector [ 2 ]
2020-08-09 20:23:45 +02:00
local obstacles =
group.surface . find_entities_filtered {
2020-11-23 23:10:32 +02:00
position = old_position ,
2020-08-09 20:23:45 +02:00
radius = step_length / 2 ,
type = { ' simple-entity ' , ' tree ' } ,
2020-10-22 13:32:18 +02:00
limit = 50
2020-08-09 20:23:45 +02:00
}
if obstacles then
2020-11-23 23:10:32 +02:00
shuffle_distance ( obstacles , old_position )
2021-03-24 21:14:55 +02:00
for ii = 1 , # obstacles , 1 do
if obstacles [ ii ] . valid then
2020-08-09 20:23:45 +02:00
commands [ # commands + 1 ] = {
type = defines.command . attack ,
2021-03-24 21:14:55 +02:00
target = obstacles [ ii ] ,
2020-11-23 23:10:32 +02:00
distraction = defines.distraction . by_anything
2020-08-09 20:23:45 +02:00
}
end
end
end
2020-09-17 09:14:07 +02:00
local position = group.surface . find_non_colliding_position ( ' behemoth-biter ' , group_position , step_length , 1 )
2020-08-09 20:23:45 +02:00
if position then
commands [ # commands + 1 ] = {
type = defines.command . attack_area ,
destination = { x = position.x , y = position.y } ,
2020-11-23 23:10:32 +02:00
radius = 16 ,
2020-08-09 20:23:45 +02:00
distraction = defines.distraction . by_anything
}
end
end
commands [ # commands + 1 ] = {
type = defines.command . attack_area ,
destination = { x = target_position.x , y = target_position.y } ,
2020-11-23 23:10:32 +02:00
radius = 8 ,
distraction = defines.distraction . by_anything
2020-08-09 20:23:45 +02:00
}
commands [ # commands + 1 ] = {
type = defines.command . attack ,
2020-11-17 13:45:47 +02:00
target = target ,
2020-11-23 23:10:32 +02:00
distraction = defines.distraction . by_anything
2020-08-09 20:23:45 +02:00
}
return commands
2019-10-08 01:17:00 +02:00
end
2020-11-23 23:10:32 +02:00
local function command_to_main_target ( group , bypass )
if not valid ( group ) then
return
end
local unit_group_last_command = WD.get ( ' unit_group_last_command ' )
local unit_group_command_delay = WD.get ( ' unit_group_command_delay ' )
if not bypass then
if not unit_group_last_command [ group.group_number ] then
unit_group_last_command [ group.group_number ] = game.tick - ( unit_group_command_delay + 1 )
end
if unit_group_last_command [ group.group_number ] then
if unit_group_last_command [ group.group_number ] + unit_group_command_delay > game.tick then
return
end
end
end
local fill_tiles_so_biter_can_path = WD.get ( ' fill_tiles_so_biter_can_path ' )
if fill_tiles_so_biter_can_path then
fill_tiles ( group , 10 )
end
local tile = group.surface . get_tile ( group.position )
if tile.valid and tile.collides_with ( ' player-layer ' ) then
group = reform_group ( group )
end
if not valid ( group ) then
return
end
local commands = get_main_command ( group )
debug_print ( ' get_main_command - got commands ' )
2021-07-07 12:22:22 +02:00
local surface_index = WD.get ( ' surface_index ' )
if group.surface . index ~= surface_index then
return
end
2020-11-23 23:10:32 +02:00
group.set_command (
{
type = defines.command . compound ,
structure_type = defines.compound_command . return_last ,
commands = commands
}
)
debug_print ( ' get_main_command - sent commands ' )
if valid ( group ) then
unit_group_last_command [ group.group_number ] = game.tick
end
end
local function command_to_side_target ( group )
2020-11-17 13:45:47 +02:00
local unit_group_last_command = WD.get ( ' unit_group_last_command ' )
local unit_group_command_delay = WD.get ( ' unit_group_command_delay ' )
if not unit_group_last_command [ group.group_number ] then
unit_group_last_command [ group.group_number ] = game.tick - ( unit_group_command_delay + 1 )
2020-08-09 20:23:45 +02:00
end
2020-10-18 12:45:18 +02:00
2020-11-17 13:45:47 +02:00
if unit_group_last_command [ group.group_number ] then
if unit_group_last_command [ group.group_number ] + unit_group_command_delay > game.tick then
2020-08-09 20:23:45 +02:00
return
end
end
2020-10-18 12:45:18 +02:00
2020-08-09 20:23:45 +02:00
local tile = group.surface . get_tile ( group.position )
if tile.valid and tile.collides_with ( ' player-layer ' ) then
group = reform_group ( group )
end
2020-11-23 23:10:32 +02:00
local commands = get_side_targets ( group )
2020-08-09 20:23:45 +02:00
group.set_command (
{
type = defines.command . compound ,
2020-10-22 13:32:18 +02:00
structure_type = defines.compound_command . return_last ,
2020-11-23 23:10:32 +02:00
commands = commands
2020-08-09 20:23:45 +02:00
}
)
2020-10-30 23:05:05 +02:00
2020-11-17 13:45:47 +02:00
unit_group_last_command [ group.group_number ] = game.tick
2019-10-07 22:40:05 +02:00
end
2020-11-23 23:10:32 +02:00
local function give_side_commands_to_group ( )
local enable_side_target = WD.get ( ' enable_side_target ' )
if not enable_side_target then
2020-08-09 20:23:45 +02:00
return
end
2020-11-23 23:10:32 +02:00
2020-11-17 13:45:47 +02:00
local target = WD.get ( ' target ' )
2020-11-23 23:10:32 +02:00
if not valid ( target ) then
2020-08-09 20:23:45 +02:00
return
end
2020-11-17 13:45:47 +02:00
local unit_groups = WD.get ( ' unit_groups ' )
for k , group in pairs ( unit_groups ) do
2020-10-22 12:44:14 +02:00
if type ( group ) ~= ' number ' then
if group.valid then
2020-11-23 23:10:32 +02:00
command_to_side_target ( group )
else
get_active_unit_groups_count ( )
2020-10-22 12:44:14 +02:00
end
2020-08-09 20:23:45 +02:00
end
end
2019-10-07 04:37:23 +02:00
end
2020-11-23 23:10:32 +02:00
local function give_main_command_to_group ( )
2020-11-18 16:24:22 +02:00
local target = WD.get ( ' target ' )
2020-11-23 23:10:32 +02:00
if not valid ( target ) then
2020-11-18 16:24:22 +02:00
return
end
local unit_groups = WD.get ( ' unit_groups ' )
for k , group in pairs ( unit_groups ) do
2020-11-23 23:10:32 +02:00
if type ( group ) ~= ' number ' then
if group.valid then
2021-02-01 21:01:07 +02:00
if group.surface . index == target.surface . index then
command_to_main_target ( group )
end
2020-11-23 23:10:32 +02:00
else
get_active_unit_groups_count ( )
2020-11-18 16:24:22 +02:00
end
end
end
end
2021-11-11 02:56:03 +02:00
local function spawn_unit_group ( fs )
if fs then
debug_print ( ' spawn_unit_group - forcing new biters ' )
else
if not can_units_spawn ( ) then
debug_print ( ' spawn_unit_group - Cant spawn units? ' )
return
end
2020-08-09 20:23:45 +02:00
end
2020-11-17 13:45:47 +02:00
local target = WD.get ( ' target ' )
2020-11-23 23:10:32 +02:00
if not valid ( target ) then
2020-11-18 16:24:22 +02:00
debug_print ( ' spawn_unit_group - Target was not valid? ' )
2020-08-09 20:23:45 +02:00
return
end
2020-11-17 13:45:47 +02:00
local max_active_unit_groups = WD.get ( ' max_active_unit_groups ' )
2021-11-11 02:56:03 +02:00
if fs then
debug_print ( ' spawn_unit_group - forcing new biters ' )
else
if get_active_unit_groups_count ( ) >= max_active_unit_groups then
debug_print ( ' spawn_unit_group - unit_groups at max ' )
return
end
2020-08-09 20:23:45 +02:00
end
2020-11-17 13:45:47 +02:00
local surface_index = WD.get ( ' surface_index ' )
2020-11-23 23:10:32 +02:00
local remove_entities = WD.get ( ' remove_entities ' )
2020-11-17 13:45:47 +02:00
local surface = game.surfaces [ surface_index ]
2020-08-09 20:23:45 +02:00
set_group_spawn_position ( surface )
2020-11-18 16:24:22 +02:00
local spawn_position = get_spawn_pos ( )
2020-11-23 23:10:32 +02:00
if not spawn_position then
return
2020-08-09 20:23:45 +02:00
end
local radius = 10
2020-11-18 16:24:22 +02:00
local area = {
left_top = { spawn_position.x - radius , spawn_position.y - radius } ,
right_bottom = { spawn_position.x + radius , spawn_position.y + radius }
}
2020-08-09 20:23:45 +02:00
for k , v in pairs ( surface.find_entities_filtered { area = area , name = ' land-mine ' } ) do
if v and v.valid then
2020-11-18 16:24:22 +02:00
debug_print ( ' spawn_unit_group - found land-mines ' )
2020-08-09 20:23:45 +02:00
v.die ( )
end
end
2020-11-23 23:10:32 +02:00
if remove_entities then
remove_trees ( { surface = surface , position = spawn_position , valid = true } )
remove_rocks ( { surface = surface , position = spawn_position , valid = true } )
fill_tiles ( { surface = surface , position = spawn_position , valid = true } )
end
2020-11-17 13:45:47 +02:00
local wave_number = WD.get ( ' wave_number ' )
BiterRolls.wave_defense_set_unit_raffle ( wave_number )
2020-08-09 20:23:45 +02:00
2020-11-17 13:45:47 +02:00
debug_print ( ' Spawning unit group at x ' .. spawn_position.x .. ' y ' .. spawn_position.y )
2020-08-09 20:23:45 +02:00
2020-11-23 23:10:32 +02:00
local unit_group_pos = WD.get ( ' unit_group_pos ' )
2021-11-11 02:56:03 +02:00
local unit_group = surface.create_unit_group ( { position = spawn_position , force = ' enemy ' } )
2020-11-23 23:10:32 +02:00
unit_group_pos.positions [ unit_group.group_number ] = { position = unit_group.position , index = 0 }
2020-11-17 13:45:47 +02:00
local average_unit_group_size = WD.get ( ' average_unit_group_size ' )
local group_size = math_floor ( average_unit_group_size * group_size_modifier_raffle [ math_random ( 1 , group_size_modifier_raffle_size ) ] )
2020-08-09 20:23:45 +02:00
for _ = 1 , group_size , 1 do
2021-11-11 02:56:03 +02:00
local biter = spawn_biter ( surface , spawn_position , fs )
2020-08-09 20:23:45 +02:00
if not biter then
2020-11-18 16:24:22 +02:00
debug_print ( ' spawn_unit_group - No biters were found? ' )
2020-08-09 20:23:45 +02:00
break
end
unit_group.add_member ( biter )
2020-11-23 23:10:32 +02:00
-- command_to_side_target(unit_group)
2020-08-09 20:23:45 +02:00
end
2020-11-17 13:45:47 +02:00
local boss_wave = WD.get ( ' boss_wave ' )
2021-11-13 00:27:51 +02:00
if boss_wave then
2020-11-17 13:45:47 +02:00
local count = math_random ( 1 , math_floor ( wave_number * 0.01 ) + 2 )
if count > 16 then
count = 16
end
if count <= 1 then
count = 4
2020-08-09 20:23:45 +02:00
end
for _ = 1 , count , 1 do
2021-11-11 02:56:03 +02:00
local biter = spawn_biter ( surface , spawn_position , fs , true )
2020-08-09 20:23:45 +02:00
if not biter then
2020-11-18 16:24:22 +02:00
debug_print ( ' spawn_unit_group - No biters were found? ' )
2020-08-09 20:23:45 +02:00
break
end
unit_group.add_member ( biter )
end
2020-11-17 13:45:47 +02:00
WD.set ( ' boss_wave ' , false )
end
local unit_groups = WD.get ( ' unit_groups ' )
unit_groups [ unit_group.group_number ] = unit_group
2021-11-11 02:56:03 +02:00
local unit_groups_size = WD.get ( ' unit_groups_size ' )
WD.set ( ' unit_groups_size ' , unit_groups_size + 1 )
2020-10-22 12:44:14 +02:00
if math_random ( 1 , 2 ) == 1 then
2021-11-11 02:56:03 +02:00
WD.set ( ' random_group ' , unit_group )
2020-10-22 12:44:14 +02:00
end
2020-11-23 23:10:32 +02:00
WD.set ( ' spot ' , ' nil ' )
2020-08-09 20:23:45 +02:00
return true
2019-10-27 20:26:55 +02:00
end
2020-11-23 23:10:32 +02:00
local function check_group_positions ( )
local unit_groups = WD.get ( ' unit_groups ' )
local unit_group_pos = WD.get ( ' unit_group_pos ' )
local target = WD.get ( ' target ' )
if not valid ( target ) then
return
end
for k , group in pairs ( unit_groups ) do
if group.valid then
local ugp = unit_group_pos.positions
if group.state == defines.group_state . finished then
return command_to_main_target ( group , true )
end
if ugp [ group.group_number ] then
local success = is_position_near ( group.position , ugp [ group.group_number ] . position )
if success then
ugp [ group.group_number ] . index = ugp [ group.group_number ] . index + 1
if ugp [ group.group_number ] . index >= 2 then
command_to_main_target ( group , true )
fill_tiles ( group , 30 )
remove_rocks ( group )
remove_trees ( group )
if ugp [ group.group_number ] . index >= 4 then
unit_group_pos.positions [ group.group_number ] = nil
reform_group ( group )
end
end
end
end
end
end
end
2019-10-28 14:29:15 +02:00
local function log_threat ( )
2020-11-17 13:45:47 +02:00
local threat_log_index = WD.get ( ' threat_log_index ' )
WD.set ( ' threat_log_index ' , threat_log_index + 1 )
local threat_log = WD.get ( ' threat_log ' )
local threat = WD.get ( ' threat ' )
threat_log_index = WD.get ( ' threat_log_index ' )
threat_log [ threat_log_index ] = threat
if threat_log_index > 900 then
threat_log [ threat_log_index - 901 ] = nil
2020-08-09 20:23:45 +02:00
end
end
2019-10-15 04:20:40 +02:00
local tick_tasks = {
2020-08-09 20:23:45 +02:00
[ 30 ] = set_main_target ,
[ 60 ] = set_enemy_evolution ,
2020-11-23 23:10:32 +02:00
[ 120 ] = give_main_command_to_group ,
2020-08-09 20:23:45 +02:00
[ 150 ] = ThreatEvent.build_nest ,
[ 180 ] = ThreatEvent.build_worm ,
2020-11-23 23:10:32 +02:00
[ 1200 ] = give_side_commands_to_group ,
2020-08-09 20:23:45 +02:00
[ 3600 ] = time_out_biters ,
2020-10-18 12:45:18 +02:00
[ 7200 ] = refresh_active_unit_threat
2019-10-15 04:20:40 +02:00
}
2021-11-11 02:56:03 +02:00
local function t1 ( )
2020-11-17 13:45:47 +02:00
local tick = game.tick
local game_lost = WD.get ( ' game_lost ' )
if game_lost then
2020-08-09 20:23:45 +02:00
return
end
2021-10-29 00:06:35 +02:00
local paused = WD.get ( ' paused ' )
if paused then
return
end
2020-11-17 13:45:47 +02:00
local next_wave = WD.get ( ' next_wave ' )
if tick > next_wave then
2020-08-09 20:23:45 +02:00
set_next_wave ( )
end
2020-11-17 13:45:47 +02:00
local t = tick % 300
local t2 = tick % 18000
2020-08-09 20:23:45 +02:00
if tick_tasks [ t ] then
2021-11-11 02:56:03 +02:00
tick_tasks [ t ] ( true )
2020-08-09 20:23:45 +02:00
end
if tick_tasks [ t2 ] then
2021-11-11 02:56:03 +02:00
tick_tasks [ t2 ] ( true )
2020-08-09 20:23:45 +02:00
end
2020-11-23 23:10:32 +02:00
local resolve_pathing = WD.get ( ' resolve_pathing ' )
if resolve_pathing then
if tick % 60 == 0 then
check_group_positions ( )
end
end
2020-11-17 13:45:47 +02:00
local enable_threat_log = WD.get ( ' enable_threat_log ' )
if enable_threat_log then
if tick % 60 == 0 then
2020-08-09 20:23:45 +02:00
log_threat ( )
end
end
2020-10-22 12:44:14 +02:00
local players = game.connected_players
for _ , player in pairs ( players ) do
2020-08-09 20:23:45 +02:00
update_gui ( player )
end
2019-10-07 01:46:26 +02:00
end
2021-11-11 02:56:03 +02:00
local function t2 ( )
local game_lost = WD.get ( ' game_lost ' )
if game_lost then
return
end
local paused = WD.get ( ' paused ' )
if paused then
return
end
spawn_unit_group ( )
end
Public.spawn_unit_group = spawn_unit_group
Event.on_nth_tick ( 30 , t1 )
Event.on_nth_tick ( 130 , t2 )
2019-10-07 01:46:26 +02:00
2020-04-08 20:28:02 +02:00
return Public