1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-10 00:43:27 +02:00
ComfyFactorio/maps/mountain_fortress_v3/functions.lua
2024-07-11 20:02:10 +02:00

1948 lines
63 KiB
Lua

local Event = require 'utils.event'
local Public = require 'maps.mountain_fortress_v3.table'
local Server = require 'utils.server'
local Task = require 'utils.task_token'
local Color = require 'utils.color_presets'
local ICW = require 'maps.mountain_fortress_v3.icw.main'
local Global = require 'utils.global'
local Alert = require 'utils.alert'
local WD = require 'modules.wave_defense.table'
local RPG = require 'modules.rpg.main'
local Collapse = require 'modules.collapse'
local Difficulty = require 'modules.difficulty_vote_by_amount'
local ICW_Func = require 'maps.mountain_fortress_v3.icw.functions'
local math2d = require 'math2d'
local Misc = require 'utils.commands.misc'
local Core = require 'utils.core'
local Beams = require 'modules.render_beam'
local BottomFrame = require 'utils.gui.bottom_frame'
local Modifiers = require 'utils.player_modifiers'
local Session = require 'utils.datastore.session_data'
local zone_settings = Public.zone_settings
local remove_boost_movement_speed_on_respawn
local de = defines.events
local this = {
power_sources = { index = 1 },
refill_turrets = { index = 1 },
magic_crafters = { index = 1 },
magic_fluid_crafters = { index = 1 },
art_table = { index = 1 },
editor_mode = {},
techs = {},
limit_types = {},
starting_items = {
['pistol'] = {
count = 1
},
['firearm-magazine'] = {
count = 16
},
['rail'] = {
count = 16
},
['wood'] = {
count = 16
},
['explosives'] = {
count = 32
}
}
}
local exit_editor_mode_token =
Task.register(
function (event)
local player_index = event.player_index
this.editor_mode[player_index] = nil
end
)
local random_respawn_messages = {
'The doctors stitched you up as best they could.',
'Ow! Your right leg hurts.',
'Ow! Your left leg hurts.',
'You can feel your whole body aching.',
"You still have some bullet wounds that aren't patched up.",
'You feel dizzy but adrenalin is granting you speed.',
'Adrenalin is kicking in, but your body is damaged.'
}
local health_values = {
'0.35',
'0.40',
'0.45',
'0.50',
'0.55',
'0.60',
'0.65',
'0.70',
'0.75',
'0.80',
'0.85',
'0.90',
'0.95',
'1'
}
Global.register(
this,
function (t)
this = t
end
)
local random = math.random
local floor = math.floor
local round = math.round
local sqrt = math.sqrt
local remove = table.remove
local magic_crafters_per_tick = 3
local magic_fluid_crafters_per_tick = 8
local tile_damage = 50
local artillery_target_entities = {
'character',
'tank',
'car',
'radar',
'lab',
'furnace',
'locomotive',
'cargo-wagon',
'fluid-wagon',
'artillery-wagon',
'artillery-turret',
'laser-turret',
'gun-turret',
'flamethrower-turret',
'silo',
'spidertron'
}
local function debug_str(msg)
local debug = Public.get('debug')
if not debug then
return
end
print('Mtn: ' .. msg)
end
local function show_text(msg, pos, color, surface)
if color == nil then
surface.create_entity({ name = 'flying-text', position = pos, text = msg })
else
surface.create_entity({ name = 'flying-text', position = pos, text = msg, color = color })
end
end
local function fast_remove(tbl, index)
local count = #tbl
if index > count then
return
elseif index < count then
tbl[index] = tbl[count]
end
tbl[count] = nil
end
local pause_waves_custom_callback_token =
Task.register(
function (event)
local wave_number = WD.get_wave()
if wave_number >= 200 then
Collapse.start_now(event.start, not event.start)
local status_str = event.start and 'is active once again!' or 'has stopped!'
Alert.alert_all_players(30, 'Collapse ' .. status_str, nil, 'achievement/tech-maniac', 0.6)
end
end
)
local function do_refill_turrets()
local refill_turrets = this.refill_turrets
local index = refill_turrets.index
if index > #refill_turrets then
refill_turrets.index = 1
return
end
local turret_data = refill_turrets[index]
local turret = turret_data.turret
if not turret.valid then
fast_remove(refill_turrets, index)
return
end
refill_turrets.index = index + 1
local data = turret_data.data
if data.liquid then
turret.fluidbox[1] = data
elseif data then
turret.insert(data)
end
end
local function do_magic_crafters()
local magic_crafters = this.magic_crafters
local limit = #magic_crafters
if limit == 0 then
return
end
local index = magic_crafters.index
for _ = 1, magic_crafters_per_tick do
if index > limit then
index = 1
end
local data = magic_crafters[index]
local entity = data.entity
if not entity.valid then
fast_remove(magic_crafters, index)
limit = limit - 1
if limit == 0 then
return
end
else
index = index + 1
local tick = game.tick
local last_tick = data.last_tick
local rate = data.rate
local count = (tick - last_tick) * rate
local fcount = floor(count)
if fcount > 1 then
fcount = 1
end
if fcount > 0 then
if entity.get_output_inventory().can_insert({ name = data.item, count = fcount }) then
entity.get_output_inventory().insert { name = data.item, count = fcount }
entity.products_finished = entity.products_finished + fcount
data.last_tick = round(tick - (count - fcount) / rate)
end
end
end
end
magic_crafters.index = index
end
local function do_magic_fluid_crafters()
local magic_fluid_crafters = this.magic_fluid_crafters
local limit = #magic_fluid_crafters
if limit == 0 then
return
end
local index = magic_fluid_crafters.index
for _ = 1, magic_fluid_crafters_per_tick do
if index > limit then
index = 1
end
local data = magic_fluid_crafters[index]
local entity = data.entity
if not entity.valid then
fast_remove(magic_fluid_crafters, index)
limit = limit - 1
if limit == 0 then
return
end
else
index = index + 1
local tick = game.tick
local last_tick = data.last_tick
local rate = data.rate
local count = (tick - last_tick) * rate
local fcount = floor(count)
if fcount > 0 then
local fluidbox_index = data.fluidbox_index
local fb = entity.fluidbox
local fb_data = fb[fluidbox_index] or { name = data.item, amount = 0 }
fb_data.amount = fb_data.amount + fcount
fb[fluidbox_index] = fb_data
entity.products_finished = entity.products_finished + fcount
data.last_tick = tick - (count - fcount) / rate
end
end
end
magic_fluid_crafters.index = index
end
local artillery_target_callback =
Task.register(
function (data)
local position = data.position
local entity = data.entity
local index = data.index
local art_table = this.art_table
local outpost = art_table[index]
if not outpost then
return
end
if not entity.valid then
outpost.last_fire_tick = 0
return
end
local tx, ty = position.x, position.y
local fired_at_target = false
local pos = entity.position
local x, y = pos.x, pos.y
local dx, dy = tx - x, ty - y
local d = dx * dx + dy * dy
if d >= 1024 and d <= 441398 then -- 704 in depth~
if entity.name == 'character' then
entity.surface.create_entity {
name = 'artillery-projectile',
position = position,
target = entity,
force = 'enemy',
speed = 1.5
}
fired_at_target = true
elseif entity.name ~= 'character' then
entity.surface.create_entity {
name = 'rocket',
position = position,
target = entity,
force = 'enemy',
speed = 1.5
}
fired_at_target = true
end
end
if not fired_at_target then
outpost.last_fire_tick = 0
end
end
)
local function do_beams_away()
local wave_number = WD.get_wave()
local orbital_strikes = Public.get('orbital_strikes')
if not orbital_strikes or not orbital_strikes.enabled then
return
end
if wave_number > 500 then
local difficulty_index = Difficulty.get('index')
local wave_nth = 9999
if difficulty_index == 1 then
wave_nth = 500
elseif difficulty_index == 2 then
wave_nth = 250
elseif difficulty_index == 3 then
wave_nth = 100
end
if wave_number % wave_nth == 0 then
local active_surface_index = Public.get('active_surface_index')
local surface = game.get_surface(active_surface_index)
if not orbital_strikes[wave_number] then
orbital_strikes[wave_number] = true
Beams.new_beam_delayed(surface, random(500, 3000))
end
end
end
end
local function do_clear_enemy_spawners()
local tick = game.tick
if tick % 1000 ~= 0 then
return
end
local enemy_spawners = Public.get('enemy_spawners')
if not enemy_spawners or not enemy_spawners.enabled then
return
end
for unit_number, spawner in pairs(enemy_spawners.spawners) do
if not spawner.entity or not spawner.entity.valid then
enemy_spawners.spawners[unit_number] = nil
end
end
end
local function do_clear_rocks_slowly()
local rocks_to_remove = Public.get('rocks_to_remove')
if not rocks_to_remove or not next(rocks_to_remove) then
return
end
for _ = 1, 300 do
local entity = table.remove(rocks_to_remove, #rocks_to_remove)
if entity and entity.valid then
entity.destroy()
end
end
end
local function do_replace_tiles_slowly()
local active_surface_index = Public.get('active_surface_index')
if not active_surface_index then return end
local surface = game.get_surface(active_surface_index)
if not (surface and surface.valid) then
return
end
local tiles_to_replace = Public.get('tiles_to_replace')
if not tiles_to_replace or not next(tiles_to_replace) then
return
end
for _ = 1, 300 do
local tile = table.remove(tiles_to_replace, #tiles_to_replace)
if tile and tile.valid then
surface.set_tiles({ { name = 'water-shallow', position = tile.position } }, true)
end
end
end
local function do_season_fix()
local active_surface_index = Public.get('active_surface_index')
local surface = game.surfaces[active_surface_index]
if not (surface and surface.valid) then
return
end
local current_season = Public.get('current_season')
if current_season then
rendering.destroy(current_season)
end
Public.set(
'current_season',
rendering.draw_text {
text = 'Season: ' .. Public.get_stateful('season'),
surface = surface,
target = { -0, 12 },
color = { r = 0.98, g = 0.77, b = 0.22 },
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
)
end
local do_season_fix_token = Task.register(do_season_fix)
local function do_artillery_turrets_targets()
local art_table = this.art_table
local index = art_table.index
if index > #art_table then
art_table.index = 1
return
end
art_table.index = index + 1
local outpost = art_table[index]
local now = game.tick
if now - outpost.last_fire_tick < 480 then
return
end
local turrets = outpost.artillery_turrets
for i = #turrets, 1, -1 do
local turret = turrets[i]
if not turret.valid then
fast_remove(turrets, i)
end
end
local count = #turrets
if count == 0 then
fast_remove(art_table, index)
return
end
local turret = turrets[1]
local area = outpost.artillery_area
local surface = turret.surface
local entities = surface.find_entities_filtered { area = area, name = artillery_target_entities, force = 'player' }
if #entities == 0 then
return
end
local position = turret.position
outpost.last_fire_tick = now
for i = 1, count do
local entity = entities[random(#entities)]
if entity and entity.valid then
local data = { position = position, entity = entity, index = index }
Task.set_timeout_in_ticks(i * 60, artillery_target_callback, data)
end
end
end
local function add_magic_crafter_output(entity, output, distance)
local magic_fluid_crafters = this.magic_fluid_crafters
local magic_crafters = this.magic_crafters
local rate = output.min_rate + output.distance_factor * distance
local fluidbox_index = output.fluidbox_index
local data = {
entity = entity,
last_tick = game.tick,
base_rate = round(rate, 8),
rate = round(rate, 8),
item = output.item,
fluidbox_index = fluidbox_index
}
if fluidbox_index then
magic_fluid_crafters[#magic_fluid_crafters + 1] = data
else
magic_crafters[#magic_crafters + 1] = data
end
end
local function tick()
do_refill_turrets()
do_magic_crafters()
do_magic_fluid_crafters()
do_artillery_turrets_targets()
do_beams_away()
do_clear_enemy_spawners()
do_clear_rocks_slowly()
do_replace_tiles_slowly()
end
Public.deactivate_callback =
Task.register(
function (entity)
if entity and entity.valid then
entity.active = false
entity.operable = false
entity.destructible = false
end
end
)
Public.neutral_force =
Task.register(
function (entity)
if entity and entity.valid then
entity.force = 'neutral'
end
end
)
Public.enemy_force =
Task.register(
function (entity)
if entity and entity.valid then
entity.force = 'enemy'
end
end
)
Public.active_not_destructible_callback =
Task.register(
function (entity)
if entity and entity.valid then
entity.active = true
entity.operable = false
entity.destructible = false
end
end
)
Public.disable_minable_callback =
Task.register(
function (entity)
if entity and entity.valid then
entity.minable = false
end
end
)
Public.disable_minable_and_ICW_callback =
Task.register(
function (entity)
if entity and entity.valid then
local wagons_in_the_wild = Public.get('wagons_in_the_wild')
entity.minable = false
entity.destructible = false
ICW.register_wagon(entity)
wagons_in_the_wild[entity.unit_number] = entity
end
end
)
Public.disable_destructible_callback =
Task.register(
function (entity)
if entity and entity.valid then
entity.destructible = false
entity.minable = false
end
end
)
Public.disable_active_callback =
Task.register(
function (entity)
if entity and entity.valid then
entity.active = false
end
end
)
local disable_active_callback = Public.disable_active_callback
Public.refill_turret_callback =
Task.register(
function (turret, data)
local refill_turrets = this.refill_turrets
local callback_data = data.callback_data
turret.direction = 3
refill_turrets[#refill_turrets + 1] = { turret = turret, data = callback_data }
end
)
Public.refill_artillery_turret_callback =
Task.register(
function (turret, data)
local refill_turrets = this.refill_turrets
local art_table = this.art_table
local index = art_table.index
turret.active = false
turret.direction = 3
refill_turrets[#refill_turrets + 1] = { turret = turret, data = data.callback_data }
local artillery_data = art_table[index]
if not artillery_data then
artillery_data = {}
end
local artillery_turrets = artillery_data.artillery_turrets
if not artillery_turrets then
artillery_turrets = {}
artillery_data.artillery_turrets = artillery_turrets
local pos = turret.position
local x, y = pos.x, pos.y
local adjusted_zones = Public.get('adjusted_zones')
if adjusted_zones.reversed then
artillery_data.artillery_area = { { x - 112, y - 212 }, { x + 112, y } }
else
artillery_data.artillery_area = { { x - 112, y }, { x + 112, y + 212 } }
end
artillery_data.last_fire_tick = 0
art_table[#art_table + 1] = artillery_data
end
artillery_turrets[#artillery_turrets + 1] = turret
end
)
Public.refill_liquid_turret_callback =
Task.register(
function (turret, data)
local refill_turrets = this.refill_turrets
local callback_data = data.callback_data
callback_data.liquid = true
refill_turrets[#refill_turrets + 1] = { turret = turret, data = callback_data }
end
)
Public.power_source_callback =
Task.register(
function (turret)
local power_sources = this.power_sources
power_sources[#power_sources + 1] = turret
end
)
Public.magic_item_crafting_callback =
Task.register(
function (entity, data)
local callback_data = data.callback_data
if not (entity and entity.valid) then
return
end
entity.minable = false
entity.destructible = false
entity.operable = false
local force = game.forces.player
local tech = callback_data.tech
if not callback_data.testing then
if tech then
if not force.technologies[tech].researched then
entity.destroy()
return
end
end
end
local recipe = callback_data.recipe
if recipe then
entity.set_recipe(recipe)
else
local furance_item = callback_data.furance_item
if furance_item then
local inv = entity.get_inventory(defines.inventory.furnace_result)
inv.insert(furance_item)
end
end
local p = entity.position
local x, y = p.x, p.y
local distance = sqrt(x * x + y * y)
local output = callback_data.output
if #output == 0 then
add_magic_crafter_output(entity, output, distance)
else
for i = 1, #output do
local o = output[i]
add_magic_crafter_output(entity, o, distance)
end
end
if not callback_data.keep_active then
Task.set_timeout_in_ticks(2, disable_active_callback, entity) -- causes problems with refineries.
end
end
)
Public.magic_item_crafting_callback_weighted =
Task.register(
function (entity, data)
local callback_data = data.callback_data
if not (entity and entity.valid) then
return
end
local weights = callback_data.weights
local loot = callback_data.loot
local destructible = callback_data.destructible
if not destructible then
entity.destructible = false
end
entity.minable = false
entity.operable = false
local p = entity.position
local i = random() * weights.total
local index = table.binary_search(weights, i)
if (index < 0) then
index = bit32.bnot(index)
end
local stack = loot[index].stack
if not stack then
return
end
local force = game.forces.player
local tech = stack.tech
if not callback_data.testing then
if tech then
if force.technologies[tech] then
if not force.technologies[tech].researched then
entity.destroy()
return
end
end
end
end
local recipe = stack.recipe
if recipe then
entity.set_recipe(recipe)
else
local furance_item = stack.furance_item
if furance_item then
local inv = entity.get_inventory(defines.inventory.furnace_result)
inv.insert(furance_item)
end
end
local x, y = p.x, p.y
local distance = sqrt(x * x + y * y)
local output = stack.output
if #output == 0 then
add_magic_crafter_output(entity, output, distance)
else
for o_i = 1, #output do
local o = output[o_i]
add_magic_crafter_output(entity, o, distance)
end
end
if not callback_data.keep_active then
Task.set_timeout_in_ticks(2, disable_active_callback, entity) -- causes problems with refineries.
end
end
)
function Public.prepare_weighted_loot(loot)
local total = 0
local weights = {}
for i = 1, #loot do
local v = loot[i]
total = total + v.weight
weights[#weights + 1] = total
end
weights.total = total
return weights
end
function Public.do_random_loot(entity, weights, loot)
if not entity.valid then
return
end
entity.operable = false
--entity.destructible = false
local i = random() * weights.total
local index = table.binary_search(weights, i)
if (index < 0) then
index = bit32.bnot(index)
end
local stack = loot[index].stack
if not stack then
return
end
local df = stack.distance_factor
local count
if df then
local p = entity.position
local x, y = p.x, p.y
local d = sqrt(x * x + y * y)
count = stack.count + d * df
else
count = stack.count
end
entity.insert { name = stack.name, count = count }
end
local function calc_players()
local players = game.connected_players
local check_afk_players = Public.get('check_afk_players')
if not check_afk_players then
return #players
end
local total = 0
Core.iter_connected_players(
function (player)
if player.afk_time < 36000 then
total = total + 1
end
end
)
if total <= 0 then
total = #players
end
return total
end
remove_boost_movement_speed_on_respawn =
Task.register(
function (data)
local player = data.player
if not player or not player.valid then
return
end
Modifiers.update_single_modifier(player, 'character_running_speed_modifier', 'v3_move_boost')
Modifiers.update_player_modifiers(player)
player.print('Movement speed bonus removed!', Color.info)
local rpg_t = RPG.get_value_from_player(player.index)
rpg_t.has_boost_on_respawn = nil
end
)
local boost_movement_speed_on_respawn =
Task.register(
function (data)
local player = data.player
if not player or not player.valid then
return
end
if not player.character or not player.character.valid then
return
end
local rpg_t = RPG.get_value_from_player(player.index)
if rpg_t.has_boost_on_respawn then
return
end
rpg_t.has_boost_on_respawn = true
Modifiers.update_single_modifier(player, 'character_running_speed_modifier', 'v3_move_boost', 1)
Modifiers.update_player_modifiers(player)
Task.set_timeout_in_ticks(800, remove_boost_movement_speed_on_respawn, { player = player })
player.print('Movement speed bonus applied! Be quick and fetch your corpse!', Color.info)
end
)
local function on_wave_created(event)
if not event or not event.wave_number then
return
end
local wave_number = event.wave_number
if wave_number % 50 == 0 then
if wave_number >= 1500 then
WD.set_pause_wave_in_ticks(random(36000, 54000))
else
WD.set_pause_wave_in_ticks(random(18000, 54000))
end
end
end
function Public.find_rocks_and_slowly_remove()
local active_surface_index = Public.get('active_surface_index')
local surface = game.get_surface(active_surface_index)
if not (surface and surface.valid) then
return
end
local ents = surface.find_entities_filtered({ type = 'simple-entity' })
if ents and #ents > 0 then
Public.set('rocks_to_remove', ents)
end
end
function Public.find_void_tiles_and_replace()
local active_surface_index = Public.get('active_surface_index')
local surface = game.get_surface(active_surface_index)
if not (surface and surface.valid) then
return
end
local cp = Collapse.get_position()
local rp = Collapse.get_reverse_position()
local area = {
left_top = { x = (-zone_settings.zone_width / 2) + 10, y = cp.y },
right_bottom = { x = (zone_settings.zone_width / 2) - 10, y = rp.y }
}
local adjusted_zones = Public.get('adjusted_zones')
if adjusted_zones.reversed then
area = {
left_top = { x = ((zone_settings.zone_width / 2) + 10) * -1, y = rp.y },
right_bottom = { x = math.abs((-zone_settings.zone_width / 2) - 10), y = cp.y },
}
end
local tiles = surface.find_tiles_filtered({ area = area, name = { 'out-of-map', 'water', 'deepwater', 'water-green', 'deepwater-green' } })
if tiles and #tiles > 0 then
Public.set('tiles_to_replace', tiles)
end
end
function Public.set_difficulty()
local final_battle = Public.get('final_battle')
if final_battle then
return
end
local pre_final_battle = Public.get('pre_final_battle')
if pre_final_battle then
return
end
local game_lost = Public.get('game_lost')
if game_lost then
return
end
local Diff = Difficulty.get()
if not Diff then
return
end
local wave_defense_table = WD.get_table()
local check_if_threat_below_zero = Public.get('check_if_threat_below_zero')
local collapse_amount = Public.get('collapse_amount')
local collapse_speed = Public.get('collapse_speed')
local difficulty = Public.get('difficulty')
local mining_bonus_till_wave = Public.get('mining_bonus_till_wave')
local mining_bonus = Public.get('mining_bonus')
local disable_mining_boost = Public.get('disable_mining_boost')
local wave_number = WD.get_wave()
local player_count = calc_players()
if not Diff.value then
Diff.value = 0.1
end
if not wave_defense_table.enforce_max_active_biters then
if Diff.index == 1 then
wave_defense_table.max_active_biters = 768 + player_count * (90 * Diff.value)
elseif Diff.index == 2 then
wave_defense_table.max_active_biters = 845 + player_count * (90 * Diff.value)
elseif Diff.index == 3 then
wave_defense_table.max_active_biters = 1000 + player_count * (90 * Diff.value)
end
if wave_defense_table.max_active_biters >= 4000 then
wave_defense_table.max_active_biters = 4000
end
end
-- threat gain / wave
if Diff.index == 1 then
wave_defense_table.threat_gain_multiplier = 1.2 + player_count * Diff.value * 0.1
elseif Diff.index == 2 then
wave_defense_table.threat_gain_multiplier = 2 + player_count * Diff.value * 0.1
elseif Diff.index == 3 then
wave_defense_table.threat_gain_multiplier = 4 + player_count * Diff.value * 0.1
end
-- local amount = player_count * 0.40 + 2 -- too high?
local amount = player_count * difficulty.multiply + 2
amount = floor(amount)
if amount < difficulty.lowest then
amount = difficulty.lowest
elseif amount > difficulty.highest then
amount = difficulty.highest -- lowered from 20 to 10
end
local wave = WD.get('wave_number')
local threat_check = nil
if check_if_threat_below_zero then
threat_check = wave_defense_table.threat <= 0
end
if Diff.index == 1 then
if wave < 100 then
wave_defense_table.wave_interval = 4500
else
wave_defense_table.wave_interval = 3600 - player_count * 60
end
if wave_defense_table.wave_interval < 2000 or threat_check then
wave_defense_table.wave_interval = 2000
end
elseif Diff.index == 2 then
if wave < 100 then
wave_defense_table.wave_interval = 3000
else
wave_defense_table.wave_interval = 2600 - player_count * 60
end
if wave_defense_table.wave_interval < 1800 or threat_check then
wave_defense_table.wave_interval = 1800
end
elseif Diff.index == 3 then
if wave < 100 then
wave_defense_table.wave_interval = 3000
else
wave_defense_table.wave_interval = 1600 - player_count * 60
end
wave_defense_table.wave_interval = 1600 - player_count * 60
if wave_defense_table.wave_interval < 1600 or threat_check then
wave_defense_table.wave_interval = 1600
end
end
if collapse_amount then
Collapse.set_amount(collapse_amount)
else
Collapse.set_amount(amount)
end
if collapse_speed then
Collapse.set_speed(collapse_speed)
else
if player_count >= 1 and player_count <= 8 then
Collapse.set_speed(8)
elseif player_count > 8 and player_count <= 20 then
Collapse.set_speed(7)
elseif player_count > 20 and player_count <= 35 then
Collapse.set_speed(6)
elseif player_count > 35 then
Collapse.set_speed(5)
end
end
if player_count >= 1 and not disable_mining_boost then
local force = game.forces.player
if wave_number < mining_bonus_till_wave then
-- the mining speed of the players will increase drastically since RPG is also loaded.
-- additional mining speed comes from steel axe research: 100%, and difficulty settings: too young to die 50%, hurt me plenty 25%
force.manual_mining_speed_modifier = force.manual_mining_speed_modifier - mining_bonus
if player_count <= 5 then
mining_bonus = 3 -- set a static 300% bonus if there are <= 5 players.
elseif player_count >= 6 and player_count <= 10 then
mining_bonus = 1 -- set a static 100% bonus if there are <= 10 players.
elseif player_count >= 11 then
mining_bonus = 0 -- back to 0% with more than 11 players
end
force.manual_mining_speed_modifier = force.manual_mining_speed_modifier + mining_bonus
Public.set('mining_bonus', mining_bonus) -- Setting mining_bonus globally so it remembers how much to reduce
else
force.manual_mining_speed_modifier = force.manual_mining_speed_modifier - mining_bonus
Public.set('disable_mining_boost', true)
end
end
end
function Public.render_direction(surface, reversed)
local counter = Public.get('soft_reset_counter')
local winter_mode = Public.get('winter_mode')
local text = 'Welcome to Mountain Fortress v3!'
if winter_mode then
text = 'Welcome to Wintery Mountain Fortress v3!'
end
Public.set(
'current_season',
rendering.draw_text {
text = 'Season: ' .. Public.get_stateful('season'),
surface = surface,
target = { -0, 12 },
color = { r = 0.98, g = 0.77, b = 0.22 },
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
)
Task.set_timeout_in_ticks(25, do_season_fix_token, {})
Task.set_timeout_in_ticks(50, do_season_fix_token, {})
if counter then
rendering.draw_text {
text = text .. '\nRun: ' .. counter,
surface = surface,
target = { -0, 10 },
color = { r = 0.98, g = 0.66, b = 0.22 },
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
else
rendering.draw_text {
text = text,
surface = surface,
target = { -0, 10 },
color = { r = 0.98, g = 0.66, b = 0.22 },
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
end
local x_min = -zone_settings.zone_width / 2
local x_max = zone_settings.zone_width / 2
if reversed then
local inc = 0
for _ = 1, 5 do
rendering.draw_text {
text = '',
surface = surface,
target = { -0, -20 - inc },
color = { r = 0.98, g = 0.66, b = 0.22 },
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
inc = inc + 10
end
rendering.draw_text {
text = 'Biters will attack this area.',
surface = surface,
target = { -0, -70 },
color = { r = 0.98, g = 0.66, b = 0.22 },
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
surface.create_entity({ name = 'electric-beam', position = { x_min, -74 }, source = { x_min, -74 }, target = { x_max, -74 } })
surface.create_entity({ name = 'electric-beam', position = { x_min, -74 }, source = { x_min, -74 }, target = { x_max, -74 } })
else
local inc = 0
for _ = 1, 5 do
rendering.draw_text {
text = '',
surface = surface,
target = { -0, 20 + inc },
color = { r = 0.98, g = 0.66, b = 0.22 },
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
inc = inc + 10
end
rendering.draw_text {
text = 'Biters will attack this area.',
surface = surface,
target = { -0, 70 },
color = { r = 0.98, g = 0.66, b = 0.22 },
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
surface.create_entity({ name = 'electric-beam', position = { x_min, 74 }, source = { x_min, 74 }, target = { x_max, 74 } })
surface.create_entity({ name = 'electric-beam', position = { x_min, 74 }, source = { x_min, 74 }, target = { x_max, 74 } })
end
end
function Public.boost_difficulty()
local difficulty_set = Public.get('difficulty_set')
if difficulty_set then
return
end
local force = game.forces.player
force.manual_mining_speed_modifier = force.manual_mining_speed_modifier + 0.5
force.character_running_speed_modifier = 0.15
force.manual_crafting_speed_modifier = 0.15
Public.set('coin_amount', 1)
Public.set('upgrades').flame_turret.limit = 12
Public.set('upgrades').landmine.limit = 50
Public.set('locomotive_health', 10000)
Public.set('locomotive_max_health', 10000)
Public.set('bonus_xp_on_join', 500)
WD.set('next_wave', game.tick + 3600 * 15)
Public.set('spidertron_unlocked_at_zone', 10)
WD.set_normal_unit_current_health(1.2)
WD.set_unit_health_increment_per_wave(0.30)
WD.set_boss_unit_current_health(2)
WD.set_boss_health_increment_per_wave(1.5)
WD.set('death_mode', false)
Public.set('difficulty_set', true)
end
function Public.set_spawn_position()
local collapse_pos = Collapse.get_position()
local locomotive = Public.get('locomotive')
local final_battle = Public.get('final_battle')
if final_battle then
return
end
if not locomotive or not locomotive.valid then
return
end
--TODO check if final battle
local l = locomotive.position
local retries = 0
local function check_tile(surface, tile, tbl, inc)
if not (surface and surface.valid) then
return false
end
if not tile then
return false
end
local get_tile = surface.get_tile(tile)
if get_tile.valid and get_tile.name == 'out-of-map' then
remove(tbl, inc - inc + 1)
return true
else
return false
end
end
::retry::
local y_value_position = Public.get('y_value_position')
local locomotive_positions = Public.get('locomotive_pos')
local total_pos = #locomotive_positions.tbl
local active_surface_index = Public.get('active_surface_index')
local surface = game.surfaces[active_surface_index]
if not (surface and surface.valid) then
return
end
local spawn_near_collapse = Public.get('spawn_near_collapse')
if spawn_near_collapse.active then
local collapse_position = surface.find_non_colliding_position('rocket-silo', collapse_pos, 64, 2)
if not collapse_position then
collapse_position = surface.find_non_colliding_position('solar-panel', collapse_pos, 32, 2)
end
if not collapse_position then
collapse_position = surface.find_non_colliding_position('steel-chest', collapse_pos, 32, 2)
end
local sizeof = locomotive_positions.tbl[total_pos - total_pos + 1]
if not sizeof then
goto continue
end
if check_tile(surface, sizeof, locomotive_positions.tbl, total_pos) then
retries = retries + 1
if retries == 2 then
goto continue
end
goto retry
end
local locomotive_position = surface.find_non_colliding_position('steel-chest', sizeof, 128, 1)
local distance_from = floor(math2d.position.distance(locomotive_position, locomotive.position))
local l_y = l.y
local t_y = locomotive_position.y
local c_y = collapse_pos.y
local adjusted_zones = Public.get('adjusted_zones')
local compare_pos
local compare_next
if adjusted_zones.reversed then
compare_pos = l_y - t_y >= spawn_near_collapse.compare
compare_next = c_y - t_y >= spawn_near_collapse.compare_next
else
compare_pos = l_y - t_y <= spawn_near_collapse.compare
compare_next = c_y - t_y <= spawn_near_collapse.compare_next
end
if total_pos > spawn_near_collapse.total_pos then
if compare_pos then
if locomotive_position then
if check_tile(surface, sizeof, locomotive_positions.tbl, total_pos) then
debug_str('total_pos was higher - found oom')
retries = retries + 1
if retries == 2 then
goto continue
end
goto retry
end
debug_str('total_pos was higher - spawning at locomotive_position')
WD.set_spawn_position(locomotive_position)
end
elseif compare_next then
if distance_from >= spawn_near_collapse.distance_from then
local success = check_tile(surface, locomotive_position, locomotive_positions.tbl, total_pos)
if success then
debug_str('distance_from was higher - found oom')
return
end
debug_str('distance_from was higher - spawning at locomotive_position')
WD.set_spawn_position({ x = locomotive_position.x, y = collapse_pos.y - y_value_position })
else
debug_str('distance_from was lower - spawning at locomotive_position')
WD.set_spawn_position({ x = locomotive_position.x, y = collapse_pos.y - y_value_position })
end
else
if collapse_position then
debug_str('total_pos was higher - spawning at collapse_position')
collapse_position = { x = collapse_position.x, y = collapse_position.y - y_value_position }
WD.set_spawn_position(collapse_position)
end
end
else
if collapse_position then
debug_str('total_pos was lower - spawning at collapse_position')
collapse_position = { x = collapse_position.x, y = collapse_position.y - y_value_position }
WD.set_spawn_position(collapse_position)
end
end
end
::continue::
end
function Public.on_player_joined_game(event)
local active_surface_index = Public.get('active_surface_index')
if not active_surface_index then
return
end
local players = Public.get('players')
local player = game.players[event.player_index]
local surface = game.surfaces[active_surface_index]
Public.set_difficulty()
ICW_Func.is_minimap_valid(player, surface)
Modifiers.update_player_modifiers(player)
if player.online_time < 1 then
if not players[player.index] then
players[player.index] = {}
end
local message = ({ 'main.greeting', player.name })
Alert.alert_player(player, 15, message)
if Public.get('death_mode') then
local death_message = ({ 'main.death_mode_warning' })
Alert.alert_player(player, 15, death_message)
end
player.clear_items_inside()
for item, data in pairs(this.starting_items) do
player.insert({ name = item, count = data.count })
end
end
if player.surface.index ~= active_surface_index then
local pos = surface.find_non_colliding_position('character', game.forces.player.get_spawn_position(surface), 3, 0, 5)
if pos then
player.teleport(pos, surface)
else
pos = game.forces.player.get_spawn_position(surface)
player.teleport(pos, surface)
end
else
local p = { x = player.position.x, y = player.position.y }
local get_tile = surface.get_tile(p)
if get_tile.valid and get_tile.name == 'out-of-map' then
local pos = surface.find_non_colliding_position('character', game.forces.player.get_spawn_position(surface), 3, 0, 5)
if pos then
player.teleport(pos, surface)
else
pos = game.forces.player.get_spawn_position(surface)
player.teleport(pos, surface)
end
end
end
local locomotive = Public.get('locomotive')
if not locomotive or not locomotive.valid then
return
end
local adjusted_zones = Public.get('adjusted_zones')
local distance_from_train
if adjusted_zones.reversed then
distance_from_train = player.position.y < locomotive.position.y
else
distance_from_train = player.position.y > locomotive.position.y
end
if distance_from_train then
local pos = surface.find_non_colliding_position('character', game.forces.player.get_spawn_position(surface), 3, 0, 5)
if pos then
player.teleport(pos, surface)
else
pos = game.forces.player.get_spawn_position(surface)
player.teleport(pos, surface)
end
end
end
function Public.on_player_left_game()
Public.set_difficulty()
end
function Public.is_creativity_mode_on()
local creative_enabled = Misc.get('creative_enabled')
if creative_enabled then
WD.set('next_wave', 1000)
Collapse.start_now(true)
Public.set_difficulty()
end
end
function Public.disable_creative()
local creative_enabled = Misc.get('creative_enabled')
if creative_enabled then
Misc.set('creative_enabled', false)
end
end
function Public.on_player_respawned(event)
local player = game.get_player(event.player_index)
if not player or not player.valid then
return
end
if player.character and player.character.valid then
Task.set_timeout_in_ticks(15, boost_movement_speed_on_respawn, { player = player })
player.character.health = round(player.character.health * health_values[random(1, #health_values)])
player.print(random_respawn_messages[random(1, #random_respawn_messages)])
end
end
function Public.on_player_driving_changed_state(event)
local player = game.get_player(event.player_index)
if not player or not player.valid then
return
end
local entity = event.entity
if not entity or not entity.valid then
return
end
local unit_number = entity.unit_number
local wagons_in_the_wild = Public.get('wagons_in_the_wild')
if not wagons_in_the_wild or not next(wagons_in_the_wild) then
return
end
local wagon = wagons_in_the_wild[unit_number]
if not wagon or not wagon.valid then
wagons_in_the_wild[unit_number] = nil
return
end
wagon.destructible = true
wagons_in_the_wild[unit_number] = nil
end
function Public.on_pre_player_toggled_map_editor(event)
local player = game.get_player(event.player_index)
if not player or not player.valid then
return
end
if player.name == 'Gerkiz' or not game.is_multiplayer() then
return
end
if this.editor_mode[player.index] then
return
end
this.editor_mode[player.index] = true
player.toggle_map_editor()
Task.set_timeout_in_ticks(5, exit_editor_mode_token, { player_index = player.index })
end
function Public.on_player_changed_position(event)
local active_surface_index = Public.get('active_surface_index')
if not active_surface_index then
return
end
local player = game.players[event.player_index]
if not player or not player.valid then
return
end
if player.controller_type == defines.controllers.spectator then
return
end
local map_name = 'mtn_v3'
if string.sub(player.surface.name, 0, #map_name) ~= map_name then
return
end
local position = player.position
local surface = game.surfaces[active_surface_index]
local adjusted_zones = Public.get('adjusted_zones')
local p = { x = player.position.x, y = player.position.y }
local config_tile = Public.get('void_or_tile')
if config_tile == 'lab-dark-2' then
local get_tile = surface.get_tile(p)
if get_tile.valid and get_tile.name == 'lab-dark-2' then
if random(1, 2) == 1 then
if random(1, 2) == 1 then
show_text('This path is not for players!', p, { r = 0.98, g = 0.66, b = 0.22 }, surface)
end
player.surface.create_entity({ name = 'fire-flame', position = player.position })
player.character.health = player.character.health - tile_damage
if player.character.health == 0 then
player.character.die()
local message = ({ 'main.death_message_' .. random(1, 7), player.name })
game.print(message, { r = 0.98, g = 0.66, b = 0.22 })
end
end
end
end
if adjusted_zones.reversed then
if position.y < -74 then
player.teleport({ position.x, position.y + 1 }, surface)
player.print(({ 'main.forcefield' }), { r = 0.98, g = 0.66, b = 0.22 })
if player.character then
player.character.health = player.character.health - 5
player.character.surface.create_entity({ name = 'water-splash', position = position })
if player.character.health <= 0 then
player.character.die('enemy')
end
end
end
else
if position.y >= 74 then
player.teleport({ position.x, position.y - 1 }, surface)
player.print(({ 'main.forcefield' }), { r = 0.98, g = 0.66, b = 0.22 })
if player.character then
player.character.health = player.character.health - 5
player.character.surface.create_entity({ name = 'water-splash', position = position })
if player.character.health <= 0 then
player.character.die('enemy')
end
end
end
end
end
local disable_recipes = function (force)
force.recipes['cargo-wagon'].enabled = false
force.recipes['fluid-wagon'].enabled = false
force.recipes['car'].enabled = false
force.recipes['tank'].enabled = false
force.recipes['artillery-wagon'].enabled = false
force.recipes['artillery-turret'].enabled = false
force.recipes['artillery-shell'].enabled = false
force.recipes['artillery-targeting-remote'].enabled = false
force.recipes['locomotive'].enabled = false
force.recipes['pistol'].enabled = false
force.recipes['spidertron-remote'].enabled = false
force.recipes['discharge-defense-equipment'].enabled = false
force.recipes['discharge-defense-remote'].enabled = false
end
function Public.disable_tech()
local force = game.forces.player
force.technologies['landfill'].enabled = false
force.technologies['spidertron'].enabled = false
force.technologies['spidertron'].researched = false
force.technologies['atomic-bomb'].enabled = false
force.technologies['atomic-bomb'].researched = false
force.technologies['artillery'].enabled = false
force.technologies['artillery'].researched = false
force.technologies['artillery-shell-range-1'].enabled = false
force.technologies['artillery-shell-range-1'].researched = false
force.technologies['artillery-shell-speed-1'].enabled = false
force.technologies['artillery-shell-speed-1'].researched = false
force.technologies['optics'].researched = true
force.technologies['railway'].researched = true
force.technologies['land-mine'].enabled = false
force.technologies['fluid-wagon'].enabled = false
force.technologies['cliff-explosives'].enabled = false
disable_recipes(force)
end
local disable_tech = Public.disable_tech
---@param event EventData.on_research_finished
function Public.on_research_finished(event)
disable_tech()
local research = event.research
local bonus_drill = game.forces.bonus_drill
local player = game.forces.player
local research_name = research.name
local force = research.force
if event.tick > 100 then
if Public.get('print_tech_to_discord') and force.name == 'player' then
Server.to_discord_embed_raw('<a:Modded:835932131036364810> ' .. research_name:gsub('^%l', string.upper) .. ' has been researched!')
end
end
if research.name == 'toolbelt' then
Public.set('toolbelt_researched_count', 10)
end
research.force.character_inventory_slots_bonus = (player.mining_drill_productivity_bonus * 50) + (Public.get('toolbelt_researched_count') or 0)
if bonus_drill then
bonus_drill.mining_drill_productivity_bonus = bonus_drill.mining_drill_productivity_bonus + 0.03
if bonus_drill.mining_drill_productivity_bonus >= 3 then
bonus_drill.mining_drill_productivity_bonus = 3
end
end
local players = game.connected_players
for i = 1, #players do
local p = players[i]
Modifiers.update_player_modifiers(p)
end
if research.name == 'steel-axe' then
local msg = 'Steel-axe technology has been researched, 100% has been applied.\nBuy Pickaxe-upgrades in the market to boost it even more!'
Alert.alert_all_players(30, msg, nil, 'achievement/tech-maniac', 0.6)
end
local force_name = research.force.name
if not force_name then
return
end
local flamethrower_damage = Public.get('flamethrower_damage')
flamethrower_damage[force_name] = -0.85
if research.name == 'military' then
game.forces[force_name].set_turret_attack_modifier('flamethrower-turret', flamethrower_damage[force_name])
game.forces[force_name].set_ammo_damage_modifier('flamethrower', flamethrower_damage[force_name])
end
if string.sub(research.name, 0, 18) == 'refined-flammables' then
flamethrower_damage[force_name] = flamethrower_damage[force_name] + 0.10
game.forces[force_name].set_turret_attack_modifier('flamethrower-turret', flamethrower_damage[force_name])
game.forces[force_name].set_ammo_damage_modifier('flamethrower', flamethrower_damage[force_name])
end
end
function Public.set_player_to_god(player)
if player.character and player.character.valid then
return false
end
if not player.character and player.controller_type ~= defines.controllers.spectator then
player.print('[color=blue][Spectate][/color] It seems that you are not in the realm of the living.', Color.warning)
return false
end
local spectate = Public.get('spectate')
if spectate[player.index] and spectate[player.index].delay and spectate[player.index].delay > game.tick then
local cooldown = floor((spectate[player.index].delay - game.tick) / 60) + 1 .. ' seconds!'
player.print('[color=blue][Spectate][/color] Retry again in ' .. cooldown, Color.warning)
return false
end
spectate[player.index] = nil
player.set_controller({ type = defines.controllers.god })
player.create_character()
local active_surface_index = Public.get('active_surface_index')
local surface = game.get_surface(active_surface_index)
if not surface or not surface.valid then
return false
end
if string.sub(player.surface.name, 0, #surface.name) == surface.name then
local pos = surface.find_non_colliding_position('character', game.forces.player.get_spawn_position(surface), 3, 0, 5)
if pos then
player.teleport(pos, surface)
else
pos = game.forces.player.get_spawn_position(surface)
player.teleport(pos, surface)
end
else
local pos = player.surface.find_non_colliding_position('character', { 0, 0 }, 3, 0, 5)
if pos then
player.teleport(pos, player.surface)
else
player.teleport({ pos }, player.surface)
end
end
Event.raise(
BottomFrame.events.bottom_quickbar_respawn_raise,
{
player_index = player.index
}
)
player.tag = ''
game.print('[color=blue][Spectate][/color] ' .. player.name .. ' is no longer spectating!')
Server.to_discord_bold(table.concat { '*** ', '[Spectate] ' .. player.name .. ' is no longer spectating!', ' ***' })
return true
end
function Public.set_player_to_spectator(player)
if player.in_combat then
player.print('[color=blue][Spectate][/color] You are in combat. Try again soon.', Color.warning)
return false
end
if player.driving then
return player.print('[color=blue][Spectate][/color] Please exit the vehicle before continuing', Color.warning)
end
local spectate = Public.get('spectate')
if not spectate[player.index] then
spectate[player.index] = {
verify = false
}
player.print('[color=blue][Spectate][/color] Please click the spectate button again if you really want to this.', Color.warning)
return false
end
if player.character and player.character.valid then
player.character.die()
end
player.character = nil
player.spectator = true
player.tag = '[img=utility/ghost_time_to_live_modifier_icon]'
player.set_controller({ type = defines.controllers.spectator })
game.print('[color=blue][Spectate][/color] ' .. player.name .. ' is now spectating.')
Server.to_discord_bold(table.concat { '*** ', '[Spectate] ' .. player.name .. ' is now spectating.', ' ***' })
if spectate[player.index] and not spectate[player.index].delay then
spectate[player.index].verify = true
spectate[player.index].delay = game.tick + 3600
end
return true
end
Public.firearm_magazine_ammo = { name = 'firearm-magazine', count = 200 }
Public.piercing_rounds_magazine_ammo = { name = 'piercing-rounds-magazine', count = 200 }
Public.uranium_rounds_magazine_ammo = { name = 'uranium-rounds-magazine', count = 200 }
Public.light_oil_ammo = { name = 'light-oil', amount = 100 }
Public.artillery_shell_ammo = { name = 'artillery-shell', count = 15 }
Public.laser_turrent_power_source = { buffer_size = 2400000, power_production = 40000 }
Public.pause_waves_custom_callback_token = pause_waves_custom_callback_token
function Public.get_func(key)
if key then
return this[key]
else
return this
end
end
function Public.show_all_gui(player)
for _, child in pairs(player.gui.top.children) do
child.visible = true
end
end
function Public.clear_spec_tag(player)
if player.tag == '[Spectator]' then
player.tag = ''
end
end
function Public.equip_players(starting_items, recreate)
local players = Public.get('players')
for _, player in pairs(game.players) do
if player.character and player.character.valid then
player.character.destroy()
end
player.clear_items_inside()
if player.connected then
if not player.character then
player.set_controller({ type = defines.controllers.god })
player.create_character()
end
player.character.destructible = true
Modifiers.update_player_modifiers(player)
if not recreate then
starting_items = starting_items or this.starting_items
for item, item_data in pairs(this.starting_items) do
player.insert({ name = item, count = item_data.count })
end
end
Public.show_all_gui(player)
Public.clear_spec_tag(player)
else
if player.character then
player.character.destructible = true
end
players[player.index] = nil
Session.clear_player(player)
game.remove_offline_players({ player.index })
end
end
end
function Public.reset_func_table()
this.power_sources = { index = 1 }
this.refill_turrets = { index = 1 }
this.magic_crafters = { index = 1 }
this.magic_fluid_crafters = { index = 1 }
this.techs = {}
this.limit_types = {}
this.starting_items = {
['pistol'] = {
count = 1
},
['firearm-magazine'] = {
count = 16
},
['rail'] = {
count = 16
},
['wood'] = {
count = 16
},
['explosives'] = {
count = 32
}
}
end
local on_player_joined_game = Public.on_player_joined_game
local on_player_left_game = Public.on_player_left_game
local on_research_finished = Public.on_research_finished
local on_player_changed_position = Public.on_player_changed_position
local on_player_respawned = Public.on_player_respawned
local on_player_driving_changed_state = Public.on_player_driving_changed_state
local on_pre_player_toggled_map_editor = Public.on_pre_player_toggled_map_editor
Event.add(de.on_player_joined_game, on_player_joined_game)
Event.add(de.on_player_left_game, on_player_left_game)
Event.add(de.on_research_finished, on_research_finished)
Event.add(de.on_player_changed_position, on_player_changed_position)
Event.add(de.on_player_respawned, on_player_respawned)
Event.add(de.on_player_driving_changed_state, on_player_driving_changed_state)
Event.add(de.on_pre_player_toggled_map_editor, on_pre_player_toggled_map_editor)
Event.on_nth_tick(10, tick)
Event.add(WD.events.on_wave_created, on_wave_created)
return Public