1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-24 03:47:58 +02:00
2023-06-22 22:57:07 +02:00

1627 lines
49 KiB
Lua

require 'modules.rocks_broken_paint_tiles'
local Event = require 'utils.event'
local Public = require 'maps.mountain_fortress_v3.table'
local Server = require 'utils.server'
local RPG = require 'modules.rpg.main'
local Collapse = require 'modules.collapse'
local Alert = require 'utils.alert'
local Task = require 'utils.task'
local Score = require 'utils.gui.score'
local Token = require 'utils.token'
local Discord = require 'utils.discord'
local Core = require 'utils.core'
local Diff = require 'modules.difficulty_vote_by_amount'
local format_number = require 'util'.format_number
local RPG_Progression = require 'utils.datastore.rpg_data'
local WD = require 'modules.wave_defense.table'
local zone_settings = Public.zone_settings
local random = math.random
local floor = math.floor
local abs = math.abs
local round = math.round
-- Use these settings for live
local send_ping_to_channel = Discord.channel_names.mtn_channel
-- Use these settings for testing
-- bot-lounge
-- local send_ping_to_channel = Discord.channel_names.bot_quarters
-- dev
-- local send_ping_to_channel = Discord.channel_names.bot_quarters
-- local role_to_mention = Discord.role_mentions.test_role
local chests = {
'wooden-chest',
'iron-chest',
'steel-chest',
'crash-site-chest-1',
'crash-site-chest-2',
'crash-site-spaceship-wreck-big-1',
'crash-site-spaceship-wreck-big-2',
'crash-site-spaceship-wreck-medium-1',
'crash-site-spaceship-wreck-medium-2',
'crash-site-spaceship-wreck-medium-3'
}
local size_chests = #chests
local treasure_chest_messages = {
({'entity.treasure_1'}),
({'entity.treasure_2'}),
({'entity.treasure_3'})
}
local rare_treasure_chest_messages = {
({'entity.treasure_rare_1'}),
({'entity.treasure_rare_2'}),
({'entity.treasure_rare_3'})
}
local disabled_threats = {
['entity-ghost'] = true,
['raw-fish'] = true
}
local defeated_messages = {
({'entity.defeated_1'}),
({'entity.defeated_2'}),
({'entity.defeated_3'}),
({'entity.defeated_4'})
}
local protect_types = {
['cargo-wagon'] = true,
['artillery-wagon'] = true,
['fluid-wagon'] = true,
['locomotive'] = true,
['reactor'] = true,
['spider-vehicle'] = true,
['car'] = true
}
local reset_game =
Token.register(
function(data)
local this = data.this
if this.soft_reset then
Public.set_scores()
this.game_reset_tick = nil
Public.reset_map()
return
end
if this.restart then
Public.set_scores()
local message = ({'entity.reset_game'})
Server.to_discord_bold(message, true)
Server.start_scenario('Mountain_Fortress_v3')
return
end
if this.shutdown then
Public.set_scores()
local message = ({'entity.shutdown_game'})
Server.to_discord_bold(message, true)
Server.stop_scenario()
return
end
end
)
local function get_random_weighted(weighted_table, item_index, weight_index)
local total_weight = 0
item_index = item_index or 1
weight_index = weight_index or 2
for _, w in pairs(weighted_table) do
total_weight = total_weight + w[weight_index]
end
local index = random() * total_weight
local weight_sum = 0
for _, w in pairs(weighted_table) do
weight_sum = weight_sum + w[weight_index]
if weight_sum >= index then
return w[item_index]
end
end
end
local function on_entity_removed(data)
local entity = data.entity
local upgrades = Public.get('upgrades')
local built = {
['land-mine'] = upgrades.landmine.built,
['flamethrower-turret'] = upgrades.flame_turret.built
}
local validator = {
['land-mine'] = 'landmine',
['flamethrower-turret'] = 'flame_turret'
}
local name = validator[entity.name]
if built[entity.name] and entity.force.index == 1 then
upgrades[name].built = upgrades[name].built - 1
if upgrades[name].built <= 0 then
upgrades[name].built = 0
end
end
end
local function check_health()
local locomotive_health = Public.get('locomotive_health')
local locomotive_max_health = Public.get('locomotive_max_health')
local carriages = Public.get('carriages')
if locomotive_health <= 0 then
Public.set('locomotive_health', 0)
end
local m = locomotive_health / locomotive_max_health
if carriages then
for i = 1, #carriages do
local entity = carriages[i]
if not (entity and entity.valid) then
return
end
local cargo_health = 600
if entity.type == 'locomotive' then
entity.health = 1000 * m
else
entity.health = cargo_health * m
end
end
end
end
local function check_health_final_damage(final_damage_amount)
local carriages = Public.get('carriages')
if carriages then
for i = 1, #carriages do
local entity = carriages[i]
if not (entity and entity.valid) then
return
end
entity.health = entity.health + final_damage_amount
end
end
end
local function set_train_final_health(final_damage_amount, repair)
if final_damage_amount == 0 then
return
end
local locomotive = Public.get('locomotive')
if not (locomotive and locomotive.valid) then
return
end
local locomotive_health = Public.get('locomotive_health')
local locomotive_max_health = Public.get('locomotive_max_health')
if not repair then
local poison_deployed = Public.get('poison_deployed')
local robotics_deployed = Public.get('robotics_deployed')
local lower_high = locomotive_max_health * 0.4
local higher_high = locomotive_max_health * 0.5
local lower_low = locomotive_max_health * 0.2
local higher_low = locomotive_max_health * 0.3
if locomotive_health >= lower_high and locomotive_health <= higher_high then
if not poison_deployed then
local carriages = Public.get('carriages')
if WD.get('wave_number') <= 800 then
if carriages then
for i = 1, #carriages do
local entity = carriages[i]
Public.enable_poison_defense(entity.position)
end
end
end
local p = {
position = locomotive.position
}
local msg = ({'entity.train_taking_damage'})
Alert.alert_all_players_location(p, msg)
Public.set().poison_deployed = true
end
elseif locomotive_health >= lower_low and locomotive_health <= higher_low then
if not robotics_deployed then
local carriages = Public.get('carriages')
if carriages then
for i = 1, #carriages do
local entity = carriages[i]
Public.enable_robotic_defense(entity.position)
end
end
local p = {
position = locomotive.position
}
local msg = ({'entity.train_taking_damage'})
Alert.alert_all_players_location(p, msg)
Public.set().robotics_deployed = true
end
elseif locomotive_health >= locomotive_max_health then
Public.set().poison_deployed = false
end
end
if locomotive_health <= 0 or locomotive.health <= 5 then
locomotive.destructible = false
locomotive.health = 1
Public.set('game_lost', true)
Public.loco_died()
end
if locomotive_health <= 0 then
check_health_final_damage(final_damage_amount)
return
end
Public.set('locomotive_health', floor(locomotive_health - final_damage_amount))
if locomotive_health > locomotive_max_health then
Public.set('locomotive_health', locomotive_max_health)
end
locomotive_health = Public.get('locomotive_health')
check_health()
local health_text = Public.get('health_text')
rendering.set_text(health_text, 'HP: ' .. round(locomotive_health) .. ' / ' .. round(locomotive_max_health))
end
local function is_protected(e)
local map_name = 'mtn_v3'
if string.sub(e.surface.name, 0, #map_name) ~= map_name then
return true
end
if protect_types[e.type] then
return true
end
return false
end
local function protect_entities(data)
local cause = data.cause
local entity = data.entity
local force = data.force
local dmg = data.final_damage_amount
if not dmg then
return
end
local check_heavy_damage = Public.get('check_heavy_damage')
if check_heavy_damage then
if (entity.type == 'simple-entity' or entity.type == 'simple-entity-with-owner') and dmg >= 500 then
entity.health = entity.health + dmg
end
end
if entity.force.index ~= 1 then
return
end
local carriages_numbers = Public.get('carriages_numbers')
if is_protected(entity) then
if (cause and cause.valid) then
if Public.valid_enemy_forces[cause.force.name] then
if carriages_numbers and carriages_numbers[entity.unit_number] then
set_train_final_health(dmg, false)
return
else
entity.health = entity.health - dmg
return
end
end
elseif not (cause and cause.valid) then
if force and Public.valid_enemy_forces[force.name] then
if carriages_numbers and carriages_numbers[entity.unit_number] then
set_train_final_health(dmg, false)
return
else
entity.health = entity.health - dmg
return
end
end
end
entity.health = entity.health + dmg
end
end
local function hidden_treasure(player, entity)
local rpg = RPG.get_value_from_player(player.index)
if not rpg then
return
end
local magic = rpg.magicka
if magic >= 50 then
local msg = rare_treasure_chest_messages[random(1, #rare_treasure_chest_messages)]
Alert.alert_player(player, 5, msg)
Public.add_loot_rare(entity.surface, entity.position, 'wooden-chest', magic)
return
end
local msg = treasure_chest_messages[random(1, #treasure_chest_messages)]
Alert.alert_player(player, 5, msg, nil, nil, 0.3)
Public.add_loot(entity.surface, entity.position, chests[random(1, size_chests)])
end
local function biters_chew_rocks_faster(data)
local cause = data.cause
local entity = data.entity
local final_damage_amount = data.final_damage_amount
if entity.force.index ~= 3 then
return
end --Neutral Force
if not cause then
return
end
if not cause.valid then
return
end
if cause.force.index ~= 2 then
return
end --Enemy Force
entity.health = entity.health - final_damage_amount * 7
end
local projectiles = {'grenade', 'explosive-rocket', 'grenade', 'explosive-rocket', 'explosive-cannon-projectile'}
local function angry_tree(entity, cause, player)
if entity.type ~= 'tree' then
return
end
if abs(entity.position.y) < zone_settings.zone_depth then
return
end
if random(1, 6) == 1 then
Public.buried_biter(entity.surface, entity.position)
end
if random(1, 8) == 1 then
Public.buried_worm(entity.surface, entity.position)
end
if random(1, 32) ~= 1 then
return
end
local position = false
if cause then
if cause.valid then
position = cause.position
end
end
if not position then
position = {entity.position.x + (-20 + random(0, 40)), entity.position.y + (-20 + random(0, 40))}
end
if player then
local forest_zone = RPG.get_value_from_player(player.index, 'forest_zone')
if forest_zone and random(1, 32) == 1 then
local cbl = Public.refill_turret_callback
local data = {callback_data = Public.piercing_rounds_magazine_ammo}
local e =
entity.surface.create_entity(
{
name = 'gun-turret',
position = entity.position,
force = 'enemy'
}
)
if e.can_insert(Public.piercing_rounds_magazine_ammo) then
e.insert(Public.piercing_rounds_magazine_ammo)
end
local callback = Token.get(cbl)
callback(e, data)
return
end
end
entity.surface.create_entity(
{
name = projectiles[random(1, 5)],
position = entity.position,
force = 'neutral',
source = entity.position,
target = position,
max_range = 16,
speed = 0.01
}
)
end
local function give_coin(player)
local coin_amount = Public.get('coin_amount')
local coin_override = Public.get('coin_override')
local forest_zone = RPG.get_value_from_player(player.index, 'forest_zone')
if forest_zone then
if random(1, 3) ~= 1 then
return
end
end
if coin_amount >= 1 then
if coin_override then
player.insert({name = 'coin', count = coin_override})
else
player.insert({name = 'coin', count = random(1, coin_amount)})
end
end
end
local immunity_spawner =
Token.register(
function(data)
local entity = data.entity
if not entity or not entity.valid then
return
end
entity.destructible = true
end
)
local mining_events = {
{
function()
end,
300000,
'Nothing'
},
{
function()
end,
16384,
'Nothing'
},
{
function()
end,
4096,
'Nothing'
},
{
function(entity)
if Public.is_around_train(entity) then
entity.destroy()
return
end
Public.buried_biter(entity.surface, entity.position)
entity.destroy()
end,
4096,
'Angry Biter #2'
},
{
function(entity)
if Public.is_around_train(entity) then
entity.destroy()
return
end
Public.buried_biter(entity.surface, entity.position)
entity.destroy()
end,
512,
'Angry Biter #2'
},
{
function(entity)
if Public.is_around_train(entity) then
entity.destroy()
return
end
Public.buried_worm(entity.surface, entity.position)
entity.destroy()
end,
2048,
'Angry Worm'
},
{
function(entity)
if Public.is_around_train(entity) then
entity.destroy()
return
end
Public.tick_tack_trap(entity.surface, entity.position)
entity.destroy()
end,
2048,
'Dangerous Trap'
},
{
function(entity, index)
if Public.is_around_train(entity) then
entity.destroy()
return
end
local player = game.get_player(index)
if entity.type == 'tree' then
angry_tree(entity, player.character, player)
entity.destroy()
end
end,
1024,
'Angry Tree'
},
{
function(entity, index)
local player = game.get_player(index)
hidden_treasure(player, entity)
end,
1024,
'Treasure_Tier_1'
},
{
function(entity, index)
local player = game.get_player(index)
hidden_treasure(player, entity)
end,
512,
'Treasure_Tier_2'
},
{
function(entity, index)
local player = game.get_player(index)
hidden_treasure(player, entity)
end,
256,
'Treasure_Tier_3'
},
{
function(entity, index)
local player = game.get_player(index)
hidden_treasure(player, entity)
end,
128,
'Treasure_Tier_4'
},
{
function(entity, index)
local player = game.get_player(index)
hidden_treasure(player, entity)
end,
64,
'Treasure_Tier_5'
},
{
function(entity, index)
local player = game.get_player(index)
hidden_treasure(player, entity)
end,
32,
'Treasure_Tier_6'
},
{
function(entity, index)
local player = game.get_player(index)
hidden_treasure(player, entity)
end,
16,
'Treasure_Tier_7'
},
{
function(entity, index)
if Public.is_around_train(entity) then
entity.destroy()
return
end
local ent_to_create = {'biter-spawner', 'spitter-spawner'}
local position = entity.position
local surface = entity.surface
local e = surface.create_entity({name = ent_to_create[random(1, #ent_to_create)], position = position, force = 'enemy'})
e.destructible = false
Task.set_timeout_in_ticks(300, immunity_spawner, {entity = e})
Public.unstuck_player(index)
end,
512,
'Nest'
},
{
function(entity, index)
if Public.is_around_train(entity) then
entity.destroy()
return
end
local ent_to_create = {'biter-spawner', 'spitter-spawner'}
local position = entity.position
local surface = entity.surface
local e = surface.create_entity({name = ent_to_create[random(1, #ent_to_create)], position = position, force = 'enemy'})
e.destructible = false
Task.set_timeout_in_ticks(300, immunity_spawner, {entity = e})
Public.unstuck_player(index)
end,
512,
'Nest'
},
{
function(entity)
local position = entity.position
local surface = entity.surface
surface.create_entity({name = 'compilatron', position = position, force = 'player'})
end,
64,
'Friendly Compilatron'
},
{
function(entity)
if Public.is_around_train(entity) then
entity.destroy()
return
end
local position = entity.position
local surface = entity.surface
surface.create_entity({name = 'compilatron', position = position, force = 'enemy'})
end,
128,
'Enemy Compilatron'
},
{
function(entity)
local chest = 'crash-site-chest-' .. random(1, 2)
local container = entity.surface.create_entity({name = chest, position = entity.position, force = 'neutral'})
if container and container.health then
container.insert({name = 'vehicle-machine-gun', count = 1})
container.health = random(1, container.health)
end
end,
64,
'VSMG'
},
{
function(entity, index)
local position = entity.position
local surface = entity.surface
surface.create_entity({name = 'car', position = position, force = 'player'})
Public.unstuck_player(index)
local player = game.players[index]
local msg = ({'entity.found_car', player.name})
Alert.alert_player(player, 15, msg)
end,
32,
'Car'
}
}
local function on_player_mined_entity(event)
local entity = event.entity
local player = game.get_player(event.player_index)
if not player.valid then
return
end
if not entity.valid then
return
end
local rpg_char = RPG.get_value_from_player(player.index)
local map_name = 'mtn_v3'
if string.sub(entity.surface.name, 0, #map_name) ~= map_name then
return
end
local d = {
entity = entity
}
on_entity_removed(d)
if disabled_threats[entity.name] then
return
end
local mined_scrap = Public.get('mined_scrap')
if entity.type == 'simple-entity' or entity.type == 'simple-entity-with-owner' or entity.type == 'tree' then
Public.set().mined_scrap = mined_scrap + 1
Public.on_player_mined_entity(event)
if entity.type == 'tree' then
if random(1, 3) == 1 then
give_coin(player)
end
elseif entity.type == 'simple-entity-with-owner' then
if random(1, 6) == 1 then
give_coin(player)
end
else
give_coin(player)
end
if rpg_char.stone_path then
entity.surface.set_tiles({{name = 'stone-path', position = entity.position}}, true)
end
local func = get_random_weighted(mining_events)
func(entity, player.index)
end
end
local function on_robot_mined_entity(event)
local entity = event.entity
if not entity.valid then
return
end
local map_name = 'mtn_v3'
if string.sub(entity.surface.name, 0, #map_name) ~= map_name then
return
end
local d = {
entity = entity
}
on_entity_removed(d)
end
local function get_damage(data)
local entity = data.entity
local original_damage_amount = data.original_damage_amount
local damage = original_damage_amount + original_damage_amount * random(1, 100)
if entity.prototype.resistances then
if entity.prototype.resistances.physical then
damage = damage - entity.prototype.resistances.physical.decrease
damage = damage - damage * entity.prototype.resistances.physical.percent
end
end
damage = round(damage, 3)
if damage < 1 then
damage = 1
end
return damage
end
local function kaboom(entity, target, damage)
local base_vector = {target.position.x - entity.position.x, target.position.y - entity.position.y}
local vector = {base_vector[1], base_vector[2]}
vector[1] = vector[1] * 512
vector[2] = vector[2] * 256
local msg = {'TASTY', 'MUNCH', 'SNACK_TIME', 'OVER 9000!'}
entity.surface.create_entity(
{
name = 'flying-text',
position = {entity.position.x + base_vector[1] * 0.5, entity.position.y + base_vector[2] * 0.5},
text = msg[random(1, #msg)],
color = {255, 0, 0}
}
)
if abs(vector[1]) > abs(vector[2]) then
local d = abs(vector[1])
if abs(vector[1]) > 0 then
vector[1] = vector[1] / d
end
if abs(vector[2]) > 0 then
vector[2] = vector[2] / d
end
else
local d = abs(vector[2])
if abs(vector[2]) > 0 then
vector[2] = vector[2] / d
end
if abs(vector[1]) > 0 and d > 0 then
vector[1] = vector[1] / d
end
end
vector[1] = vector[1] * 1.6
vector[2] = vector[2] * 1.6
local a = 0.30
for i = 1, 8, 1 do
for x = i * -1 * a, i * a, 1 do
for y = i * -1 * a, i * a, 1 do
local p = {entity.position.x + x + vector[1] * i, entity.position.y + y + vector[2] * i}
entity.surface.create_trivial_smoke({name = 'fire-smoke', position = p})
for _, e in pairs(entity.surface.find_entities({{p[1] - a, p[2] - a}, {p[1] + a, p[2] + a}})) do
if e.valid then
if e.health then
if e.destructible and e.minable then
if e.force.index ~= entity.force.index then
e.health = e.health - damage * 0.05
if e.health <= 0 then
e.die(e.force.name, entity)
end
end
end
end
end
end
end
end
end
end
local function boss_puncher(data)
local cause = data.cause
local entity = data.entity
if not cause then
return
end
if not cause.valid then
return
end
if cause.force.index ~= 2 then
return
end
if entity.force.index ~= 1 then
return
end
if not entity then
return
end
if not entity.valid then
return
end
if random(1, 10) == 1 then
kaboom(cause, entity, get_damage(data))
end
end
local function on_entity_damaged(event)
local entity = event.entity
if not (entity and entity.valid) then
return
end
local cause = event.cause
local force = event.force
local final_damage_amount = event.final_damage_amount
local original_damage_amount = event.original_damage_amount
local wave_number = WD.get_wave()
local boss_wave_warning = WD.get_alert_boss_wave()
local munch_time = Public.get('munch_time')
local data = {
cause = cause,
entity = entity,
final_damage_amount = final_damage_amount,
original_damage_amount = original_damage_amount,
force = force
}
protect_entities(data)
biters_chew_rocks_faster(data)
if munch_time then
if boss_wave_warning or wave_number >= 1000 then
if random(0, 512) == 1 then
boss_puncher(data)
end
end
end
end
local function on_player_repaired_entity(event)
if not event.entity then
return
end
if not event.entity.valid then
return
end
if not event.entity.health then
return
end
local entity = event.entity
local carriages_numbers = Public.get('carriages_numbers')
if carriages_numbers[entity.unit_number] then
local player = game.get_player(event.player_index)
local repair_speed = RPG.get_magicka(player)
if repair_speed <= 1 then
set_train_final_health(-1, true)
return
else
set_train_final_health(-repair_speed, true)
return
end
end
end
local function on_entity_died(event)
local entity = event.entity
if not entity.valid then
return
end
local cause = event.cause
local map_name = 'mtn_v3'
if string.sub(entity.surface.name, 0, #map_name) ~= map_name then
return
end
local d = {
entity = entity
}
on_entity_removed(d)
local player
local valid_enemy_forces = Public.valid_enemy_forces
if cause then
if cause.valid then
if (cause and cause.name == 'character' and cause.player) then
player = cause.player
end
if valid_enemy_forces[cause.force.name] or cause.force.index == 3 then
entity.destroy()
return
end
end
end
if disabled_threats[entity.name] then
return
end
local biters_killed = Public.get('biters_killed')
local biters = Public.get('biters')
if entity.type == 'unit' or entity.type == 'unit-spawner' then
Public.set().biters_killed = biters_killed + 1
biters.amount = biters.amount - 1
if biters.amount <= 0 then
biters.amount = 0
end
if Public.is_around_train(entity) then
return
end
if random(1, 512) == 1 then
Public.tick_tack_trap(entity.surface, entity.position)
return
end
end
if entity.type == 'tree' then
for _, e in pairs(
entity.surface.find_entities_filtered(
{
area = {
{entity.position.x - 4, entity.position.y - 4},
{entity.position.x + 4, entity.position.y + 4}
},
name = 'fire-flame-on-tree'
}
)
) do
if e.valid then
e.destroy()
return
end
end
if Public.is_around_train(entity) then
return
end
angry_tree(entity, cause, player)
return
end
if entity.type == 'simple-entity' then
if Public.is_around_train(entity) then
entity.destroy()
return
end
if random(1, 32) == 1 then
Public.buried_biter(entity.surface, entity.position)
entity.destroy()
return
end
if random(1, 64) == 1 then
Public.buried_worm(entity.surface, entity.position)
entity.destroy()
return
end
if random(1, 512) == 1 then
Public.tick_tack_trap(entity.surface, entity.position)
return
end
entity.destroy()
return
end
end
local function get_sorted_list(column_name, score_list)
for _ = 1, #score_list, 1 do
for y = 1, #score_list, 1 do
if not score_list[y + 1] then
break
end
if score_list[y][column_name] < score_list[y + 1][column_name] then
local key = score_list[y]
score_list[y] = score_list[y + 1]
score_list[y + 1] = key
end
end
end
return score_list
end
local function get_mvps(force)
local get_score = Score.get_table().score_table
if not get_score[force] then
return false
end
local score = get_score[force]
local score_list = {}
for _, p in pairs(game.players) do
if score.players[p.name] then
local killscore = 0
if score.players[p.name].killscore then
killscore = score.players[p.name].killscore
end
local built_entities = 0
if score.players[p.name].built_entities then
built_entities = score.players[p.name].built_entities
end
local mined_entities = 0
if score.players[p.name].mined_entities then
mined_entities = score.players[p.name].mined_entities
end
table.insert(score_list, {name = p.name, killscore = killscore, built_entities = built_entities, mined_entities = mined_entities})
end
end
local mvp = {}
score_list = get_sorted_list('killscore', score_list)
mvp.killscore = {name = score_list[1].name, score = score_list[1].killscore}
score_list = get_sorted_list('mined_entities', score_list)
mvp.mined_entities = {name = score_list[1].name, score = score_list[1].mined_entities}
score_list = get_sorted_list('built_entities', score_list)
mvp.built_entities = {name = score_list[1].name, score = score_list[1].built_entities}
return mvp
end
local function show_mvps(player)
local get_score = Score.get_table().score_table
local wave_defense_table = WD.get_table()
if not get_score then
return
end
if player.gui.left['mvps'] then
return
end
local frame = player.gui.left.add({type = 'frame', name = 'mvps', direction = 'vertical'})
local l = frame.add({type = 'label', caption = 'MVPs:'})
l.style.font = 'default-listbox'
l.style.font_color = {r = 0.55, g = 0.55, b = 0.99}
local t = frame.add({type = 'table', column_count = 2})
local mvp = get_mvps('player')
if mvp then
local wave_defense = t.add({type = 'label', caption = 'Highest Wave >> '})
wave_defense.style.font = 'default-listbox'
wave_defense.style.font_color = {r = 0.22, g = 0.77, b = 0.44}
local wave_defense_text = t.add({type = 'label', caption = 'This rounds highest wave was: ' .. wave_defense_table.wave_number})
wave_defense_text.style.font = 'default-bold'
wave_defense_text.style.font_color = {r = 0.33, g = 0.66, b = 0.9}
local fighter_label = t.add({type = 'label', caption = 'Fighter >> '})
fighter_label.style.font = 'default-listbox'
fighter_label.style.font_color = {r = 0.22, g = 0.77, b = 0.44}
local fighter_label_text = t.add({type = 'label', caption = mvp.killscore.name .. ' with a killing score of ' .. mvp.killscore.score .. ' kills!'})
fighter_label_text.style.font = 'default-bold'
fighter_label_text.style.font_color = {r = 0.33, g = 0.66, b = 0.9}
local builder_label = t.add({type = 'label', caption = 'Builder >> '})
builder_label.style.font = 'default-listbox'
builder_label.style.font_color = {r = 0.22, g = 0.77, b = 0.44}
local builder_label_text = t.add({type = 'label', caption = mvp.built_entities.name .. ' built ' .. mvp.built_entities.score .. ' things!'})
builder_label_text.style.font = 'default-bold'
builder_label_text.style.font_color = {r = 0.33, g = 0.66, b = 0.9}
local miners_label = t.add({type = 'label', caption = 'Miners >> '})
miners_label.style.font = 'default-listbox'
miners_label.style.font_color = {r = 0.22, g = 0.77, b = 0.44}
local miners_label_text = t.add({type = 'label', caption = mvp.mined_entities.name .. ' mined a total of ' .. mvp.mined_entities.score .. ' entities!'})
miners_label_text.style.font = 'default-bold'
miners_label_text.style.font_color = {r = 0.33, g = 0.66, b = 0.9}
local sent_to_discord = Public.get('sent_to_discord')
local server_name_matches = Server.check_server_name('Mtn Fortress')
if not sent_to_discord and server_name_matches then
local message = {
title = 'Game over',
description = 'Player statistics is below',
color = 'failure',
field1 = {
text1 = 'Highest Wave:',
text2 = wave_defense_table.wave_number,
inline = 'false'
},
field2 = {
text1 = 'MVP Fighter:',
text2 = mvp.killscore.name .. ' with a killing score of ' .. mvp.killscore.score .. ' kills!',
inline = 'false'
},
field3 = {
text1 = 'MVP Builder:',
text2 = mvp.built_entities.name .. ' built ' .. mvp.built_entities.score .. ' things!',
inline = 'false'
},
field4 = {
text1 = 'MVP Miners:',
text2 = mvp.mined_entities.name .. ' mined a total of ' .. mvp.mined_entities.score .. ' entities!',
inline = 'false'
}
}
Server.to_discord_embed_parsed(message)
local wave = WD.get_wave()
local threat = WD.get('threat')
local collapse_speed = Collapse.get_speed()
local collapse_amount = Collapse.get_amount()
local diff = Diff.get()
if not diff then
return
end
local time_played = Core.format_time(game.ticks_played)
local total_players = #game.players
local pickaxe_upgrades = Public.pickaxe_upgrades
local upgrades = Public.get('upgrades')
local pick_tier = pickaxe_upgrades[upgrades.pickaxe_tier]
if Public.get('prestige_system_enabled') then
RPG_Progression.save_all_players()
end
local date = Server.get_start_time()
game.server_save('Final_Mtn_v3_' .. tostring(date) .. '_wave' .. tostring(wave))
local text = {
title = 'Game over!',
description = 'Game statistics from the game is below',
color = 'failure',
field1 = {
text1 = 'Time played:',
text2 = time_played,
inline = 'true'
},
field2 = {
text1 = 'Game Difficulty:',
text2 = diff.name,
inline = 'true',
emptyField = 'true',
emptyInline = 'true'
},
field3 = {
text1 = 'Highest wave:',
text2 = format_number(wave, true),
inline = 'true'
},
field4 = {
text1 = 'Total connected players:',
text2 = total_players,
inline = 'true',
emptyField = 'true',
emptyInline = 'true'
},
field5 = {
text1 = 'Threat:',
text2 = format_number(threat, true),
inline = 'true'
},
field6 = {
text1 = 'Pickaxe Upgrade:',
text2 = pick_tier .. ' (' .. upgrades.pickaxe_tier .. ')',
inline = 'true',
emptyField = 'true',
emptyInline = 'true'
},
field7 = {
text1 = 'Collapse Speed:',
text2 = collapse_speed,
inline = 'true'
},
field8 = {
text1 = 'Collapse Amount:',
text2 = collapse_amount,
inline = 'true',
emptyField = 'true',
emptyInline = 'true'
}
}
if server_name_matches then
if wave >= 1000 then
Server.to_discord_named_parsed_embed(send_ping_to_channel, text)
end
else
Server.to_discord_embed_parsed(text)
end
Public.set('sent_to_discord', true)
end
end
end
function Public.unstuck_player(index)
local player = game.get_player(index)
local surface = player.surface
local position = surface.find_non_colliding_position('character', player.position, 32, 0.5)
if not position then
return
end
player.teleport(position, surface)
end
function Public.loco_died(invalid_locomotive)
local game_lost = Public.get('game_lost')
if not game_lost then
return
end
local announced_message = Public.get('announced_message')
if announced_message then
return
end
local active_surface_index = Public.get('active_surface_index')
local locomotive = Public.get('locomotive')
local surface = game.surfaces[active_surface_index]
local wave_defense_table = WD.get_table()
if wave_defense_table.game_lost and not invalid_locomotive then
return
end
Collapse.start_now(false)
for _, player in pairs(game.connected_players) do
player.play_sound {path = 'utility/game_lost', volume_modifier = 0.75}
show_mvps(player)
end
if not locomotive.valid then
local this = Public.get()
local data = {}
if this.locomotive and this.locomotive.valid then
data.position = this.locomotive.position
else
data.position = {x = 0, y = 0}
end
local msg = defeated_messages[random(1, #defeated_messages)]
Alert.alert_all_players_location(data, msg, nil, 15)
wave_defense_table.game_lost = true
wave_defense_table.target = nil
local params = {
this = this
}
if this.soft_reset then
this.game_reset_tick = nil
Task.set_timeout_in_ticks(600, reset_game, params)
this.announced_message = true
return
end
if this.restart then
game.print(({'entity.notify_restart'}), {r = 0.22, g = 0.88, b = 0.22})
Task.set_timeout_in_ticks(600, reset_game, params)
this.announced_message = true
return
end
if this.shutdown then
game.print(({'entity.notify_shutdown'}), {r = 0.22, g = 0.88, b = 0.22})
Task.set_timeout_in_ticks(600, reset_game, params)
this.announced_message = true
return
end
return
end
local this = Public.get()
this.locomotive_health = 0
this.locomotive.color = {0.49, 0, 255, 1}
rendering.set_text(this.health_text, 'HP: ' .. round(this.locomotive_health) .. ' / ' .. round(this.locomotive_max_health))
wave_defense_table.game_lost = true
wave_defense_table.target = nil
local msg = defeated_messages[random(1, #defeated_messages)]
local pos = {
position = this.locomotive.position
}
Alert.alert_all_players_location(pos, msg)
game.forces.enemy.set_friend('player', true)
game.forces.player.set_friend('enemy', true)
local fake_shooter = surface.create_entity({name = 'character', position = this.locomotive.position, force = 'enemy'})
surface.create_entity(
{
name = 'atomic-rocket',
position = this.locomotive.position,
force = 'enemy',
speed = 1,
max_range = 1200,
target = this.locomotive,
source = fake_shooter
}
)
surface.spill_item_stack(this.locomotive.position, {name = 'coin', count = 512}, false)
this.game_reset_tick = 5400
for _, player in pairs(game.connected_players) do
player.play_sound {path = 'utility/game_lost', volume_modifier = 0.75}
show_mvps(player)
end
end
local function on_built_entity(event)
local entity = event.created_entity
if not entity.valid then
return
end
local map_name = 'mtn_v3'
if string.sub(entity.surface.name, 0, #map_name) ~= map_name then
return
end
local position = entity.position
local player = game.get_player(event.player_index)
if entity.name == 'radar' then
if entity.surface.count_entities_filtered({type = 'radar', position = position, radius = 64}) > 1 then
player.surface.create_entity(
{
name = 'flying-text',
position = entity.position,
text = ({'entity.radar_limit'}),
color = {255, 0, 0}
}
)
player.surface.spill_item_stack(position, {name = entity.name, count = 1, true})
entity.destroy()
return
end
end
local valid_drills = {
['burner-mining-drill'] = true,
['electric-mining-drill'] = true
}
if valid_drills[entity.name] then
entity.force = 'bonus_drill'
return
end
local upgrades = Public.get('upgrades')
local upg = upgrades
local surface = entity.surface
local built = {
['land-mine'] = upg.landmine.built,
['flamethrower-turret'] = upg.flame_turret.built
}
local limit = {
['land-mine'] = upg.landmine.limit,
['flamethrower-turret'] = upg.flame_turret.limit
}
local validator = {
['land-mine'] = 'landmine',
['flamethrower-turret'] = 'flame_turret'
}
local name = validator[entity.name]
if built[entity.name] and entity.force.index == 1 then
if built[entity.name] < limit[entity.name] then
upgrades[name].built = built[entity.name] + 1
upgrades.unit_number[name][entity] = entity
upgrades.showed_text = false
surface.create_entity(
{
name = 'flying-text',
position = entity.position,
text = upgrades[name].built .. ' / ' .. limit[entity.name] .. ' ' .. entity.name,
color = {r = 0.82, g = 0.11, b = 0.11}
}
)
else
if not upgrades.showed_text then
surface.create_entity(
{
name = 'flying-text',
position = entity.position,
text = ({'entity.entity_limit_reached', entity.name}),
color = {r = 0.82, g = 0.11, b = 0.11}
}
)
upgrades.showed_text = true
end
player.insert({name = entity.name, count = 1})
entity.destroy()
end
end
end
local function on_robot_built_entity(event)
local entity = event.created_entity
if not entity.valid then
return
end
local map_name = 'mtn_v3'
if string.sub(entity.surface.name, 0, #map_name) ~= map_name then
return
end
local position = entity.position
if entity.name == 'radar' then
if entity.surface.count_entities_filtered({type = 'radar', position = position, radius = 64}) > 1 then
entity.surface.create_entity(
{
name = 'flying-text',
position = entity.position,
text = ({'entity.radar_limit'}),
color = {255, 0, 0}
}
)
entity.surface.spill_item_stack(position, {name = entity.name, count = 1, true})
entity.destroy()
return
end
end
local valid_drills = {
['burner-mining-drill'] = true,
['electric-mining-drill'] = true
}
if valid_drills[entity.name] then
entity.force = 'bonus_drill'
return
end
local upgrades = Public.get('upgrades')
local upg = upgrades
local surface = entity.surface
local built = {
['land-mine'] = upg.landmine.built,
['flamethrower-turret'] = upg.flame_turret.built
}
local limit = {
['land-mine'] = upg.landmine.limit,
['flamethrower-turret'] = upg.flame_turret.limit
}
local validator = {
['land-mine'] = 'landmine',
['flamethrower-turret'] = 'flame_turret'
}
local name = validator[entity.name]
if built[entity.name] and entity.force.index == 1 then
if built[entity.name] < limit[entity.name] then
upgrades[name].built = built[entity.name] + 1
upgrades.unit_number[name][entity] = entity
upgrades.showed_text = false
surface.create_entity(
{
name = 'flying-text',
position = entity.position,
text = upgrades[name].built .. ' / ' .. limit[entity.name] .. ' ' .. entity.name,
color = {r = 0.82, g = 0.11, b = 0.11}
}
)
else
if not upgrades.showed_text then
surface.create_entity(
{
name = 'flying-text',
position = entity.position,
text = ({'entity.entity_limit_reached', entity.name}),
color = {r = 0.82, g = 0.11, b = 0.11}
}
)
upgrades.showed_text = true
end
local inventory = event.robot.get_inventory(defines.inventory.robot_cargo)
inventory.insert({name = entity.name, count = 1})
entity.destroy()
end
end
end
local on_player_or_robot_built_tile = function(event)
local surface = game.surfaces[event.surface_index]
local map_name = 'mtn_v3'
if string.sub(surface.name, 0, #map_name) ~= map_name then
return
end
local tiles = event.tiles
if not tiles then
return
end
for k, v in pairs(tiles) do
local old_tile = v.old_tile
if old_tile.name == 'black-refined-concrete' then
surface.set_tiles({{name = 'black-refined-concrete', position = v.position}}, true)
end
if old_tile.name == 'blue-refined-concrete' then
surface.set_tiles({{name = 'blue-refined-concrete', position = v.position}}, true)
end
if old_tile.name == 'cyan-refined-concrete' then
surface.set_tiles({{name = 'cyan-refined-concrete', position = v.position}}, true)
end
if old_tile.name == 'hazard-concrete-right' then
surface.set_tiles({{name = 'hazard-concrete-right', position = v.position}}, true)
end
if old_tile.name == 'lab-dark-2' then
surface.set_tiles({{name = 'lab-dark-2', position = v.position}}, true)
end
end
end
Public.get_random_weighted = get_random_weighted
Event.add_event_filter(defines.events.on_entity_damaged, {filter = 'final-damage-amount', comparison = '>', value = 0})
Event.add(defines.events.on_entity_damaged, on_entity_damaged)
Event.add(defines.events.on_player_repaired_entity, on_player_repaired_entity)
Event.add(defines.events.on_player_mined_entity, on_player_mined_entity)
Event.add(defines.events.on_robot_mined_entity, on_robot_mined_entity)
Event.add(defines.events.on_entity_died, on_entity_died)
Event.add(defines.events.on_built_entity, on_built_entity)
Event.add(defines.events.on_robot_built_entity, on_robot_built_entity)
Event.add(defines.events.on_player_built_tile, on_player_or_robot_built_tile)
Event.add(defines.events.on_robot_built_tile, on_player_or_robot_built_tile)
return Public