2013-06-23 18:30:21 +02:00
|
|
|
--Minetest
|
|
|
|
--Copyright (C) 2013 sapier
|
|
|
|
--
|
|
|
|
--This program is free software; you can redistribute it and/or modify
|
|
|
|
--it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
--the Free Software Foundation; either version 2.1 of the License, or
|
|
|
|
--(at your option) any later version.
|
|
|
|
--
|
|
|
|
--This program is distributed in the hope that it will be useful,
|
|
|
|
--but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
--MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
--GNU Lesser General Public License for more details.
|
|
|
|
--
|
|
|
|
--You should have received a copy of the GNU Lesser General Public License along
|
|
|
|
--with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
--51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2020-03-06 21:19:14 +01:00
|
|
|
local function get_last_folder(text,count)
|
|
|
|
local parts = text:split(DIR_DELIM)
|
|
|
|
|
|
|
|
if count == nil then
|
|
|
|
return parts[#parts]
|
|
|
|
end
|
|
|
|
|
|
|
|
local retval = ""
|
|
|
|
for i=1,count,1 do
|
|
|
|
retval = retval .. parts[#parts - (count-i)] .. DIR_DELIM
|
|
|
|
end
|
|
|
|
|
|
|
|
return retval
|
|
|
|
end
|
|
|
|
|
|
|
|
local function cleanup_path(temppath)
|
|
|
|
|
|
|
|
local parts = temppath:split("-")
|
|
|
|
temppath = ""
|
|
|
|
for i=1,#parts,1 do
|
|
|
|
if temppath ~= "" then
|
|
|
|
temppath = temppath .. "_"
|
|
|
|
end
|
|
|
|
temppath = temppath .. parts[i]
|
|
|
|
end
|
|
|
|
|
|
|
|
parts = temppath:split(".")
|
|
|
|
temppath = ""
|
|
|
|
for i=1,#parts,1 do
|
|
|
|
if temppath ~= "" then
|
|
|
|
temppath = temppath .. "_"
|
|
|
|
end
|
|
|
|
temppath = temppath .. parts[i]
|
|
|
|
end
|
|
|
|
|
|
|
|
parts = temppath:split("'")
|
|
|
|
temppath = ""
|
|
|
|
for i=1,#parts,1 do
|
|
|
|
if temppath ~= "" then
|
|
|
|
temppath = temppath .. ""
|
|
|
|
end
|
|
|
|
temppath = temppath .. parts[i]
|
|
|
|
end
|
|
|
|
|
|
|
|
parts = temppath:split(" ")
|
|
|
|
temppath = ""
|
|
|
|
for i=1,#parts,1 do
|
|
|
|
if temppath ~= "" then
|
|
|
|
temppath = temppath
|
|
|
|
end
|
|
|
|
temppath = temppath .. parts[i]
|
|
|
|
end
|
|
|
|
|
|
|
|
return temppath
|
|
|
|
end
|
|
|
|
|
2021-01-21 20:01:37 +01:00
|
|
|
local function load_texture_packs(txtpath, retval)
|
|
|
|
local list = core.get_dir_list(txtpath, true)
|
|
|
|
local current_texture_path = core.settings:get("texture_path")
|
|
|
|
|
|
|
|
for _, item in ipairs(list) do
|
|
|
|
if item ~= "base" then
|
|
|
|
local path = txtpath .. DIR_DELIM .. item .. DIR_DELIM
|
|
|
|
local conf = Settings(path .. "texture_pack.conf")
|
2022-04-24 23:59:19 +02:00
|
|
|
local enabled = path == current_texture_path
|
2022-04-24 21:09:11 +02:00
|
|
|
|
2022-05-08 00:05:24 +02:00
|
|
|
local title = conf:get("title") or item
|
2021-01-21 20:01:37 +01:00
|
|
|
|
2022-05-08 00:05:24 +02:00
|
|
|
-- list_* is only used if non-nil, else the regular versions are used.
|
2021-01-21 20:01:37 +01:00
|
|
|
retval[#retval + 1] = {
|
|
|
|
name = item,
|
2022-04-24 21:09:11 +02:00
|
|
|
title = title,
|
|
|
|
list_name = enabled and fgettext("$1 (Enabled)", item) or nil,
|
|
|
|
list_title = enabled and fgettext("$1 (Enabled)", title) or nil,
|
2021-01-21 20:01:37 +01:00
|
|
|
author = conf:get("author"),
|
2021-02-28 18:10:40 +01:00
|
|
|
release = tonumber(conf:get("release")) or 0,
|
2021-01-21 20:01:37 +01:00
|
|
|
type = "txp",
|
|
|
|
path = path,
|
2022-04-24 21:09:11 +02:00
|
|
|
enabled = enabled,
|
2021-01-21 20:01:37 +01:00
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-01-30 23:40:53 +01:00
|
|
|
function get_mods(path, virtual_path, retval, modpack)
|
2015-05-04 20:59:13 +02:00
|
|
|
local mods = core.get_dir_list(path, true)
|
2016-12-28 00:26:36 +01:00
|
|
|
|
2015-12-07 04:49:01 +01:00
|
|
|
for _, name in ipairs(mods) do
|
|
|
|
if name:sub(1, 1) ~= "." then
|
2022-01-30 23:40:53 +01:00
|
|
|
local mod_path = path .. DIR_DELIM .. name
|
|
|
|
local mod_virtual_path = virtual_path .. "/" .. name
|
2019-02-03 02:49:53 +01:00
|
|
|
local toadd = {
|
|
|
|
dir_name = name,
|
|
|
|
parent_dir = path,
|
|
|
|
}
|
2016-03-06 16:53:45 +01:00
|
|
|
retval[#retval + 1] = toadd
|
2013-06-23 18:30:21 +02:00
|
|
|
|
2019-01-06 10:23:35 +01:00
|
|
|
-- Get config file
|
|
|
|
local mod_conf
|
2022-01-30 23:40:53 +01:00
|
|
|
local modpack_conf = io.open(mod_path .. DIR_DELIM .. "modpack.conf")
|
2019-01-06 10:23:35 +01:00
|
|
|
if modpack_conf then
|
|
|
|
toadd.is_modpack = true
|
|
|
|
modpack_conf:close()
|
|
|
|
|
2022-01-30 23:40:53 +01:00
|
|
|
mod_conf = Settings(mod_path .. DIR_DELIM .. "modpack.conf"):to_table()
|
2019-01-06 10:23:35 +01:00
|
|
|
if mod_conf.name then
|
|
|
|
name = mod_conf.name
|
2019-02-03 02:49:53 +01:00
|
|
|
toadd.is_name_explicit = true
|
2019-01-06 10:23:35 +01:00
|
|
|
end
|
|
|
|
else
|
2022-01-30 23:40:53 +01:00
|
|
|
mod_conf = Settings(mod_path .. DIR_DELIM .. "mod.conf"):to_table()
|
2019-01-06 10:23:35 +01:00
|
|
|
if mod_conf.name then
|
|
|
|
name = mod_conf.name
|
2019-02-03 02:49:53 +01:00
|
|
|
toadd.is_name_explicit = true
|
2019-01-06 10:23:35 +01:00
|
|
|
end
|
2014-12-07 11:14:52 +01:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2019-01-06 10:23:35 +01:00
|
|
|
-- Read from config
|
2015-12-07 04:49:01 +01:00
|
|
|
toadd.name = name
|
2022-04-24 21:09:11 +02:00
|
|
|
toadd.title = mod_conf.title
|
2018-04-17 15:54:50 +02:00
|
|
|
toadd.author = mod_conf.author
|
2021-02-28 18:10:40 +01:00
|
|
|
toadd.release = tonumber(mod_conf.release) or 0
|
2022-01-30 23:40:53 +01:00
|
|
|
toadd.path = mod_path
|
|
|
|
toadd.virtual_path = mod_virtual_path
|
2018-04-17 15:54:50 +02:00
|
|
|
toadd.type = "mod"
|
2015-12-07 04:49:01 +01:00
|
|
|
|
2019-01-06 10:23:35 +01:00
|
|
|
-- Check modpack.txt
|
2021-01-21 20:01:37 +01:00
|
|
|
-- Note: modpack.conf is already checked above
|
2022-01-30 23:40:53 +01:00
|
|
|
local modpackfile = io.open(mod_path .. DIR_DELIM .. "modpack.txt")
|
2019-01-06 10:23:35 +01:00
|
|
|
if modpackfile then
|
|
|
|
modpackfile:close()
|
|
|
|
toadd.is_modpack = true
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Deal with modpack contents
|
|
|
|
if modpack and modpack ~= "" then
|
2015-12-07 04:49:01 +01:00
|
|
|
toadd.modpack = modpack
|
2019-01-06 10:23:35 +01:00
|
|
|
elseif toadd.is_modpack then
|
|
|
|
toadd.type = "modpack"
|
|
|
|
toadd.is_modpack = true
|
2022-01-30 23:40:53 +01:00
|
|
|
get_mods(mod_path, mod_virtual_path, retval, name)
|
2014-12-07 11:14:52 +01:00
|
|
|
end
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
--modmanager implementation
|
2018-04-17 15:54:50 +02:00
|
|
|
pkgmgr = {}
|
|
|
|
|
|
|
|
function pkgmgr.get_texture_packs()
|
|
|
|
local txtpath = core.get_texturepath()
|
2021-01-21 20:01:37 +01:00
|
|
|
local txtpath_system = core.get_texturepath_share()
|
2018-04-17 15:54:50 +02:00
|
|
|
local retval = {}
|
|
|
|
|
2021-01-21 20:01:37 +01:00
|
|
|
load_texture_packs(txtpath, retval)
|
|
|
|
-- on portable versions these two paths coincide. It avoids loading the path twice
|
|
|
|
if txtpath ~= txtpath_system then
|
|
|
|
load_texture_packs(txtpath_system, retval)
|
2018-04-17 15:54:50 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
table.sort(retval, function(a, b)
|
2022-09-10 12:20:33 +02:00
|
|
|
return a.title:lower() < b.title:lower()
|
2018-04-17 15:54:50 +02:00
|
|
|
end)
|
|
|
|
|
|
|
|
return retval
|
|
|
|
end
|
2013-06-23 18:30:21 +02:00
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2018-04-17 15:54:50 +02:00
|
|
|
function pkgmgr.get_folder_type(path)
|
|
|
|
local testfile = io.open(path .. DIR_DELIM .. "init.lua","r")
|
|
|
|
if testfile ~= nil then
|
|
|
|
testfile:close()
|
|
|
|
return { type = "mod", path = path }
|
|
|
|
end
|
2013-06-23 18:30:21 +02:00
|
|
|
|
2019-01-06 10:23:35 +01:00
|
|
|
testfile = io.open(path .. DIR_DELIM .. "modpack.conf","r")
|
|
|
|
if testfile ~= nil then
|
|
|
|
testfile:close()
|
|
|
|
return { type = "modpack", path = path }
|
|
|
|
end
|
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
testfile = io.open(path .. DIR_DELIM .. "modpack.txt","r")
|
|
|
|
if testfile ~= nil then
|
|
|
|
testfile:close()
|
|
|
|
return { type = "modpack", path = path }
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
testfile = io.open(path .. DIR_DELIM .. "game.conf","r")
|
2013-06-23 18:30:21 +02:00
|
|
|
if testfile ~= nil then
|
|
|
|
testfile:close()
|
2018-04-17 15:54:50 +02:00
|
|
|
return { type = "game", path = path }
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
testfile = io.open(path .. DIR_DELIM .. "texture_pack.conf","r")
|
2013-06-23 18:30:21 +02:00
|
|
|
if testfile ~= nil then
|
|
|
|
testfile:close()
|
2018-04-17 15:54:50 +02:00
|
|
|
return { type = "txp", path = path }
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
return nil
|
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
-------------------------------------------------------------------------------
|
|
|
|
function pkgmgr.get_base_folder(temppath)
|
|
|
|
if temppath == nil then
|
|
|
|
return { type = "invalid", path = "" }
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
local ret = pkgmgr.get_folder_type(temppath)
|
|
|
|
if ret then
|
|
|
|
return ret
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
local subdirs = core.get_dir_list(temppath, true)
|
|
|
|
if #subdirs == 1 then
|
|
|
|
ret = pkgmgr.get_folder_type(temppath .. DIR_DELIM .. subdirs[1])
|
|
|
|
if ret then
|
|
|
|
return ret
|
|
|
|
else
|
|
|
|
return { type = "invalid", path = temppath .. DIR_DELIM .. subdirs[1] }
|
|
|
|
end
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
return nil
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2022-08-15 09:31:01 +02:00
|
|
|
function pkgmgr.is_valid_modname(modpath)
|
2022-08-15 10:08:24 +02:00
|
|
|
return modpath:match("[^a-z0-9_]") == nil
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2022-05-07 17:45:17 +02:00
|
|
|
function pkgmgr.render_packagelist(render_list, use_technical_names, with_error)
|
2019-09-26 21:03:54 +02:00
|
|
|
if not render_list then
|
|
|
|
if not pkgmgr.global_mods then
|
2018-04-17 15:54:50 +02:00
|
|
|
pkgmgr.refresh_globals()
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
2018-04-17 15:54:50 +02:00
|
|
|
render_list = pkgmgr.global_mods
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2014-04-18 15:39:15 +02:00
|
|
|
local list = render_list:get_list()
|
2016-12-28 00:26:36 +01:00
|
|
|
local retval = {}
|
|
|
|
for i, v in ipairs(list) do
|
2013-08-19 11:26:51 +02:00
|
|
|
local color = ""
|
2022-05-07 17:45:17 +02:00
|
|
|
local icon = 0
|
|
|
|
local error = with_error and with_error[v.virtual_path]
|
|
|
|
local function update_error(val)
|
|
|
|
if val and (not error or (error.type == "warning" and val.type == "error")) then
|
|
|
|
error = val
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
if v.is_modpack then
|
2014-04-18 15:39:15 +02:00
|
|
|
local rawlist = render_list:get_raw_list()
|
2016-12-28 00:26:36 +01:00
|
|
|
color = mt_color_dark_green
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2022-05-07 17:45:17 +02:00
|
|
|
for j = 1, #rawlist do
|
|
|
|
if rawlist[j].modpack == list[i].name then
|
|
|
|
if with_error then
|
|
|
|
update_error(with_error[rawlist[j].virtual_path])
|
|
|
|
end
|
|
|
|
|
|
|
|
if rawlist[j].enabled then
|
|
|
|
icon = 1
|
|
|
|
else
|
|
|
|
-- Modpack not entirely enabled so showing as grey
|
|
|
|
color = mt_color_grey
|
|
|
|
end
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
|
|
|
end
|
2018-04-17 15:54:50 +02:00
|
|
|
elseif v.is_game_content or v.type == "game" then
|
2022-05-07 17:45:17 +02:00
|
|
|
icon = 1
|
2013-08-19 11:26:51 +02:00
|
|
|
color = mt_color_blue
|
2022-05-07 17:45:17 +02:00
|
|
|
|
|
|
|
local rawlist = render_list:get_raw_list()
|
|
|
|
if v.type == "game" and with_error then
|
|
|
|
for j = 1, #rawlist do
|
|
|
|
if rawlist[j].is_game_content then
|
|
|
|
update_error(with_error[rawlist[j].virtual_path])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-04-17 15:54:50 +02:00
|
|
|
elseif v.enabled or v.type == "txp" then
|
2022-05-07 17:45:17 +02:00
|
|
|
icon = 1
|
2016-12-28 00:26:36 +01:00
|
|
|
color = mt_color_green
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
2013-08-19 11:26:51 +02:00
|
|
|
|
2022-05-07 17:45:17 +02:00
|
|
|
if error then
|
|
|
|
if error.type == "warning" then
|
|
|
|
color = mt_color_orange
|
|
|
|
icon = 2
|
|
|
|
else
|
|
|
|
color = mt_color_red
|
|
|
|
icon = 3
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-12-28 00:26:36 +01:00
|
|
|
retval[#retval + 1] = color
|
2018-04-17 15:54:50 +02:00
|
|
|
if v.modpack ~= nil or v.loc == "game" then
|
2016-12-28 00:26:36 +01:00
|
|
|
retval[#retval + 1] = "1"
|
|
|
|
else
|
|
|
|
retval[#retval + 1] = "0"
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
2022-04-24 21:09:11 +02:00
|
|
|
|
2022-05-07 17:45:17 +02:00
|
|
|
if with_error then
|
|
|
|
retval[#retval + 1] = icon
|
|
|
|
end
|
|
|
|
|
2022-04-24 21:09:11 +02:00
|
|
|
if use_technical_names then
|
|
|
|
retval[#retval + 1] = core.formspec_escape(v.list_name or v.name)
|
|
|
|
else
|
|
|
|
retval[#retval + 1] = core.formspec_escape(v.list_title or v.list_name or v.title or v.name)
|
|
|
|
end
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2016-12-28 00:26:36 +01:00
|
|
|
return table.concat(retval, ",")
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2018-04-17 15:54:50 +02:00
|
|
|
function pkgmgr.get_dependencies(path)
|
2018-03-28 23:14:16 +02:00
|
|
|
if path == nil then
|
2019-03-07 08:23:03 +01:00
|
|
|
return {}, {}
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
local info = core.get_content_info(path)
|
2019-03-07 08:23:03 +01:00
|
|
|
return info.depends or {}, info.optional_depends or {}
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
|
|
|
|
2018-05-29 18:37:51 +02:00
|
|
|
----------- tests whether all of the mods in the modpack are enabled -----------
|
|
|
|
function pkgmgr.is_modpack_entirely_enabled(data, name)
|
|
|
|
local rawlist = data.list:get_raw_list()
|
|
|
|
for j = 1, #rawlist do
|
|
|
|
if rawlist[j].modpack == name and not rawlist[j].enabled then
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
|
2022-01-30 23:40:53 +01:00
|
|
|
local function disable_all_by_name(list, name, except)
|
|
|
|
for i=1, #list do
|
|
|
|
if list[i].name == name and list[i] ~= except then
|
|
|
|
list[i].enabled = false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-09-26 21:03:54 +02:00
|
|
|
---------- toggles or en/disables a mod or modpack and its dependencies --------
|
2021-03-01 12:13:47 +01:00
|
|
|
local function toggle_mod_or_modpack(list, toggled_mods, enabled_mods, toset, mod)
|
2018-05-29 18:37:51 +02:00
|
|
|
if not mod.is_modpack then
|
2019-09-26 21:03:54 +02:00
|
|
|
-- Toggle or en/disable the mod
|
2018-05-29 18:37:51 +02:00
|
|
|
if toset == nil then
|
2019-09-26 21:03:54 +02:00
|
|
|
toset = not mod.enabled
|
|
|
|
end
|
|
|
|
if mod.enabled ~= toset then
|
|
|
|
toggled_mods[#toggled_mods+1] = mod.name
|
2018-05-29 18:37:51 +02:00
|
|
|
end
|
2019-09-26 21:03:54 +02:00
|
|
|
if toset then
|
|
|
|
-- Mark this mod for recursive dependency traversal
|
|
|
|
enabled_mods[mod.name] = true
|
2022-02-24 17:01:22 +01:00
|
|
|
|
|
|
|
-- Disable other mods with the same name
|
|
|
|
disable_all_by_name(list, mod.name, mod)
|
2019-09-26 21:03:54 +02:00
|
|
|
end
|
2022-02-24 17:01:22 +01:00
|
|
|
mod.enabled = toset
|
2019-09-26 21:03:54 +02:00
|
|
|
else
|
|
|
|
-- Toggle or en/disable every mod in the modpack,
|
|
|
|
-- interleaved unsupported
|
|
|
|
for i = 1, #list do
|
|
|
|
if list[i].modpack == mod.name then
|
2021-03-01 12:13:47 +01:00
|
|
|
toggle_mod_or_modpack(list, toggled_mods, enabled_mods, toset, list[i])
|
2019-09-26 21:03:54 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2021-03-01 12:13:47 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
function pkgmgr.enable_mod(this, toset)
|
|
|
|
local list = this.data.list:get_list()
|
|
|
|
local mod = list[this.data.selected_mod]
|
|
|
|
|
|
|
|
-- Game mods can't be enabled or disabled
|
|
|
|
if mod.is_game_content then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
local toggled_mods = {}
|
|
|
|
local enabled_mods = {}
|
|
|
|
toggle_mod_or_modpack(list, toggled_mods, enabled_mods, toset, mod)
|
|
|
|
|
2022-05-08 17:21:59 +02:00
|
|
|
if next(enabled_mods) == nil then
|
2019-09-26 21:03:54 +02:00
|
|
|
-- Mod(s) were disabled, so no dependencies need to be enabled
|
|
|
|
table.sort(toggled_mods)
|
2021-02-10 12:02:34 +01:00
|
|
|
core.log("info", "Following mods were disabled: " ..
|
2019-09-26 21:03:54 +02:00
|
|
|
table.concat(toggled_mods, ", "))
|
2018-05-29 18:37:51 +02:00
|
|
|
return
|
|
|
|
end
|
|
|
|
|
2019-09-26 21:03:54 +02:00
|
|
|
-- Enable mods' depends after activation
|
|
|
|
|
2022-05-08 15:14:14 +02:00
|
|
|
-- Make a list of mod ids indexed by their names. Among mods with the
|
|
|
|
-- same name, enabled mods take precedence, after which game mods take
|
|
|
|
-- precedence, being last in the mod list.
|
2019-09-26 21:03:54 +02:00
|
|
|
local mod_ids = {}
|
2019-12-07 00:16:33 +01:00
|
|
|
for id, mod2 in pairs(list) do
|
|
|
|
if mod2.type == "mod" and not mod2.is_modpack then
|
2022-05-08 15:14:14 +02:00
|
|
|
local prev_id = mod_ids[mod2.name]
|
|
|
|
if not prev_id or not list[prev_id].enabled then
|
|
|
|
mod_ids[mod2.name] = id
|
|
|
|
end
|
2019-09-26 21:03:54 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- to_enable is used as a DFS stack with sp as stack pointer
|
|
|
|
local to_enable = {}
|
|
|
|
local sp = 0
|
|
|
|
for name in pairs(enabled_mods) do
|
|
|
|
local depends = pkgmgr.get_dependencies(list[mod_ids[name]].path)
|
|
|
|
for i = 1, #depends do
|
|
|
|
local dependency_name = depends[i]
|
|
|
|
if not enabled_mods[dependency_name] then
|
|
|
|
sp = sp+1
|
|
|
|
to_enable[sp] = dependency_name
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2022-02-24 17:01:22 +01:00
|
|
|
|
2019-09-26 21:03:54 +02:00
|
|
|
-- If sp is 0, every dependency is already activated
|
|
|
|
while sp > 0 do
|
|
|
|
local name = to_enable[sp]
|
|
|
|
sp = sp-1
|
|
|
|
|
|
|
|
if not enabled_mods[name] then
|
|
|
|
enabled_mods[name] = true
|
2019-12-07 00:16:33 +01:00
|
|
|
local mod_to_enable = list[mod_ids[name]]
|
|
|
|
if not mod_to_enable then
|
2021-02-10 12:02:34 +01:00
|
|
|
core.log("warning", "Mod dependency \"" .. name ..
|
2019-09-26 21:03:54 +02:00
|
|
|
"\" not found!")
|
2022-05-07 17:45:17 +02:00
|
|
|
elseif not mod_to_enable.is_game_content then
|
2022-02-24 17:01:22 +01:00
|
|
|
if not mod_to_enable.enabled then
|
2019-12-07 00:16:33 +01:00
|
|
|
mod_to_enable.enabled = true
|
|
|
|
toggled_mods[#toggled_mods+1] = mod_to_enable.name
|
2019-09-26 21:03:54 +02:00
|
|
|
end
|
|
|
|
-- Push the dependencies of the dependency onto the stack
|
2019-12-07 00:16:33 +01:00
|
|
|
local depends = pkgmgr.get_dependencies(mod_to_enable.path)
|
2019-09-26 21:03:54 +02:00
|
|
|
for i = 1, #depends do
|
2022-05-08 15:01:32 +02:00
|
|
|
if not enabled_mods[depends[i]] then
|
2019-09-26 21:03:54 +02:00
|
|
|
sp = sp+1
|
|
|
|
to_enable[sp] = depends[i]
|
|
|
|
end
|
|
|
|
end
|
2018-05-29 18:37:51 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2019-09-26 21:03:54 +02:00
|
|
|
|
|
|
|
-- Log the list of enabled mods
|
|
|
|
table.sort(toggled_mods)
|
2021-02-10 12:02:34 +01:00
|
|
|
core.log("info", "Following mods were enabled: " ..
|
2019-09-26 21:03:54 +02:00
|
|
|
table.concat(toggled_mods, ", "))
|
2018-05-29 18:37:51 +02:00
|
|
|
end
|
|
|
|
|
2013-06-23 18:30:21 +02:00
|
|
|
--------------------------------------------------------------------------------
|
2018-04-17 15:54:50 +02:00
|
|
|
function pkgmgr.get_worldconfig(worldpath)
|
2013-06-23 18:30:21 +02:00
|
|
|
local filename = worldpath ..
|
|
|
|
DIR_DELIM .. "world.mt"
|
|
|
|
|
2013-09-10 21:09:21 +02:00
|
|
|
local worldfile = Settings(filename)
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-06-23 18:30:21 +02:00
|
|
|
local worldconfig = {}
|
|
|
|
worldconfig.global_mods = {}
|
|
|
|
worldconfig.game_mods = {}
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-09-10 21:09:21 +02:00
|
|
|
for key,value in pairs(worldfile:to_table()) do
|
|
|
|
if key == "gameid" then
|
|
|
|
worldconfig.id = value
|
2015-03-15 14:52:10 +01:00
|
|
|
elseif key:sub(0, 9) == "load_mod_" then
|
2019-01-13 16:22:32 +01:00
|
|
|
-- Compatibility: Check against "nil" which was erroneously used
|
|
|
|
-- as value for fresh configured worlds
|
|
|
|
worldconfig.global_mods[key] = value ~= "false" and value ~= "nil"
|
|
|
|
and value
|
2015-03-15 14:52:10 +01:00
|
|
|
else
|
|
|
|
worldconfig[key] = value
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-06-23 18:30:21 +02:00
|
|
|
--read gamemods
|
2018-04-17 15:54:50 +02:00
|
|
|
local gamespec = pkgmgr.find_by_gameid(worldconfig.id)
|
|
|
|
pkgmgr.get_game_mods(gamespec, worldconfig.game_mods)
|
2013-06-23 18:30:21 +02:00
|
|
|
|
|
|
|
return worldconfig
|
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2022-08-15 09:31:01 +02:00
|
|
|
function pkgmgr.install_dir(expected_type, path, basename, targetpath)
|
|
|
|
assert(type(expected_type) == "string")
|
|
|
|
assert(type(path) == "string")
|
|
|
|
assert(basename == nil or type(basename) == "string")
|
|
|
|
assert(targetpath == nil or type(targetpath) == "string")
|
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
local basefolder = pkgmgr.get_base_folder(path)
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2022-08-15 09:31:01 +02:00
|
|
|
if expected_type == "txp" then
|
2022-08-15 10:08:24 +02:00
|
|
|
assert(basename)
|
|
|
|
|
2022-08-15 09:31:01 +02:00
|
|
|
-- There's no good way to detect a texture pack, so let's just assume
|
|
|
|
-- it's correct for now.
|
2018-04-17 15:54:50 +02:00
|
|
|
if basefolder and basefolder.type ~= "invalid" and basefolder.type ~= "txp" then
|
|
|
|
return nil, fgettext("Unable to install a $1 as a texture pack", basefolder.type)
|
|
|
|
end
|
|
|
|
|
|
|
|
local from = basefolder and basefolder.path or path
|
2022-08-15 09:30:46 +02:00
|
|
|
if not targetpath then
|
2018-05-16 22:52:12 +02:00
|
|
|
targetpath = core.get_texturepath() .. DIR_DELIM .. basename
|
|
|
|
end
|
2022-08-15 09:30:46 +02:00
|
|
|
core.delete_dir(targetpath)
|
2021-09-19 18:16:53 +02:00
|
|
|
if not core.copy_dir(from, targetpath, false) then
|
2018-05-16 22:52:12 +02:00
|
|
|
return nil,
|
|
|
|
fgettext("Failed to install $1 to $2", basename, targetpath)
|
|
|
|
end
|
2018-04-17 15:54:50 +02:00
|
|
|
return targetpath, nil
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
elseif not basefolder then
|
2022-08-15 09:31:01 +02:00
|
|
|
return nil, fgettext("Unable to find a valid mod, modpack, or game")
|
2018-04-17 15:54:50 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2022-08-15 09:31:01 +02:00
|
|
|
-- Check type
|
|
|
|
if basefolder.type ~= expected_type and (basefolder.type ~= "modpack" or expected_type ~= "mod") then
|
2022-08-15 10:08:24 +02:00
|
|
|
return nil, fgettext("Unable to install a $1 as a $2", basefolder.type, expected_type)
|
2022-08-15 09:31:01 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2022-08-15 09:31:01 +02:00
|
|
|
-- Set targetpath if not predetermined
|
|
|
|
if not targetpath then
|
|
|
|
local content_path
|
|
|
|
if basefolder.type == "modpack" or basefolder.type == "mod" then
|
|
|
|
if not basename then
|
|
|
|
basename = get_last_folder(cleanup_path(basefolder.path))
|
2018-05-16 22:52:12 +02:00
|
|
|
end
|
2022-08-15 09:31:01 +02:00
|
|
|
content_path = core.get_modpath()
|
|
|
|
elseif basefolder.type == "game" then
|
|
|
|
content_path = core.get_gamepath()
|
2018-05-16 22:52:12 +02:00
|
|
|
else
|
2022-08-15 09:31:01 +02:00
|
|
|
error("Unknown content type")
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
2018-04-17 15:54:50 +02:00
|
|
|
|
2022-08-15 09:31:01 +02:00
|
|
|
if basename and (basefolder.type ~= "mod" or pkgmgr.is_valid_modname(basename)) then
|
|
|
|
targetpath = content_path .. DIR_DELIM .. basename
|
2018-05-16 22:52:12 +02:00
|
|
|
else
|
2022-08-15 09:31:01 +02:00
|
|
|
return nil,
|
|
|
|
fgettext("Install: Unable to find suitable folder name for $1", path)
|
2018-05-16 22:52:12 +02:00
|
|
|
end
|
2013-06-23 18:30:21 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2018-05-16 22:52:12 +02:00
|
|
|
-- Copy it
|
2022-08-15 09:30:46 +02:00
|
|
|
core.delete_dir(targetpath)
|
2021-09-19 18:16:53 +02:00
|
|
|
if not core.copy_dir(basefolder.path, targetpath, false) then
|
2018-05-16 22:52:12 +02:00
|
|
|
return nil,
|
|
|
|
fgettext("Failed to install $1 to $2", basename, targetpath)
|
|
|
|
end
|
|
|
|
|
2019-01-09 14:14:24 +01:00
|
|
|
if basefolder.type == "game" then
|
|
|
|
pkgmgr.update_gamelist()
|
|
|
|
else
|
|
|
|
pkgmgr.refresh_globals()
|
|
|
|
end
|
2018-05-16 22:52:12 +02:00
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
return targetpath, nil
|
|
|
|
end
|
|
|
|
|
2013-06-23 18:30:21 +02:00
|
|
|
--------------------------------------------------------------------------------
|
2018-04-17 15:54:50 +02:00
|
|
|
function pkgmgr.preparemodlist(data)
|
2013-07-15 22:56:53 +02:00
|
|
|
local retval = {}
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
local global_mods = {}
|
|
|
|
local game_mods = {}
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
--read global mods
|
2021-10-07 00:19:41 +02:00
|
|
|
local modpaths = core.get_modpaths()
|
2022-01-30 23:40:53 +01:00
|
|
|
for key, modpath in pairs(modpaths) do
|
|
|
|
get_mods(modpath, key, global_mods)
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
for i=1,#global_mods,1 do
|
2018-04-17 15:54:50 +02:00
|
|
|
global_mods[i].type = "mod"
|
|
|
|
global_mods[i].loc = "global"
|
2022-02-24 17:01:22 +01:00
|
|
|
global_mods[i].enabled = false
|
2016-03-06 16:53:45 +01:00
|
|
|
retval[#retval + 1] = global_mods[i]
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
--read game mods
|
2018-04-17 15:54:50 +02:00
|
|
|
local gamespec = pkgmgr.find_by_gameid(data.gameid)
|
|
|
|
pkgmgr.get_game_mods(gamespec, game_mods)
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2017-05-17 19:31:57 +02:00
|
|
|
if #game_mods > 0 then
|
|
|
|
-- Add title
|
|
|
|
retval[#retval + 1] = {
|
2018-04-17 15:54:50 +02:00
|
|
|
type = "game",
|
2017-05-19 07:45:47 +02:00
|
|
|
is_game_content = true,
|
2022-05-21 17:23:30 +02:00
|
|
|
name = fgettext("$1 mods", gamespec.title),
|
2018-12-03 00:41:05 +01:00
|
|
|
path = gamespec.path
|
2017-05-17 19:31:57 +02:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
for i=1,#game_mods,1 do
|
2018-04-17 15:54:50 +02:00
|
|
|
game_mods[i].type = "mod"
|
|
|
|
game_mods[i].loc = "game"
|
2017-05-19 07:45:47 +02:00
|
|
|
game_mods[i].is_game_content = true
|
2016-03-06 16:53:45 +01:00
|
|
|
retval[#retval + 1] = game_mods[i]
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
if data.worldpath == nil then
|
|
|
|
return retval
|
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
--read world mod configuration
|
|
|
|
local filename = data.worldpath ..
|
|
|
|
DIR_DELIM .. "world.mt"
|
|
|
|
|
2013-09-10 21:09:21 +02:00
|
|
|
local worldfile = Settings(filename)
|
2022-01-30 23:40:53 +01:00
|
|
|
for key, value in pairs(worldfile:to_table()) do
|
2013-09-10 21:09:21 +02:00
|
|
|
if key:sub(1, 9) == "load_mod_" then
|
|
|
|
key = key:sub(10)
|
2022-01-30 23:40:53 +01:00
|
|
|
local mod_found = false
|
|
|
|
|
|
|
|
local fallback_found = false
|
|
|
|
local fallback_mod = nil
|
|
|
|
|
|
|
|
for i=1, #retval do
|
2014-04-18 15:39:15 +02:00
|
|
|
if retval[i].name == key and
|
2022-01-30 23:40:53 +01:00
|
|
|
not retval[i].is_modpack then
|
|
|
|
if core.is_yes(value) or retval[i].virtual_path == value then
|
|
|
|
retval[i].enabled = true
|
|
|
|
mod_found = true
|
|
|
|
break
|
|
|
|
elseif fallback_found then
|
|
|
|
-- Only allow fallback if only one mod matches
|
|
|
|
fallback_mod = nil
|
|
|
|
else
|
|
|
|
fallback_found = true
|
|
|
|
fallback_mod = retval[i]
|
|
|
|
end
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
|
|
|
end
|
2022-01-30 23:40:53 +01:00
|
|
|
|
|
|
|
if not mod_found then
|
|
|
|
if fallback_mod and value:find("/") then
|
|
|
|
fallback_mod.enabled = true
|
|
|
|
else
|
|
|
|
core.log("info", "Mod: " .. key .. " " .. dump(value) .. " but not found")
|
|
|
|
end
|
2013-09-10 21:09:21 +02:00
|
|
|
end
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return retval
|
|
|
|
end
|
2013-06-23 18:30:21 +02:00
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
function pkgmgr.compare_package(a, b)
|
|
|
|
return a and b and a.name == b.name and a.path == b.path
|
|
|
|
end
|
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
--------------------------------------------------------------------------------
|
2018-04-17 15:54:50 +02:00
|
|
|
function pkgmgr.comparemod(elem1,elem2)
|
2013-07-15 22:56:53 +02:00
|
|
|
if elem1 == nil or elem2 == nil then
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
if elem1.name ~= elem2.name then
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
if elem1.is_modpack ~= elem2.is_modpack then
|
|
|
|
return false
|
|
|
|
end
|
2018-04-17 15:54:50 +02:00
|
|
|
if elem1.type ~= elem2.type then
|
2013-07-15 22:56:53 +02:00
|
|
|
return false
|
|
|
|
end
|
|
|
|
if elem1.modpack ~= elem2.modpack then
|
|
|
|
return false
|
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
if elem1.path ~= elem2.path then
|
|
|
|
return false
|
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
return true
|
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2018-04-17 15:54:50 +02:00
|
|
|
function pkgmgr.mod_exists(basename)
|
2013-07-15 22:56:53 +02:00
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
if pkgmgr.global_mods == nil then
|
|
|
|
pkgmgr.refresh_globals()
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
if pkgmgr.global_mods:raw_index_by_uid(basename) > 0 then
|
2013-07-15 22:56:53 +02:00
|
|
|
return true
|
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2013-07-15 22:56:53 +02:00
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2018-04-17 15:54:50 +02:00
|
|
|
function pkgmgr.get_global_mod(idx)
|
2013-07-15 22:56:53 +02:00
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
if pkgmgr.global_mods == nil then
|
2013-07-15 22:56:53 +02:00
|
|
|
return nil
|
|
|
|
end
|
2013-11-26 18:15:31 +01:00
|
|
|
|
2014-04-18 15:39:15 +02:00
|
|
|
if idx == nil or idx < 1 or
|
2018-04-17 15:54:50 +02:00
|
|
|
idx > pkgmgr.global_mods:size() then
|
2013-07-15 22:56:53 +02:00
|
|
|
return nil
|
|
|
|
end
|
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
return pkgmgr.global_mods:get_list()[idx]
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2018-04-17 15:54:50 +02:00
|
|
|
function pkgmgr.refresh_globals()
|
|
|
|
local function is_equal(element,uid) --uid match
|
|
|
|
if element.name == uid then
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
pkgmgr.global_mods = filterlist.create(pkgmgr.preparemodlist,
|
|
|
|
pkgmgr.comparemod, is_equal, nil, {})
|
|
|
|
pkgmgr.global_mods:add_sort_mechanism("alphabetic", sort_mod_list)
|
|
|
|
pkgmgr.global_mods:set_sortmode("alphabetic")
|
2013-07-15 22:56:53 +02:00
|
|
|
end
|
2013-07-16 20:56:18 +02:00
|
|
|
|
2018-04-17 15:54:50 +02:00
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
function pkgmgr.find_by_gameid(gameid)
|
|
|
|
for i=1,#pkgmgr.games,1 do
|
|
|
|
if pkgmgr.games[i].id == gameid then
|
|
|
|
return pkgmgr.games[i], i
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return nil, nil
|
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
function pkgmgr.get_game_mods(gamespec, retval)
|
|
|
|
if gamespec ~= nil and
|
|
|
|
gamespec.gamemods_path ~= nil and
|
|
|
|
gamespec.gamemods_path ~= "" then
|
2022-01-30 23:40:53 +01:00
|
|
|
get_mods(gamespec.gamemods_path, ("games/%s/mods"):format(gamespec.id), retval)
|
2018-04-17 15:54:50 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
function pkgmgr.get_game_modlist(gamespec)
|
|
|
|
local retval = ""
|
|
|
|
local game_mods = {}
|
|
|
|
pkgmgr.get_game_mods(gamespec, game_mods)
|
|
|
|
for i=1,#game_mods,1 do
|
|
|
|
if retval ~= "" then
|
|
|
|
retval = retval..","
|
|
|
|
end
|
|
|
|
retval = retval .. game_mods[i].name
|
|
|
|
end
|
|
|
|
return retval
|
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
function pkgmgr.get_game(index)
|
|
|
|
if index > 0 and index <= #pkgmgr.games then
|
|
|
|
return pkgmgr.games[index]
|
|
|
|
end
|
|
|
|
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
function pkgmgr.update_gamelist()
|
|
|
|
pkgmgr.games = core.get_games()
|
2022-09-10 12:20:33 +02:00
|
|
|
table.sort(pkgmgr.games, function(a, b)
|
|
|
|
return a.title:lower() < b.title:lower()
|
|
|
|
end)
|
2018-04-17 15:54:50 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
function pkgmgr.gamelist()
|
|
|
|
local retval = ""
|
|
|
|
if #pkgmgr.games > 0 then
|
2022-05-21 17:23:30 +02:00
|
|
|
retval = retval .. core.formspec_escape(pkgmgr.games[1].title)
|
2018-04-17 15:54:50 +02:00
|
|
|
|
|
|
|
for i=2,#pkgmgr.games,1 do
|
2022-05-21 17:23:30 +02:00
|
|
|
retval = retval .. "," .. core.formspec_escape(pkgmgr.games[i].title)
|
2018-04-17 15:54:50 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
return retval
|
|
|
|
end
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
-- read initial data
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
pkgmgr.update_gamelist()
|