mirror of
https://github.com/luanti-org/luanti.git
synced 2025-10-24 05:15:22 +02:00
476 lines
13 KiB
Lua
476 lines
13 KiB
Lua
-- Luanti
|
|
-- Copyright (C) 2014 sapier
|
|
-- SPDX-License-Identifier: LGPL-2.1-or-later
|
|
|
|
|
|
local current_game, singleplayer_refresh_gamebar
|
|
local valid_disabled_settings = {
|
|
["enable_damage"]=true,
|
|
["creative_mode"]=true,
|
|
["enable_server"]=true,
|
|
}
|
|
|
|
-- Name and port stored to persist when updating the formspec
|
|
local current_name = core.settings:get("name")
|
|
local current_port = core.settings:get("port")
|
|
|
|
-- Currently chosen game in gamebar for theming and filtering
|
|
function current_game()
|
|
local gameid = core.settings:get("menu_last_game")
|
|
local game = gameid and pkgmgr.find_by_gameid(gameid)
|
|
-- Fall back to first game installed if one exists.
|
|
if not game and #pkgmgr.games > 0 then
|
|
|
|
-- If devtest is the first game in the list and there is another
|
|
-- game available, pick the other game instead.
|
|
local picked_game
|
|
if pkgmgr.games[1].id == "devtest" and #pkgmgr.games > 1 then
|
|
picked_game = 2
|
|
else
|
|
picked_game = 1
|
|
end
|
|
|
|
game = pkgmgr.games[picked_game]
|
|
gameid = game.id
|
|
core.settings:set("menu_last_game", gameid)
|
|
end
|
|
|
|
return game
|
|
end
|
|
|
|
-- Apply menu changes from given game
|
|
function apply_game(game)
|
|
core.settings:set("menu_last_game", game.id)
|
|
menudata.worldlist:set_filtercriteria(game.id)
|
|
|
|
mm_game_theme.set_game(game)
|
|
|
|
local index = filterlist.get_current_index(menudata.worldlist,
|
|
tonumber(core.settings:get("mainmenu_last_selected_world")))
|
|
if not index or index < 1 then
|
|
local selected = core.get_textlist_index("sp_worlds")
|
|
if selected ~= nil and selected < #menudata.worldlist:get_list() then
|
|
index = selected
|
|
else
|
|
index = #menudata.worldlist:get_list()
|
|
end
|
|
end
|
|
menu_worldmt_legacy(index)
|
|
end
|
|
|
|
function singleplayer_refresh_gamebar()
|
|
|
|
local old_bar = ui.find_by_name("game_button_bar")
|
|
if old_bar ~= nil then
|
|
old_bar:delete()
|
|
end
|
|
|
|
-- Hide gamebar if no games are installed
|
|
if #pkgmgr.games == 0 then
|
|
return false
|
|
end
|
|
|
|
local function game_buttonbar_button_handler(fields)
|
|
for _, game in ipairs(pkgmgr.games) do
|
|
if fields["game_btnbar_" .. game.id] then
|
|
apply_game(game)
|
|
return true
|
|
end
|
|
end
|
|
end
|
|
|
|
local TOUCH_GUI = core.settings:get_bool("touch_gui")
|
|
|
|
local gamebar_pos_y = MAIN_TAB_H
|
|
+ TABHEADER_H -- tabheader included in formspec size
|
|
+ (TOUCH_GUI and GAMEBAR_OFFSET_TOUCH or GAMEBAR_OFFSET_DESKTOP)
|
|
|
|
local btnbar = buttonbar_create(
|
|
"game_button_bar",
|
|
{x = 0, y = gamebar_pos_y},
|
|
{x = MAIN_TAB_W, y = GAMEBAR_H},
|
|
"#000000",
|
|
game_buttonbar_button_handler)
|
|
|
|
for _, game in ipairs(pkgmgr.games) do
|
|
local btn_name = "game_btnbar_" .. game.id
|
|
|
|
local image = nil
|
|
local text = nil
|
|
local tooltip = core.formspec_escape(game.title)
|
|
|
|
if (game.menuicon_path or "") ~= "" then
|
|
image = core.formspec_escape(game.menuicon_path)
|
|
else
|
|
local part1 = game.id:sub(1,5)
|
|
local part2 = game.id:sub(6,10)
|
|
local part3 = game.id:sub(11)
|
|
|
|
text = part1 .. "\n" .. part2
|
|
if part3 ~= "" then
|
|
text = text .. "\n" .. part3
|
|
end
|
|
end
|
|
btnbar:add_button(btn_name, text, image, tooltip)
|
|
end
|
|
|
|
local plus_image = core.formspec_escape(defaulttexturedir .. "plus.png")
|
|
btnbar:add_button("game_open_cdb", "", plus_image, fgettext("Install games from ContentDB"))
|
|
return true
|
|
end
|
|
|
|
local function get_disabled_settings(game)
|
|
if not game then
|
|
return {}
|
|
end
|
|
|
|
local gameconfig = Settings(game.path .. "/game.conf")
|
|
local disabled_settings = {}
|
|
if gameconfig then
|
|
local disabled_settings_str = (gameconfig:get("disabled_settings") or ""):split()
|
|
for _, value in pairs(disabled_settings_str) do
|
|
local state = false
|
|
value = value:trim()
|
|
if string.sub(value, 1, 1) == "!" then
|
|
state = true
|
|
value = string.sub(value, 2)
|
|
end
|
|
if valid_disabled_settings[value] then
|
|
disabled_settings[value] = state
|
|
else
|
|
core.log("error", "Invalid disabled setting in game.conf: "..tostring(value))
|
|
end
|
|
end
|
|
end
|
|
return disabled_settings
|
|
end
|
|
|
|
local function get_formspec(tabview, name, tabdata)
|
|
|
|
-- Point the player to ContentDB when no games are found
|
|
if #pkgmgr.games == 0 then
|
|
local W = tabview.width
|
|
local H = tabview.height
|
|
|
|
local hypertext = "<global valign=middle halign=center size=18>" ..
|
|
fgettext_ne("Luanti is a game-creation platform that allows you to play many different games.") .. "\n" ..
|
|
fgettext_ne("Luanti doesn't come with a game by default.") .. " " ..
|
|
fgettext_ne("You need to install a game before you can create a world.")
|
|
|
|
local button_y = H * 2/3 - 0.6
|
|
return table.concat({
|
|
"hypertext[0.375,0;", W - 2*0.375, ",", button_y, ";ht;", core.formspec_escape(hypertext), "]",
|
|
"button[5.25,", button_y, ";5,1.2;game_open_cdb;", fgettext("Install a game"), "]"})
|
|
end
|
|
|
|
local retval = ""
|
|
|
|
local index = core.get_textlist_index("sp_worlds") or filterlist.get_current_index(menudata.worldlist,
|
|
tonumber(core.settings:get("mainmenu_last_selected_world"))) or 0
|
|
|
|
local list = menudata.worldlist:get_list()
|
|
-- When changing tabs to a world list with fewer entries, the last index is selected (visually).
|
|
-- However, the formspec fields lag behind, thus 'index > #list' can be a valid choice.
|
|
local world = list and list[math.min(index, #list)]
|
|
local game
|
|
|
|
if world then
|
|
game = pkgmgr.find_by_gameid(world.gameid)
|
|
else
|
|
game = current_game()
|
|
end
|
|
local disabled_settings = get_disabled_settings(game)
|
|
|
|
local creative, damage, host = "", "", ""
|
|
|
|
-- Y offsets for game settings checkboxes
|
|
local y = 0.2
|
|
local yo = 0.5625
|
|
|
|
if world then
|
|
if disabled_settings["creative_mode"] == nil then
|
|
creative = "checkbox[0,"..y..";cb_creative_mode;".. fgettext("Creative Mode") .. ";" ..
|
|
dump(core.settings:get_bool("creative_mode")) .. "]"
|
|
y = y + yo
|
|
end
|
|
if disabled_settings["enable_damage"] == nil then
|
|
damage = "checkbox[0,"..y..";cb_enable_damage;".. fgettext("Enable Damage") .. ";" ..
|
|
dump(core.settings:get_bool("enable_damage")) .. "]"
|
|
y = y + yo
|
|
end
|
|
if disabled_settings["enable_server"] == nil then
|
|
host = "checkbox[0,"..y..";cb_server;".. fgettext("Host Server") ..";" ..
|
|
dump(core.settings:get_bool("enable_server")) .. "]"
|
|
y = y + yo
|
|
end
|
|
end
|
|
|
|
retval = retval ..
|
|
"container[5.25,4.875]" ..
|
|
"button[6.65,0;3.225,0.8;world_create;".. fgettext("New") .. "]"
|
|
if world then
|
|
retval = retval ..
|
|
"button[0,0;3.225,0.8;world_delete;".. fgettext("Delete") .. "]" ..
|
|
"button[3.325,0;3.225,0.8;world_configure;".. fgettext("Select Mods") .. "]"
|
|
end
|
|
retval = retval ..
|
|
"container_end[]" ..
|
|
"container[0.375,0.375]" ..
|
|
creative ..
|
|
damage ..
|
|
host ..
|
|
"container_end[]" ..
|
|
"container[5.25,0.375]" ..
|
|
"label[0,0.2;".. fgettext("Select World:") .. "]"..
|
|
"textlist[0,0.5;9.875,3.9;sp_worlds;" ..
|
|
menu_render_worldlist() ..
|
|
";" .. index .. "]" ..
|
|
"container_end[]"
|
|
|
|
if core.settings:get_bool("enable_server") and disabled_settings["enable_server"] == nil then
|
|
retval = retval ..
|
|
"button[10.1875,5.925;4.9375,0.8;play;".. fgettext("Host Game") .. "]" ..
|
|
"container[0.375,0.375]" ..
|
|
"checkbox[0,"..y..";cb_server_announce;" .. fgettext("Announce Server") .. ";" ..
|
|
dump(core.settings:get_bool("server_announce")) .. "]"
|
|
|
|
-- Reset y so that the text fields always start at the same position,
|
|
-- regardless of whether some of the checkboxes are hidden.
|
|
y = 0.2 + 4 * yo + 0.35
|
|
|
|
retval = retval .. "field[0," .. y .. ";4.5,0.75;te_playername;" .. fgettext("Name") .. ";" ..
|
|
core.formspec_escape(current_name) .. "]"
|
|
|
|
y = y + 1.15 + 0.25
|
|
|
|
retval = retval .. "pwdfield[0," .. y .. ";4.5,0.75;te_passwd;" .. fgettext("Password") .. "]"
|
|
|
|
y = y + 1.15 + 0.25
|
|
|
|
local bind_addr = core.settings:get("bind_address")
|
|
if bind_addr ~= nil and bind_addr ~= "" then
|
|
retval = retval ..
|
|
"field[0," .. y .. ";3,0.75;te_serveraddr;" .. fgettext("Bind Address") .. ";" ..
|
|
core.formspec_escape(core.settings:get("bind_address")) .. "]" ..
|
|
"field[3.25," .. y .. ";1.25,0.75;te_serverport;" .. fgettext("Port") .. ";" ..
|
|
core.formspec_escape(current_port) .. "]"
|
|
else
|
|
retval = retval ..
|
|
"field[0," .. y .. ";4.5,0.75;te_serverport;" .. fgettext("Server Port") .. ";" ..
|
|
core.formspec_escape(current_port) .. "]"
|
|
end
|
|
|
|
retval = retval .. "container_end[]"
|
|
elseif world then
|
|
retval = retval ..
|
|
"button[10.1875,5.925;4.9375,0.8;play;" .. fgettext("Play Game") .. "]"
|
|
end
|
|
|
|
return retval
|
|
end
|
|
|
|
local function main_button_handler(this, fields, name, tabdata)
|
|
|
|
assert(name == "local")
|
|
|
|
if fields.game_open_cdb then
|
|
local maintab = ui.find_by_name("maintab")
|
|
local dlg = create_contentdb_dlg("game")
|
|
dlg:set_parent(maintab)
|
|
maintab:hide()
|
|
dlg:show()
|
|
return true
|
|
end
|
|
|
|
if this.dlg_create_world_closed_at == nil then
|
|
this.dlg_create_world_closed_at = 0
|
|
end
|
|
|
|
local world_doubleclick = false
|
|
|
|
if fields["te_playername"] then
|
|
current_name = fields["te_playername"]
|
|
end
|
|
|
|
if fields["te_serverport"] then
|
|
current_port = fields["te_serverport"]
|
|
end
|
|
|
|
if fields["sp_worlds"] ~= nil then
|
|
local event = core.explode_textlist_event(fields["sp_worlds"])
|
|
local selected = core.get_textlist_index("sp_worlds")
|
|
|
|
menu_worldmt_legacy(selected)
|
|
|
|
if event.type == "DCL" then
|
|
world_doubleclick = true
|
|
end
|
|
|
|
if event.type == "CHG" and selected ~= nil then
|
|
core.settings:set("mainmenu_last_selected_world",
|
|
menudata.worldlist:get_raw_index(selected))
|
|
return true
|
|
end
|
|
end
|
|
|
|
if menu_handle_key_up_down(fields,"sp_worlds","mainmenu_last_selected_world") then
|
|
return true
|
|
end
|
|
|
|
if fields["cb_creative_mode"] then
|
|
core.settings:set("creative_mode", fields["cb_creative_mode"])
|
|
local selected = core.get_textlist_index("sp_worlds")
|
|
menu_worldmt(selected, "creative_mode", fields["cb_creative_mode"])
|
|
|
|
return true
|
|
end
|
|
|
|
if fields["cb_enable_damage"] then
|
|
core.settings:set("enable_damage", fields["cb_enable_damage"])
|
|
local selected = core.get_textlist_index("sp_worlds")
|
|
menu_worldmt(selected, "enable_damage", fields["cb_enable_damage"])
|
|
|
|
return true
|
|
end
|
|
|
|
if fields["cb_server"] then
|
|
core.settings:set("enable_server", fields["cb_server"])
|
|
|
|
return true
|
|
end
|
|
|
|
if fields["cb_server_announce"] then
|
|
core.settings:set("server_announce", fields["cb_server_announce"])
|
|
local selected = core.get_textlist_index("srv_worlds")
|
|
menu_worldmt(selected, "server_announce", fields["cb_server_announce"])
|
|
|
|
return true
|
|
end
|
|
|
|
if fields["play"] ~= nil or world_doubleclick or fields["key_enter"] then
|
|
local enter_key_duration = core.get_us_time() - this.dlg_create_world_closed_at
|
|
if world_doubleclick and enter_key_duration <= 200000 then -- 200 ms
|
|
this.dlg_create_world_closed_at = 0
|
|
return true
|
|
end
|
|
|
|
local selected = core.get_textlist_index("sp_worlds")
|
|
gamedata.selected_world = menudata.worldlist:get_raw_index(selected)
|
|
|
|
if selected == nil or gamedata.selected_world == 0 then
|
|
return true
|
|
end
|
|
|
|
-- Update last game
|
|
local world = menudata.worldlist:get_raw_element(gamedata.selected_world)
|
|
local game_obj
|
|
if world then
|
|
game_obj = pkgmgr.find_by_gameid(world.gameid)
|
|
core.settings:set("menu_last_game", game_obj.id)
|
|
end
|
|
|
|
local disabled_settings = get_disabled_settings(game_obj)
|
|
for k, _ in pairs(valid_disabled_settings) do
|
|
local v = disabled_settings[k]
|
|
if v ~= nil then
|
|
if k == "enable_server" and v == true then
|
|
error("Setting 'enable_server' cannot be force-enabled! The game.conf needs to be fixed.")
|
|
end
|
|
core.settings:set_bool(k, disabled_settings[k])
|
|
end
|
|
end
|
|
|
|
if core.settings:get_bool("enable_server") then
|
|
gamedata.playername = fields["te_playername"]
|
|
gamedata.password = fields["te_passwd"]
|
|
gamedata.port = fields["te_serverport"]
|
|
gamedata.address = ""
|
|
|
|
core.settings:set("port",gamedata.port)
|
|
if fields["te_serveraddr"] ~= nil then
|
|
core.settings:set("bind_address",fields["te_serveraddr"])
|
|
end
|
|
else
|
|
gamedata.singleplayer = true
|
|
end
|
|
|
|
core.start()
|
|
return true
|
|
end
|
|
|
|
if fields["world_create"] ~= nil then
|
|
this.dlg_create_world_closed_at = 0
|
|
local create_world_dlg = create_create_world_dlg()
|
|
create_world_dlg:set_parent(this)
|
|
this:hide()
|
|
create_world_dlg:show()
|
|
return true
|
|
end
|
|
|
|
if fields["world_delete"] ~= nil then
|
|
local selected = core.get_textlist_index("sp_worlds")
|
|
if selected ~= nil and
|
|
selected <= menudata.worldlist:size() then
|
|
local world = menudata.worldlist:get_list()[selected]
|
|
if world ~= nil and
|
|
world.name ~= nil and
|
|
world.name ~= "" then
|
|
local index = menudata.worldlist:get_raw_index(selected)
|
|
local delete_world_dlg = create_delete_world_dlg(world.name,index)
|
|
delete_world_dlg:set_parent(this)
|
|
this:hide()
|
|
delete_world_dlg:show()
|
|
end
|
|
end
|
|
|
|
return true
|
|
end
|
|
|
|
if fields["world_configure"] ~= nil then
|
|
local selected = core.get_textlist_index("sp_worlds")
|
|
if selected ~= nil then
|
|
local configdialog =
|
|
create_configure_world_dlg(
|
|
menudata.worldlist:get_raw_index(selected))
|
|
|
|
if (configdialog ~= nil) then
|
|
configdialog:set_parent(this)
|
|
this:hide()
|
|
configdialog:show()
|
|
end
|
|
end
|
|
|
|
return true
|
|
end
|
|
end
|
|
|
|
local function on_change(type)
|
|
if type == "ENTER" then
|
|
local game = current_game()
|
|
if game then
|
|
apply_game(game)
|
|
else
|
|
mm_game_theme.set_engine()
|
|
end
|
|
|
|
if singleplayer_refresh_gamebar() then
|
|
ui.find_by_name("game_button_bar"):show()
|
|
end
|
|
elseif type == "LEAVE" then
|
|
menudata.worldlist:set_filtercriteria(nil)
|
|
local gamebar = ui.find_by_name("game_button_bar")
|
|
if gamebar then
|
|
gamebar:hide()
|
|
end
|
|
end
|
|
end
|
|
|
|
--------------------------------------------------------------------------------
|
|
return {
|
|
name = "local",
|
|
caption = fgettext("Start Game"),
|
|
cbf_formspec = get_formspec,
|
|
cbf_button_handler = main_button_handler,
|
|
on_change = on_change
|
|
}
|