local filter = smart_inventory.filter local cache = smart_inventory.cache local txt = smart_inventory.txt local creative = minetest.setting_getbool("creative_mode") local armor_type = {} if smart_inventory.armor_mod then for _,v in ipairs(armor.elements) do armor_type["group:armor:"..v] = true end end local function update_grid(state, listname) -- Update the users inventory grid local list = {} state.param["armor_"..listname.."_list"] = list local name = state.location.rootState.location.player local inventory = minetest.get_player_by_name(name):get_inventory() local invlist = inventory:get_list(listname) for stack_index, stack in ipairs(invlist) do local itemdef = stack:get_definition() local is_armor = false if itemdef and cache.citems[itemdef.name] then for groupname, groupinfo in pairs(cache.citems[itemdef.name].cgroups) do if groupinfo.parent and armor_type[groupinfo.parent.name] then local wear = stack:get_wear() if wear == 0 then wear = "" else wear = " "..wear end table.insert(list, { itemdef = itemdef, stack_index = stack_index, -- buttons_grid related item = itemdef.name..wear, is_button = true }) break end end end end table.sort(list, function(a,b) return a.item < b.item end) local grid = state:get(listname.."_grid") grid:setList(list) end local function update_selected_item(state, listentry) local name = state.location.rootState.location.player if listentry then state.param.armor_selected_item = listentry else listentry = state.param.armor_selected_item end if not listentry then return end local i_list = state:get("i_list") i_list:clearItems() for groupname, groupinfo in pairs(cache.citems[listentry.itemdef.name].cgroups) do if groupinfo.group_desc and cache.cgroups[groupname].parent then local parent = cache.cgroups[groupname].parent local parentvalue = cache.cgroups[groupname].parent.childs[groupname] if parentvalue and parentvalue ~= "0" and cache.cgroups[parent.name] and cache.cgroups[parent.name].parent and ( cache.cgroups[parent.name].parent.name == "group:armor" or cache.cgroups[parent.name].parent.name == "group:physics" ) then i_list:addItem(parent.group_desc..": "..parentvalue) end end end state:get("item_name"):setText(listentry.itemdef.description) state:get("item_image"):setImage(listentry.item) end local function update_page(state) local name = state.location.rootState.location.player if smart_inventory.armor_mod then if creative == false then update_grid(state, "main") end update_grid(state, "armor") state:get("preview"):setImage(armor.textures[name].preview) state.location.parentState:get("player_button"):setImage(armor.textures[name].preview) local a_list = state:get("a_list") a_list:clearItems() for k, v in pairs(armor.def[name]) do local skipval = 1 local groupinfo = cache.cgroups["group:physics:"..k] local grouptext if groupinfo then grouptext = groupinfo.group_desc else grouptext = filter.get("group"):get_description({name = "group:armor:"..k}) skipval = 0 end if grouptext and skipval ~= v then a_list:addItem(grouptext..": "..v) end end update_selected_item(state) elseif smart_inventory.skins_mod then state.location.parentState:get("player_button"):setImage(skins.skins[name].."_preview.png") state:get("preview"):setImage(skins.skins[name].."_preview.png") end if smart_inventory.skins_mod then local skin = skins.skins[name] if skin and skins.meta[skin] then state:get("skinname"):setText("Skin name: "..(skins.meta[skin].name or "")) state:get("skinauthor"):setText("Author: "..(skins.meta[skin].author or "")) state:get("skinlicense"):setText("License: "..(skins.meta[skin].license or "")) else state:get("skinname"):setText("") state:get("skinauthor"):setText("") state:get("skinlicense"):setText("") end end end local function move_item_to_armor(state, item) local name = state.location.rootState.location.player local inventory = minetest.get_player_by_name(name):get_inventory() local itemstack -- get item and try to move if creative == true then itemstack = inventory:add_item("armor", item.item) else itemstack = inventory:get_stack("main", item.stack_index) itemstack = inventory:add_item("armor", itemstack) inventory:set_stack("main", item.stack_index, itemstack) end -- second try: empty place and move again if not itemstack:is_empty() then local itemname = itemstack:get_name() local removestack, removeindex local newgroups = {} for groupname, groupdef in pairs(cache.citems[itemname].cgroups) do if string.sub(groupname, 1, 12) == "group:armor_" then newgroups[groupname] = groupdef end end -- check items group assignment. Prefer same group as new item to remove local oldgroups = {} for stack_index, stack in ipairs(inventory:get_list("armor")) do local itemname = stack:get_name() for groupname, groupdef in pairs(cache.citems[itemname].cgroups) do if newgroups[groupname] then removestack = stack removeindex = stack_index break end if string.sub(groupname, 1, 6) == "group:armor_" then if not oldgroups[groupname] then oldgroups[groupname] = 1 else oldgroups[groupname] = oldgroups[groupname] + 1 end end end if removestack then break end end -- no same group found. Check for biggest group (duplicates) to remove the item if not removestack then local maxcounter local removegroup for k, v in pairs(oldgroups) do if not maxcounter or maxcounter < v then maxcounter = v removegroup = k end end if removegroup then for stack_index, stack in ipairs(inventory:get_list("armor")) do local itemname = stack:get_name() if cache.citems[itemname].cgroups[removegroup] then removestack = stack removeindex = stack_index break end end end end -- replace the item if removestack then if creative == true then itemstack = inventory:set_stack("armor", removeindex, itemstack) else removestack = inventory:get_stack("armor", removeindex) removestack = inventory:add_item("main", removestack) inventory:set_stack("armor", removeindex, removestack) itemstack = inventory:get_stack("main", item.stack_index) itemstack = inventory:add_item("armor", itemstack) inventory:set_stack("main", item.stack_index, itemstack) end end end armor:set_player_armor(minetest.get_player_by_name(name)) end local function move_item_to_inv(state, item) local name = state.location.rootState.location.player local inventory = minetest.get_player_by_name(name):get_inventory() if creative == true then inventory:set_stack("armor", item.stack_index, {}) else local itemstack = inventory:get_stack("armor", item.stack_index) itemstack = inventory:add_item("main", itemstack) inventory:set_stack("armor", item.stack_index, itemstack) end armor:set_player_armor(minetest.get_player_by_name(name)) end local function player_callback(state) local name = state.location.rootState.location.player state:background(0, 2.3, 6, 4.6, "it_bg", "minimap_overlay_square.png") state:label(2,5,"item_name", "") state:listbox(2.2,2.5,3.2,2.5,"i_list", nil, true) state:item_image(0,3.5,2,2,"item_image","") state:background(6.7, 2.3, 6, 4.6, "pl_bg", "minimap_overlay_square.png") state:image(7,3.0,2,4,"preview","") state:listbox(9.2,2.5,3.2,2.5,"a_list", nil, true) state:label(9,5.0,"skinname","") state:label(9,5.5,"skinauthor", "") state:label(9,6.0, "skinlicense", "") state:background(0, 0, 20, 1, "top_bg", "halo.png") state:background(0, 8, 20, 2, "bottom_bg", "halo.png") if smart_inventory.armor_mod then local grid_armor = smart_inventory.smartfs_elements.buttons_grid(state, 0, 0, 8, 1, "armor_grid") grid_armor:onClick(function(self, state, index, player) update_selected_item(state, state.param.armor_armor_list[index]) move_item_to_inv(state, state.param.armor_armor_list[index]) update_page(state) end) local grid_main = smart_inventory.smartfs_elements.buttons_grid(state, 0, 8, 20, 2, "main_grid") grid_main:onClick(function(self, state, index, player) update_selected_item(state, state.param.armor_main_list[index]) move_item_to_armor(state, state.param.armor_main_list[index]) update_page(state) end) armor:set_player_armor(minetest.get_player_by_name(name)) if creative == true then -- fill creative list once, not each page update local list = {} local list_dedup = {} for armor_group, _ in pairs(armor_type) do for _, itemdef in pairs(cache.cgroups[armor_group].items) do list_dedup[itemdef.name] = itemdef end end for _, itemdef in pairs(list_dedup) do table.insert(list, { itemdef = itemdef, -- buttons_grid related item = itemdef.name, is_button = true }) end table.sort(list, function(a,b) return a.item < b.item end) grid_main:setList(list) state.param.armor_main_list = list end end if smart_inventory.skins_mod then -- Skins Grid local grid_skins = smart_inventory.smartfs_elements.buttons_grid(state, 13.1, 1.3, 7 , 7, "skins_grid", 0.87, 1.30) state:background(13, 1, 7 , 7, "bg_skins", "minimap_overlay_square.png") grid_skins:onClick(function(self, state, index, player) local skin = skins.list[index] local player_obj = minetest.get_player_by_name(name) skins.skins[player] = skin skins.file_save = true skins.update_player_skin(player_obj) if smart_inventory.armor_mod then armor.textures[name].skin = skin..".png" armor:set_player_armor(player_obj) end update_page(state) end) local skins_grid_data = {} for idx, skin in pairs(skins.list) do table.insert(skins_grid_data, { image = skin.."_preview.png", tooltip = skins.meta[skin].name, is_button = true, size = { w = 0.87, h = 1.30 } }) if skin == skins.skins[name] then grid_skins:setFirstVisible(idx - 19) --8x5 (grid size) / 2 -1 end end grid_skins:setList(skins_grid_data) end -- not visible update plugin for updates from outsite (API) state:element("code", { name = "update_hook" }):onSubmit(function(self, state) update_page(state) state.location.rootState:show() end) update_page(state) end smart_inventory.register_page({ name = "player", icon = "player.png", tooltip = "Customize yourself", smartfs_callback = player_callback, sequence = 20, on_button_click = update_page })