1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-16 02:47:48 +02:00
ComfyFactorio/maps/tank_battles_old.lua
Gerkiz 690c1b6e48 2.0 changes
Change global -> storage
Rework how rendering works
Game prototypes are now stored inside 'prototypes.#'
Renamed entity names
2024-09-24 19:37:11 +02:00

610 lines
24 KiB
Lua

--luacheck: ignore
--tank battles (royale)-- mewmew made this --
local event = require 'utils.event'
local table_insert = table.insert
local math_random = math.random
local map_functions = require 'utils.tools.map_functions'
local arena_size = 160
local function shuffle(tbl)
local size = #tbl
for i = size, 1, -1 do
local rand = math.random(size)
tbl[i], tbl[rand] = tbl[rand], tbl[i]
end
return tbl
end
local function create_tank_battle_score_gui()
for _, player in pairs(game.connected_players) do
if player.gui.left['tank_battle_score'] then
player.gui.left['tank_battle_score'].destroy()
end
local frame = player.gui.left.add({ type = 'frame', name = 'tank_battle_score', direction = 'vertical' })
local l = frame.add({ type = 'label', caption = 'Won rounds' })
l.style.font_color = { r = 0.98, g = 0.66, b = 0.22 }
l.style.font = 'default-listbox'
local t = frame.add({ type = 'table', column_count = 2 })
local scores = {}
for _, player in pairs(game.connected_players) do
if storage.tank_battles_score[player.index] then
table_insert(scores, { name = player.name, score = storage.tank_battles_score[player.index] })
end
end
for x = 1, #scores, 1 do
for y = 1, #scores, 1 do
if not scores[y + 1] then
break
end
if scores[y]['score'] < scores[y + 1]['score'] then
local key = scores[y]
scores[y] = scores[y + 1]
scores[y + 1] = key
end
end
end
for i = 1, 8, 1 do
if scores[i] then
local l = t.add({ type = 'label', caption = scores[i].name })
local color = game.players[scores[i].name].color
color = { r = color.r * 0.6 + 0.4, g = color.g * 0.6 + 0.4, b = color.b * 0.6 + 0.4, a = 1 }
l.style.font_color = color
l.style.font = 'default-bold'
t.add({ type = 'label', caption = scores[i].score })
end
end
end
end
local loot = {
--{{name = "submachine-gun", count = 1}, weight = 2},
--{{name = "combat-shotgun", count = 1}, weight = 2},
--{{name = "flamethrower", count = 1}, weight = 1},
--{{name = "rocket-launcher", count = 1}, weight = 2},
{ { name = 'flamethrower-ammo', count = 16 }, weight = 2 },
{ { name = 'piercing-shotgun-shell', count = 16 }, weight = 2 },
--{{name = "piercing-rounds-magazine", count = 16}, weight = 2},
--{{name = "uranium-rounds-magazine", count = 8}, weight = 1},
{ { name = 'explosive-rocket', count = 8 }, weight = 2 },
{ { name = 'rocket', count = 8 }, weight = 2 },
{ { name = 'grenade', count = 16 }, weight = 2 },
{ { name = 'cluster-grenade', count = 8 }, weight = 2 },
--{{name = "poison-capsule", count = 4}, weight = 1},
{ { name = 'defender-capsule', count = 6 }, weight = 1 },
{ { name = 'distractor-capsule', count = 3 }, weight = 1 },
--{{name = "destroyer-capsule", count = 1}, weight = 1},
{ { name = 'cannon-shell', count = 8 }, weight = 16 },
{ { name = 'explosive-cannon-shell', count = 8 }, weight = 16 },
{ { name = 'uranium-cannon-shell', count = 8 }, weight = 6 },
{ { name = 'explosive-uranium-cannon-shell', count = 8 }, weight = 6 },
--{{name = "light-armor", count = 1}, weight = 5},
--{{name = "heavy-armor", count = 1}, weight = 2},
{ { name = 'modular-armor', count = 1 }, weight = 3 },
{ { name = 'power-armor', count = 1 }, weight = 2 },
{ { name = 'power-armor-mk2', count = 1 }, weight = 1 },
--{{name = "battery-mk2-equipment", count = 1}, weight = 2},
{ { name = 'energy-shield-equipment', count = 1 }, weight = 2 },
{ { name = 'exoskeleton-equipment', count = 1 }, weight = 2 },
{ { name = 'fusion-reactor-equipment', count = 1 }, weight = 2 },
{ { name = 'repair-pack', count = 1 }, weight = 6 },
{ { name = 'coal', count = 16 }, weight = 3 },
--{{name = "solid-fuel", count = 8}, weight = 2},
{ { name = 'nuclear-fuel', count = 1 }, weight = 1 },
{ { name = 'gate', count = 16 }, weight = 2 },
{ { name = 'stone-wall', count = 16 }, weight = 2 }
}
local loot_raffle = {}
for _, item in pairs(loot) do
for x = 1, item.weight, 1 do
table.insert(loot_raffle, item[1])
end
end
local function get_valid_random_spawn_position(surface)
local chunks = {}
for chunk in surface.get_chunks() do
table_insert(chunks, { x = chunk.x, y = chunk.y })
end
chunks = shuffle(chunks)
for _, chunk in pairs(chunks) do
if chunk.x * 32 < arena_size and chunk.y * 32 < arena_size and chunk.x * 32 >= arena_size * -1 and chunk.y * 32 >= arena_size * -1 then
local area = { { chunk.x * 32 - 64, chunk.y * 32 - 64 }, { chunk.x * 32 + 64, chunk.y * 32 + 64 } }
if surface.count_entities_filtered({ name = 'tank', area = area }) == 0 then
local pos = surface.find_non_colliding_position('tank', { chunk.x * 32 + 16, chunk.y * 32 + 16 }, 16, 8)
return pos
end
end
end
local pos = surface.find_non_colliding_position('tank', { 0, 0 }, 32, 4)
if pos then
return pos
end
return { 0, 0 }
end
local function put_players_into_arena()
for _, player in pairs(game.connected_players) do
local permissions_group = game.permissions.get_group('Default')
permissions_group.add_player(player.name)
player.character.destroy()
player.character = nil
player.create_character()
player.insert({ name = 'combat-shotgun', count = 1 })
player.insert({ name = 'rocket-launcher', count = 1 })
player.insert({ name = 'flamethrower', count = 1 })
--player.insert({name = "firearm-magazine", count = 512})
local surface = game.surfaces['tank_battles']
local pos = get_valid_random_spawn_position(surface)
player.teleport(pos, surface)
local tank = surface.create_entity({ name = 'tank', force = game.forces[player.index], position = pos })
tank.insert({ name = 'coal', count = 24 })
tank.insert({ name = 'cannon-shell', count = 16 })
tank.set_driver(player)
end
end
function create_new_arena()
local map_gen_settings = {}
map_gen_settings.seed = math_random(1, 2097152)
map_gen_settings.water = 'none'
map_gen_settings.cliff_settings = { cliff_elevation_interval = 4, cliff_elevation_0 = 0.1 }
map_gen_settings.autoplace_controls = {
['coal'] = { frequency = 'none', size = 'none', richness = 'none' },
['stone'] = { frequency = 'none', size = 'none', richness = 'none' },
['copper-ore'] = { frequency = 'none', size = 'none', richness = 'none' },
['iron-ore'] = { frequency = 'none', size = 'none', richness = 'none' },
['crude-oil'] = { frequency = 'none', size = 'none', richness = 'none' },
['trees'] = { frequency = 'normal', size = 'normal', richness = 'normal' },
['enemy-base'] = { frequency = 'none', size = 'none', richness = 'none' }
}
game.create_surface('tank_battles', map_gen_settings)
local surface = game.surfaces['tank_battles']
surface.request_to_generate_chunks({ 0, 0 }, math.ceil(arena_size / 32) + 2)
surface.force_generate_chunk_requests()
surface.daytime = 1
surface.freeze_daytime = 1
storage.current_arena_size = arena_size
put_players_into_arena()
storage.game_stage = 'ongoing_game'
end
local function set_unique_player_force(player)
if not game.forces[player.index] then
game.create_force(player.index)
game.forces[player.index].technologies['follower-robot-count-1'].researched = true
game.forces[player.index].technologies['follower-robot-count-2'].researched = true
game.forces[player.index].technologies['follower-robot-count-3'].researched = true
game.forces[player.index].technologies['follower-robot-count-4'].researched = true
game.forces[player.index].technologies['follower-robot-count-5'].researched = true
end
player.force = game.forces[player.index]
end
local function on_player_joined_game(event)
local player = game.players[event.player_index]
set_unique_player_force(player)
if not storage.map_init_done then
local spectator_permission_group = game.permissions.create_group('Spectator')
for action_name, _ in pairs(defines.input_action) do
spectator_permission_group.set_allows_action(defines.input_action[action_name], false)
end
spectator_permission_group.set_allows_action(defines.input_action.write_to_console, true)
spectator_permission_group.set_allows_action(defines.input_action.gui_click, true)
spectator_permission_group.set_allows_action(defines.input_action.gui_selection_state_changed, true)
spectator_permission_group.set_allows_action(defines.input_action.start_walking, true)
spectator_permission_group.set_allows_action(defines.input_action.open_kills_gui, true)
spectator_permission_group.set_allows_action(defines.input_action.open_character_gui, true)
--spectator_permission_group.set_allows_action(defines.input_action.open_equipment_gui, true)
spectator_permission_group.set_allows_action(defines.input_action.edit_permission_group, true)
spectator_permission_group.set_allows_action(defines.input_action.toggle_show_entity_info, true)
storage.tank_battles_score = {}
storage.game_stage = 'lobby'
storage.map_init_done = true
end
if #storage.tank_battles_score > 0 then
create_tank_battle_score_gui()
end
if game.surfaces['tank_battles'] then
player.character.destroy()
player.character = nil
local permissions_group = game.permissions.get_group('Spectator')
permissions_group.add_player(player.name)
player.teleport({ 0, 0 }, game.surfaces['tank_battles'])
end
if not game.surfaces['tank_battles'] then
player.character.destroy()
player.character = nil
if storage.lobby_timer then
storage.lobby_timer = 1800
end
if player.online_time < 1 then
player.insert({ name = 'concrete', count = 500 })
player.insert({ name = 'hazard-concrete', count = 500 })
player.insert({ name = 'stone-brick', count = 500 })
player.insert({ name = 'refined-concrete', count = 500 })
player.insert({ name = 'refined-hazard-concrete', count = 500 })
end
return
end
end
local function on_marked_for_deconstruction(event)
event.entity.cancel_deconstruction(game.players[event.player_index].force.name)
end
function shrink_arena()
local surface = game.surfaces['tank_battles']
if storage.current_arena_size < 0 then
return
end
local shrink_width = 8
local current_arena_size = storage.current_arena_size
local tiles = {}
for x = arena_size * -1, arena_size, 1 do
for y = current_arena_size * -1 - shrink_width, current_arena_size * -1, 1 do
local pos = { x = x, y = y }
if surface.get_tile(pos).name ~= 'water' and surface.get_tile(pos).name ~= 'deepwater' then
if x > current_arena_size or y > current_arena_size or x < current_arena_size * -1 or y < current_arena_size * -1 then
if math_random(1, 3) ~= 1 then
table_insert(tiles, { name = 'water', position = pos })
end
end
end
end
end
for x = arena_size * -1, arena_size, 1 do
for y = current_arena_size, current_arena_size + shrink_width, 1 do
local pos = { x = x, y = y }
if surface.get_tile(pos).name ~= 'water' and surface.get_tile(pos).name ~= 'deepwater' then
if x > current_arena_size or y > current_arena_size or x < current_arena_size * -1 or y < current_arena_size * -1 then
if math_random(1, 3) ~= 1 then
table_insert(tiles, { name = 'water', position = pos })
end
end
end
end
end
for x = current_arena_size * -1 - shrink_width, current_arena_size * -1, 1 do
for y = arena_size * -1, arena_size, 1 do
local pos = { x = x, y = y }
if surface.get_tile(pos).name ~= 'water' and surface.get_tile(pos).name ~= 'deepwater' then
if x > current_arena_size or y > current_arena_size or x < current_arena_size * -1 or y < current_arena_size * -1 then
if math_random(1, 3) ~= 1 then
table_insert(tiles, { name = 'water', position = pos })
end
end
end
end
end
for x = current_arena_size, current_arena_size + shrink_width, 1 do
for y = arena_size * -1, arena_size, 1 do
local pos = { x = x, y = y }
if surface.get_tile(pos).name ~= 'water' and surface.get_tile(pos).name ~= 'deepwater' then
if x > current_arena_size or y > current_arena_size or x < current_arena_size * -1 or y < current_arena_size * -1 then
if math_random(1, 3) ~= 1 then
table_insert(tiles, { name = 'water', position = pos })
end
end
end
end
end
storage.current_arena_size = storage.current_arena_size - 1
surface.set_tiles(tiles, true)
end
local function render_arena_chunk(event)
if event.surface.name ~= 'tank_battles' then
return
end
local surface = event.surface
local left_top = event.area.left_top
local tiles = {}
for x = 0, 31, 1 do
for y = 0, 31, 1 do
local pos = { x = left_top.x + x, y = left_top.y + y }
if pos.x > arena_size or pos.y > arena_size or pos.x < arena_size * -1 or pos.y < arena_size * -1 then
table_insert(tiles, { name = 'water', position = pos })
else
if math_random(1, 256) == 1 then
if surface.can_place_entity({ name = 'wooden-chest', position = pos, force = 'enemy' }) then
surface.create_entity({ name = 'wooden-chest', position = pos, force = 'enemy' })
end
end
if math_random(1, 1024) == 1 then
if math_random(1, 64) == 1 then
if surface.can_place_entity({ name = 'assembling-machine-1', position = pos, force = 'enemy' }) then
surface.create_entity({ name = 'assembling-machine-1', position = pos, force = 'enemy' })
end
end
if math_random(1, 64) == 1 then
if surface.can_place_entity({ name = 'big-worm-turret', position = pos, force = 'enemy' }) then
surface.create_entity({ name = 'big-worm-turret', position = pos, force = 'enemy' })
end
end
if math_random(1, 32) == 1 then
if surface.can_place_entity({ name = 'medium-worm-turret', position = pos, force = 'enemy' }) then
surface.create_entity({ name = 'medium-worm-turret', position = pos, force = 'enemy' })
end
end
if math_random(1, 512) == 1 then
if surface.can_place_entity({ name = 'behemoth-biter', position = pos, force = 'enemy' }) then
surface.create_entity({ name = 'behemoth-biter', position = pos, force = 'enemy' })
end
end
if math_random(1, 64) == 1 then
if surface.can_place_entity({ name = 'big-biter', position = pos, force = 'enemy' }) then
surface.create_entity({ name = 'big-biter', position = pos, force = 'enemy' })
end
end
end
end
end
end
surface.set_tiles(tiles, true)
end
local function render_spawn_chunk(event)
if event.surface.name ~= 'nauvis' then
return
end
local surface = event.surface
local left_top = event.area.left_top
for _, entity in pairs(surface.find_entities_filtered({ area = event.area })) do
if entity.name ~= 'character' then
entity.destroy()
end
end
local tiles = {}
for x = 0, 31, 1 do
for y = 0, 31, 1 do
local pos = { x = left_top.x + x, y = left_top.y + y }
table_insert(tiles, { name = 'grass-2', position = pos })
end
end
surface.set_tiles(tiles, true)
end
local function kill_idle_players()
for _, player in pairs(game.connected_players) do
if player.character then
if player.afk_time > 600 then
local area = { { player.position.x - 1, player.position.y - 1 }, { player.position.x + 1, player.position.y + 1 } }
local water_tile_count = player.surface.count_tiles_filtered({ name = { 'water', 'deepwater' }, area = area })
if water_tile_count > 3 then
player.character.die()
game.print(player.name .. ' drowned.', { r = 150, g = 150, b = 0 })
else
if player.afk_time > 9000 then
player.character.die()
game.print(player.name .. ' was idle for too long.', { r = 150, g = 150, b = 0 })
end
end
end
end
end
end
local function check_for_game_over()
local surface = game.surfaces['tank_battles']
kill_idle_players()
local alive_players = 0
for _, player in pairs(game.connected_players) do
if player.character and player.surface.name == 'tank_battles' then
alive_players = alive_players + 1
end
end
if alive_players > 1 then
return
end -----------------
local player
for _, p in pairs(game.connected_players) do
if p.character and p.surface.name == 'tank_battles' then
player = p
end
end
if alive_players == 1 then
if not storage.tank_battles_score[player.index] then
storage.tank_battles_score[player.index] = 1
else
storage.tank_battles_score[player.index] = storage.tank_battles_score[player.index] + 1
end
game.print(player.name .. ' has won the battle!', { r = 150, g = 150, b = 0 })
create_tank_battle_score_gui()
end
if alive_players == 0 then
game.print('No alive players! Round ends in a draw!', { r = 150, g = 150, b = 0 })
end
storage.game_stage = 'lobby'
end
local function on_chunk_generated(event)
render_arena_chunk(event)
render_spawn_chunk(event)
end
local function on_player_respawned(event)
local player = game.players[event.player_index]
local permissions_group = game.permissions.get_group('Spectator')
permissions_group.add_player(player.name)
player.character.destroy()
player.character = nil
end
local function lobby()
if game.surfaces['tank_battles'] then
game.delete_surface(game.surfaces['tank_battles'])
for _, player in pairs(game.connected_players) do
if player.character then
player.character.destroy()
player.character = nil
end
end
end
local connected_players_count = 0
local permissions_group = game.permissions.get_group('Default')
for _, player in pairs(game.connected_players) do
permissions_group.add_player(player.name)
if not player.character and player.ticks_to_respawn == nil then
player.create_character()
local pos = player.surface.find_non_colliding_position('character', { 0, 0 }, 16, 3)
player.insert({ name = 'concrete', count = 500 })
player.insert({ name = 'hazard-concrete', count = 500 })
player.insert({ name = 'stone-brick', count = 500 })
player.insert({ name = 'refined-concrete', count = 500 })
player.insert({ name = 'refined-hazard-concrete', count = 500 })
player.teleport({ math_random(1, 32), math_random(1, 32) }, game.surfaces[1])
end
connected_players_count = connected_players_count + 1
end
if connected_players_count < 2 then
--game.print("Waiting for players.", {r = 0, g = 150, b = 150})
return
end
if not storage.lobby_timer then
storage.lobby_timer = 1800
end
if storage.lobby_timer % 600 == 0 then
if storage.lobby_timer <= 0 then
game.print('Round has started!', { r = 0, g = 150, b = 150 })
else
game.print('Round will begin in ' .. storage.lobby_timer / 60 .. ' seconds.', { r = 0, g = 150, b = 150 })
end
end
storage.lobby_timer = storage.lobby_timer - 300
if storage.lobby_timer >= 0 then
return
end
storage.lobby_timer = nil
storage.game_stage = 'create_arena'
end
local function on_tick(event)
if game.tick % 300 == 0 then
if storage.game_stage == 'lobby' then
lobby()
end
if storage.game_stage == 'create_arena' then
create_new_arena()
end
if storage.game_stage == 'ongoing_game' then
shrink_arena()
check_for_game_over()
end
end
end
local function on_entity_died(event)
if event.entity.name == 'wooden-chest' then
local loot = loot_raffle[math_random(1, #loot_raffle)]
event.entity.surface.spill_item_stack(event.entity.position, loot, true)
--event.entity.surface.create_entity({name = "flying-text", position = event.entity.position, text = loot.name, color = {r=0.98, g=0.66, b=0.22}})
end
end
local function on_player_died(event)
local player = game.players[event.player_index]
local str = ' '
if event.cause then
if event.cause.name ~= nil then
str = ' by ' .. event.cause.name
end
if event.cause.name == 'character' then
str = ' by ' .. event.cause.player.name
end
if event.cause.name == 'tank' then
local driver = event.cause.get_driver()
if driver.player then
str = ' by ' .. driver.player.name
end
end
end
for _, target_player in pairs(game.connected_players) do
if target_player.name ~= player.name then
player.print(player.name .. ' was killed' .. str, { r = 0.99, g = 0.0, b = 0.0 })
end
end
end
----------share chat with player and spectator force-------------------
local function on_console_chat(event)
if not event.message then
return
end
if not event.player_index then
return
end
local player = game.players[event.player_index]
local color = {}
color = player.color
color.r = color.r * 0.6 + 0.35
color.g = color.g * 0.6 + 0.35
color.b = color.b * 0.6 + 0.35
color.a = 1
for _, target_player in pairs(game.connected_players) do
if target_player.name ~= player.name then
target_player.print(player.name .. ': ' .. event.message, color)
end
end
end
event.add(defines.events.on_tick, on_tick)
event.add(defines.events.on_console_chat, on_console_chat)
event.add(defines.events.on_entity_died, on_entity_died)
event.add(defines.events.on_entity_damaged, on_entity_damaged)
event.add(defines.events.on_player_died, on_player_died)
event.add(defines.events.on_marked_for_deconstruction, on_marked_for_deconstruction)
event.add(defines.events.on_player_joined_game, on_player_joined_game)
event.add(defines.events.on_player_respawned, on_player_respawned)
event.add(defines.events.on_chunk_generated, on_chunk_generated)