2019-10-07 01:46:26 +02:00
local math_random = math.random
local threat_values = {
[ " behemoth-biter " ] = 10 ,
[ " behemoth-spitter " ] = 10 ,
[ " big-biter " ] = 5 ,
[ " big-spitter " ] = 5 ,
[ " medium-biter " ] = 3 ,
[ " medium-spitter " ] = 3 ,
[ " small-biter " ] = 1 ,
[ " small-spitter " ] = 1 ,
}
2019-10-08 01:17:00 +02:00
local function debug_print ( msg )
if global.wave_defense . debug then
print ( " WaveDefense>> " .. msg )
end
end
2019-10-07 16:40:52 +02:00
local function roll_biter_name ( )
2019-10-07 01:46:26 +02:00
local max_chance = 0
for k , v in pairs ( global.wave_defense . biter_raffle ) do
max_chance = max_chance + v
end
local r = math.random ( 1 , max_chance )
local current_chance = 0
for k , v in pairs ( global.wave_defense . biter_raffle ) do
current_chance = current_chance + v
if r <= current_chance then return k end
end
end
2019-10-07 16:40:52 +02:00
local function set_biter_raffle ( level )
2019-10-07 01:46:26 +02:00
global.wave_defense . biter_raffle = {
[ " small-biter " ] = 1000 - level * 2 ,
[ " small-spitter " ] = 1000 - level * 2 ,
2019-10-07 04:37:23 +02:00
[ " medium-biter " ] = level ,
[ " medium-spitter " ] = level ,
2019-10-07 01:46:26 +02:00
[ " big-biter " ] = 0 ,
[ " big-spitter " ] = 0 ,
[ " behemoth-biter " ] = 0 ,
[ " behemoth-spitter " ] = 0 ,
}
if level > 500 then
global.wave_defense . biter_raffle [ " big-biter " ] = ( level - 500 ) * 5
global.wave_defense . biter_raffle [ " big-spitter " ] = ( level - 500 ) * 5
end
if level > 800 then
global.wave_defense . biter_raffle [ " behemoth-biter " ] = ( level - 800 ) * 10
global.wave_defense . biter_raffle [ " behemoth-spitter " ] = ( level - 800 ) * 10
end
for k , v in pairs ( global.wave_defense . biter_raffle ) do
if global.wave_defense . biter_raffle [ k ] < 0 then global.wave_defense . biter_raffle [ k ] = 0 end
end
end
2019-10-07 22:40:05 +02:00
local function is_unit_valid ( biter )
2019-10-08 01:17:00 +02:00
if not biter.entity then debug_print ( " is_unit_valid - Unit did no longer exist " ) return false end
if not biter.entity . valid then debug_print ( " is_unit_valid - Unit invalid " ) return false end
if not biter.entity . unit_group then debug_print ( " is_unit_valid - Unit had no unitgroup " ) return false end
if biter.spawn_tick + global.wave_defense . max_biter_age < game.tick then debug_print ( " is_unit_valid - Unit timed out " ) return false end
2019-10-07 22:40:05 +02:00
return true
end
2019-10-07 04:37:23 +02:00
local function time_out_biters ( )
for k , biter in pairs ( global.wave_defense . active_biters ) do
2019-10-07 22:40:05 +02:00
if not is_unit_valid ( biter ) then
global.wave_defense . active_biter_count = global.wave_defense . active_biter_count - 1
if biter.entity then
if biter.entity . valid then
global.wave_defense . threat = global.wave_defense . threat + threat_values [ biter.entity . name ]
biter.entity . destroy ( )
end
end
2019-10-07 04:37:23 +02:00
global.wave_defense . active_biters [ k ] = nil
end
end
end
2019-10-07 01:46:26 +02:00
local function get_random_close_spawner ( )
local spawners = global.wave_defense . surface.find_entities_filtered ( { type = " unit-spawner " } )
if not spawners [ 1 ] then return false end
local center = global.wave_defense . target.position
local spawner = spawners [ math_random ( 1 , # spawners ) ]
for i = 1 , 5 , 1 do
local spawner_2 = spawners [ math_random ( 1 , # spawners ) ]
if ( center.x - spawner_2.position . x ) ^ 2 + ( center.y - spawner_2.position . y ) ^ 2 < ( center.x - spawner.position . x ) ^ 2 + ( center.y - spawner.position . y ) ^ 2 then spawner = spawner_2 end
end
return spawner
end
local function set_target ( )
2019-10-08 01:17:00 +02:00
if global.wave_defense . target then
if global.wave_defense . target.valid then
if global.wave_defense . target.name ~= " character " then return end
end
2019-10-07 01:46:26 +02:00
end
local characters = { }
for i = 1 , # game.connected_players , 1 do
if game.connected_players [ i ] . character then
if game.connected_players [ i ] . character.valid then
characters [ # characters + 1 ] = game.connected_players [ i ] . character
end
end
end
2019-10-08 01:17:00 +02:00
if # characters == 0 then return end
2019-10-07 01:46:26 +02:00
global.wave_defense . target = characters [ math_random ( 1 , # characters ) ]
end
local function set_group_spawn_position ( )
local spawner = get_random_close_spawner ( )
if not spawner then return end
2019-10-07 04:37:23 +02:00
local position = global.wave_defense . surface.find_non_colliding_position ( " rocket-silo " , spawner.position , 48 , 1 )
2019-10-07 01:46:26 +02:00
if not position then return end
global.wave_defense . spawn_position = position
end
local function set_enemy_evolution ( )
local evolution = global.wave_defense . wave_number * 0.001
2019-10-07 09:48:17 +02:00
if evolution > 1 then
global.biter_evasion_health_increase_factor = ( evolution - 1 ) * 4
evolution = 1
2019-10-07 16:40:52 +02:00
else
global.biter_evasion_health_increase_factor = 1
2019-10-07 09:48:17 +02:00
end
2019-10-07 01:46:26 +02:00
game.forces . enemy.evolution_factor = evolution
end
local function spawn_biter ( )
if global.wave_defense . threat <= 0 then return false end
if global.wave_defense . active_biter_count >= global.wave_defense . max_active_biters then return false end
local name = roll_biter_name ( )
local position = global.wave_defense . surface.find_non_colliding_position ( name , global.wave_defense . spawn_position , 32 , 1 )
if not position then return false end
local biter = global.wave_defense . surface.create_entity ( { name = name , position = position , force = " enemy " } )
biter.ai_settings . allow_destroy_when_commands_fail = false
biter.ai_settings . allow_try_return_to_spawner = false
global.wave_defense . active_biters [ biter.unit_number ] = { entity = biter , spawn_tick = game.tick }
global.wave_defense . active_biter_count = global.wave_defense . active_biter_count + 1
global.wave_defense . threat = global.wave_defense . threat - threat_values [ name ]
return biter
end
local function spawn_unit_group ( )
if global.wave_defense . threat <= 0 then return false end
if global.wave_defense . active_biter_count >= global.wave_defense . max_active_biters then return false end
set_group_spawn_position ( )
local unit_group = global.wave_defense . surface.create_unit_group ( { position = global.wave_defense . spawn_position , force = " enemy " } )
for a = 1 , global.wave_defense . group_size , 1 do
local biter = spawn_biter ( )
if not biter then break end
unit_group.add_member ( biter )
end
2019-10-07 04:37:23 +02:00
global.wave_defense . unit_groups [ # global.wave_defense . unit_groups + 1 ] = unit_group
2019-10-07 01:46:26 +02:00
return true
end
2019-10-07 22:40:05 +02:00
local function set_unit_group_count ( )
c = 0
2019-10-08 01:17:00 +02:00
for k , group in pairs ( global.wave_defense . unit_groups ) do
if group.valid then
if # group.members > 0 then
c = c + 1
else
group.destroy ( )
global.wave_defense . unit_groups [ k ] = nil
end
2019-10-07 22:40:05 +02:00
else
global.wave_defense . unit_groups [ k ] = nil
end
end
global.wave_defense . active_unit_group_count = c
end
local function spawn_attack_groups ( )
2019-10-07 04:37:23 +02:00
if global.wave_defense . active_biter_count >= global.wave_defense . max_active_biters then return false end
2019-10-07 01:46:26 +02:00
set_biter_raffle ( global.wave_defense . wave_number )
2019-10-07 22:40:05 +02:00
for a = 1 , global.wave_defense . max_active_unit_groups - global.wave_defense . active_unit_group_count , 1 do
2019-10-07 01:46:26 +02:00
if not spawn_unit_group ( ) then break end
end
end
2019-10-07 22:40:05 +02:00
local function set_next_wave ( )
global.wave_defense . wave_number = global.wave_defense . wave_number + 1
2019-10-08 01:17:00 +02:00
global.wave_defense . group_size = global.wave_defense . wave_number * 3
2019-10-07 22:40:05 +02:00
if global.wave_defense . group_size > global.wave_defense . max_group_size then global.wave_defense . group_size = global.wave_defense . max_group_size end
global.wave_defense . threat = global.wave_defense . threat + global.wave_defense . wave_number * 3
global.wave_defense . last_wave = global.wave_defense . next_wave
global.wave_defense . next_wave = game.tick + global.wave_defense . wave_interval
end
2019-10-08 01:17:00 +02:00
local function get_commmands ( group )
local commands = { }
local target_position = global.wave_defense . target.position
local group_position = { x = group.position . x , y = group.position . y }
local step_length = global.wave_defense . unit_group_command_step_length
local distance_to_target = math.floor ( math.sqrt ( ( target_position.x - group_position.x ) ^ 2 + ( target_position.y - group_position.y ) ^ 2 ) )
local steps = math.floor ( distance_to_target / step_length ) + 1
local vector = { math.round ( ( target_position.x - group_position.x ) / steps , 3 ) , math.round ( ( target_position.y - group_position.y ) / steps , 3 ) }
if global.wave_defense . debug then
print ( " get_commmands " )
print ( " distance_to_target " .. distance_to_target )
print ( " steps " .. steps )
print ( " vector " .. vector [ 1 ] .. " _ " .. vector [ 2 ] )
end
for i = 1 , steps , 1 do
group_position.x = group_position.x + vector [ 1 ]
group_position.y = group_position.y + vector [ 2 ]
local position = group.surface . find_non_colliding_position ( " small-biter " , group_position , 64 , 2 )
if position then
commands [ # commands + 1 ] = {
type = defines.command . attack_area ,
destination = { x = position.x , y = position.y } ,
radius = 16 ,
distraction = defines.distraction . by_enemy
}
if global.wave_defense . debug then print ( position ) end
end
end
commands [ # commands + 1 ] = {
type = defines.command . attack_area ,
destination = { x = global.wave_defense . target.position . x , y = global.wave_defense . target.position . y } ,
radius = 8 ,
distraction = defines.distraction . by_enemy
}
commands [ # commands + 1 ] = {
type = defines.command . attack ,
target = global.wave_defense . target ,
distraction = defines.distraction . by_enemy ,
}
return commands
end
2019-10-07 22:40:05 +02:00
local function command_unit_group ( group )
if not global.wave_defense . unit_group_last_command [ group.group_number ] then global.wave_defense . unit_group_last_command [ group.group_number ] = game.tick - ( global.wave_defense . unit_group_command_delay + 1 ) end
if global.wave_defense . unit_group_last_command [ group.group_number ] + global.wave_defense . unit_group_command_delay > game.tick then return end
global.wave_defense . unit_group_last_command [ group.group_number ] = game.tick
group.set_command ( {
type = defines.command . compound ,
structure_type = defines.compound_command . return_last ,
2019-10-08 01:17:00 +02:00
commands = get_commmands ( group )
2019-10-07 22:40:05 +02:00
} )
end
2019-10-07 04:37:23 +02:00
local function give_commands_to_unit_groups ( )
if # global.wave_defense . unit_groups == 0 then return end
if not global.wave_defense . target then return end
if not global.wave_defense . target.valid then return end
for k , group in pairs ( global.wave_defense . unit_groups ) do
if group.valid then
2019-10-07 22:40:05 +02:00
command_unit_group ( group )
2019-10-07 04:37:23 +02:00
else
global.wave_defense . unit_groups [ k ] = nil
end
end
end
local function create_gui ( player )
local frame = player.gui . top.add ( { type = " frame " , name = " wave_defense " , tooltip = " Click to show map info " } )
frame.style . maximal_height = 38
local label = frame.add ( { type = " label " , caption = " " , name = " label " } )
label.style . font_color = { r = 0.88 , g = 0.88 , b = 0.88 }
2019-10-07 22:40:05 +02:00
label.style . font = " default-bold "
2019-10-07 04:37:23 +02:00
label.style . left_padding = 4
label.style . right_padding = 4
label.style . minimal_width = 68
label.style . font_color = { r = 0.33 , g = 0.66 , b = 0.9 }
local progressbar = frame.add ( { type = " progressbar " , name = " progressbar " , value = 0 } )
progressbar.style . minimal_width = 128
progressbar.style . maximal_width = 128
progressbar.style . top_padding = 10
2019-10-07 22:40:05 +02:00
local line = frame.add ( { type = " line " , direction = " vertical " } )
line.style . left_padding = 4
line.style . right_padding = 4
local label = frame.add ( { type = " label " , caption = " " , name = " threat " } )
label.style . font_color = { r = 0.88 , g = 0.88 , b = 0.88 }
label.style . font = " default-bold "
label.style . left_padding = 4
label.style . right_padding = 4
label.style . minimal_width = 10
label.style . font_color = { r = 0.99 , g = 0.0 , b = 0.5 }
2019-10-07 04:37:23 +02:00
end
local function update_gui ( player )
if not player.gui . top.wave_defense then create_gui ( player ) end
player.gui . top.wave_defense . label.caption = " Wave: " .. global.wave_defense . wave_number
if global.wave_defense . wave_number == 0 then player.gui . top.wave_defense . label.caption = " First wave in " .. math.floor ( ( global.wave_defense . next_wave - game.tick ) / 60 ) + 1 end
2019-10-07 16:40:52 +02:00
local interval = global.wave_defense . next_wave - global.wave_defense . last_wave
player.gui . top.wave_defense . progressbar.value = 1 - ( global.wave_defense . next_wave - game.tick ) / interval
2019-10-07 22:40:05 +02:00
player.gui . top.wave_defense . threat.caption = " Threat: " .. global.wave_defense . threat
2019-10-07 04:37:23 +02:00
end
2019-10-07 01:46:26 +02:00
local function on_entity_died ( event )
if not event.entity . valid then return end
if event.entity . type ~= " unit " then return end
if not global.wave_defense . active_biters [ event.entity . unit_number ] then return end
global.wave_defense . active_biters [ event.entity . unit_number ] = nil
global.wave_defense . active_biter_count = global.wave_defense . active_biter_count - 1
end
local function on_tick ( )
2019-10-07 04:37:23 +02:00
if global.wave_defense . game_lost then return end
for _ , player in pairs ( game.connected_players ) do update_gui ( player ) end
2019-10-07 22:40:05 +02:00
if game.tick > global.wave_defense . next_wave then set_next_wave ( ) end
2019-10-07 04:37:23 +02:00
2019-10-07 22:40:05 +02:00
if game.tick % 180 == 0 then
if game.tick % 1800 == 0 then
time_out_biters ( )
end
2019-10-07 04:37:23 +02:00
set_target ( )
2019-10-08 01:17:00 +02:00
set_enemy_evolution ( )
2019-10-07 22:40:05 +02:00
spawn_attack_groups ( )
set_unit_group_count ( )
2019-10-07 04:37:23 +02:00
give_commands_to_unit_groups ( )
2019-10-07 22:40:05 +02:00
end
2019-10-07 01:46:26 +02:00
end
2019-10-07 16:40:52 +02:00
function reset_wave_defense ( )
2019-10-07 01:46:26 +02:00
global.wave_defense = {
2019-10-07 22:40:05 +02:00
debug = false ,
2019-10-07 01:46:26 +02:00
surface = game.surfaces [ " nauvis " ] ,
active_biters = { } ,
2019-10-07 04:37:23 +02:00
unit_groups = { } ,
2019-10-07 22:40:05 +02:00
unit_group_last_command = { } ,
unit_group_command_delay = 3600 * 5 ,
2019-10-08 01:17:00 +02:00
unit_group_command_step_length = 64 ,
max_active_unit_groups = 6 ,
2019-10-07 04:37:23 +02:00
max_active_biters = 1024 ,
2019-10-07 01:46:26 +02:00
max_group_size = 256 ,
2019-10-07 04:37:23 +02:00
max_biter_age = 3600 * 30 ,
2019-10-07 22:40:05 +02:00
active_unit_group_count = 0 ,
2019-10-07 01:46:26 +02:00
active_biter_count = 0 ,
spawn_position = { x = 0 , y = 48 } ,
2019-10-07 16:40:52 +02:00
last_wave = game.tick ,
2019-10-07 17:37:57 +02:00
next_wave = game.tick + 3600 * 5 ,
2019-10-07 04:37:23 +02:00
wave_interval = 1800 ,
2019-10-07 01:46:26 +02:00
wave_number = 0 ,
2019-10-07 04:37:23 +02:00
game_lost = false ,
2019-10-07 01:46:26 +02:00
threat = 0 ,
}
end
2019-10-07 16:40:52 +02:00
local function on_init ( )
reset_wave_defense ( )
end
2019-10-07 01:46:26 +02:00
local event = require ' utils.event '
2019-10-07 04:37:23 +02:00
event.on_nth_tick ( 30 , on_tick )
2019-10-07 01:46:26 +02:00
event.on_init ( on_init )
event.add ( defines.events . on_entity_died , on_entity_died )