2019-10-25 08:09:39 +02:00
require " modules.biter_health_booster "
2019-10-08 06:54:42 +02:00
require " modules.wave_defense.biter_rolls "
2019-10-08 17:41:15 +02:00
require " modules.wave_defense.threat_events "
2019-10-22 10:18:28 +02:00
local update_gui = require " modules.wave_defense.gui "
2019-10-08 17:41:15 +02:00
local threat_values = require " modules.wave_defense.threat_values "
2019-10-07 01:46:26 +02:00
local math_random = math.random
2019-10-25 08:09:39 +02:00
local side_target_types = { " accumulator " , " assembling-machine " , " beacon " , " boiler " , " container " , " furnace " , " lamp " , " lab " , " logistic-container " , " mining-drill " , " container " , " pump " , " radar " , " reactor " , " roboport " , " rocket-silo " , " solar-panel " , " storage-tank " , }
--local side_target_types = {"assembling-machine", "electric-pole", "furnace", "mining-drill", "pump", "radar", "reactor", "roboport"}
2019-10-07 01:46:26 +02:00
2019-10-08 01:17:00 +02:00
local function debug_print ( msg )
2019-10-11 21:52:32 +02:00
if not global.wave_defense . debug then return end
print ( " WaveDefense: " .. msg )
2019-10-08 01:17:00 +02:00
end
2019-10-15 04:20:40 +02:00
local function is_unit_valid ( biter )
2019-10-08 17:41:15 +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
if biter.spawn_tick + global.wave_defense . max_biter_age < game.tick then debug_print ( " is_unit_valid - unit destroyed - timed out " ) return false end
2019-10-07 22:40:05 +02:00
return true
end
2019-10-25 08:09:39 +02:00
local function refresh_active_unit_threat ( )
debug_print ( " refresh_active_unit_threat - current value " .. global.wave_defense . active_biter_threat )
local active_biter_threat = 0
for k , biter in pairs ( global.wave_defense . active_biters ) do
if biter.entity then
if biter.entity . valid then
active_biter_threat = active_biter_threat + threat_values [ biter.entity . name ]
end
end
end
global.wave_defense . active_biter_threat = math.round ( active_biter_threat * global.biter_health_boost , 2 )
debug_print ( " refresh_active_unit_threat - new value " .. global.wave_defense . active_biter_threat )
end
2019-10-14 06:52:17 +02:00
local function time_out_biters ( )
2019-10-15 04:20:40 +02:00
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
2019-10-25 08:09:39 +02:00
global.wave_defense . active_biter_threat = global.wave_defense . active_biter_threat - math.round ( threat_values [ biter.entity . name ] * global.biter_health_boost , 2 )
2019-10-14 06:52:17 +02:00
if biter.entity . force.index == 2 then
biter.entity . destroy ( )
end
2019-10-07 22:40:05 +02:00
end
end
2019-10-15 04:20:40 +02:00
global.wave_defense . active_biters [ k ] = nil
2019-10-07 04:37:23 +02:00
end
end
end
2019-10-12 04:06:48 +02:00
local function get_random_close_spawner ( surface )
local spawners = surface.find_entities_filtered ( { type = " unit-spawner " } )
2019-10-07 01:46:26 +02:00
if not spawners [ 1 ] then return false end
local center = global.wave_defense . target.position
local spawner = spawners [ math_random ( 1 , # spawners ) ]
2019-10-09 03:25:00 +02:00
for i = 1 , global.wave_defense . get_random_close_spawner_attempts , 1 do
2019-10-07 01:46:26 +02:00
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
2019-10-25 08:09:39 +02:00
debug_print ( " get_random_close_spawner - Found at x " .. spawner.position . x .. " y " .. spawner.position . y )
2019-10-07 01:46:26 +02:00
return spawner
end
2019-10-25 08:09:39 +02:00
local function set_side_target_list ( )
local surface = game.surfaces [ global.wave_defense . surface_index ]
local position = false
local force = false
2019-10-08 01:17:00 +02:00
if global.wave_defense . target then
2019-10-25 08:09:39 +02:00
if global.wave_defense . target.valid then
position = global.wave_defense . target.position
force = global.wave_defense . target.force
2019-10-07 01:46:26 +02:00
end
end
2019-10-25 08:09:39 +02:00
if not position then
local r = math_random ( 1 , # game.connected_players )
position = { x = game.connected_players [ r ] . position.x , y = game.connected_players [ r ] . position.y }
force = game.connected_players [ r ] . force
end
global.wave_defense . side_targets = surface.find_entities_filtered ( {
2019-10-10 01:42:38 +02:00
area = {
2019-10-25 08:09:39 +02:00
{ position.x - global.wave_defense . side_target_search_radius , position.y - global.wave_defense . side_target_search_radius } ,
{ position.x + global.wave_defense . side_target_search_radius , position.y + global.wave_defense . side_target_search_radius }
2019-10-10 01:42:38 +02:00
} ,
2019-10-25 08:09:39 +02:00
force = force ,
type = side_target_types ,
2019-10-10 01:42:38 +02:00
} )
2019-10-25 08:09:39 +02:00
debug_print ( " set_side_target_list -- " .. # global.wave_defense . side_targets .. " targets around position x " .. position.x .. " y " .. position.y .. " saved. " )
2019-10-10 05:40:39 +02:00
end
2019-10-10 01:42:38 +02:00
2019-10-10 05:40:39 +02:00
local function get_side_target ( )
if not global.wave_defense . side_targets then return false end
if # global.wave_defense . side_targets < 2 then return false end
local side_target = global.wave_defense . side_targets [ math_random ( 1 , # global.wave_defense . side_targets ) ]
if not side_target then return false end
if not side_target.valid then return false end
for _ = 1 , 4 , 1 do
local new_target = global.wave_defense . side_targets [ math_random ( 1 , # global.wave_defense . side_targets ) ]
if new_target then
if new_target.valid then
local side_target_distance = ( global.wave_defense . target.position . x - side_target.position . x ) ^ 2 + ( global.wave_defense . target.position . y - side_target.position . y ) ^ 2
local new_target_distance = ( global.wave_defense . target.position . x - new_target.position . x ) ^ 2 + ( global.wave_defense . target.position . y - new_target.position . y ) ^ 2
if new_target_distance > side_target_distance then side_target = new_target end
end
end
2019-10-25 08:09:39 +02:00
end
debug_print ( " get_side_target -- " .. side_target.name .. " at position x " .. side_target.position . x .. " y " .. side_target.position . y .. " selected. " )
2019-10-10 01:42:38 +02:00
return side_target
end
2019-10-25 08:09:39 +02:00
--[[
local function set_main_target ( )
if global.wave_defense . target then
if global.wave_defense . target.valid then return end
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
if game.connected_players [ i ] . surface.index == global.wave_defense . surface_index then
characters [ # characters + 1 ] = game.connected_players [ i ] . character
end
end
end
end
if # characters == 0 then return end
global.wave_defense . target = characters [ math_random ( 1 , # characters ) ]
end
] ]
local function set_main_target ( )
if global.wave_defense . target then
if global.wave_defense . target.valid then return end
end
if # global.wave_defense . side_targets == 0 then return end
local target = global.wave_defense . side_targets [ math_random ( 1 , # global.wave_defense . side_targets ) ]
if not target then return end
if not target.valid then return end
global.wave_defense . target = target
debug_print ( " set_main_target -- New main target " .. target.name .. " at position x " .. target.position . x .. " y " .. target.position . y .. " selected. " )
end
2019-10-12 04:06:48 +02:00
local function set_group_spawn_position ( surface )
local spawner = get_random_close_spawner ( surface )
2019-10-07 01:46:26 +02:00
if not spawner then return end
2019-10-12 04:06:48 +02:00
local position = 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
2019-10-11 21:52:32 +02:00
global.wave_defense . spawn_position = { x = math.floor ( position.x ) , y = math.floor ( position.y ) }
2019-10-25 08:09:39 +02:00
debug_print ( " set_group_spawn_position -- Changed position to x " .. global.wave_defense . spawn_position.x .. " y " .. global.wave_defense . spawn_position.y .. " . " )
2019-10-07 01:46:26 +02:00
end
local function set_enemy_evolution ( )
2019-10-13 05:24:29 +02:00
local evolution_factor = global.wave_defense . wave_number * 0.001
2019-10-23 21:17:15 +02:00
local biter_health_boost = 1
2019-10-13 05:24:29 +02:00
local damage_increase = 0
if evolution_factor > 1 then
2019-10-23 21:17:15 +02:00
damage_increase = damage_increase + ( evolution_factor - 1 )
2019-10-25 08:09:39 +02:00
biter_health_boost = biter_health_boost + ( evolution_factor - 1 ) * 2
2019-10-13 05:24:29 +02:00
evolution_factor = 1
end
2019-10-23 21:17:15 +02:00
2019-10-13 05:24:29 +02:00
if global.wave_defense . threat > 0 then
2019-10-25 08:09:39 +02:00
biter_health_boost = math.round ( biter_health_boost + global.wave_defense . threat * 0.00005 , 3 )
damage_increase = math.round ( damage_increase + global.wave_defense . threat * 0.000005 , 3 )
2019-10-07 09:48:17 +02:00
end
2019-10-13 05:24:29 +02:00
2019-10-23 21:17:15 +02:00
global.biter_health_boost = biter_health_boost
2019-10-13 05:24:29 +02:00
game.forces . enemy.set_ammo_damage_modifier ( " melee " , damage_increase )
game.forces . enemy.set_ammo_damage_modifier ( " biological " , damage_increase )
game.forces . enemy.evolution_factor = evolution_factor
2019-10-23 21:39:35 +02:00
if global.biter_health_boost then
for _ , player in pairs ( game.connected_players ) do
player.gui . top.wave_defense . threat.tooltip = " High threat may empower biters. \n Biter health " .. biter_health_boost * 100 .. " % | damage " .. ( damage_increase + 1 ) * 100 .. " % "
end
end
2019-10-25 08:09:39 +02:00
end
local function can_units_spawn ( )
if global.wave_defense . threat <= 0 then
debug_print ( " can_units_spawn - threat too low " )
return false
end
if global.wave_defense . active_biter_count >= global.wave_defense . max_active_biters then
debug_print ( " can_units_spawn - active biter count too high " )
return false
end
if global.wave_defense . active_biter_threat >= global.wave_defense . threat then
debug_print ( " can_units_spawn - active biter threat too high ( " .. global.wave_defense . active_biter_threat .. " ) " )
return false
end
return true
end
local function get_active_unit_groups_count ( )
local count = 0
for _ , g in pairs ( global.wave_defense . unit_groups ) do
if g.valid then
count = count + 1
end
end
debug_print ( " Active unit group count: " .. count )
return count
2019-10-07 01:46:26 +02:00
end
2019-10-12 04:06:48 +02:00
local function spawn_biter ( surface )
2019-10-25 08:09:39 +02:00
if not can_units_spawn ( ) then return end
2019-10-12 04:06:48 +02:00
local name
if math.random ( 1 , 100 ) > 73 then
name = wave_defense_roll_spitter_name ( )
else
name = wave_defense_roll_biter_name ( )
end
local position = surface.find_non_colliding_position ( name , global.wave_defense . spawn_position , 48 , 2 )
2019-10-07 01:46:26 +02:00
if not position then return false end
2019-10-12 04:06:48 +02:00
local biter = surface.create_entity ( { name = name , position = position , force = " enemy " } )
2019-10-07 01:46:26 +02:00
biter.ai_settings . allow_destroy_when_commands_fail = false
biter.ai_settings . allow_try_return_to_spawner = false
2019-10-15 04:20:40 +02:00
global.wave_defense . active_biters [ biter.unit_number ] = { entity = biter , spawn_tick = game.tick }
2019-10-07 01:46:26 +02:00
global.wave_defense . active_biter_count = global.wave_defense . active_biter_count + 1
2019-10-25 08:09:39 +02:00
global.wave_defense . active_biter_threat = global.wave_defense . active_biter_threat + math.round ( threat_values [ name ] * global.biter_health_boost , 2 )
2019-10-14 06:52:17 +02:00
return biter
2019-10-07 01:46:26 +02:00
end
2019-10-15 04:20:40 +02:00
local function spawn_unit_group ( )
2019-10-25 08:09:39 +02:00
if not can_units_spawn ( ) then return end
if not global.wave_defense . target then return end
if not global.wave_defense . target.valid then return end
if get_active_unit_groups_count ( ) >= global.wave_defense . max_active_unit_groups then return end
wave_defense_set_unit_raffle ( global.wave_defense . wave_number )
2019-10-15 04:20:40 +02:00
local surface = game.surfaces [ global.wave_defense . surface_index ]
set_group_spawn_position ( surface )
2019-10-25 08:09:39 +02:00
debug_print ( " Spawning unit group at x " .. global.wave_defense . spawn_position.x .. " y " .. global.wave_defense . spawn_position.y )
2019-10-15 04:20:40 +02:00
local unit_group = 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 ( surface )
if not biter then break end
unit_group.add_member ( biter )
end
for i = 1 , # global.wave_defense . unit_groups , 1 do
if not global.wave_defense . unit_groups [ i ] then
global.wave_defense . unit_groups [ i ] = unit_group
return true
end
end
global.wave_defense . unit_groups [ # global.wave_defense . unit_groups + 1 ] = unit_group
return true
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 05:50:32 +02:00
global.wave_defense . group_size = global.wave_defense . wave_number * 2
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
2019-10-25 11:37:56 +02:00
global.wave_defense . threat = global.wave_defense . threat + math.floor ( global.wave_defense . wave_number * global.wave_defense . threat_gain_multiplier )
2019-10-07 22:40:05 +02:00
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 group_position = { x = group.position . x , y = group.position . y }
local step_length = global.wave_defense . unit_group_command_step_length
2019-10-10 01:42:38 +02:00
2019-10-11 21:52:32 +02:00
if math_random ( 1 , 3 ) ~= 1 then
local side_target = false
for _ = 1 , 3 , 1 do
side_target = get_side_target ( )
if side_target then break end
2019-10-10 01:42:38 +02:00
end
2019-10-11 21:52:32 +02:00
if side_target then
local target_position = side_target.position
local distance_to_target = math.floor ( math.sqrt ( ( target_position.x - group_position.x ) ^ 2 + ( target_position.y - group_position.y ) ^ 2 ) )
local steps = math.floor ( distance_to_target / step_length ) + 1
local vector = { math.round ( ( target_position.x - group_position.x ) / steps , 3 ) , math.round ( ( target_position.y - group_position.y ) / steps , 3 ) }
2019-10-15 04:20:40 +02:00
if global.wave_defense . debug then
2019-10-25 08:09:39 +02:00
debug_print ( " get_commmands - to side_target x " .. side_target.position . x .. " y " .. side_target.position . y )
debug_print ( " get_commmands - distance_to_target: " .. distance_to_target .. " steps: " .. steps )
debug_print ( " get_commmands - vector " .. vector [ 1 ] .. " _ " .. vector [ 2 ] )
2019-10-15 04:20:40 +02:00
end
2019-10-11 21:52:32 +02:00
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_anything
}
end
end
commands [ # commands + 1 ] = {
type = defines.command . attack ,
target = side_target ,
distraction = defines.distraction . by_enemy ,
}
2019-10-10 01:42:38 +02:00
end
end
2019-10-11 21:52:32 +02:00
2019-10-10 01:42:38 +02:00
local target_position = global.wave_defense . target.position
2019-10-08 01:17:00 +02:00
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 ) }
2019-10-15 04:20:40 +02:00
if global.wave_defense . debug then
2019-10-25 08:09:39 +02:00
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 ] )
2019-10-15 04:20:40 +02:00
end
2019-10-08 01:17:00 +02:00
for i = 1 , steps , 1 do
group_position.x = group_position.x + vector [ 1 ]
2019-10-10 01:42:38 +02:00
group_position.y = group_position.y + vector [ 2 ]
2019-10-11 07:13:08 +02:00
local position = group.surface . find_non_colliding_position ( " small-biter " , group_position , 64 , 1 )
2019-10-08 01:17:00 +02:00
if position then
commands [ # commands + 1 ] = {
type = defines.command . attack_area ,
destination = { x = position.x , y = position.y } ,
radius = 16 ,
2019-10-08 05:50:32 +02:00
distraction = defines.distraction . by_anything
2019-10-08 01:17:00 +02:00
}
2019-10-10 01:42:38 +02:00
end
2019-10-08 01:17:00 +02:00
end
commands [ # commands + 1 ] = {
type = defines.command . attack_area ,
2019-10-18 07:01:14 +02:00
destination = { x = target_position.x , y = target_position.y } ,
2019-10-08 01:17:00 +02:00
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 )
2019-10-16 00:23:54 +02:00
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
2019-10-15 04:20:40 +02:00
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
2019-10-07 22:40:05 +02:00
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-15 04:20:40 +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
command_unit_group ( group )
else
global.wave_defense . unit_groups [ k ] = nil
end
end
2019-10-07 04:37:23 +02:00
end
2019-10-15 04:20:40 +02:00
local tick_tasks = {
[ 30 ] = set_main_target ,
[ 60 ] = set_enemy_evolution ,
2019-10-25 08:09:39 +02:00
[ 90 ] = spawn_unit_group ,
2019-10-15 04:20:40 +02:00
[ 120 ] = give_commands_to_unit_groups ,
[ 150 ] = build_nest ,
[ 180 ] = build_worm ,
2019-10-25 08:09:39 +02:00
[ 1800 ] = set_side_target_list ,
[ 3600 ] = time_out_biters ,
[ 7200 ] = refresh_active_unit_threat ,
2019-10-15 04:20:40 +02:00
}
2019-10-07 01:46:26 +02:00
local function on_tick ( )
2019-10-15 04:20:40 +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 04:37:23 +02:00
2019-10-07 22:40:05 +02:00
if game.tick > global.wave_defense . next_wave then set_next_wave ( ) end
2019-10-15 04:20:40 +02:00
local t = game.tick % 300
2019-10-25 08:09:39 +02:00
local t2 = game.tick % 18000
2019-10-15 04:20:40 +02:00
if tick_tasks [ t ] then tick_tasks [ t ] ( ) end
2019-10-25 08:09:39 +02:00
if tick_tasks [ t2 ] then tick_tasks [ t2 ] ( ) 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-25 11:37:56 +02:00
active_biter_count = 0 ,
2019-10-25 08:09:39 +02:00
active_biter_threat = 0 ,
2019-10-25 11:37:56 +02:00
active_biters = { } ,
debug = false ,
game_lost = false ,
get_random_close_spawner_attempts = 5 ,
2019-10-13 05:24:29 +02:00
group_size = 2 ,
2019-10-25 11:37:56 +02:00
last_wave = game.tick ,
2019-10-10 19:33:32 +02:00
max_active_biters = 1024 ,
2019-10-25 11:37:56 +02:00
max_active_unit_groups = 8 ,
2019-10-08 05:50:32 +02:00
max_biter_age = 3600 * 60 ,
2019-10-25 11:37:56 +02:00
max_group_size = 192 ,
nest_building_chance = 4 , --high value = less chance
nest_building_density = 64 , --lower values = more dense building
next_wave = game.tick + 3600 * 15 ,
2019-10-16 03:57:53 +02:00
side_target_search_radius = 768 ,
2019-10-25 11:37:56 +02:00
simple_entity_shredding_cost_modifier = 0.01 , --threat cost for one health
simple_entity_shredding_count_modifier = 0.0003 ,
2019-10-08 17:41:15 +02:00
spawn_position = { x = 0 , y = 64 } ,
2019-10-25 11:37:56 +02:00
surface_index = 1 ,
threat = 0 ,
threat_gain_multiplier = 2 ,
unit_group_command_delay = 3600 * 8 ,
unit_group_command_step_length = 64 ,
unit_group_last_command = { } ,
unit_groups = { } ,
2019-10-22 08:29:30 +02:00
wave_interval = 3600 ,
2019-10-07 01:46:26 +02:00
wave_number = 0 ,
2019-10-10 21:33:54 +02:00
worm_building_chance = 2 , --high value = less chance
2019-10-25 11:37:56 +02:00
worm_building_density = 8 , --lower values = more dense building
2019-10-07 01:46:26 +02:00
}
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-08 17:41:15 +02:00
event.on_init ( on_init )