forked from mtcontrib/3d_armor
564 lines
16 KiB
Lua
564 lines
16 KiB
Lua
local skin_previews = {}
|
|
local use_player_monoids = minetest.global_exists("player_monoids")
|
|
local use_armor_monoid = minetest.global_exists("armor_monoid")
|
|
local armor_def = setmetatable({}, {
|
|
__index = function()
|
|
return setmetatable({
|
|
groups = setmetatable({}, {
|
|
__index = function()
|
|
return 0
|
|
end})
|
|
}, {
|
|
__index = function()
|
|
return 0
|
|
end
|
|
})
|
|
end,
|
|
})
|
|
local armor_textures = setmetatable({}, {
|
|
__index = function()
|
|
return setmetatable({}, {
|
|
__index = function()
|
|
return "blank.png"
|
|
end
|
|
})
|
|
end
|
|
})
|
|
|
|
armor = {
|
|
timer = 0,
|
|
elements = {"head", "torso", "legs", "feet"},
|
|
physics = {"jump", "speed", "gravity"},
|
|
attributes = {"heal", "fire", "water"},
|
|
formspec = "image[2.5,0;2,4;armor_preview]"..
|
|
default.gui_bg..
|
|
default.gui_bg_img..
|
|
default.gui_slots..
|
|
default.get_hotbar_bg(0, 4.7)..
|
|
"list[current_player;main;0,4.7;8,1;]"..
|
|
"list[current_player;main;0,5.85;8,3;8]",
|
|
def = armor_def,
|
|
textures = armor_textures,
|
|
default_skin = "character",
|
|
materials = {
|
|
wood = "group:wood",
|
|
cactus = "default:cactus",
|
|
steel = "default:steel_ingot",
|
|
bronze = "default:bronze_ingot",
|
|
diamond = "default:diamond",
|
|
gold = "default:gold_ingot",
|
|
mithril = "moreores:mithril_ingot",
|
|
crystal = "ethereal:crystal_ingot",
|
|
},
|
|
fire_nodes = {
|
|
{"default:lava_source", 5, 8},
|
|
{"default:lava_flowing", 5, 8},
|
|
{"fire:basic_flame", 3, 4},
|
|
{"fire:permanent_flame", 3, 4},
|
|
{"ethereal:crystal_spike", 2, 1},
|
|
{"ethereal:fire_flower", 2, 1},
|
|
{"default:torch", 1, 1},
|
|
{"default:torch_ceiling", 1, 1},
|
|
{"default:torch_wall", 1, 1},
|
|
},
|
|
registered_groups = {["fleshy"]=100},
|
|
registered_callbacks = {
|
|
on_update = {},
|
|
on_equip = {},
|
|
on_unequip = {},
|
|
on_damage = {},
|
|
on_destroy = {},
|
|
},
|
|
version = "0.4.8",
|
|
}
|
|
|
|
armor.config = {
|
|
init_delay = 2,
|
|
init_times = 10,
|
|
bones_delay = 1,
|
|
update_time = 1,
|
|
drop = minetest.get_modpath("bones") ~= nil,
|
|
destroy = false,
|
|
level_multiplier = 1,
|
|
heal_multiplier = 1,
|
|
material_wood = true,
|
|
material_cactus = true,
|
|
material_steel = true,
|
|
material_bronze = true,
|
|
material_diamond = true,
|
|
material_gold = true,
|
|
material_mithril = true,
|
|
material_crystal = true,
|
|
water_protect = true,
|
|
fire_protect = minetest.get_modpath("ethereal") ~= nil,
|
|
punch_damage = true,
|
|
}
|
|
|
|
-- Armor Registration
|
|
|
|
armor.register_armor = function(self, name, def)
|
|
minetest.register_tool(name, def)
|
|
end
|
|
|
|
armor.register_armor_group = function(self, group, base)
|
|
base = base or 100
|
|
self.registered_groups[group] = base
|
|
if use_armor_monoid then
|
|
armor_monoid.register_armor_group(group, base)
|
|
end
|
|
end
|
|
|
|
-- Armor callbacks
|
|
|
|
armor.register_on_update = function(self, func)
|
|
if type(func) == "function" then
|
|
table.insert(self.registered_callbacks.on_update, func)
|
|
end
|
|
end
|
|
|
|
armor.register_on_equip = function(self, func)
|
|
if type(func) == "function" then
|
|
table.insert(self.registered_callbacks.on_equip, func)
|
|
end
|
|
end
|
|
|
|
armor.register_on_unequip = function(self, func)
|
|
if type(func) == "function" then
|
|
table.insert(self.registered_callbacks.on_unequip, func)
|
|
end
|
|
end
|
|
|
|
armor.register_on_damage = function(self, func)
|
|
if type(func) == "function" then
|
|
table.insert(self.registered_callbacks.on_damage, func)
|
|
end
|
|
end
|
|
|
|
armor.register_on_destroy = function(self, func)
|
|
if type(func) == "function" then
|
|
table.insert(self.registered_callbacks.on_destroy, func)
|
|
end
|
|
end
|
|
|
|
armor.run_callbacks = function(self, callback, player, stack, index)
|
|
if stack then
|
|
local def = stack:get_definition() or {}
|
|
if type(def[callback]) == "function" then
|
|
def[callback](player, stack, index)
|
|
end
|
|
end
|
|
local callbacks = self.registered_callbacks[callback]
|
|
if callbacks then
|
|
for _, func in pairs(callbacks) do
|
|
func(player, stack, index)
|
|
end
|
|
end
|
|
end
|
|
|
|
armor.update_player_visuals = function(self, player)
|
|
if not player then
|
|
return
|
|
end
|
|
local name = player:get_player_name()
|
|
if self.textures[name] then
|
|
default.player_set_textures(player, {
|
|
self.textures[name].skin,
|
|
self.textures[name].armor,
|
|
self.textures[name].wielditem,
|
|
})
|
|
end
|
|
end
|
|
|
|
armor.init_player_armor = function(self, player)
|
|
local name = player:get_player_name()
|
|
local player_inv = player:get_inventory()
|
|
local pos = player:getpos()
|
|
if not name or not player_inv or not pos then
|
|
return false
|
|
end
|
|
local armor_inv = minetest.create_detached_inventory(name.."_armor", {
|
|
on_put = function(inv, listname, index, stack, player)
|
|
player:get_inventory():set_stack(listname, index, stack)
|
|
armor:set_player_armor(player)
|
|
armor:run_callbacks("on_equip", player, stack, index)
|
|
end,
|
|
on_take = function(inv, listname, index, stack, player)
|
|
player:get_inventory():set_stack(listname, index, nil)
|
|
armor:set_player_armor(player)
|
|
armor:run_callbacks("on_unequip", player, stack, index)
|
|
end,
|
|
on_move = function(inv, from_list, from_index, to_list, to_index, count, player)
|
|
local plaver_inv = player:get_inventory()
|
|
local stack = inv:get_stack(to_list, to_index)
|
|
player_inv:set_stack(to_list, to_index, stack)
|
|
player_inv:set_stack(from_list, from_index, nil)
|
|
armor:set_player_armor(player)
|
|
end,
|
|
allow_put = function(inv, listname, index, stack, player)
|
|
local def = stack:get_definition() or {}
|
|
local allowed = 0
|
|
for _, element in pairs(armor.elements) do
|
|
if def.groups["armor_"..element] then
|
|
allowed = 1
|
|
for i = 1, 6 do
|
|
local item = inv:get_stack("armor", i):get_name()
|
|
if minetest.get_item_group(item, "armor_"..element) > 0 then
|
|
return 0
|
|
end
|
|
end
|
|
end
|
|
end
|
|
return allowed
|
|
end,
|
|
allow_take = function(inv, listname, index, stack, player)
|
|
return stack:get_count()
|
|
end,
|
|
allow_move = function(inv, from_list, from_index, to_list, to_index, count, player)
|
|
return count
|
|
end,
|
|
}, name)
|
|
armor_inv:set_size("armor", 6)
|
|
player_inv:set_size("armor", 6)
|
|
for i=1, 6 do
|
|
local stack = player_inv:get_stack("armor", i)
|
|
armor_inv:set_stack("armor", i, stack)
|
|
self:run_callbacks("on_equip", player, stack, i)
|
|
end
|
|
self.def[name] = {
|
|
init_time = minetest.get_gametime(),
|
|
level = 0,
|
|
state = 0,
|
|
count = 0,
|
|
groups = {},
|
|
}
|
|
for _, phys in pairs(armor.physics) do
|
|
self.def[name][phys] = 1
|
|
end
|
|
for _, attr in pairs(armor.attributes) do
|
|
self.def[name][attr] = 0
|
|
end
|
|
for group, _ in pairs(armor.registered_groups) do
|
|
self.def[name].groups[group] = 0
|
|
end
|
|
local skin = self:get_player_skin(name)
|
|
self.textures[name] = {
|
|
skin = skin..".png",
|
|
armor = "3d_armor_trans.png",
|
|
wielditem = "3d_armor_trans.png",
|
|
preview = armor.default_skin.."_preview.png",
|
|
}
|
|
local texture_path = minetest.get_modpath("player_textures")
|
|
if texture_path then
|
|
local dir_list = minetest.get_dir_list(texture_path.."/textures")
|
|
for _, fn in pairs(dir_list) do
|
|
if fn == "player_"..name..".png" then
|
|
self.textures[name].skin = fn
|
|
break
|
|
end
|
|
end
|
|
end
|
|
self:set_player_armor(player)
|
|
return true
|
|
end
|
|
|
|
armor.set_player_armor = function(self, player)
|
|
local name, player_inv = self:get_valid_player(player, "[set_player_armor]")
|
|
if not name then
|
|
return
|
|
end
|
|
local state = 0
|
|
local count = 0
|
|
local material = {count=1}
|
|
local preview = armor:get_preview(name)
|
|
local texture = "3d_armor_trans.png"
|
|
local textures = {}
|
|
local physics = {}
|
|
local attributes = {}
|
|
local levels = {}
|
|
local groups = {}
|
|
local change = {}
|
|
for _, phys in pairs(self.physics) do
|
|
physics[phys] = 1
|
|
end
|
|
for _, attr in pairs(self.attributes) do
|
|
attributes[attr] = 0
|
|
end
|
|
for group, _ in pairs(self.registered_groups) do
|
|
change[group] = 1
|
|
levels[group] = 0
|
|
end
|
|
local list = player_inv:get_list("armor")
|
|
for i, stack in pairs(list) do
|
|
if stack:get_count() == 1 then
|
|
local def = stack:get_definition()
|
|
for _, element in pairs(self.elements) do
|
|
if def.groups["armor_"..element] then
|
|
if def.armor_groups then
|
|
for group, level in pairs(def.armor_groups) do
|
|
if levels[group] then
|
|
levels[group] = levels[group] + level
|
|
end
|
|
end
|
|
else
|
|
local level = def.groups["armor_"..element]
|
|
levels["fleshy"] = levels["fleshy"] + level
|
|
end
|
|
end
|
|
-- DEPRECATED, use armor_groups instead
|
|
if def.groups["armor_radiation"] and levels["radiation"] then
|
|
levels["radiation"] = def.groups["armor_radiation"]
|
|
end
|
|
end
|
|
local item = stack:get_name()
|
|
local tex = def.texture or item:gsub("%:", "_")
|
|
tex = tex:gsub(".png$", "")
|
|
local prev = def.preview or tex.."_preview"
|
|
prev = prev:gsub(".png$", "")
|
|
texture = texture.."^"..tex..".png"
|
|
preview = preview.."^"..prev..".png"
|
|
state = state + stack:get_wear()
|
|
count = count + 1
|
|
for _, phys in pairs(self.physics) do
|
|
local value = def.groups["physics_"..phys] or 0
|
|
physics[phys] = physics[phys] + value
|
|
end
|
|
for _, attr in pairs(self.attributes) do
|
|
local value = def.groups["armor_"..attr] or 0
|
|
attributes[attr] = attributes[attr] + value
|
|
end
|
|
local mat = string.match(item, "%:.+_(.+)$")
|
|
if material.name then
|
|
if material.name == mat then
|
|
material.count = material.count + 1
|
|
end
|
|
else
|
|
material.name = mat
|
|
end
|
|
end
|
|
end
|
|
for group, level in pairs(levels) do
|
|
if level > 0 then
|
|
level = level * armor.config.level_multiplier
|
|
if material.name and material.count == #self.elements then
|
|
level = level * 1.1
|
|
end
|
|
end
|
|
local base = self.registered_groups[group]
|
|
self.def[name].groups[group] = level
|
|
if level > base then
|
|
level = base
|
|
end
|
|
groups[group] = base - level
|
|
change[group] = groups[group] / base
|
|
end
|
|
for _, attr in pairs(self.attributes) do
|
|
self.def[name][attr] = attributes[attr]
|
|
end
|
|
for _, phys in pairs(self.physics) do
|
|
self.def[name][phys] = physics[phys]
|
|
end
|
|
if use_armor_monoid then
|
|
armor_monoid.monoid:add_change(player, change, "3d_armor:armor")
|
|
else
|
|
player:set_armor_groups(groups)
|
|
end
|
|
if use_player_monoids then
|
|
player_monoids.speed:add_change(player, physics.speed,
|
|
"3d_armor:physics")
|
|
player_monoids.jump:add_change(player, physics.jump,
|
|
"3d_armor:physics")
|
|
player_monoids.gravity:add_change(player, physics.gravity,
|
|
"3d_armor:physics")
|
|
else
|
|
player:set_physics_override(physics)
|
|
end
|
|
self.textures[name].armor = texture
|
|
self.textures[name].preview = preview
|
|
self.def[name].level = self.def[name].groups.fleshy or 0
|
|
self.def[name].state = state
|
|
self.def[name].count = count
|
|
self:update_player_visuals(player)
|
|
self:run_callbacks("on_update", player)
|
|
end
|
|
|
|
armor.punch = function(self, player, hitter, time_from_last_punch, tool_capabilities)
|
|
local name, player_inv = self:get_valid_player(player, "[punch]")
|
|
if not name then
|
|
return
|
|
end
|
|
local state = 0
|
|
local count = 0
|
|
local recip = true
|
|
local default_groups = {cracky=3, snappy=3, choppy=3, crumbly=3, level=1}
|
|
local list = player_inv:get_list("armor")
|
|
for i, stack in pairs(list) do
|
|
if stack:get_count() == 1 then
|
|
local name = stack:get_name()
|
|
local use = minetest.get_item_group(name, "armor_use") or 0
|
|
local damage = use > 0
|
|
local def = stack:get_definition() or {}
|
|
if type(def.on_punch) == "function" then
|
|
damage = def.on_punch(player, hitter, time_from_last_punch,
|
|
tool_capabilities) ~= false and damage == true
|
|
end
|
|
if damage == true and tool_capabilities then
|
|
local damage_groups = def.damage_groups or default_groups
|
|
local level = damage_groups.level or 0
|
|
local groupcaps = tool_capabilities.groupcaps or {}
|
|
local uses = 0
|
|
damage = false
|
|
for group, caps in pairs(groupcaps) do
|
|
local maxlevel = caps.maxlevel or 0
|
|
local diff = maxlevel - level
|
|
if diff == 0 then
|
|
diff = 1
|
|
end
|
|
if diff > 0 and caps.times then
|
|
local group_level = damage_groups[group]
|
|
if group_level then
|
|
local time = caps.times[group_level]
|
|
if time then
|
|
local dt = time_from_last_punch or 0
|
|
if dt > time / diff then
|
|
if caps.uses then
|
|
uses = caps.uses * math.pow(3, diff)
|
|
end
|
|
damage = true
|
|
break
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
if damage == true and recip == true and hitter and
|
|
def.reciprocate_damage == true and uses > 0 then
|
|
local item = hitter:get_wielded_item()
|
|
if item and item:get_name() ~= "" then
|
|
item:add_wear(65535 / uses)
|
|
hitter:set_wielded_item(item)
|
|
end
|
|
-- reciprocate tool damage only once
|
|
recip = false
|
|
end
|
|
end
|
|
if damage == true and hitter == "fire" then
|
|
damage = minetest.get_item_group(name, "flammable") > 0
|
|
end
|
|
if damage == true then
|
|
local old_stack = ItemStack(stack)
|
|
stack:add_wear(use)
|
|
self:set_inventory_stack(player, i, stack)
|
|
self:run_callbacks("on_damage", player, stack, i)
|
|
if stack:get_count() == 0 then
|
|
self:run_callbacks("on_unequip", player, old_stack, i)
|
|
self:run_callbacks("on_destroy", player, old_stack, i)
|
|
self:set_player_armor(player)
|
|
end
|
|
end
|
|
state = state + stack:get_wear()
|
|
count = count + 1
|
|
end
|
|
end
|
|
self.def[name].state = state
|
|
self.def[name].count = count
|
|
end
|
|
|
|
armor.get_player_skin = function(self, name)
|
|
local skin = nil
|
|
if self.skin_mod == "skins" or self.skin_mod == "simple_skins" then
|
|
skin = skins.skins[name]
|
|
elseif self.skin_mod == "u_skins" then
|
|
skin = u_skins.u_skins[name]
|
|
elseif self.skin_mod == "wardrobe" then
|
|
local skins = wardrobe.playerSkins or {}
|
|
if skins[name] then
|
|
skin = string.gsub(skins[name], "%.png$","")
|
|
end
|
|
end
|
|
return skin or armor.default_skin
|
|
end
|
|
|
|
armor.add_preview = function(self, preview)
|
|
skin_previews[preview] = true
|
|
end
|
|
|
|
armor.get_preview = function(self, name)
|
|
local preview = armor:get_player_skin(name).."_preview.png"
|
|
if skin_previews[preview] then
|
|
return preview
|
|
end
|
|
return "character_preview.png"
|
|
end
|
|
|
|
armor.get_armor_formspec = function(self, name, listring)
|
|
if armor.def[name].init_time == 0 then
|
|
return "label[0,0;Armor not initialized!]"
|
|
end
|
|
local formspec = armor.formspec..
|
|
"list[detached:"..name.."_armor;armor;0,0.5;2,3;]"
|
|
if listring == true then
|
|
formspec = formspec.."listring[current_player;main]"..
|
|
"listring[detached:"..name.."_armor;armor]"
|
|
end
|
|
formspec = formspec:gsub("armor_preview", armor.textures[name].preview)
|
|
formspec = formspec:gsub("armor_level", armor.def[name].level)
|
|
for _, attr in pairs(self.attributes) do
|
|
formspec = formspec:gsub("armor_attr_"..attr, armor.def[name][attr])
|
|
end
|
|
for _, group in pairs(self.attributes) do
|
|
formspec = formspec:gsub("armor_group_"..group, armor.def[name][group])
|
|
end
|
|
return formspec
|
|
end
|
|
|
|
armor.update_inventory = function(self, player)
|
|
-- DEPRECATED: Legacy inventory support
|
|
end
|
|
|
|
armor.set_inventory_stack = function(self, player, i, stack)
|
|
local msg = "[set_inventory_stack]"
|
|
local name = player:get_player_name()
|
|
if not name then
|
|
minetest.log("warning", "3d_armor: Player name is nil "..msg)
|
|
return
|
|
end
|
|
local player_inv = player:get_inventory()
|
|
local armor_inv = minetest.get_inventory({type="detached", name=name.."_armor"})
|
|
if not player_inv then
|
|
minetest.log("warning", "3d_armor: Player inventory is nil "..msg)
|
|
return
|
|
elseif not armor_inv then
|
|
minetest.log("warning", "3d_armor: Detached armor inventory is nil "..msg)
|
|
return
|
|
end
|
|
player_inv:set_stack("armor", i, stack)
|
|
armor_inv:set_stack("armor", i, stack)
|
|
end
|
|
|
|
armor.get_valid_player = function(self, player, msg)
|
|
msg = msg or ""
|
|
if not player then
|
|
minetest.log("warning", "3d_armor: Player reference is nil "..msg)
|
|
return
|
|
end
|
|
local name = player:get_player_name()
|
|
if not name then
|
|
minetest.log("warning", "3d_armor: Player name is nil "..msg)
|
|
return
|
|
end
|
|
local inv = player:get_inventory()
|
|
if not inv then
|
|
minetest.log("warning", "3d_armor: Player inventory is nil "..msg)
|
|
return
|
|
end
|
|
return name, inv
|
|
end
|
|
|
|
armor.drop_armor = function(pos, stack)
|
|
local obj = minetest.add_item(pos, stack)
|
|
if obj then
|
|
obj:setvelocity({x=math.random(-1, 1), y=5, z=math.random(-1, 1)})
|
|
end
|
|
end
|
|
|