mirror of
https://git.minetest.land/VoxeLibre/VoxeLibre.git
synced 2025-01-10 00:59:35 +01:00
571 lines
16 KiB
Lua
571 lines
16 KiB
Lua
local minetest, math, vector = minetest, math, vector
|
|
local modname = minetest.get_current_modname()
|
|
local S = minetest.get_translator(modname)
|
|
|
|
mcl_shields = {
|
|
types = {
|
|
mob = true,
|
|
player = true,
|
|
arrow = true,
|
|
generic = true,
|
|
explosion = true,
|
|
dragon_breath = true,
|
|
},
|
|
enchantments = {"mending", "unbreaking"},
|
|
players = {},
|
|
}
|
|
|
|
local interact_priv = minetest.registered_privileges.interact
|
|
interact_priv.give_to_singleplayer = false
|
|
interact_priv.give_to_admin = false
|
|
|
|
local overlay = mcl_enchanting.overlay
|
|
local hud = "mcl_shield_hud.png"
|
|
|
|
minetest.register_tool("mcl_shields:shield", {
|
|
description = S("Shield"),
|
|
_doc_items_longdesc = S("A shield is a tool used for protecting the player against attacks."),
|
|
inventory_image = "mcl_shield.png",
|
|
stack_max = 1,
|
|
groups = {
|
|
shield = 1,
|
|
weapon = 1,
|
|
enchantability = -1,
|
|
offhand_item = 1,
|
|
},
|
|
sound = {breaks = "default_tool_breaks"},
|
|
_repair_material = "group:wood",
|
|
wield_scale = vector.new(2, 2, 2),
|
|
_mcl_wieldview_item = "",
|
|
})
|
|
|
|
local function wielded_item(obj, i)
|
|
local itemstack = obj:get_wielded_item()
|
|
if i == 1 then
|
|
itemstack = obj:get_inventory():get_stack("offhand", 1)
|
|
end
|
|
return itemstack:get_name()
|
|
end
|
|
|
|
function mcl_shields.wielding_shield(obj, i)
|
|
return wielded_item(obj, i):find("mcl_shields:shield")
|
|
end
|
|
|
|
local function shield_is_enchanted(obj, i)
|
|
return mcl_enchanting.is_enchanted(wielded_item(obj, i))
|
|
end
|
|
|
|
minetest.register_entity("mcl_shields:shield_entity", {
|
|
initial_properties = {
|
|
visual = "mesh",
|
|
mesh = "mcl_shield.obj",
|
|
physical = false,
|
|
pointable = false,
|
|
collide_with_objects = false,
|
|
textures = {"mcl_shield_base_nopattern.png"},
|
|
visual_size = vector.new(1, 1, 1),
|
|
},
|
|
_blocking = false,
|
|
_shield_number = 2,
|
|
_texture_copy = "",
|
|
on_step = function(self, dtime, moveresult)
|
|
local player = self.object:get_attach()
|
|
if not player then
|
|
self.object:remove()
|
|
return
|
|
end
|
|
local shield_texture = "mcl_shield_base_nopattern.png"
|
|
local i = self._shield_number
|
|
local item = wielded_item(player, i)
|
|
|
|
if item ~= "mcl_shields:shield" and item ~= "mcl_shields:shield_enchanted" then
|
|
local itemstack = player:get_wielded_item()
|
|
if i == 1 then
|
|
itemstack = player:get_inventory():get_stack("offhand", 1)
|
|
end
|
|
local meta_texture = itemstack:get_meta():get_string("mcl_shields:shield_custom_pattern_texture")
|
|
if meta_texture ~= "" then
|
|
shield_texture = meta_texture
|
|
else
|
|
local color = minetest.registered_items[item]._shield_color
|
|
if color then
|
|
shield_texture = "mcl_shield_base_nopattern.png^(mcl_shield_pattern_base.png^[colorize:" .. color .. ")"
|
|
end
|
|
end
|
|
end
|
|
|
|
if shield_is_enchanted(player, i) then
|
|
shield_texture = shield_texture .. overlay
|
|
end
|
|
|
|
if self._texture_copy ~= shield_texture then
|
|
self.object:set_properties({textures = {shield_texture}})
|
|
end
|
|
|
|
self._texture_copy = shield_texture
|
|
end,
|
|
})
|
|
|
|
for _, e in pairs(mcl_shields.enchantments) do
|
|
mcl_enchanting.enchantments[e].secondary.shield = true
|
|
end
|
|
|
|
function mcl_shields.is_blocking(obj)
|
|
if not obj:is_player() then return end
|
|
if mcl_shields.players[obj] then
|
|
local blocking = mcl_shields.players[obj].blocking
|
|
if blocking <= 0 then
|
|
return
|
|
end
|
|
|
|
local shieldstack = obj:get_wielded_item()
|
|
if blocking == 1 then
|
|
shieldstack = obj:get_inventory():get_stack("offhand", 1)
|
|
end
|
|
return blocking, shieldstack
|
|
end
|
|
end
|
|
|
|
mcl_damage.register_modifier(function(obj, damage, reason)
|
|
local type = reason.type
|
|
local damager = reason.direct
|
|
local blocking, shieldstack = mcl_shields.is_blocking(obj)
|
|
|
|
if not (obj:is_player() and blocking and mcl_shields.types[type] and damager) then
|
|
return
|
|
end
|
|
|
|
local entity = damager:get_luaentity()
|
|
if entity and entity._shooter then
|
|
damager = entity._shooter
|
|
end
|
|
|
|
local dpos = damager:get_pos()
|
|
|
|
-- Used for removed / killed entities before the projectile hits the player
|
|
if entity and not entity._shooter and entity._saved_shooter_pos then
|
|
dpos = entity._saved_shooter_pos
|
|
end
|
|
|
|
if not dpos or vector.dot(obj:get_look_dir(), vector.subtract(dpos, obj:get_pos())) < 0 then
|
|
return
|
|
end
|
|
|
|
local durability = 336
|
|
local unbreaking = mcl_enchanting.get_enchantment(shieldstack, mcl_shields.enchantments[2])
|
|
if unbreaking > 0 then
|
|
durability = durability * (unbreaking + 1)
|
|
end
|
|
|
|
if not minetest.is_creative_enabled(obj:get_player_name()) and damage >= 3 then
|
|
shieldstack:add_wear(65535 / durability)
|
|
if blocking == 2 then
|
|
obj:set_wielded_item(shieldstack)
|
|
else
|
|
obj:get_inventory():set_stack("offhand", 1, shieldstack)
|
|
mcl_inventory.update_inventory_formspec(obj)
|
|
end
|
|
end
|
|
minetest.sound_play({name = "mcl_block"}, {pos = obj:get_pos(), max_hear_distance = 16})
|
|
return 0
|
|
end)
|
|
|
|
local function modify_shield(player, vpos, vrot, i)
|
|
local arm = "Right"
|
|
if i == 1 then
|
|
arm = "Left"
|
|
end
|
|
local shield = mcl_shields.players[player].shields[i]
|
|
if shield then
|
|
shield:set_attach(player, "Arm_" .. arm, vpos, vrot, false)
|
|
end
|
|
end
|
|
|
|
local function set_shield(player, block, i)
|
|
if block then
|
|
if i == 1 then
|
|
modify_shield(player, vector.new(-9, 4, 0.5), vector.new(80, 100, 0), i) -- TODO
|
|
else
|
|
modify_shield(player, vector.new(-8, 4, -2.5), vector.new(80, 80, 0), i)
|
|
end
|
|
else
|
|
if i == 1 then
|
|
modify_shield(player, vector.new(-3, -5, 0), vector.new(0, 180, 0), i)
|
|
else
|
|
modify_shield(player, vector.new(3, -5, 0), vector.new(0, 0, 0), i)
|
|
end
|
|
end
|
|
local shield = mcl_shields.players[player].shields[i]
|
|
if not shield then return end
|
|
|
|
local luaentity = shield:get_luaentity()
|
|
if not luaentity then return end
|
|
|
|
luaentity._blocking = block
|
|
end
|
|
|
|
local function set_interact(player, interact)
|
|
local player_name = player:get_player_name()
|
|
local privs = minetest.get_player_privs(player_name)
|
|
if privs.interact == interact then
|
|
return
|
|
end
|
|
local meta = player:get_meta()
|
|
|
|
if interact and meta:get_int("mcl_shields:interact_revoked") ~= 0 then
|
|
meta:set_int("mcl_shields:interact_revoked", 0)
|
|
privs.interact = true
|
|
elseif not interact then
|
|
meta:set_int("mcl_shields:interact_revoked", privs.interact and 1 or 0)
|
|
privs.interact = nil
|
|
end
|
|
|
|
minetest.set_player_privs(player_name, privs)
|
|
end
|
|
|
|
-- Prevent player from being able to circumvent interact privilage removal by
|
|
-- using shield.
|
|
minetest.register_on_priv_revoke(function(name, revoker, priv)
|
|
if priv == "interact" and revoker then
|
|
local player = minetest.get_player_by_name(name)
|
|
if not player then
|
|
return
|
|
end
|
|
local meta = player:get_meta()
|
|
meta:set_int("mcl_shields:interact_revoked", 0)
|
|
end
|
|
end)
|
|
|
|
local shield_hud = {}
|
|
|
|
local function remove_shield_hud(player)
|
|
set_interact(player, true)
|
|
playerphysics.remove_physics_factor(player, "speed", "shield_speed")
|
|
|
|
if not shield_hud[player] then return end --this function takes a long time. only run it when necessary
|
|
player:hud_remove(shield_hud[player])
|
|
shield_hud[player] = nil
|
|
set_shield(player, false, 1)
|
|
set_shield(player, false, 2)
|
|
|
|
local hf = player:hud_get_flags()
|
|
if not hf.wielditem then
|
|
player:hud_set_flags({wielditem = true})
|
|
end
|
|
end
|
|
|
|
local function add_shield_entity(player, i)
|
|
local shield = minetest.add_entity(player:get_pos(), "mcl_shields:shield_entity")
|
|
shield:get_luaentity()._shield_number = i
|
|
mcl_shields.players[player].shields[i] = shield
|
|
set_shield(player, false, i)
|
|
end
|
|
|
|
local function remove_shield_entity(player, i)
|
|
local shields = mcl_shields.players[player].shields
|
|
if shields[i] then
|
|
shields[i]:remove()
|
|
shields[i] = nil
|
|
end
|
|
end
|
|
|
|
local function is_rmb_conflicting_node(nodename)
|
|
local nodedef = minetest.registered_nodes[nodename] or {}
|
|
return nodedef.on_rightclick
|
|
end
|
|
|
|
local function handle_blocking(player)
|
|
local player_shield = mcl_shields.players[player]
|
|
local rmb = player:get_player_control().RMB
|
|
if not rmb then
|
|
player_shield.blocking = 0
|
|
return
|
|
end
|
|
|
|
local pointed_thing = mcl_util.get_pointed_thing(player, true)
|
|
local wielded_stack = player:get_wielded_item()
|
|
|
|
local shield_in_offhand = mcl_shields.wielding_shield(player, 1)
|
|
local shield_in_hand = mcl_shields.wielding_shield(player)
|
|
local not_blocking = player_shield.blocking == 0
|
|
|
|
if pointed_thing and pointed_thing.type == "node" then
|
|
local pointed_node = minetest.get_node(pointed_thing.under)
|
|
if minetest.get_item_group(pointed_node.name, "container") > 1
|
|
or is_rmb_conflicting_node(pointed_node.name)
|
|
or wielded_stack:get_definition().type == "node" then
|
|
return
|
|
end
|
|
end
|
|
|
|
if shield_in_hand then
|
|
if not_blocking then
|
|
minetest.after(0.05, function()
|
|
if (not_blocking or not shield_in_offhand) and shield_in_hand and rmb then
|
|
player_shield.blocking = 2
|
|
set_shield(player, true, 2)
|
|
end
|
|
end)
|
|
elseif not shield_in_offhand then
|
|
player_shield.blocking = 2
|
|
end
|
|
elseif shield_in_offhand then
|
|
local offhand_can_block = minetest.get_item_group(wielded_item(player), "cannot_block") ~= 1
|
|
|
|
if not offhand_can_block then
|
|
return
|
|
end
|
|
if not_blocking then
|
|
minetest.after(0.05, function()
|
|
if (not_blocking or not shield_in_hand) and shield_in_offhand
|
|
and rmb and offhand_can_block then
|
|
player_shield.blocking = 1
|
|
set_shield(player, true, 1)
|
|
end
|
|
end)
|
|
elseif not shield_in_hand then
|
|
player_shield.blocking = 1
|
|
end
|
|
else
|
|
player_shield.blocking = 0
|
|
end
|
|
end
|
|
|
|
local function update_shield_entity(player, blocking, i)
|
|
local shield = mcl_shields.players[player].shields[i]
|
|
if mcl_shields.wielding_shield(player, i) then
|
|
if not shield then
|
|
add_shield_entity(player, i)
|
|
else
|
|
if blocking == i then
|
|
if shield:get_luaentity() and not shield:get_luaentity()._blocking then
|
|
set_shield(player, true, i)
|
|
end
|
|
else
|
|
set_shield(player, false, i)
|
|
end
|
|
end
|
|
elseif shield then
|
|
remove_shield_entity(player, i)
|
|
end
|
|
end
|
|
|
|
local function add_shield_hud(shieldstack, player, blocking)
|
|
local texture = hud
|
|
if mcl_enchanting.is_enchanted(shieldstack:get_name()) then
|
|
texture = texture .. overlay
|
|
end
|
|
local offset = 100
|
|
if blocking == 1 then
|
|
texture = texture .. "^[transform4"
|
|
offset = -100
|
|
else
|
|
player:hud_set_flags({wielditem = false})
|
|
end
|
|
shield_hud[player] = player:hud_add({
|
|
hud_elem_type = "image",
|
|
position = {x = 0.5, y = 0.5},
|
|
scale = {x = -101, y = -101},
|
|
offset = {x = offset, y = 0},
|
|
text = texture,
|
|
z_index = -200,
|
|
})
|
|
playerphysics.add_physics_factor(player, "speed", "shield_speed", 0.5)
|
|
set_interact(player, false)
|
|
end
|
|
|
|
local function update_shield_hud(player, blocking, shieldstack)
|
|
local shieldhud = shield_hud[player]
|
|
if not shieldhud then
|
|
add_shield_hud(shieldstack, player, blocking)
|
|
return
|
|
end
|
|
|
|
local wielditem = player:hud_get_flags().wielditem
|
|
if blocking == 1 then
|
|
if not wielditem then
|
|
player:hud_change(shieldhud, "text", hud .. "^[transform4")
|
|
player:hud_change(shieldhud, "offset", {x = -100, y = 0})
|
|
player:hud_set_flags({wielditem = true})
|
|
end
|
|
elseif wielditem then
|
|
player:hud_change(shieldhud, "text", hud)
|
|
player:hud_change(shieldhud, "offset", {x = 100, y = 0})
|
|
player:hud_set_flags({wielditem = false})
|
|
end
|
|
|
|
local image = player:hud_get(shieldhud).text
|
|
local enchanted = hud .. overlay
|
|
local enchanted1 = image == enchanted
|
|
local enchanted2 = image == enchanted .. "^[transform4"
|
|
if mcl_enchanting.is_enchanted(shieldstack:get_name()) then
|
|
if not enchanted1 and not enchanted2 then
|
|
if blocking == 1 then
|
|
player:hud_change(shieldhud, "text", hud .. overlay .. "^[transform4")
|
|
else
|
|
player:hud_change(shieldhud, "text", hud .. overlay)
|
|
end
|
|
end
|
|
elseif enchanted1 or enchanted2 then
|
|
if blocking == 1 then
|
|
player:hud_change(shieldhud, "text", hud .. "^[transform4")
|
|
else
|
|
player:hud_change(shieldhud, "text", hud)
|
|
end
|
|
end
|
|
end
|
|
|
|
minetest.register_globalstep(function(dtime)
|
|
for _, player in pairs(minetest.get_connected_players()) do
|
|
|
|
handle_blocking(player)
|
|
|
|
local blocking, shieldstack = mcl_shields.is_blocking(player)
|
|
|
|
if blocking then
|
|
update_shield_hud(player, blocking, shieldstack)
|
|
else
|
|
remove_shield_hud(player)
|
|
end
|
|
|
|
for i = 1, 2 do
|
|
update_shield_entity(player, blocking, i)
|
|
end
|
|
end
|
|
end)
|
|
|
|
local keep_inventory = vl_tuning.setting("gamerule:keepInventory")
|
|
minetest.register_on_dieplayer(function(player)
|
|
remove_shield_hud(player)
|
|
if not keep_inventory[1] then
|
|
remove_shield_entity(player, 1)
|
|
remove_shield_entity(player, 2)
|
|
end
|
|
end)
|
|
|
|
minetest.register_on_leaveplayer(function(player)
|
|
shield_hud[player] = nil
|
|
mcl_shields.players[player] = nil
|
|
end)
|
|
|
|
minetest.register_craft({
|
|
output = "mcl_shields:shield",
|
|
recipe = {
|
|
{"group:wood", "mcl_core:iron_ingot", "group:wood"},
|
|
{"group:wood", "group:wood", "group:wood"},
|
|
{"", "group:wood", ""},
|
|
}
|
|
})
|
|
|
|
local color_names = {
|
|
["white"] = S("White Shield"),
|
|
["grey"] = S("Grey Shield"),
|
|
["silver"] = S("Light Grey Shield"),
|
|
["black"] = S("Black Shield"),
|
|
["red"] = S("Red Shield"),
|
|
["yellow"] = S("Yellow Shield"),
|
|
["green"] = S("Green Shield"),
|
|
["cyan"] = S("Cyan Shield"),
|
|
["blue"] = S("Blue Shield"),
|
|
["magenta"] = S("Magenta Shield"),
|
|
["orange"] = S("Orange Shield"),
|
|
["purple"] = S("Purple Shield"),
|
|
["brown"] = S("Brown Shield"),
|
|
["pink"] = S("Pink Shield"),
|
|
["lime"] = S("Lime Shield"),
|
|
["light_blue"] = S("Light Blue Shield"),
|
|
}
|
|
|
|
|
|
for _, colortab in pairs(mcl_banners.colors) do
|
|
local color = colortab[1]
|
|
minetest.register_tool("mcl_shields:shield_" .. color, {
|
|
description = color_names[color],
|
|
_doc_items_longdesc = S("A shield is a tool used for protecting the player against attacks."),
|
|
inventory_image = "mcl_shield.png^(mcl_shield_item_overlay.png^[colorize:" .. colortab[4] ..")",
|
|
stack_max = 1,
|
|
groups = {
|
|
shield = 1,
|
|
weapon = 1,
|
|
enchantability = -1,
|
|
not_in_creative_inventory = 1,
|
|
offhand_item = 1,
|
|
},
|
|
sound = {breaks = "default_tool_breaks"},
|
|
_repair_material = "group:wood",
|
|
wield_scale = vector.new(2, 2, 2),
|
|
_shield_color = colortab[4],
|
|
_mcl_wieldview_item = "",
|
|
})
|
|
|
|
local banner = "mcl_banners:banner_item_" .. color
|
|
minetest.register_craft({
|
|
type = "shapeless",
|
|
output = "mcl_shields:shield_" .. color,
|
|
recipe = {"mcl_shields:shield", banner},
|
|
})
|
|
minetest.register_craft({
|
|
type = "shapeless",
|
|
output = "mcl_shields:shield_" .. color .. "_enchanted",
|
|
recipe = {"mcl_shields:shield_enchanted", banner},
|
|
})
|
|
end
|
|
|
|
local function to_shield_texture(banner_texture)
|
|
return banner_texture
|
|
:gsub("mcl_banners_base_inverted.png", "mcl_shield_base_nopattern.png^mcl_shield_pattern_base.png")
|
|
:gsub("mcl_banners_banner_base.png", "mcl_shield_base_nopattern.png^mcl_shield_pattern_base.png")
|
|
:gsub("mcl_banners_base", "mcl_shield_pattern_base")
|
|
:gsub("mcl_banners", "mcl_shield_pattern")
|
|
end
|
|
|
|
local function craft_banner_on_shield(itemstack, player, old_craft_grid, craft_inv)
|
|
if not string.find(itemstack:get_name(), "mcl_shields:shield_") then
|
|
return itemstack
|
|
end
|
|
|
|
local shield_stack
|
|
for i = 1, player:get_inventory():get_size("craft") do
|
|
local stack = old_craft_grid[i]
|
|
local name = stack:get_name()
|
|
if minetest.get_item_group(name, "shield") then
|
|
shield_stack = stack
|
|
break
|
|
end
|
|
end
|
|
|
|
for i = 1, player:get_inventory():get_size("craft") do
|
|
local banner_stack = old_craft_grid[i]
|
|
local banner_name = banner_stack:get_name()
|
|
if string.find(banner_name, "mcl_banners:banner") and shield_stack then
|
|
local banner_meta = banner_stack:get_meta()
|
|
local layers_meta = banner_meta:get_string("layers")
|
|
local new_shield_meta = itemstack:get_meta()
|
|
if layers_meta ~= "" then
|
|
local color = mcl_banners.color_reverse(banner_name)
|
|
local layers = minetest.deserialize(layers_meta)
|
|
local texture = mcl_banners.make_banner_texture(color, layers)
|
|
new_shield_meta:set_string("description", mcl_banners.make_advanced_banner_description(itemstack:get_description(), layers))
|
|
new_shield_meta:set_string("mcl_shields:shield_custom_pattern_texture", to_shield_texture(texture))
|
|
end
|
|
itemstack:set_wear(shield_stack:get_wear())
|
|
break
|
|
end
|
|
end
|
|
end
|
|
|
|
minetest.register_craft_predict(function(itemstack, player, old_craft_grid, craft_inv)
|
|
return craft_banner_on_shield(itemstack, player, old_craft_grid, craft_inv)
|
|
end)
|
|
|
|
minetest.register_on_craft(function(itemstack, player, old_craft_grid, craft_inv)
|
|
return craft_banner_on_shield(itemstack, player, old_craft_grid, craft_inv)
|
|
end)
|
|
|
|
minetest.register_on_joinplayer(function(player)
|
|
mcl_shields.players[player] = {
|
|
shields = {},
|
|
blocking = 0,
|
|
}
|
|
remove_shield_hud(player)
|
|
end)
|