1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-16 02:47:48 +02:00
ComfyFactorio/maps/mountain_fortress_v3/functions.lua

1425 lines
44 KiB
Lua
Raw Normal View History

2020-05-21 23:08:23 +02:00
local Token = require 'utils.token'
2020-06-25 17:59:16 +02:00
local Task = require 'utils.task'
2021-05-23 21:13:21 +02:00
local Color = require 'utils.color_presets'
2020-05-21 23:08:23 +02:00
local ICW = require 'maps.mountain_fortress_v3.icw.main'
local Event = require 'utils.event'
2020-07-14 21:50:56 +02:00
local Global = require 'utils.global'
2020-12-04 02:09:52 +02:00
local Alert = require 'utils.alert'
local WPT = require 'maps.mountain_fortress_v3.table'
local WD = require 'modules.wave_defense.table'
2022-05-12 22:21:15 +02:00
local RPG = require 'modules.rpg.main'
2020-12-04 02:09:52 +02:00
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'
2021-12-05 23:15:49 +02:00
local Misc = require 'utils.commands.misc'
2022-04-07 16:11:18 +02:00
local Core = require 'utils.core'
2022-04-07 00:17:41 +02:00
local zone_settings = WPT.zone_settings
2020-07-14 21:50:56 +02:00
local this = {
power_sources = {index = 1},
refill_turrets = {index = 1},
magic_crafters = {index = 1},
2020-07-28 11:24:16 +02:00
magic_fluid_crafters = {index = 1},
2020-08-09 20:31:50 +02:00
art_table = {index = 1},
surface_cleared = false
2020-07-14 21:50:56 +02:00
}
2021-05-23 17:03:52 +02:00
local starting_items = {
['pistol'] = 1,
['firearm-magazine'] = 16,
['rail'] = 16,
['wood'] = 16,
['explosives'] = 32
}
2020-12-04 02:09:52 +02:00
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'
}
2020-07-14 21:50:56 +02:00
Global.register(
this,
function(t)
this = t
end
)
2020-05-21 23:08:23 +02:00
local Public = {}
2020-12-04 02:09:52 +02:00
2020-08-09 20:22:33 +02:00
local random = math.random
local floor = math.floor
local round = math.round
2020-12-04 02:09:52 +02:00
local remove = table.remove
2020-08-09 20:22:33 +02:00
local sqrt = math.sqrt
2020-05-21 23:08:23 +02:00
local magic_crafters_per_tick = 3
local magic_fluid_crafters_per_tick = 8
2020-12-04 02:09:52 +02:00
local tile_damage = 50
2020-05-21 23:08:23 +02:00
2020-07-28 11:24:16 +02:00
local artillery_target_entities = {
'character',
'tank',
'car',
2020-12-05 19:09:09 +02:00
'radar',
'lab',
2020-07-28 11:24:16 +02:00
'furnace',
2020-10-19 20:21:27 +02:00
'locomotive',
'cargo-wagon',
'fluid-wagon',
2020-12-05 19:09:09 +02:00
'artillery-wagon',
'artillery-turret',
'laser-turret',
'gun-turret',
'flamethrower-turret',
'silo',
'spidertron'
2020-07-28 11:24:16 +02:00
}
2020-12-04 02:09:52 +02:00
local function debug_str(msg)
local debug = WPT.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
2020-05-21 23:08:23 +02:00
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 function do_refill_turrets()
2020-07-14 21:50:56 +02:00
local refill_turrets = this.refill_turrets
2020-05-21 23:08:23 +02:00
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()
2020-07-14 21:50:56 +02:00
local magic_crafters = this.magic_crafters
2020-05-21 23:08:23 +02:00
local limit = #magic_crafters
if limit == 0 then
return
end
local index = magic_crafters.index
2022-04-07 16:11:18 +02:00
for _ = 1, magic_crafters_per_tick do
2020-05-21 23:08:23 +02:00
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)
2020-11-17 13:45:27 +02:00
if fcount > 1 then
fcount = 1
end
2020-05-21 23:08:23 +02:00
if fcount > 0 then
2021-05-12 20:52:29 +02:00
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
2021-11-23 21:28:23 +02:00
data.last_tick = round(tick - (count - fcount) / rate)
2021-05-12 20:52:29 +02:00
end
2020-05-21 23:08:23 +02:00
end
end
end
magic_crafters.index = index
end
local function do_magic_fluid_crafters()
2020-07-14 21:50:56 +02:00
local magic_fluid_crafters = this.magic_fluid_crafters
2020-05-21 23:08:23 +02:00
local limit = #magic_fluid_crafters
if limit == 0 then
return
end
local index = magic_fluid_crafters.index
2022-04-07 16:11:18 +02:00
for _ = 1, magic_fluid_crafters_per_tick do
2020-05-21 23:08:23 +02:00
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
2021-05-12 20:52:29 +02:00
entity.products_finished = entity.products_finished + fcount
2020-05-21 23:08:23 +02:00
data.last_tick = tick - (count - fcount) / rate
end
end
end
magic_fluid_crafters.index = index
end
2020-07-28 11:24:16 +02:00
local artillery_target_callback =
Token.register(
function(data)
local position = data.position
local entity = data.entity
if not entity.valid then
return
end
local tx, ty = position.x, position.y
local pos = entity.position
local x, y = pos.x, pos.y
local dx, dy = tx - x, ty - y
local d = dx * dx + dy * dy
2020-11-17 13:45:27 +02:00
if d >= 1024 and d <= 441398 then -- 704 in depth~
2020-12-05 19:09:09 +02:00
if entity.name == 'character' then
entity.surface.create_entity {
name = 'artillery-projectile',
position = position,
target = entity,
force = 'enemy',
speed = 1.5
}
elseif entity.name ~= 'character' then
entity.surface.create_entity {
name = 'rocket',
position = position,
target = entity,
force = 'enemy',
speed = 1.5
}
end
2020-07-28 11:24:16 +02:00
end
end
)
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
outpost.last_fire_tick = now
local turret = turrets[1]
local area = outpost.artillery_area
local surface = turret.surface
2020-12-05 19:09:09 +02:00
local entities = surface.find_entities_filtered {area = area, name = artillery_target_entities, force = 'player'}
2020-07-28 11:24:16 +02:00
if #entities == 0 then
return
end
local position = turret.position
for i = 1, count do
2020-08-09 20:22:33 +02:00
local entity = entities[random(#entities)]
2020-07-28 11:24:16 +02:00
if entity and entity.valid then
local data = {position = position, entity = entity}
Task.set_timeout_in_ticks(i * 60, artillery_target_callback, data)
end
end
end
2020-06-25 17:59:16 +02:00
local function add_magic_crafter_output(entity, output, distance)
2020-07-14 21:50:56 +02:00
local magic_fluid_crafters = this.magic_fluid_crafters
local magic_crafters = this.magic_crafters
2020-06-25 17:59:16 +02:00
local rate = output.min_rate + output.distance_factor * distance
local fluidbox_index = output.fluidbox_index
local data = {
entity = entity,
2022-04-05 12:00:13 +02:00
last_tick = game.tick,
2021-11-23 21:28:23 +02:00
base_rate = round(rate, 8),
rate = round(rate, 8),
2020-06-25 17:59:16 +02:00
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
2020-05-21 23:08:23 +02:00
local function tick()
do_refill_turrets()
do_magic_crafters()
do_magic_fluid_crafters()
2020-07-28 11:24:16 +02:00
do_artillery_turrets_targets()
2020-05-21 23:08:23 +02:00
end
2020-05-23 21:18:18 +02:00
Public.deactivate_callback =
2020-05-21 23:08:23 +02:00
Token.register(
2020-05-23 21:18:18 +02:00
function(entity)
2020-06-25 17:59:16 +02:00
if entity and entity.valid then
entity.active = false
entity.operable = false
entity.destructible = false
end
2020-05-23 21:18:18 +02:00
end
)
2020-05-21 23:08:23 +02:00
2020-05-23 21:18:18 +02:00
Public.neutral_force =
Token.register(
function(entity)
2020-06-25 17:59:16 +02:00
if entity and entity.valid then
entity.force = 'neutral'
end
2020-05-21 23:08:23 +02:00
end
)
2020-05-23 21:18:18 +02:00
Public.enemy_force =
2020-05-21 23:08:23 +02:00
Token.register(
function(entity)
2020-06-25 17:59:16 +02:00
if entity and entity.valid then
entity.force = 'enemy'
end
2020-05-21 23:08:23 +02:00
end
)
2020-05-23 21:18:18 +02:00
2020-05-21 23:08:23 +02:00
Public.active_not_destructible_callback =
Token.register(
function(entity)
2020-06-25 17:59:16 +02:00
if entity and entity.valid then
entity.active = true
entity.operable = false
entity.destructible = false
end
2020-05-21 23:08:23 +02:00
end
)
Public.disable_minable_callback =
Token.register(
function(entity)
2020-06-25 17:59:16 +02:00
if entity and entity.valid then
entity.minable = false
end
2020-05-21 23:08:23 +02:00
end
)
Public.disable_minable_and_ICW_callback =
Token.register(
function(entity)
2020-06-25 17:59:16 +02:00
if entity and entity.valid then
entity.minable = false
2022-04-07 16:11:18 +02:00
ICW.register_wagon(entity)
2020-06-25 17:59:16 +02:00
end
2020-05-21 23:08:23 +02:00
end
)
Public.disable_destructible_callback =
Token.register(
function(entity)
2020-06-25 17:59:16 +02:00
if entity and entity.valid then
entity.destructible = false
2020-08-04 12:10:15 +02:00
entity.minable = false
2020-06-25 17:59:16 +02:00
end
2020-05-21 23:08:23 +02:00
end
)
Public.disable_active_callback =
Token.register(
function(entity)
2020-06-25 17:59:16 +02:00
if entity and entity.valid then
entity.active = false
end
2020-05-21 23:08:23 +02:00
end
)
2020-06-25 17:59:16 +02:00
local disable_active_callback = Public.disable_active_callback
2020-05-23 21:18:18 +02:00
Public.refill_turret_callback =
Token.register(
function(turret, data)
2020-07-14 21:50:56 +02:00
local refill_turrets = this.refill_turrets
2020-05-23 21:18:18 +02:00
local callback_data = data.callback_data
turret.direction = 3
refill_turrets[#refill_turrets + 1] = {turret = turret, data = callback_data}
end
)
2020-07-28 11:24:16 +02:00
Public.refill_artillery_turret_callback =
Token.register(
function(turret, data)
local refill_turrets = this.refill_turrets
local art_table = this.art_table
local index = art_table.index
2020-12-05 19:09:09 +02:00
turret.active = false
2020-07-28 11:24:16 +02:00
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
artillery_data.artillery_area = {{x - 112, y}, {x + 112, y + 212}}
artillery_data.last_fire_tick = 0
art_table[#art_table + 1] = artillery_data
end
artillery_turrets[#artillery_turrets + 1] = turret
end
)
2020-05-21 23:08:23 +02:00
Public.refill_liquid_turret_callback =
Token.register(
function(turret, data)
2020-07-14 21:50:56 +02:00
local refill_turrets = this.refill_turrets
2020-05-21 23:08:23 +02:00
local callback_data = data.callback_data
callback_data.liquid = true
2020-05-23 21:18:18 +02:00
refill_turrets[#refill_turrets + 1] = {turret = turret, data = callback_data}
2020-05-21 23:08:23 +02:00
end
)
Public.power_source_callback =
Token.register(
function(turret)
2020-07-14 21:50:56 +02:00
local power_sources = this.power_sources
2020-11-25 20:55:51 +02:00
power_sources[#power_sources + 1] = turret
2020-05-21 23:08:23 +02:00
end
)
2020-06-25 17:59:16 +02:00
Public.magic_item_crafting_callback =
Token.register(
function(entity, data)
local callback_data = data.callback_data
2020-12-04 02:09:52 +02:00
if not (entity and entity.valid) then
return
end
2020-06-25 17:59:16 +02:00
entity.minable = false
entity.destructible = false
entity.operable = false
2020-12-04 02:09:52 +02:00
local force = game.forces.player
local tech = callback_data.tech
2022-04-05 19:27:46 +02:00
if not callback_data.testing then
if tech then
if not force.technologies[tech].researched then
entity.destroy()
return
end
2020-12-04 02:09:52 +02:00
end
end
2020-06-25 17:59:16 +02:00
local recipe = callback_data.recipe
if recipe then
entity.set_recipe(recipe)
else
local furance_item = callback_data.furance_item
if furance_item then
2020-07-25 22:22:21 +02:00
local inv = entity.get_inventory(defines.inventory.furnace_result)
2020-06-25 17:59:16 +02:00
inv.insert(furance_item)
end
end
local p = entity.position
local x, y = p.x, p.y
2020-08-09 20:22:33 +02:00
local distance = sqrt(x * x + y * y)
2020-06-25 17:59:16 +02:00
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 =
Token.register(
function(entity, data)
local callback_data = data.callback_data
2020-12-04 02:09:52 +02:00
if not (entity and entity.valid) then
return
end
2020-06-25 17:59:16 +02:00
entity.minable = false
entity.destructible = false
entity.operable = false
local weights = callback_data.weights
local loot = callback_data.loot
local p = entity.position
2020-08-09 20:22:33 +02:00
local i = random() * weights.total
2020-06-25 17:59:16 +02:00
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
2020-12-04 02:09:52 +02:00
local force = game.forces.player
local tech = stack.tech
2022-04-05 19:27:46 +02:00
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
2020-12-14 18:31:02 +02:00
end
2020-12-04 02:09:52 +02:00
end
end
2020-06-25 17:59:16 +02:00
local recipe = stack.recipe
if recipe then
entity.set_recipe(recipe)
else
local furance_item = stack.furance_item
if furance_item then
2020-07-30 22:27:38 +02:00
local inv = entity.get_inventory(defines.inventory.furnace_result)
2020-06-25 17:59:16 +02:00
inv.insert(furance_item)
end
end
local x, y = p.x, p.y
2020-08-09 20:22:33 +02:00
local distance = sqrt(x * x + y * y)
2020-06-25 17:59:16 +02:00
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
2020-08-09 20:22:33 +02:00
local i = random() * weights.total
2020-06-25 17:59:16 +02:00
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
2020-08-09 20:22:33 +02:00
local d = sqrt(x * x + y * y)
2020-06-25 17:59:16 +02:00
count = stack.count + d * df
else
count = stack.count
end
entity.insert {name = stack.name, count = count}
end
2020-12-20 20:56:26 +02:00
local function calc_players()
local players = game.connected_players
2020-12-20 21:40:12 +02:00
local check_afk_players = WPT.get('check_afk_players')
if not check_afk_players then
return #players
end
2020-12-20 20:56:26 +02:00
local total = 0
2022-04-07 16:11:18 +02:00
Core.iter_connected_players(
function(player)
if player.afk_time < 36000 then
total = total + 1
end
2020-12-20 20:56:26 +02:00
end
2022-04-07 16:11:18 +02:00
)
2020-12-20 20:56:26 +02:00
if total <= 0 then
total = #players
2020-12-20 20:56:26 +02:00
end
return total
end
2021-05-23 21:13:21 +02:00
local retry_final_boost_movement_speed_on_respawn =
Token.register(
function(data)
local player = data.player
local old_speed = data.old_speed
if not player or not player.valid then
return
end
if not player.character or not player.character.valid then
return
end
player.character.character_running_speed_modifier = old_speed
player.print('Movement speed bonus removed!', Color.info)
2022-05-12 22:21:15 +02:00
local rpg_t = RPG.get_value_from_player(player.index)
rpg_t.has_custom_spell_active = nil
2021-05-23 21:13:21 +02:00
end
)
local retry_boost_movement_speed_on_respawn =
Token.register(
function(data)
local player = data.player
local old_speed = data.old_speed
if not player or not player.valid then
return
end
if not player.character or not player.character.valid then
Task.set_timeout_in_ticks(10, retry_final_boost_movement_speed_on_respawn, {player = player, old_speed = old_speed})
return
end
player.character.character_running_speed_modifier = old_speed
player.print('Movement speed bonus removed!', Color.info)
2022-05-12 22:21:15 +02:00
local rpg_t = RPG.get_value_from_player(player.index)
rpg_t.has_custom_spell_active = nil
2021-05-23 21:13:21 +02:00
end
)
2021-05-28 20:37:06 +02:00
local remove_boost_movement_speed_on_respawn =
2021-05-23 21:13:21 +02:00
Token.register(
function(data)
local player = data.player
local old_speed = data.old_speed
if not player or not player.valid then
return
end
if not player.character or not player.character.valid then
Task.set_timeout_in_ticks(10, retry_boost_movement_speed_on_respawn, {player = player, old_speed = old_speed})
return
end
player.character.character_running_speed_modifier = old_speed
player.print('Movement speed bonus removed!', Color.info)
2022-05-12 22:21:15 +02:00
local rpg_t = RPG.get_value_from_player(player.index)
rpg_t.has_custom_spell_active = nil
2021-05-23 21:13:21 +02:00
end
)
2021-05-28 20:37:06 +02:00
local boost_movement_speed_on_respawn =
Token.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
2022-05-12 22:21:15 +02:00
local rpg_t = RPG.get_value_from_player(player.index)
rpg_t.has_custom_spell_active = true
2021-05-28 20:37:06 +02:00
local old_speed = player.character_running_speed_modifier
local new_speed = player.character_running_speed_modifier + 1
Task.set_timeout_in_ticks(800, remove_boost_movement_speed_on_respawn, {player = player, old_speed = old_speed})
player.character.character_running_speed_modifier = new_speed
player.print('Movement speed bonus applied! Be quick and fetch your corpse!', Color.info)
end
)
2020-12-04 02:09:52 +02:00
function Public.set_difficulty()
2020-12-05 19:09:09 +02:00
local game_lost = WPT.get('game_lost')
if game_lost then
return
end
2020-12-04 02:09:52 +02:00
local Diff = Difficulty.get()
2022-06-26 23:32:54 +02:00
if not Diff then
return
end
2020-12-04 02:09:52 +02:00
local wave_defense_table = WD.get_table()
2021-11-22 21:16:33 +02:00
local check_if_threat_below_zero = WPT.get('check_if_threat_below_zero')
2020-12-04 02:09:52 +02:00
local collapse_amount = WPT.get('collapse_amount')
2020-12-05 19:09:09 +02:00
local collapse_speed = WPT.get('collapse_speed')
2020-12-30 12:07:44 +02:00
local difficulty = WPT.get('difficulty')
2021-10-08 18:50:55 +02:00
local mining_bonus_till_wave = WPT.get('mining_bonus_till_wave')
2021-10-11 21:53:27 +02:00
local mining_bonus = WPT.get('mining_bonus')
2021-10-11 10:08:28 +02:00
local disable_mining_boost = WPT.get('disable_mining_boost')
2021-10-08 18:50:55 +02:00
local wave_number = WD.get_wave()
2020-12-20 21:40:12 +02:00
local player_count = calc_players()
2020-12-04 02:09:52 +02:00
if not Diff.difficulty_vote_value then
Diff.difficulty_vote_value = 0.1
end
2021-10-15 16:59:25 +02:00
if Diff.name == "I'm too young to die" then
wave_defense_table.max_active_biters = 768 + player_count * (90 * Diff.difficulty_vote_value)
elseif Diff.name == 'Hurt me plenty' then
wave_defense_table.max_active_biters = 845 + player_count * (90 * Diff.difficulty_vote_value)
elseif Diff.name == 'Ultra-violence' then
wave_defense_table.max_active_biters = 1000 + player_count * (90 * Diff.difficulty_vote_value)
end
2020-12-04 02:09:52 +02:00
2020-12-05 19:09:09 +02:00
if wave_defense_table.max_active_biters >= 4000 then
wave_defense_table.max_active_biters = 4000
2020-12-04 02:09:52 +02:00
end
-- threat gain / wave
2021-10-15 16:59:25 +02:00
if Diff.name == "I'm too young to die" then
wave_defense_table.threat_gain_multiplier = 1.2 + player_count * Diff.difficulty_vote_value * 0.1
elseif Diff.name == 'Hurt me plenty' then
wave_defense_table.threat_gain_multiplier = 2 + player_count * Diff.difficulty_vote_value * 0.1
elseif Diff.name == 'Ultra-violence' then
wave_defense_table.threat_gain_multiplier = 4 + player_count * Diff.difficulty_vote_value * 0.1
end
2020-12-04 02:09:52 +02:00
2020-12-20 20:56:26 +02:00
-- local amount = player_count * 0.40 + 2 -- too high?
2020-12-30 12:07:44 +02:00
local amount = player_count * difficulty.multiply + 2
2020-12-04 02:09:52 +02:00
amount = floor(amount)
2021-10-08 18:50:55 +02:00
if amount < difficulty.lowest then
amount = difficulty.lowest
elseif amount > difficulty.highest then
amount = difficulty.highest -- lowered from 20 to 10
2020-12-04 02:09:52 +02:00
end
2021-11-22 21:16:33 +02:00
local threat_check = nil
if check_if_threat_below_zero then
threat_check = wave_defense_table.threat <= 0
end
2021-10-15 16:59:25 +02:00
if Diff.name == "I'm too young to die" then
if player_count < 10 then
wave_defense_table.wave_interval = 4500
else
wave_defense_table.wave_interval = 3600 - player_count * 60
end
2021-11-28 21:42:38 +02:00
if wave_defense_table.wave_interval < 2200 or threat_check then
wave_defense_table.wave_interval = 2200
2021-10-15 16:59:25 +02:00
end
elseif Diff.name == 'Hurt me plenty' then
if player_count < 10 then
wave_defense_table.wave_interval = 3000
else
wave_defense_table.wave_interval = 2600 - player_count * 60
end
2021-11-28 21:42:38 +02:00
if wave_defense_table.wave_interval < 1900 or threat_check then
wave_defense_table.wave_interval = 1900
2021-10-15 16:59:25 +02:00
end
elseif Diff.name == 'Ultra-violence' then
if player_count < 10 then
wave_defense_table.wave_interval = 2000
else
wave_defense_table.wave_interval = 1600 - player_count * 60
end
2021-11-26 09:52:18 +02:00
wave_defense_table.wave_interval = 1600 - player_count * 60
2021-11-28 21:42:38 +02:00
if wave_defense_table.wave_interval < 1600 or threat_check then
wave_defense_table.wave_interval = 1600
2021-10-15 16:59:25 +02:00
end
2020-12-04 02:09:52 +02:00
end
if collapse_amount then
Collapse.set_amount(collapse_amount)
else
Collapse.set_amount(amount)
end
2020-12-05 19:09:09 +02:00
if collapse_speed then
Collapse.set_speed(collapse_speed)
else
2020-12-29 14:46:53 +02:00
if player_count >= 1 and player_count <= 8 then
2020-12-05 19:09:09 +02:00
Collapse.set_speed(8)
2021-10-08 18:50:55 +02:00
elseif player_count > 8 and player_count <= 20 then
Collapse.set_speed(7)
2020-12-29 14:46:53 +02:00
elseif player_count > 20 and player_count <= 35 then
2020-12-05 19:09:09 +02:00
Collapse.set_speed(6)
2020-12-29 14:46:53 +02:00
elseif player_count > 35 then
2020-12-05 19:09:09 +02:00
Collapse.set_speed(5)
end
end
2021-10-08 18:50:55 +02:00
2021-10-11 10:08:28 +02:00
if player_count >= 1 and not disable_mining_boost then
2021-10-08 18:50:55 +02:00
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.
2021-10-11 21:53:27 +02:00
-- 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
2021-10-11 21:53:27 +02:00
mining_bonus = 3 -- set a static 300% bonus if there are <= 5 players.
elseif player_count >= 6 and player_count <= 10 then
2021-10-11 21:53:27 +02:00
mining_bonus = 1 -- set a static 100% bonus if there are <= 10 players.
2021-10-15 16:59:25 +02:00
elseif player_count >= 11 then
mining_bonus = 0 -- back to 0% with more than 11 players
2021-10-08 18:50:55 +02:00
end
2021-10-11 21:53:27 +02:00
force.manual_mining_speed_modifier = force.manual_mining_speed_modifier + mining_bonus
WPT.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
WPT.set('disable_mining_boost', true)
2021-10-08 18:50:55 +02:00
end
end
2020-12-04 02:09:52 +02:00
end
function Public.render_direction(surface)
local counter = WPT.get('soft_reset_counter')
2020-12-20 20:56:26 +02:00
local winter_mode = WPT.get('winter_mode')
local text = 'Welcome to Mountain Fortress v3!'
if winter_mode then
text = 'Welcome to Wintery Mountain Fortress v3!'
end
2020-12-04 02:09:52 +02:00
if counter then
rendering.draw_text {
2020-12-20 20:56:26 +02:00
text = text .. '\nRun: ' .. counter,
2020-12-04 02:09:52 +02:00
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 {
2020-12-20 20:56:26 +02:00
text = text,
2020-12-04 02:09:52 +02:00
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
rendering.draw_text {
text = '',
surface = surface,
target = {-0, 20},
color = {r = 0.98, g = 0.66, b = 0.22},
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
rendering.draw_text {
text = '',
surface = surface,
target = {-0, 30},
color = {r = 0.98, g = 0.66, b = 0.22},
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
rendering.draw_text {
text = '',
surface = surface,
target = {-0, 40},
color = {r = 0.98, g = 0.66, b = 0.22},
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
rendering.draw_text {
text = '',
surface = surface,
target = {-0, 50},
color = {r = 0.98, g = 0.66, b = 0.22},
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
rendering.draw_text {
text = '',
surface = surface,
target = {-0, 60},
color = {r = 0.98, g = 0.66, b = 0.22},
scale = 3,
font = 'heading-1',
alignment = 'center',
scale_with_zoom = false
}
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
}
2022-04-07 00:17:41 +02:00
local x_min = -zone_settings.zone_width / 2
local x_max = zone_settings.zone_width / 2
2020-12-04 02:09:52 +02:00
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
function Public.boost_difficulty()
local difficulty_set = WPT.get('difficulty_set')
if difficulty_set then
return
end
local breached_wall = WPT.get('breached_wall')
local difficulty = Difficulty.get()
local name = difficulty.difficulties[difficulty.difficulty_vote_index].name
if game.tick < difficulty.difficulty_poll_closing_timeout and breached_wall <= 1 then
return
end
Difficulty.get().name = name
Difficulty.get().difficulty_poll_closing_timeout = game.tick
Difficulty.get().button_tooltip = difficulty.tooltip[difficulty.difficulty_vote_index]
Difficulty.difficulty_gui()
local message = ({'main.diff_set', name})
local data = {
position = WPT.get('locomotive').position
}
Alert.alert_all_players_location(data, message)
local force = game.forces.player
2022-04-05 12:00:13 +02:00
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
WPT.set('coin_amount', 1)
WPT.set('upgrades').flame_turret.limit = 12
WPT.set('upgrades').landmine.limit = 50
WPT.set('locomotive_health', 10000)
WPT.set('locomotive_max_health', 10000)
WPT.set('bonus_xp_on_join', 500)
WD.set('next_wave', game.tick + 3600 * 15)
WPT.set('spidertron_unlocked_at_zone', 10)
WD.set_normal_unit_current_health(1.0)
WD.set_unit_health_increment_per_wave(0.15)
WD.set_boss_unit_current_health(2)
WD.set_boss_health_increment_per_wave(1.5)
WPT.set('difficulty_set', true)
2020-12-04 02:09:52 +02:00
end
function Public.set_spawn_position()
local collapse_pos = Collapse.get_position()
local locomotive = WPT.get('locomotive')
if not locomotive or not locomotive.valid then
return
end
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
2020-12-05 19:09:09 +02:00
remove(tbl, inc - inc + 1)
2020-12-04 02:09:52 +02:00
return true
else
return false
end
end
::retry::
local y_value_position = WPT.get('y_value_position')
2020-12-04 02:09:52 +02:00
local locomotive_positions = WPT.get('locomotive_pos')
local total_pos = #locomotive_positions.tbl
local active_surface_index = WPT.get('active_surface_index')
local surface = game.surfaces[active_surface_index]
if not (surface and surface.valid) then
return
end
local spawn_near_collapse = WPT.get('spawn_near_collapse')
if spawn_near_collapse.active then
2021-03-01 23:04:55 +02:00
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)
2021-03-01 23:04:55 +02:00
end
2020-12-04 02:09:52 +02:00
local sizeof = locomotive_positions.tbl[total_pos - total_pos + 1]
2021-01-26 21:17:52 +02:00
if not sizeof then
goto continue
end
2020-12-04 02:09:52 +02:00
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)
2020-12-04 02:09:52 +02:00
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
if total_pos > spawn_near_collapse.total_pos then
if l_y - t_y <= spawn_near_collapse.compare 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 c_y - t_y <= spawn_near_collapse.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})
2020-12-04 02:09:52 +02:00
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})
2020-12-04 02:09:52 +02:00
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}
2020-12-04 02:09:52 +02:00
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}
2020-12-04 02:09:52 +02:00
WD.set_spawn_position(collapse_position)
end
end
end
::continue::
end
function Public.on_player_joined_game(event)
local active_surface_index = WPT.get('active_surface_index')
2021-05-23 17:03:52 +02:00
local players = WPT.get('players')
2020-12-04 02:09:52 +02:00
local player = game.players[event.player_index]
local surface = game.surfaces[active_surface_index]
Public.set_difficulty()
ICW_Func.is_minimap_valid(player, surface)
2021-05-23 17:03:52 +02:00
if player.online_time < 1 then
if not players[player.index] then
players[player.index] = {}
end
2020-12-04 02:09:52 +02:00
local message = ({'main.greeting', player.name})
Alert.alert_player(player, 15, message)
for item, amount in pairs(starting_items) do
player.insert({name = item, count = amount})
end
end
2022-04-05 19:28:08 +02:00
-- local top = player.gui.top
-- if top['mod_gui_top_frame'] then
-- top['mod_gui_top_frame'].destroy()
-- end
2020-12-04 02:09:52 +02:00
if player.surface.index ~= active_surface_index then
player.teleport(surface.find_non_colliding_position('character', game.forces.player.get_spawn_position(surface), 3, 0, 5), surface)
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
2021-07-07 12:22:22 +02:00
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
2020-12-04 02:09:52 +02:00
end
end
local locomotive = WPT.get('locomotive')
if not locomotive or not locomotive.valid then
return
end
if player.position.y > locomotive.position.y then
player.teleport(surface.find_non_colliding_position('character', game.forces.player.get_spawn_position(surface), 3, 0, 5), surface)
end
end
function Public.on_player_left_game()
Public.set_difficulty()
end
2020-12-05 19:09:09 +02:00
function Public.is_creativity_mode_on()
local creative_enabled = Misc.get('creative_enabled')
2020-12-05 19:09:09 +02:00
if creative_enabled then
WD.set('next_wave', 1000)
Collapse.start_now(true)
Public.set_difficulty()
end
end
2020-12-29 01:09:18 +02:00
function Public.disable_creative()
local creative_enabled = Misc.get('creative_enabled')
2020-12-29 01:09:18 +02:00
if creative_enabled then
Misc.set('creative_enabled', false)
2020-12-29 01:09:18 +02:00
end
end
2021-05-23 21:13:21 +02:00
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
2021-05-28 20:37:06 +02:00
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)])
2021-05-23 21:13:21 +02:00
end
end
2020-12-04 02:09:52 +02:00
function Public.on_player_changed_position(event)
local active_surface_index = WPT.get('active_surface_index')
if not active_surface_index then
return
end
local player = game.players[event.player_index]
local map_name = 'mtn_v3'
2020-12-04 02:09:52 +02:00
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 p = {x = player.position.x, y = player.position.y}
local config_tile = WPT.get('void_or_tile')
if config_tile == 'lab-dark-2' then
local get_tile = surface.get_tile(p)
2020-12-04 02:09:52 +02:00
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 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
local disable_recipes = function(force)
2020-12-04 02:09:52 +02:00
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
2021-05-16 13:48:50 +02:00
force.recipes['artillery-turret'].enabled = false
force.recipes['artillery-shell'].enabled = false
force.recipes['artillery-targeting-remote'].enabled = false
2020-12-04 02:09:52 +02:00
force.recipes['locomotive'].enabled = false
force.recipes['pistol'].enabled = false
2020-12-05 19:09:09 +02:00
force.recipes['spidertron-remote'].enabled = false
force.recipes['discharge-defense-equipment'].enabled = false
force.recipes['discharge-defense-remote'].enabled = false
2020-12-04 02:09:52 +02:00
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
2021-05-16 13:48:50 +02:00
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
2021-10-03 20:50:53 +02:00
force.technologies['fluid-wagon'].enabled = false
force.technologies['cliff-explosives'].enabled = false
2021-05-15 16:01:07 +02:00
disable_recipes(force)
2020-12-04 02:09:52 +02:00
end
local disable_tech = Public.disable_tech
function Public.on_research_finished(event)
disable_tech()
local research = event.research
2021-11-18 01:58:54 +02:00
local bonus_drill = game.forces.bonus_drill
local player = game.forces.player
2020-12-04 02:09:52 +02:00
2021-11-18 01:58:54 +02:00
research.force.character_inventory_slots_bonus = player.mining_drill_productivity_bonus * 50 -- +5 Slots /
bonus_drill.mining_drill_productivity_bonus = bonus_drill.mining_drill_productivity_bonus + 0.03
2021-11-18 01:58:54 +02:00
if bonus_drill.mining_drill_productivity_bonus >= 3 then
bonus_drill.mining_drill_productivity_bonus = 3
end
2020-12-04 02:09:52 +02:00
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)
2021-10-11 10:08:28 +02:00
end
2020-12-04 02:09:52 +02:00
local force_name = research.force.name
if not force_name then
return
end
local flamethrower_damage = WPT.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
2020-05-21 23:08:23 +02:00
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}
2020-05-23 21:18:18 +02:00
Public.artillery_shell_ammo = {name = 'artillery-shell', count = 15}
2020-05-21 23:08:23 +02:00
Public.laser_turrent_power_source = {buffer_size = 2400000, power_production = 40000}
2020-07-14 21:50:56 +02:00
function Public.reset_table()
this.power_sources = {index = 1}
this.refill_turrets = {index = 1}
this.magic_crafters = {index = 1}
this.magic_fluid_crafters = {index = 1}
end
2020-12-04 02:09:52 +02:00
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
2021-05-23 21:13:21 +02:00
local on_player_respawned = Public.on_player_respawned
2020-12-04 02:09:52 +02:00
Event.add(defines.events.on_player_joined_game, on_player_joined_game)
Event.add(defines.events.on_player_left_game, on_player_left_game)
Event.add(defines.events.on_research_finished, on_research_finished)
Event.add(defines.events.on_player_changed_position, on_player_changed_position)
2021-05-23 21:13:21 +02:00
Event.add(defines.events.on_player_respawned, on_player_respawned)
2020-06-25 17:59:16 +02:00
Event.on_nth_tick(10, tick)
2020-05-21 23:08:23 +02:00
return Public