mirror of
https://github.com/Oarcinae/FactorioScenarioMultiplayerSpawn.git
synced 2025-01-05 22:53:48 +02:00
521 lines
16 KiB
Lua
521 lines
16 KiB
Lua
-- A nice way to organize the GUI tabs.
|
|
|
|
local mod_gui = require("mod-gui")
|
|
require("lib/gui_tabs/server_info")
|
|
require("lib/gui_tabs/spawn_controls")
|
|
require("lib/gui_tabs/settings_controls")
|
|
require("lib/gui_tabs/mod_info_faq")
|
|
require("lib/gui_tabs/player_list")
|
|
require("lib/gui_tabs/surface_config")
|
|
require("lib/gui_tabs/item_shop")
|
|
|
|
--------------------------------------------------------------------------------
|
|
-- GUI Tab Handler
|
|
--------------------------------------------------------------------------------
|
|
|
|
-- NAME of the top level element (outer frame)
|
|
OARC_GUI = "oarc_gui"
|
|
|
|
-- All tabs and their content Functions
|
|
OARC_SERVER_INFO_TAB_NAME = "server_info"
|
|
OARC_SPAWN_CTRL_TAB_NAME = "spawn_controls"
|
|
OARC_CONFIG_CTRL_TAB_NAME = "settings"
|
|
OARC_MOD_INFO_CTRL_TAB_NAME = "mod_info"
|
|
OARC_MOD_PLAYER_LIST_TAB_NAME = "player_list"
|
|
OARC_SURFACE_CONFIG_TAB_NAME = "surface_config"
|
|
OARC_ITEM_SHOP_TAB_NAME = "item_shop"
|
|
|
|
local OARC_GUI_TAB_CONTENT = {
|
|
[OARC_SERVER_INFO_TAB_NAME] = {
|
|
create_tab_function = CreateServerInfoTab,
|
|
localized_name = {"oarc-server-info-tab-title"}
|
|
},
|
|
[OARC_SPAWN_CTRL_TAB_NAME] = {
|
|
create_tab_function = CreateSpawnControlsTab,
|
|
localized_name = {"oarc-spawn-ctrls-tab-title"}
|
|
},
|
|
[OARC_MOD_INFO_CTRL_TAB_NAME] = {
|
|
create_tab_function = CreateModInfoTab,
|
|
localized_name = {"oarc-mod-info-tab-title"}
|
|
},
|
|
[OARC_CONFIG_CTRL_TAB_NAME] = {
|
|
create_tab_function = CreateSettingsControlsTab,
|
|
localized_name = {"oarc-settings-tab-title"}
|
|
},
|
|
[OARC_MOD_PLAYER_LIST_TAB_NAME] = {
|
|
create_tab_function = CreatePlayerListTab,
|
|
localized_name = {"oarc-player-list-tab-title"}
|
|
},
|
|
[OARC_SURFACE_CONFIG_TAB_NAME] = {
|
|
create_tab_function = CreateSurfaceConfigTab,
|
|
localized_name = {"oarc-surface-config-tab-title"}
|
|
},
|
|
[OARC_ITEM_SHOP_TAB_NAME] = {
|
|
create_tab_function = CreateItemShopTab,
|
|
localized_name = {"oarc-item-shop-tab-title"}
|
|
}
|
|
}
|
|
|
|
---@param player LuaPlayer
|
|
---@return nil
|
|
function InitOarcGuiTabs(player)
|
|
|
|
-- Make safe to call multiple times
|
|
if (DoesOarcGuiExist(player)) then
|
|
return
|
|
end
|
|
|
|
CreateOarcGuiButton(player)
|
|
|
|
-- Add general info tab
|
|
AddOarcGuiTab(player, OARC_SERVER_INFO_TAB_NAME)
|
|
SetOarcGuiTabEnabled(player, OARC_SERVER_INFO_TAB_NAME, true)
|
|
|
|
-- Mod Info tab
|
|
AddOarcGuiTab(player, OARC_MOD_INFO_CTRL_TAB_NAME)
|
|
SetOarcGuiTabEnabled(player, OARC_MOD_INFO_CTRL_TAB_NAME, true)
|
|
|
|
-- Spawn control tab, enable if player is already spawned
|
|
AddOarcGuiTab(player, OARC_SPAWN_CTRL_TAB_NAME)
|
|
local player_respawns = storage.player_respawns[player.name]
|
|
local spawn_enabled = (player_respawns ~= nil) and (next(player_respawns) ~= nil) -- TODO: Maybe make a specific state flag or something more explicit?
|
|
SetOarcGuiTabEnabled(player, OARC_SPAWN_CTRL_TAB_NAME, spawn_enabled)
|
|
|
|
-- Player list tab
|
|
AddOarcGuiTab(player, OARC_MOD_PLAYER_LIST_TAB_NAME)
|
|
SetOarcGuiTabEnabled(player, OARC_MOD_PLAYER_LIST_TAB_NAME, true)
|
|
|
|
-- Item shop tab
|
|
if (storage.ocfg.gameplay.enable_coin_shop) then
|
|
AddOarcGuiTab(player, OARC_ITEM_SHOP_TAB_NAME)
|
|
SetOarcGuiTabEnabled(player, OARC_ITEM_SHOP_TAB_NAME, true)
|
|
end
|
|
|
|
-- Settings control tab
|
|
AddOarcGuiTab(player, OARC_CONFIG_CTRL_TAB_NAME)
|
|
SetOarcGuiTabEnabled(player, OARC_CONFIG_CTRL_TAB_NAME, true)
|
|
|
|
-- Surface config tab
|
|
if (player.admin) then
|
|
AddOarcGuiTab(player, OARC_SURFACE_CONFIG_TAB_NAME)
|
|
SetOarcGuiTabEnabled(player, OARC_SURFACE_CONFIG_TAB_NAME, true)
|
|
end
|
|
|
|
HideOarcGui(player)
|
|
end
|
|
|
|
---@param player LuaPlayer
|
|
---@return nil
|
|
function CreateOarcGuiButton(player)
|
|
if (mod_gui.get_button_flow(player).oarc_mod_gui_button == nil) then
|
|
local b = mod_gui.get_button_flow(player).add{
|
|
name="oarc_mod_gui_button",
|
|
type="sprite-button",
|
|
sprite = "oarc-mod-sprite-40",
|
|
style="slot_button",
|
|
tooltip={ "oarc-gui-tooltip" }
|
|
}
|
|
b.style.padding=0
|
|
end
|
|
end
|
|
|
|
---@param player LuaPlayer
|
|
---@return boolean
|
|
function DoesOarcGuiExist(player)
|
|
return (mod_gui.get_frame_flow(player)[OARC_GUI] ~= nil)
|
|
end
|
|
|
|
---@param player LuaPlayer
|
|
---@return boolean
|
|
function IsOarcGuiVisible(player)
|
|
---@type LuaGuiElement
|
|
local of = mod_gui.get_frame_flow(player)[OARC_GUI]
|
|
return (of.visible)
|
|
end
|
|
|
|
---@param player LuaPlayer
|
|
---@return nil
|
|
function ShowOarcGui(player)
|
|
---@type LuaGuiElement
|
|
local of = mod_gui.get_frame_flow(player)[OARC_GUI]
|
|
if (of == nil) then return end
|
|
of.visible = true
|
|
player.opened = of
|
|
end
|
|
|
|
---@param player LuaPlayer
|
|
---@return nil
|
|
function HideOarcGui(player)
|
|
---@type LuaGuiElement
|
|
local of = mod_gui.get_frame_flow(player)[OARC_GUI]
|
|
if (of == nil) then return end
|
|
of.visible = false
|
|
player.opened = nil
|
|
end
|
|
|
|
---@param player LuaPlayer
|
|
---@return LuaGuiElement?
|
|
function GetOarcGuiTabsPane(player)
|
|
---@type LuaGuiElement
|
|
local of = mod_gui.get_frame_flow(player)[OARC_GUI]
|
|
if (of == nil) then
|
|
return nil
|
|
else
|
|
return of.oarc_if.oarc_tabs
|
|
end
|
|
end
|
|
|
|
---@param event EventData.on_gui_click
|
|
---@return nil
|
|
function ClickOarcGuiButton(event)
|
|
if not event.element.valid then return end
|
|
local player = game.players[event.player_index]
|
|
local name = event.element.name
|
|
|
|
if (name ~= "oarc_mod_gui_button") then return end
|
|
|
|
if (not DoesOarcGuiExist(player)) then
|
|
CreateOarcGuiTabsPane(player)
|
|
else
|
|
if (IsOarcGuiVisible(player)) then
|
|
HideOarcGui(player)
|
|
else
|
|
ShowOarcGui(player)
|
|
OarcGuiCreateContentOfTab(player)
|
|
end
|
|
end
|
|
end
|
|
|
|
|
|
|
|
---Set tab content to currently selected tab, clears all other tab content and refreshes the selected tab content!
|
|
---Safe to call just to refresh the current tab.
|
|
---@param player LuaPlayer
|
|
---@return nil
|
|
function OarcGuiCreateContentOfTab(player)
|
|
local otabs = GetOarcGuiTabsPane(player)
|
|
if (otabs == nil) then return end
|
|
|
|
local tab_name = otabs.tabs[otabs.selected_tab_index].tab.name
|
|
|
|
-- log("OarcGuiCreateContentOfTab: " .. tab_name)
|
|
|
|
for _,t in ipairs(otabs.tabs) do
|
|
t.content.clear()
|
|
if (t.tab.name == tab_name) then
|
|
OARC_GUI_TAB_CONTENT[tab_name].create_tab_function(t.content, player)
|
|
end
|
|
end
|
|
end
|
|
|
|
---Just an alias for OarcGuiCreateContentOfTab
|
|
---@param player LuaPlayer
|
|
---@return nil
|
|
function OarcGuiRefreshContent(player)
|
|
-- log("Hit OarcGuiRefreshContent" .. player.name)
|
|
OarcGuiCreateContentOfTab(player)
|
|
end
|
|
|
|
---@param player LuaPlayer
|
|
---@return nil
|
|
function CreateOarcGuiTabsPane(player)
|
|
|
|
if (mod_gui.get_frame_flow(player)[OARC_GUI] == nil) then
|
|
|
|
-- OUTER FRAME (TOP GUI ELEMENT)
|
|
local frame = mod_gui.get_frame_flow(player).add{
|
|
type = 'frame',
|
|
name = OARC_GUI,
|
|
direction = "vertical"}
|
|
frame.style.padding = 5
|
|
|
|
-- INNER FRAME
|
|
local inside_frame = frame.add{
|
|
type = "frame",
|
|
name = "oarc_if",
|
|
style = "inside_deep_frame",
|
|
direction = "vertical"
|
|
}
|
|
|
|
-- SUB HEADING w/ LABEL
|
|
local subhead = inside_frame.add{
|
|
type="frame",
|
|
name="sub_header",
|
|
style = "subheader_frame"
|
|
}
|
|
subhead.style.horizontally_stretchable = true
|
|
|
|
-- changelog_subheader_frame =
|
|
-- {
|
|
-- type = "frame_style",
|
|
-- parent = "subheader_frame",
|
|
-- left_padding = 12,
|
|
-- right_padding = 12,
|
|
-- top_padding = 4,
|
|
-- horizontally_stretchable = "on"
|
|
-- }
|
|
|
|
AddLabel(subhead, nil, {"oarc-gui-tab-header-label"}, "subheader_caption_label")
|
|
|
|
-- TABBED PANE
|
|
local oarc_tabs = inside_frame.add{
|
|
name="oarc_tabs",
|
|
type="tabbed-pane",
|
|
style="tabbed_pane"}
|
|
oarc_tabs.style.top_padding = 8
|
|
end
|
|
end
|
|
|
|
-- Function creates a new tab.
|
|
-- It adds whatever it wants to the provided scroll-pane.
|
|
---@param player LuaPlayer
|
|
---@param tab_name string
|
|
function AddOarcGuiTab(player, tab_name)
|
|
if (not DoesOarcGuiExist(player)) then
|
|
CreateOarcGuiTabsPane(player)
|
|
end
|
|
|
|
-- Get the tabbed pane
|
|
local otabs = GetOarcGuiTabsPane(player)
|
|
|
|
if (otabs == nil) then return end
|
|
|
|
-- Create new tab
|
|
local new_tab = otabs.add{
|
|
type="tab",
|
|
name=tab_name,
|
|
caption=OARC_GUI_TAB_CONTENT[tab_name].localized_name,}
|
|
|
|
-- Create inside frame for content
|
|
local tab_inside_frame = otabs.add{
|
|
type="frame",
|
|
name=tab_name.."_if",
|
|
style = "inside_deep_frame",
|
|
direction="vertical"}
|
|
tab_inside_frame.style.left_margin = 10
|
|
tab_inside_frame.style.right_margin = 10
|
|
tab_inside_frame.style.top_margin = 4
|
|
tab_inside_frame.style.bottom_margin = 4
|
|
tab_inside_frame.style.padding = 5
|
|
tab_inside_frame.style.horizontally_stretchable = true
|
|
-- tab_inside_frame.style.vertically_stretchable = true
|
|
-- tab_inside_frame.style.horizontally_squashable = true
|
|
-- tab_inside_frame.style.vertically_squashable = true
|
|
|
|
-- Add the whole thing to the tab now.
|
|
otabs.add_tab(new_tab, tab_inside_frame)
|
|
|
|
-- Disable all new tabs by default
|
|
new_tab.enabled = false
|
|
|
|
-- If no other tabs are selected, select the first one.
|
|
if (otabs.selected_tab_index == nil) then
|
|
otabs.selected_tab_index = 1
|
|
end
|
|
end
|
|
|
|
-- https://forums.factorio.com/viewtopic.php?f=7&t=115901
|
|
-- ---Removes a tab from the GUI.
|
|
-- ---@param player LuaPlayer
|
|
-- ---@param tab_name string
|
|
-- function RemoveOarcGuiTab(player, tab_name)
|
|
-- if (not DoesOarcGuiExist(player)) then return end
|
|
|
|
-- local otabs = GetOarcGuiTabsPane(player)
|
|
|
|
-- local selected_tab_name = otabs.tabs[otabs.selected_tab_index].tab.name
|
|
|
|
-- for _,t in ipairs(otabs.tabs) do
|
|
-- if (t.tab.name == tab_name) then
|
|
|
|
-- local tab = t.tab
|
|
-- local content = t.content
|
|
-- otabs.remove_tab(t.tab)
|
|
-- tab.destroy()
|
|
-- content.destroy()
|
|
|
|
-- --TODO: I haven't figured out how to do this nicely, but removing tabs fucks up the tab view.
|
|
-- -- So for now, we just recreate the GUI.
|
|
|
|
-- OarcGuiCreateContentOfTab(player)
|
|
|
|
-- return
|
|
-- end
|
|
-- end
|
|
-- end
|
|
|
|
---Check if tab exists in the GUI.
|
|
---@param player LuaPlayer
|
|
---@param tab_name string
|
|
---@return boolean
|
|
function DoesOarcGuiTabExist(player, tab_name)
|
|
if (not DoesOarcGuiExist(player)) then return false end
|
|
|
|
local otabs = GetOarcGuiTabsPane(player)
|
|
|
|
for _,t in ipairs(otabs.tabs) do
|
|
if (t.tab.name == tab_name) then
|
|
return true
|
|
end
|
|
end
|
|
|
|
return false
|
|
end
|
|
|
|
---This sets the enable state of a tab.
|
|
---@param player LuaPlayer
|
|
---@param tab_name string
|
|
---@param enable boolean
|
|
function SetOarcGuiTabEnabled(player, tab_name, enable)
|
|
if (not DoesOarcGuiExist(player)) then return end
|
|
|
|
local otabs = GetOarcGuiTabsPane(player)
|
|
|
|
for _,t in ipairs(otabs.tabs) do
|
|
if (t.tab.name == tab_name) then
|
|
t.tab.enabled = enable
|
|
return
|
|
end
|
|
end
|
|
end
|
|
|
|
---Switches the tab to the one specified.
|
|
---@param player LuaPlayer
|
|
---@param tab_name string
|
|
function SwitchOarcGuiTab(player, tab_name)
|
|
if (not DoesOarcGuiExist(player)) then return end
|
|
|
|
local otabs = GetOarcGuiTabsPane(player)
|
|
|
|
for i,t in pairs(otabs.tabs) do
|
|
if (t.tab.name == tab_name) then
|
|
otabs.selected_tab_index = i
|
|
OarcGuiCreateContentOfTab(player)
|
|
return
|
|
end
|
|
end
|
|
end
|
|
|
|
---Completely destroys and recreates the OARC GUI for a player.
|
|
---@param player LuaPlayer
|
|
---@return nil
|
|
function RecreateOarcGui(player)
|
|
if (mod_gui.get_button_flow(player).oarc_button ~= nil) then
|
|
mod_gui.get_button_flow(player).oarc_button.destroy()
|
|
end
|
|
|
|
if (mod_gui.get_frame_flow(player)[OARC_GUI] ~= nil) then
|
|
mod_gui.get_frame_flow(player)[OARC_GUI].destroy()
|
|
end
|
|
|
|
InitOarcGuiTabs(player)
|
|
end
|
|
|
|
---Add or remove a tab for all players.
|
|
---@param tab_name string
|
|
---@param add boolean If true, add the tab. If false, remove the tab.
|
|
---@param enable boolean If true, enables the tab. If false, disable the tab. Only used if adding the tab.
|
|
---@return nil
|
|
function AddRemoveOarcGuiTabForAllPlayers(tab_name, add, enable)
|
|
for _,player in pairs(game.players) do
|
|
if (add and not DoesOarcGuiTabExist(player, tab_name)) then
|
|
AddOarcGuiTab(player, tab_name)
|
|
SetOarcGuiTabEnabled(player, tab_name, true)
|
|
elseif (not add and DoesOarcGuiTabExist(player, tab_name)) then
|
|
-- RemoveOarcGuiTab(player, tab_name) -- TODO: SEE https://forums.factorio.com/viewtopic.php?f=7&t=115901
|
|
RecreateOarcGui(player) -- Assumes a ocfg setting change, so just recreate the whole thing.
|
|
end
|
|
end
|
|
end
|
|
|
|
--[[
|
|
_____ _____ _ _ _____ _ _ _ _ _ ___ _ ___ ___ ___
|
|
| __\ \ / / __| \| |_ _| | || | /_\ | \| | \| | | __| _ \/ __|
|
|
| _| \ V /| _|| .` | | | | __ |/ _ \| .` | |) | |__| _|| /\__ \
|
|
|___| \_/ |___|_|\_| |_| |_||_/_/ \_\_|\_|___/|____|___|_|_\|___/
|
|
|
|
]]
|
|
|
|
---Handles the closing of the OARC GUI.
|
|
---@param event EventData.on_gui_closed
|
|
---@return nil
|
|
function OarcGuiClosed(event)
|
|
if (event.element and (event.element.name == "oarc_gui")) then
|
|
HideOarcGui(game.players[event.player_index])
|
|
end
|
|
end
|
|
|
|
---@param event EventData.on_gui_selected_tab_changed
|
|
---@return nil
|
|
function OarcGuiTabsSelectedTabChanged(event)
|
|
if (event.element.name ~= "oarc_tabs") then return end
|
|
OarcGuiCreateContentOfTab(game.players[event.player_index])
|
|
end
|
|
|
|
---All gui tabs click event handler
|
|
---@param event EventData.on_gui_click
|
|
---@return nil
|
|
function OarcGuiTabsClick(event)
|
|
if not event.element.valid then return end
|
|
ClickOarcGuiButton(event)
|
|
ServerInfoTabGuiClick(event)
|
|
SpawnCtrlTabGuiClick(event)
|
|
SettingsControlsTabGuiClick(event)
|
|
SettingsSurfaceControlsTabGuiClick(event)
|
|
PlayerListTabGuiClick(event)
|
|
SurfaceConfigTabGuiClick(event)
|
|
OarcItemShopGuiClick(event)
|
|
end
|
|
|
|
---All gui tabs on_gui_checked_state_changed event handler
|
|
---@param event EventData.on_gui_checked_state_changed
|
|
---@return nil
|
|
function OarcGuiTabsCheckedStateChanged(event)
|
|
if not event.element.valid then return end
|
|
SpawnCtrlGuiOptionsCheckedStateChanged(event)
|
|
end
|
|
|
|
|
|
---Handles the `on_gui_value_changed` event.
|
|
---@param event EventData.on_gui_value_changed
|
|
---@return nil
|
|
function OarcGuiTabsValueChanged(event)
|
|
if not event.element.valid then return end
|
|
SettingsControlsTabGuiValueChanged(event)
|
|
end
|
|
|
|
---Handles the `on_gui_selection_state_changed` event.
|
|
---@param event EventData.on_gui_selection_state_changed
|
|
---@return nil
|
|
function OarcGuiTabsSelectionStateChanged(event)
|
|
if not event.element.valid then return end
|
|
SettingsControlsTabGuiSelectionStateChanged(event)
|
|
SurfaceConfigTabGuiSelect(event)
|
|
end
|
|
|
|
---Handles the `on_gui_text_changed` event.
|
|
---@param event EventData.on_gui_text_changed
|
|
---@return nil
|
|
function OarcGuiTabsTextChanged(event)
|
|
if not event.element.valid then return end
|
|
SettingsControlsTabGuiTextChanged(event)
|
|
SurfaceConfigTabGuiTextChanged(event)
|
|
end
|
|
|
|
---Handles the `on_gui_confirmed` event.
|
|
---@param event EventData.on_gui_confirmed
|
|
---@return nil
|
|
function OarcGuiTabsConfirmed(event)
|
|
if not event.element.valid then return end
|
|
SettingsControlsTabGuiTextconfirmed(event)
|
|
SurfaceConfigTabGuiConfirmed(event)
|
|
end
|
|
|
|
---Handles the `on_gui_elem_changed` event.
|
|
---@param event EventData.on_gui_elem_changed
|
|
---@return nil
|
|
function OarcGuiTabsElemChanged(event)
|
|
if not event.element.valid then return end
|
|
SurfaceConfigTabGuiElemChanged(event)
|
|
end
|