1
0
mirror of https://github.com/Oarcinae/FactorioScenarioMultiplayerSpawn.git synced 2025-01-24 03:16:22 +02:00
FactorioScenarioMultiplayer.../lib/separate_spawns_guis.lua

1168 lines
45 KiB
Lua
Raw Normal View History

-- I made a separate file for all the GUI related functions. Yay me.
local SPAWN_GUI_MAX_WIDTH = 500
local SPAWN_GUI_MAX_HEIGHT = 1000
---A display gui message. Meant to be display the first time a player joins.
---@param player LuaPlayer
---@return boolean
function DisplayWelcomeTextGui(player)
if ((player.gui.screen["welcome_msg"] ~= nil) or
(player.gui.screen["spawn_opts"] ~= nil) or
(player.gui.screen["shared_spawn_opts"] ~= nil) or
(player.gui.screen["join_shared_spawn_wait_menu"] ~= nil) or
(player.gui.screen["buddy_spawn_opts"] ~= nil) or
(player.gui.screen["buddy_wait_menu"] ~= nil) or
(player.gui.screen["buddy_request_menu"] ~= nil) or
(player.gui.screen["wait_for_spawn_dialog"] ~= nil)) then
log("DisplayWelcomeTextGui called while some other dialog is already displayed!")
return false
end
local wGui = player.gui.screen.add { name = "welcome_msg",
type = "frame",
direction = "vertical",
caption = global.ocfg.server_info.welcome_msg_title }
wGui.auto_center = true
wGui.style.maximal_width = SPAWN_GUI_MAX_WIDTH
wGui.style.maximal_height = SPAWN_GUI_MAX_HEIGHT
-- Start with server message.
AddLabel(wGui, "server_msg_lbl1", global.ocfg.server_info.server_msg, my_label_style)
-- AddLabel(wGui, "contact_info_msg_lbl1", global.ocfg.server_contact, my_label_style)
AddSpacer(wGui)
-- Informational message about the scenario
AddLabel(wGui, "scenario_info_msg_lbl1", global.ocfg.server_info.welcome_msg, my_label_style)
AddSpacer(wGui)
-- Warning about spawn creation time
AddLabel(wGui, "spawn_time_msg_lbl1", { "oarc-spawn-time-warning-msg" }, my_warning_style)
-- Confirm button
AddSpacerLine(wGui)
local button_flow = wGui.add { type = "flow" }
button_flow.style.horizontal_align = "right"
button_flow.style.horizontally_stretchable = true
button_flow.add { name = "welcome_okay_btn",
type = "button",
caption = { "oarc-i-understand" },
style = "confirm_button" }
return true
end
---Handle the gui click of the welcome msg
---@param event EventData.on_gui_click
---@return nil
function WelcomeTextGuiClick(event)
if not event.element.valid then return end
local player = game.players[event.player_index]
local buttonClicked = event.element.name
if not player then
log("Another gui click happened with no valid player...")
return
end
if (buttonClicked == "welcome_okay_btn") then
if (player.gui.screen.welcome_msg ~= nil) then
player.gui.screen.welcome_msg.destroy()
end
DisplaySpawnOptions(player)
end
end
---Display the spawn options and explanation
---@param player LuaPlayer
---@return nil
function DisplaySpawnOptions(player)
if (player == nil) then
log("DisplaySpawnOptions with no valid player...")
return
end
if (player.gui.screen.spawn_opts ~= nil) then
log("Tried to display spawn options when it was already displayed!")
return
end
local gameplay = global.ocfg.gameplay
player.gui.screen.add { name = "spawn_opts",
type = "frame",
direction = "vertical",
caption = { "oarc-spawn-options" } }
local sGui = player.gui.screen.spawn_opts
sGui.style.maximal_width = SPAWN_GUI_MAX_WIDTH
sGui.style.maximal_height = SPAWN_GUI_MAX_HEIGHT
sGui.auto_center = true
-- Warnings and explanations...
local warn_msg = { "oarc-click-info-btn-help" }
AddLabel(sGui, "warning_lbl1", warn_msg, my_warning_style)
-- TODO: Not sure what this is for...? SPAWN_MSG1 is not defined anywhere.
-- AddLabel(sGui, "spawn_msg_lbl1", SPAWN_MSG1, my_label_style)
-- Button and message about the regular vanilla spawn
-- if ENABLE_DEFAULT_SPAWN then
-- sGui.add{name = "default_spawn_btn",
-- type = "button",
-- caption={"oarc-vanilla-spawn"}}
-- local normal_spawn_text = {"oarc-default-spawn-behavior"}
-- AddLabel(sGui, "normal_spawn_lbl1", normal_spawn_text, my_label_style)
-- -- AddSpacerLine(sGui, "normal_spawn_spacer")
-- end
-- The main spawning options. Solo near and solo far.
-- If enable, you can also choose to be on your own team.
local soloSpawnFlow = sGui.add { name = "spawn_solo_flow",
type = "frame",
direction = "vertical",
style = "bordered_frame" }
-- Pick surface
if (gameplay.enable_spawning_on_other_surfaces) then
local surfacesHorizontalFlow = soloSpawnFlow.add { name = "surfaces_horizontal_flow",
type = "flow",
direction = "horizontal" }
---@type string[]
local surfaceList = {}
for surfaceName,allowed in pairs(global.ocore.surfaces) do
if allowed then
table.insert(surfaceList, surfaceName)
end
end
AddLabel(surfacesHorizontalFlow, "surfacesHorizontalFlowLabel", "Select Surface: ", my_label_style)
surfacesHorizontalFlow.add { name = "surface_select_dropdown",
type = "drop-down",
items = surfaceList,
selected_index = 1}
end
-- Radio buttons to pick your team.
if (gameplay.enable_separate_teams) then
soloSpawnFlow.add { name = "isolated_spawn_main_team_radio",
type = "radiobutton",
caption = { "oarc-join-main-team-radio" },
state = true }
soloSpawnFlow.add { name = "isolated_spawn_new_team_radio",
type = "radiobutton",
caption = { "oarc-create-own-team-radio" },
state = false }
end
-- OPTIONS frame
-- AddLabel(soloSpawnFlow, "options_spawn_lbl1",
-- "Additional spawn options can be selected here. Not all are compatible with each other.", my_label_style)
-- Allow players to spawn with a moat around their area.
--TODO: Vanilla spawn points are not implemented yet.
-- and not global.ocfg.enable_vanilla_spawns
if (gameplay.allow_moats_around_spawns) then
soloSpawnFlow.add { name = "isolated_spawn_moat_option_checkbox",
type = "checkbox",
caption = { "oarc-moat-option" },
state = false }
end
-- if (global.ocfg.enable_vanilla_spawns and (TableLength(global.vanillaSpawns) > 0)) then
-- soloSpawnFlow.add{name = "isolated_spawn_vanilla_option_checkbox",
-- type = "checkbox",
-- caption="Use a pre-set vanilla spawn point. " .. TableLengthglobal.vanillaSpawns .. " available.",
-- state=false}
-- end
-- Isolated spawn options. The core gameplay of this scenario.
local soloSpawnbuttons = soloSpawnFlow.add { name = "spawn_solo_flow",
type = "flow",
direction = "horizontal" }
soloSpawnbuttons.style.horizontal_align = "center"
soloSpawnbuttons.style.horizontally_stretchable = true
soloSpawnbuttons.add { name = "isolated_spawn_near",
type = "button",
caption = { "oarc-solo-spawn-near" },
style = "confirm_button" }
soloSpawnbuttons.add { name = "isolated_spawn_far",
type = "button",
caption = { "oarc-solo-spawn-far" },
style = "confirm_button" }
-- if (global.ocfg.enable_vanilla_spawns) then
-- AddLabel(soloSpawnFlow, "isolated_spawn_lbl1",
-- {"oarc-starting-area-vanilla"}, my_label_style)
-- AddLabel(soloSpawnFlow, "vanilla_spawn_lbl2",
-- {"oarc-vanilla-spawns-available", TableLength(global.vanillaSpawns)}, my_label_style)
-- else
AddLabel(soloSpawnFlow, "isolated_spawn_lbl1",
{ "oarc-starting-area-normal" }, my_label_style)
-- end
-- Spawn options to join another player's base.
local sharedSpawnFrame = sGui.add { name = "spawn_shared_flow",
type = "frame",
direction = "vertical",
style = "bordered_frame" }
if gameplay.enable_shared_spawns then
local numAvailSpawns = GetNumberOfAvailableSharedSpawns()
if (numAvailSpawns > 0) then
sharedSpawnFrame.add { name = "join_other_spawn",
type = "button",
caption = { "oarc-join-someone-avail", numAvailSpawns } }
local join_spawn_text = { "oarc-join-someone-info" }
AddLabel(sharedSpawnFrame, "join_other_spawn_lbl1", join_spawn_text, my_label_style)
else
AddLabel(sharedSpawnFrame, "join_other_spawn_lbl1", { "oarc-no-shared-avail" }, my_label_style)
sharedSpawnFrame.add { name = "join_other_spawn_check",
type = "button",
caption = { "oarc-join-check-again" } }
end
else
AddLabel(sharedSpawnFrame, "join_other_spawn_lbl1",
{ "oarc-shared-spawn-disabled" }, my_warning_style)
end
-- Awesome buddy spawning system
---TODO: Vanilla spawn points are not implemented yet.
-- if (not global.ocfg.enable_vanilla_spawns) then
if gameplay.enable_shared_spawns and gameplay.enable_buddy_spawn then
local buddySpawnFrame = sGui.add { name = "spawn_buddy_flow",
type = "frame",
direction = "vertical",
style = "bordered_frame" }
-- AddSpacerLine(buddySpawnFrame, "buddy_spawn_msg_spacer")
buddySpawnFrame.add { name = "buddy_spawn",
type = "button",
caption = { "oarc-buddy-spawn" } }
AddLabel(buddySpawnFrame, "buddy_spawn_lbl1",
{ "oarc-buddy-spawn-info" }, my_label_style)
end
-- end
-- Some final notes
if (gameplay.number_of_players_per_shared_spawn > 0) then
AddLabel(sGui, "max_players_lbl2",
{ "oarc-max-players-shared-spawn", gameplay.number_of_players_per_shared_spawn - 1 },
my_note_style)
end
local spawn_distance_notes = { "oarc-spawn-dist-notes",
gameplay.near_spawn_min_distance,
gameplay.near_spawn_max_distance,
gameplay.far_spawn_min_distance,
gameplay.far_spawn_max_distance }
AddLabel(sGui, "note_lbl1", spawn_distance_notes, my_note_style)
end
---This just updates the radio buttons/checkboxes when players click them.
---@param event EventData.on_gui_checked_state_changed
---@return nil
function SpawnOptsRadioSelect(event)
if not event.element.valid then return end
local elemName = event.element.name
if (elemName == "isolated_spawn_main_team_radio") then
event.element.parent.isolated_spawn_new_team_radio.state = false
elseif (elemName == "isolated_spawn_new_team_radio") then
event.element.parent.isolated_spawn_main_team_radio.state = false
end
if (elemName == "buddy_spawn_main_team_radio") then
event.element.parent.buddy_spawn_new_team_radio.state = false
event.element.parent.buddy_spawn_buddy_team_radio.state = false
elseif (elemName == "buddy_spawn_new_team_radio") then
event.element.parent.buddy_spawn_main_team_radio.state = false
event.element.parent.buddy_spawn_buddy_team_radio.state = false
elseif (elemName == "buddy_spawn_buddy_team_radio") then
event.element.parent.buddy_spawn_main_team_radio.state = false
event.element.parent.buddy_spawn_new_team_radio.state = false
end
end
---Handle the gui click of the spawn options
---@param event EventData.on_gui_click
---@return nil
function SpawnOptsGuiClick(event)
if not event.element.valid then return end
local player = game.players[event.player_index]
local elemName = event.element.name
if not player then
log("Another gui click happened with no valid player...")
return
end
if (player.gui.screen.spawn_opts == nil) then
return -- Gui event unrelated to this gui.
end
local pgcs = player.gui.screen.spawn_opts
local joinOwnTeamRadio, moatChoice = false, false
local surfaceName = global.ocfg.gameplay.default_surface -- Default to default surface
local surface = game.surfaces[surfaceName]
-- Check if a valid button on the gui was pressed
-- and delete the GUI
if ((elemName == "default_spawn_btn") or
(elemName == "isolated_spawn_near") or
(elemName == "isolated_spawn_far") or
(elemName == "join_other_spawn") or
(elemName == "buddy_spawn") or
(elemName == "join_other_spawn_check")) then
if (global.ocfg.gameplay.enable_separate_teams) then
joinMainTeamRadio = pgcs.spawn_solo_flow.isolated_spawn_main_team_radio.state
joinOwnTeamRadio = pgcs.spawn_solo_flow.isolated_spawn_new_team_radio.state
else
joinMainTeamRadio = true
joinOwnTeamRadio = false
end
---TODO: Vanilla spawn points are not implemented yet. and not global.ocfg.enable_vanilla_spawns
if (global.ocfg.gameplay.allow_moats_around_spawns and
(pgcs.spawn_solo_flow.isolated_spawn_moat_option_checkbox ~= nil)) then
moatChoice = pgcs.spawn_solo_flow.isolated_spawn_moat_option_checkbox.state
end
-- Override the default surface if the player selected a different one.
local surfaceDropdownIndex = pgcs.spawn_solo_flow.surfaces_horizontal_flow.surface_select_dropdown.selected_index
-- Index 0 means nothing was selected!
if (surfaceDropdownIndex ~= 0) then
surfaceName = pgcs.spawn_solo_flow.surfaces_horizontal_flow.surface_select_dropdown.get_item(surfaceDropdownIndex) --[[@as string]]
surface = game.surfaces[surfaceName]
end
-- if (global.ocfg.enable_vanilla_spawns and
-- (pgcs.spawn_solo_flow.isolated_spawn_vanilla_option_checkbox ~= nil)) then
-- vanillaChoice = pgcs.spawn_solo_flow.isolated_spawn_vanilla_option_checkbox.state
-- end
pgcs.destroy()
else
return -- Do nothing, no valid element item was clicked.
end
-- Default spawn should always spawn on a default surface I think?
if (elemName == "default_spawn_btn") then
GivePlayerStarterItems(player)
local defaultSurfaceName = global.ocfg.gameplay.default_surface
local defaultSurface = game.surfaces[defaultSurfaceName]
local spawnPosition = player.force.get_spawn_position(defaultSurface)
ChangePlayerSpawn(player, defaultSurfaceName, spawnPosition)
SendBroadcastMsg({ "oarc-player-is-joining-main-force", player.name, defaultSurfaceName })
ChartArea(player.force, player.position,
math.ceil(global.ocfg.surfaces_config[defaultSurfaceName].spawn_config.general.spawn_radius_tiles / CHUNK_SIZE),
defaultSurface)
-- Unlock spawn control gui tab
SetOarcGuiTabEnabled(player, OARC_SPAWN_CTRL_TAB_NAME, true)
elseif ((elemName == "isolated_spawn_near") or (elemName == "isolated_spawn_far")) then
local newSpawn = { x = 0, y = 0 }
local gameplay = global.ocfg.gameplay
-- Create a new force for player if they choose that radio button
if gameplay.enable_separate_teams and joinOwnTeamRadio then
local newForce = CreatePlayerCustomForce(player)
end
---TODO: Vanilla spawn points are not implemented yet.
-- -- Find an unused vanilla spawn
-- -- if (vanillaChoice) then
-- if (global.ocfg.enable_vanilla_spawns) then
-- if (elemName == "isolated_spawn_far") then
-- newSpawn = FindUnusedVanillaSpawn(game.surfaces[GAME_SURFACE_NAME],
-- global.ocfg.far_dist_end*CHUNK_SIZE)
-- elseif (elemName == "isolated_spawn_near") then
-- newSpawn = FindUnusedVanillaSpawn(game.surfaces[GAME_SURFACE_NAME],
-- global.ocfg.near_dist_start*CHUNK_SIZE)
-- end
-- -- Default OARC-type pre-set layout spawn.
-- else
-- Find coordinates of a good place to spawn
if (elemName == "isolated_spawn_far") then
newSpawn = FindUngeneratedCoordinates(gameplay.far_spawn_min_distance, gameplay.far_spawn_max_distance, surface)
elseif (elemName == "isolated_spawn_near") then
newSpawn = FindUngeneratedCoordinates(gameplay.near_spawn_min_distance, gameplay.near_spawn_max_distance, surface)
end
-- end
-- If that fails, find a random map edge in a rand direction.
if ((newSpawn.x == 0) and (newSpawn.y == 0)) then
newSpawn = FindMapEdge(GetRandomVector(), surface)
log("Resorting to find map edge! x=" .. newSpawn.x .. ",y=" .. newSpawn.y)
end
-- Create that player's spawn in the global vars
ChangePlayerSpawn(player, surfaceName, newSpawn)
-- Send the player there
QueuePlayerForDelayedSpawn(player.name,
surfaceName,
newSpawn,
moatChoice,
false) -- global.ocfg.enable_vanilla_spawns --TODO: Vanilla spawn points are not implemented yet.
if (elemName == "isolated_spawn_near") then
SendBroadcastMsg({ "oarc-player-is-joining-near", player.name, surfaceName })
elseif (elemName == "isolated_spawn_far") then
SendBroadcastMsg({ "oarc-player-is-joining-far", player.name, surfaceName })
end
-- Unlock spawn control gui tab
SetOarcGuiTabEnabled(player, OARC_SPAWN_CTRL_TAB_NAME, true)
player.print({ "oarc-please-wait" })
player.print({ "", { "oarc-please-wait" }, "!" })
player.print({ "", { "oarc-please-wait" }, "!!" })
elseif (elemName == "join_other_spawn") then
DisplaySharedSpawnOptions(player)
-- Provide a way to refresh the gui to check if people have shared their
-- bases.
elseif (elemName == "join_other_spawn_check") then
DisplaySpawnOptions(player)
-- Hacky buddy spawn system
elseif (elemName == "buddy_spawn") then
table.insert(global.ocore.waitingBuddies --[[@as OarcWaitingBuddiesTable]], player.name)
SendBroadcastMsg({ "oarc-looking-for-buddy", player.name })
DisplayBuddySpawnOptions(player)
end
end
---Display the spawn options and explanation
---@param player LuaPlayer
---@return nil
function DisplaySharedSpawnOptions(player)
player.gui.screen.add { name = "shared_spawn_opts",
type = "frame",
direction = "vertical",
caption = { "oarc-avail-bases-join" } }
local shGuiFrame = player.gui.screen.shared_spawn_opts
shGuiFrame.auto_center = true
local shGui = shGuiFrame.add { type = "scroll-pane", name = "spawns_scroll_pane", caption = "" }
ApplyStyle(shGui, my_fixed_width_style)
shGui.style.maximal_width = SPAWN_GUI_MAX_WIDTH
shGui.style.maximal_height = SPAWN_GUI_MAX_HEIGHT
shGui.horizontal_scroll_policy = "never"
for spawnName, sharedSpawn in pairs(global.ocore.sharedSpawns) do
if (sharedSpawn.openAccess and
(game.players[spawnName] ~= nil) and
game.players[spawnName].connected) then
local spotsRemaining = global.ocfg.gameplay.number_of_players_per_shared_spawn - TableLength(global.ocore.sharedSpawns[spawnName].players)
if (global.ocfg.gameplay.number_of_players_per_shared_spawn == 0) then
shGui.add { type = "button", caption = spawnName, name = spawnName }
elseif (spotsRemaining > 0) then
shGui.add { type = "button", caption = { "oarc-spawn-spots-remaining", spawnName, spotsRemaining }, name = spawnName }
end
if (shGui.spawnName ~= nil) then
-- AddSpacer(buddyGui, spawnName .. "spacer_lbl")
ApplyStyle(shGui[spawnName], my_small_button_style)
end
end
end
shGui.add { name = "shared_spawn_cancel",
type = "button",
caption = { "oarc-cancel-return-to-previous" },
style = "back_button" }
end
---Handle the gui click of the shared spawn options
---@param event EventData.on_gui_click
---@return nil
function SharedSpwnOptsGuiClick(event)
if not event.element.valid then return end
local player = game.players[event.player_index]
local buttonClicked = event.element.name
if not player then
log("Another gui click happened with no valid player...")
return
end
if (event.element.parent) then
if (event.element.parent.name ~= "spawns_scroll_pane") then
return
end
end
-- Check for cancel button, return to spawn options
if (buttonClicked == "shared_spawn_cancel") then
DisplaySpawnOptions(player)
if (player.gui.screen.shared_spawn_opts ~= nil) then
player.gui.screen.shared_spawn_opts.destroy()
end
-- Else check for which spawn was selected
-- If a spawn is removed during this time, the button will not do anything
else
for spawnName, sharedSpawn in pairs(global.ocore.sharedSpawns --[[@as OarcSharedSpawnsTable]]) do
if ((buttonClicked == spawnName) and
(game.players[spawnName] ~= nil) and
(game.players[spawnName].connected)) then
-- Add the player to that shared spawns join queue.
table.insert(global.ocore.sharedSpawns[spawnName].joinQueue, player.name)
-- Clear the shared spawn options gui.
if (player.gui.screen.shared_spawn_opts ~= nil) then
player.gui.screen.shared_spawn_opts.destroy()
end
-- Display wait menu with cancel button.
DisplaySharedSpawnJoinWaitMenu(player)
-- Tell other player they are requesting a response.
game.players[spawnName].print({ "oarc-player-requesting-join-you", player.name })
break
end
end
end
end
---Display shared spawn join wait menu
---@param player LuaPlayer
---@return nil
function DisplaySharedSpawnJoinWaitMenu(player)
local sGui = player.gui.screen.add { name = "join_shared_spawn_wait_menu",
type = "frame",
direction = "vertical",
caption = { "oarc-waiting-for-spawn-owner" } }
sGui.auto_center = true
sGui.style.maximal_width = SPAWN_GUI_MAX_WIDTH
sGui.style.maximal_height = SPAWN_GUI_MAX_HEIGHT
-- Warnings and explanations...
AddLabel(sGui, "warning_lbl1", { "oarc-you-will-spawn-once-host" }, my_warning_style)
sGui.add { name = "cancel_shared_spawn_wait_menu",
type = "button",
caption = { "oarc-cancel-return-to-previous" },
style = "back_button" }
end
---Handle the gui click of the shared spawn join wait menu
---@param event EventData.on_gui_click
---@return nil
function SharedSpawnJoinWaitMenuClick(event)
if not event.element.valid then return end
local player = game.players[event.player_index]
local elemName = event.element.name
if not player then
log("Another gui click happened with no valid player...")
return
end
if (player.gui.screen.join_shared_spawn_wait_menu == nil) then
return -- Gui event unrelated to this gui.
end
-- Check if player is cancelling the request.
if (elemName == "cancel_shared_spawn_wait_menu") then
player.gui.screen.join_shared_spawn_wait_menu.destroy()
DisplaySpawnOptions(player)
-- Find and remove the player from the joinQueue they were in.
for spawnName, sharedSpawn in pairs(global.ocore.sharedSpawns --[[@as OarcSharedSpawnsTable]]) do
for index, requestingPlayer in pairs(sharedSpawn.joinQueue) do
if (requestingPlayer == player.name) then
global.ocore.sharedSpawns[spawnName].joinQueue[index] = nil
game.players[spawnName].print({ "oarc-player-cancel-join-request", player.name })
return
end
end
end
log("ERROR! Failed to remove player from joinQueue!")
end
end
---Display the buddy spawn menu
---@param player LuaPlayer
---@return nil
function DisplayBuddySpawnOptions(player)
local buddyGui = player.gui.screen.add { name = "buddy_spawn_opts",
type = "frame",
direction = "vertical",
caption = { "oarc-buddy-spawn-options" } }
buddyGui.auto_center = true
buddyGui.style.maximal_width = SPAWN_GUI_MAX_WIDTH
buddyGui.style.maximal_height = SPAWN_GUI_MAX_HEIGHT
---@type OarcConfigGameplaySettings
local gameplay = global.ocfg.gameplay
-- Warnings and explanations...
AddLabel(buddyGui, "buddy_info_msg", { "oarc-buddy-spawn-instructions" }, my_label_style)
AddSpacer(buddyGui)
-- The buddy spawning options.
local buddySpawnFlow = buddyGui.add { name = "spawn_buddy_flow",
type = "frame",
direction = "vertical",
style = "bordered_frame" }
---@type string[]
local buddyList = {}
for _, buddyName in pairs(global.ocore.waitingBuddies --[[@as OarcWaitingBuddiesTable]]) do
if (buddyName ~= player.name) then
table.insert(buddyList, buddyName)
end
end
AddLabel(buddySpawnFlow, "drop_down_msg_lbl1", { "oarc-buddy-select-info" }, my_label_style)
buddySpawnFlow.add { name = "waiting_buddies_dropdown",
type = "drop-down",
items = buddyList }
buddySpawnFlow.add { name = "refresh_buddy_list",
type = "button",
caption = { "oarc-buddy-refresh" } }
-- AddSpacerLine(buddySpawnFlow)
-- Pick surface
if (gameplay.enable_spawning_on_other_surfaces) then
local surfacesHorizontalFlow = buddySpawnFlow.add { name = "buddy_surfaces_horizontal_flow",
type = "flow",
direction = "horizontal" }
---@type string[]
local surfaceList = {}
for surfaceName,allowed in pairs(global.ocore.surfaces) do
if allowed then
table.insert(surfaceList, surfaceName)
end
end
AddLabel(surfacesHorizontalFlow, "buddySurfacesHorizontalFlowLabel", "Select Surface: ", my_label_style)
surfacesHorizontalFlow.add { name = "buddy_surface_select_dropdown",
type = "drop-down",
items = surfaceList,
selected_index = 1}
end
-- Allow picking of teams
if (gameplay.enable_separate_teams) then
buddySpawnFlow.add { name = "buddy_spawn_main_team_radio",
type = "radiobutton",
caption = { "oarc-join-main-team-radio" },
state = true }
buddySpawnFlow.add { name = "buddy_spawn_new_team_radio",
type = "radiobutton",
caption = { "oarc-create-own-team-radio" },
state = false }
buddySpawnFlow.add { name = "buddy_spawn_buddy_team_radio",
type = "radiobutton",
caption = { "oarc-create-buddy-team" },
state = false }
end
if (gameplay.allow_moats_around_spawns) then
buddySpawnFlow.add { name = "buddy_spawn_moat_option_checkbox",
type = "checkbox",
caption = { "oarc-moat-option" },
state = false }
end
-- AddSpacerLine(buddySpawnFlow)
buddySpawnFlow.add { name = "buddy_spawn_request_near",
type = "button",
caption = { "oarc-buddy-spawn-near" },
style = "confirm_button" }
buddySpawnFlow.add { name = "buddy_spawn_request_far",
type = "button",
caption = { "oarc-buddy-spawn-far" },
style = "confirm_button" }
AddSpacer(buddyGui)
buddyGui.add { name = "buddy_spawn_cancel",
type = "button",
caption = { "oarc-cancel-return-to-previous" },
style = "back_button" }
-- Some final notes
AddSpacerLine(buddyGui)
if (gameplay.number_of_players_per_shared_spawn > 0) then
AddLabel(buddyGui, "buddy_max_players_lbl1",
{ "oarc-max-players-shared-spawn", gameplay.number_of_players_per_shared_spawn - 1 },
my_note_style)
end
local spawn_distance_notes = {
"oarc-spawn-dist-notes",
gameplay.near_spawn_min_distance,
gameplay.near_spawn_max_distance,
gameplay.far_spawn_min_distance,
gameplay.far_spawn_max_distance
}
AddLabel(buddyGui, "note_lbl1", spawn_distance_notes, my_note_style)
end
---Handle the gui click of the buddy spawn options
---@param event EventData.on_gui_click
---@return nil
function BuddySpawnOptsGuiClick(event)
if not event.element.valid then return end
local player = game.players[event.player_index]
local elemName = event.element.name
if not player then
log("Another gui click happened with no valid player...")
return
end
if (player.gui.screen.buddy_spawn_opts == nil) then
return -- Gui event unrelated to this gui.
end
local waiting_buddies_dropdown = player.gui.screen.buddy_spawn_opts.spawn_buddy_flow.waiting_buddies_dropdown
-- Just refresh the buddy list dropdown values only.
if (elemName == "refresh_buddy_list") then
waiting_buddies_dropdown.clear_items()
for _, buddyName in pairs(global.ocore.waitingBuddies --[[@as OarcWaitingBuddiesTable]]) do
if (player.name ~= buddyName) then
waiting_buddies_dropdown.add_item(buddyName)
end
end
return
end
-- Handle the cancel button to exit this menu
if (elemName == "buddy_spawn_cancel") then
player.gui.screen.buddy_spawn_opts.destroy()
DisplaySpawnOptions(player)
-- Remove them from the buddy list when they cancel
for i = #global.ocore.waitingBuddies, 1, -1 do
if (global.ocore.waitingBuddies[i] == player.name) then
global.ocore.waitingBuddies[i] = nil
end
end
end
local moatChoice = false
local buddyChoice = nil
-- Handle the spawn request button clicks
if ((elemName == "buddy_spawn_request_near") or
(elemName == "buddy_spawn_request_far")) then
local buddySpawnGui = player.gui.screen.buddy_spawn_opts.spawn_buddy_flow
local dropDownIndex = buddySpawnGui.waiting_buddies_dropdown.selected_index
if ((dropDownIndex > 0) and (dropDownIndex <= #buddySpawnGui.waiting_buddies_dropdown.items)) then
buddyChoice = buddySpawnGui.waiting_buddies_dropdown.get_item(dropDownIndex) --[[@as string]]
else
player.print({ "oarc-invalid-buddy" })
return
end
local buddyIsStillWaiting = false
for _, buddyName in pairs(global.ocore.waitingBuddies --[[@as OarcWaitingBuddiesTable]]) do
if (buddyChoice == buddyName) then
if (game.players[buddyChoice]) then
buddyIsStillWaiting = true
end
break
end
end
if (not buddyIsStillWaiting) then
player.print({ "oarc-buddy-not-avail" })
player.gui.screen.buddy_spawn_opts.destroy()
DisplayBuddySpawnOptions(player)
return
end
-- Override the default surface if the player selected a different one.
local surfaceDropdownIndex = buddySpawnGui.surfaces_horizontal_flow.surface_select_dropdown.selected_index
local surfaceName = global.ocfg.gameplay.default_surface
-- Index 0 means nothing was selected!
if (surfaceDropdownIndex ~= 0) then
surfaceName = buddySpawnGui.surfaces_horizontal_flow.surface_select_dropdown.get_item(surfaceDropdownIndex) --[[@as string]]
end
---@type BuddySpawnChoice
local buddyTeamRadioSelection = nil
if (global.ocfg.gameplay.enable_separate_teams) then
if buddySpawnGui.buddy_spawn_main_team_radio.state then
buddyTeamRadioSelection = BUDDY_SPAWN_CHOICE.join_main_team
elseif buddySpawnGui.buddy_spawn_new_team_radio.state then
buddyTeamRadioSelection = BUDDY_SPAWN_CHOICE.join_own_team
elseif buddySpawnGui.buddy_spawn_buddy_team_radio.state then
buddyTeamRadioSelection = BUDDY_SPAWN_CHOICE.join_buddy_team
end
else
buddyTeamRadioSelection = BUDDY_SPAWN_CHOICE.join_main_team
end
if (global.ocfg.gameplay.allow_moats_around_spawns) then
moatChoice = buddySpawnGui.buddy_spawn_moat_option_checkbox.state
end
-- Save the chosen spawn options somewhere for later use.
---@type OarcBuddySpawnOpts
local buddySpawnOpts = {}
buddySpawnOpts.teamRadioSelection = buddyTeamRadioSelection
buddySpawnOpts.moatChoice = moatChoice
buddySpawnOpts.buddyChoice = buddyChoice
buddySpawnOpts.distChoice = elemName
buddySpawnOpts.surface = surfaceName
global.ocore.buddySpawnOpts[player.name] = buddySpawnOpts
player.gui.screen.buddy_spawn_opts.destroy()
-- Display prompts to the players
DisplayBuddySpawnWaitMenu(player)
DisplayBuddySpawnRequestMenu(game.players[buddyChoice], player.name)
if (game.players[buddyChoice].gui.screen.buddy_spawn_opts ~= nil) then
game.players[buddyChoice].gui.screen.buddy_spawn_opts.destroy()
end
-- Remove them from the buddy list while they make up their minds.
for i = #global.ocore.waitingBuddies, 1, -1 do
name = global.ocore.waitingBuddies[i]
if ((name == player.name) or (name == buddyChoice)) then
global.ocore.waitingBuddies[i] = nil
end
end
else
return -- Do nothing, no valid element item was clicked.
end
end
---Display the buddy spawn wait menu
---@param player LuaPlayer
---@return nil
function DisplayBuddySpawnWaitMenu(player)
local sGui = player.gui.screen.add { name = "buddy_wait_menu",
type = "frame",
direction = "vertical",
caption = { "oarc-waiting-for-buddy" } }
sGui.auto_center = true
sGui.style.maximal_width = SPAWN_GUI_MAX_WIDTH
sGui.style.maximal_height = SPAWN_GUI_MAX_HEIGHT
-- Warnings and explanations...
AddLabel(sGui, "warning_lbl1", { "oarc-wait-buddy-select-yes" }, my_warning_style)
AddSpacer(sGui)
sGui.add { name = "cancel_buddy_wait_menu",
type = "button",
caption = { "oarc-cancel-return-to-previous" } }
end
---Handle the gui click of the buddy wait menu
---@param event EventData.on_gui_click
---@return nil
function BuddySpawnWaitMenuClick(event)
if not event.element.valid then return end
local player = game.players[event.player_index]
local elemName = event.element.name
if not player then
log("Another gui click happened with no valid player...")
return
end
if (player.gui.screen.buddy_wait_menu == nil) then
return -- Gui event unrelated to this gui.
end
-- Check if player is cancelling the request.
if (elemName == "cancel_buddy_wait_menu") then
player.gui.screen.buddy_wait_menu.destroy()
DisplaySpawnOptions(player)
---@type OarcBuddySpawnOpts
local buddySpawnOpts = global.ocore.buddySpawnOpts[player.name]
local buddy = game.players[buddySpawnOpts.buddyChoice]
-- Catch a case where the buddy has left the game early and no longer exists.
if (buddy == nil) then
return
end
if (buddy.gui.screen.buddy_request_menu ~= nil) then
buddy.gui.screen.buddy_request_menu.destroy()
end
if (buddy.gui.screen.buddy_spawn ~= nil) then
buddy.gui.screen.buddy_spawn_opts.destroy()
end
DisplaySpawnOptions(buddy)
buddy.print({ "oarc-buddy-cancel-request", player.name })
end
end
---Display the buddy spawn request menu
---@param player LuaPlayer
---@param requestingBuddyName string
---@return nil
function DisplayBuddySpawnRequestMenu(player, requestingBuddyName)
if not player then
log("Another gui click happened with no valid player...")
return
end
local sGui = player.gui.screen.add { name = "buddy_request_menu",
type = "frame",
direction = "vertical",
caption = "Buddy Request!" }
sGui.auto_center = true
sGui.style.maximal_width = SPAWN_GUI_MAX_WIDTH
sGui.style.maximal_height = SPAWN_GUI_MAX_HEIGHT
-- Warnings and explanations...
AddLabel(sGui, "warning_lbl1", { "oarc-buddy-requesting-from-you", requestingBuddyName }, my_warning_style)
---@type OarcBuddySpawnOpts
local buddySpawnOpts = global.ocore.buddySpawnOpts[requestingBuddyName]
---@type LocalisedString
local teamText = "error!"
if (buddySpawnOpts.teamRadioSelection == BUDDY_SPAWN_CHOICE.join_main_team) then
teamText = { "oarc-buddy-txt-main-team" }
elseif (buddySpawnOpts.teamRadioSelection == BUDDY_SPAWN_CHOICE.join_own_team) then
teamText = { "oarc-buddy-txt-new-teams" }
elseif (buddySpawnOpts.teamRadioSelection == BUDDY_SPAWN_CHOICE.join_buddy_team) then
teamText = { "oarc-buddy-txt-buddy-team" }
end
---@type LocalisedString
local moatText = " "
if (global.ocore.buddySpawnOpts[requestingBuddyName].moatChoice) then
moatText = { "oarc-buddy-txt-moat" }
end
---@type LocalisedString
local distText = "error!"
if (global.ocore.buddySpawnOpts[requestingBuddyName].distChoice == "buddy_spawn_request_near") then
distText = { "oarc-buddy-txt-near" }
elseif (global.ocore.buddySpawnOpts[requestingBuddyName].distChoice == "buddy_spawn_request_far") then
distText = { "oarc-buddy-txt-far" }
end
local requestText = { "", requestingBuddyName, { "oarc-buddy-txt-would-like" }, teamText, { "oarc-buddy-txt-next-to-you" },
moatText, distText }
AddLabel(sGui, "note_lbl1", requestText, my_warning_style)
AddSpacer(sGui)
sGui.add { name = "accept_buddy_request",
type = "button",
caption = { "oarc-accept" } }
sGui.add { name = "decline_buddy_request",
type = "button",
caption = { "oarc-reject" } }
end
---Handle the gui click of the buddy request menu
---@param event EventData.on_gui_click
---@return nil
function BuddySpawnRequestMenuClick(event)
if not event.element.valid then return end
local player = game.players[event.player_index]
local elemName = event.element.name
local requesterName = nil
---@type OarcBuddySpawnOpts
local requesterOptions = {}
requesterOptions.surface = global.ocfg.gameplay.default_surface
if not player then
log("Another gui click happened with no valid player...")
return
end
if (player.gui.screen.buddy_request_menu == nil) then
return -- Gui event unrelated to this gui.
end
-- Check if it's a button press and lookup the matching buddy info
if ((elemName == "accept_buddy_request") or (elemName == "decline_buddy_request")) then
for name, opts in pairs(global.ocore.buddySpawnOpts --[[@as OarcBuddySpawnOptsTable]]) do
if (opts.buddyChoice == player.name) then
requesterName = name
requesterOptions = opts
end
end
-- Not sure about this error condition...
if (requesterName == nil) then
SendBroadcastMsg("Error! Invalid buddy info???")
log("Error! Invalid buddy info...")
player.gui.screen.buddy_request_menu.destroy()
DisplaySpawnOptions(player)
return
end
else
return -- Not a button click
end
-- Handle player accepted
if (elemName == "accept_buddy_request") then
if (game.players[requesterName].gui.screen.buddy_wait_menu ~= nil) then
game.players[requesterName].gui.screen.buddy_wait_menu.destroy()
end
if (player.gui.screen.buddy_request_menu ~= nil) then
player.gui.screen.buddy_request_menu.destroy()
end
-- Create a new spawn point
local newSpawn = { x = 0, y = 0 }
-- Create a new force for each player if they chose that option
if requesterOptions.teamRadioSelection == BUDDY_SPAWN_CHOICE.join_own_team then
CreatePlayerCustomForce(player)
CreatePlayerCustomForce(game.players[requesterName])
-- Create a new force for the combined players if they chose that option
elseif requesterOptions.teamRadioSelection == BUDDY_SPAWN_CHOICE.join_buddy_team then
local buddyForce = CreatePlayerCustomForce(game.players[requesterName])
player.force = buddyForce
end
---@type OarcConfigGameplaySettings
local gameplay = global.ocfg.gameplay
local surface = game.surfaces[requesterOptions.surface]
-- Find coordinates of a good place to spawn
---TODO: Add support for multiple surfaces.
if (requesterOptions.distChoice == "buddy_spawn_request_far") then
newSpawn = FindUngeneratedCoordinates(gameplay.far_spawn_min_distance,
gameplay.far_spawn_max_distance,
surface)
elseif (requesterOptions.distChoice == "buddy_spawn_request_near") then
newSpawn = FindUngeneratedCoordinates(
gameplay.near_spawn_min_distance,
gameplay.near_spawn_max_distance,
surface)
end
-- If that fails, find a random map edge in a rand direction.
if ((newSpawn.x == 0) and (newSpawn.x == 0)) then
newSpawn = FindMapEdge(GetRandomVector(), surface)
log("Resorting to find map edge! x=" .. newSpawn.x .. ",y=" .. newSpawn.y)
end
-- Create that spawn in the global vars
local buddySpawn = { x = 0, y = 0 }
if (requesterOptions.moatChoice) then
buddySpawn = {
x = newSpawn.x + (global.ocfg.surfaces_config[requesterOptions.surface].spawn_config.general.spawn_radius_tiles * 2) + 10,
y = newSpawn.y
}
else
buddySpawn = { x = newSpawn.x + (global.ocfg.surfaces_config[requesterOptions.surface].spawn_config.general.spawn_radius_tiles * 2), y = newSpawn.y }
end
ChangePlayerSpawn(player, requesterOptions.surface, newSpawn) --TODO: Add support for multiple surfaces
ChangePlayerSpawn(game.players[requesterName], requesterOptions.surface, buddySpawn)
-- Send the player there
QueuePlayerForDelayedSpawn(player.name, requesterOptions.surface, newSpawn, requesterOptions.moatChoice, false)
QueuePlayerForDelayedSpawn(requesterName, requesterOptions.surface, buddySpawn, requesterOptions.moatChoice, false)
SendBroadcastMsg(requesterName .. " and " .. player.name .. " are joining the game together!")
-- Unlock spawn control gui tab
SetOarcGuiTabEnabled(player, OARC_SPAWN_CTRL_TAB_NAME, true)
SetOarcGuiTabEnabled(game.players[requesterName], OARC_SPAWN_CTRL_TAB_NAME, true)
player.print({ "oarc-please-wait" })
player.print({ "", { "oarc-please-wait" }, "!" })
player.print({ "", { "oarc-please-wait" }, "!!" })
game.players[requesterName].print({ "oarc-please-wait" })
game.players[requesterName].print({ "", { "oarc-please-wait" }, "!" })
game.players[requesterName].print({ "", { "oarc-please-wait" }, "!!" })
global.ocore.buddyPairs[player.name] = requesterName
global.ocore.buddyPairs[requesterName] = player.name
-- Check if player is cancelling the request.
elseif (elemName == "decline_buddy_request") then
player.gui.screen.buddy_request_menu.destroy()
DisplaySpawnOptions(player)
local requesterBuddy = game.players[requesterName]
if (requesterBuddy.gui.screen.buddy_wait_menu ~= nil) then
requesterBuddy.gui.screen.buddy_wait_menu.destroy()
end
if (requesterBuddy.gui.screen.buddy_spawn ~= nil) then
requesterBuddy.gui.screen.buddy_spawn_opts.destroy()
end
DisplaySpawnOptions(requesterBuddy)
requesterBuddy.print({ "oarc-buddy-declined", player.name })
end
global.ocore.buddySpawnOpts[requesterName] = nil
end
---Display the please wait dialog
---@param player LuaPlayer
---@param delay_seconds integer
---@return nil
function DisplayPleaseWaitForSpawnDialog(player, delay_seconds)
local pleaseWaitGui = player.gui.screen.add { name = "wait_for_spawn_dialog",
type = "frame",
direction = "vertical",
caption = { "oarc-spawn-wait" } }
pleaseWaitGui.auto_center = true
pleaseWaitGui.style.maximal_width = SPAWN_GUI_MAX_WIDTH
pleaseWaitGui.style.maximal_height = SPAWN_GUI_MAX_HEIGHT
-- Warnings and explanations...
local wait_warning_text = { "oarc-wait-text", delay_seconds }
AddLabel(pleaseWaitGui, "warning_lbl1", wait_warning_text, my_warning_style)
-- Show a minimap of the spawn location :)
---@type OarcPlayerSpawn
local player_spawn = global.ocore.playerSpawns[player.name]
pleaseWaitGui.add {
type = "minimap",
position = player_spawn.position,
surface_index = game.surfaces[player_spawn.surface].index,
force = player.force.name
}
end
---Gui click event handlers
---@param event EventData.on_gui_click
---@return nil
function SeparateSpawnsGuiClick(event)
WelcomeTextGuiClick(event)
SpawnOptsGuiClick(event)
SharedSpwnOptsGuiClick(event)
BuddySpawnOptsGuiClick(event)
BuddySpawnWaitMenuClick(event)
BuddySpawnRequestMenuClick(event)
SharedSpawnJoinWaitMenuClick(event)
end
---Gui checked state changed event handlers
---@param event EventData.on_gui_checked_state_changed
---@return nil
function SeparateSpawnsGuiCheckedStateChanged(event)
SpawnOptsRadioSelect(event)
end