forked from MineClone5/MineClone5
455 lines
18 KiB
Lua
455 lines
18 KiB
Lua
local S = minetest.get_translator(minetest.get_current_modname())
|
|
|
|
mcl_bows_s = {}
|
|
|
|
-- local arrows = {
|
|
-- ["mcl_bows:arrow"] = "mcl_bows:arrow_entity",
|
|
-- }
|
|
|
|
local GRAVITY = 9.81
|
|
local BOW_DURABILITY = 385
|
|
|
|
-- Charging time in microseconds
|
|
local _BOW_CHARGE_TIME_HALF = 350000 -- bow level 1
|
|
local _BOW_CHARGE_TIME_FULL = 900000 -- bow level 2 (full charge)
|
|
|
|
local BOW_CHARGE_TIME_HALF = 350000 -- bow level 1
|
|
local BOW_CHARGE_TIME_FULL = 900000 -- bow level 2 (full charge)
|
|
|
|
-- Factor to multiply with player speed while player uses bow
|
|
-- This emulates the sneak speed.
|
|
local PLAYER_USE_CROSSBOW_SPEED = tonumber(minetest.settings:get("movement_speed_crouch")) / tonumber(minetest.settings:get("movement_speed_walk"))
|
|
|
|
-- TODO: Use Minecraft speed (ca. 53 m/s)
|
|
-- Currently nerfed because at full speed the arrow would easily get out of the range of the loaded map.
|
|
local BOW_MAX_SPEED = 68
|
|
|
|
local function play_load_sound(id, pos)
|
|
minetest.sound_play("mcl_bows_crossbow_drawback_"..id, {pos=pos, max_hear_distance=12}, true)
|
|
end
|
|
|
|
--[[ Store the charging state of each player.
|
|
keys: player name
|
|
value:
|
|
nil = not charging or player not existing
|
|
number: currently charging, the number is the time from minetest.get_us_time
|
|
in which the charging has started
|
|
]]
|
|
local bow_load = {}
|
|
|
|
-- Another player table, this one stores the wield index of the bow being charged
|
|
local bow_index = {}
|
|
|
|
function mcl_bows_s.shoot_arrow_crossbow(arrow_item, pos, dir, yaw, shooter, power, damage, is_critical, crossbow_stack, collectable)
|
|
local obj = minetest.add_entity({x=pos.x,y=pos.y,z=pos.z}, arrow_item.."_entity")
|
|
if power == nil then
|
|
power = BOW_MAX_SPEED --19
|
|
end
|
|
if damage == nil then
|
|
damage = 3
|
|
end
|
|
local knockback
|
|
if crossbow_stack then
|
|
local enchantments = mcl_enchanting.get_enchantments(crossbow_stack)
|
|
if enchantments.piercing then
|
|
obj:get_luaentity()._piercing = 1 * enchantments.piercing
|
|
else
|
|
obj:get_luaentity()._piercing = 0
|
|
end
|
|
end
|
|
obj:set_velocity({x=dir.x*power, y=dir.y*power, z=dir.z*power})
|
|
obj:set_acceleration({x=0, y=-GRAVITY, z=0})
|
|
obj:set_yaw(yaw-math.pi/2)
|
|
local le = obj:get_luaentity()
|
|
le._shooter = shooter
|
|
le._source_object = shooter
|
|
le._damage = damage
|
|
le._is_critical = is_critical
|
|
le._startpos = pos
|
|
le._knockback = knockback
|
|
le._collectable = collectable
|
|
minetest.sound_play("mcl_bows_crossbow_shoot", {pos=pos, max_hear_distance=16}, true)
|
|
if shooter and shooter:is_player() then
|
|
if obj:get_luaentity().player == "" then
|
|
obj:get_luaentity().player = shooter
|
|
end
|
|
obj:get_luaentity().node = shooter:get_inventory():get_stack("main", 1):get_name()
|
|
end
|
|
return obj
|
|
end
|
|
|
|
local function get_arrow(player)
|
|
local inv = player:get_inventory()
|
|
local arrow_stack, arrow_stack_id
|
|
for i=1, inv:get_size("main") do
|
|
local it = inv:get_stack("main", i)
|
|
if not it:is_empty() and minetest.get_item_group(it:get_name(), "ammo_crossbow") ~= 0 then
|
|
arrow_stack = it
|
|
arrow_stack_id = i
|
|
break
|
|
end
|
|
end
|
|
return arrow_stack, arrow_stack_id
|
|
end
|
|
|
|
local function player_shoot_arrow(itemstack, player, power, damage, is_critical)
|
|
local has_multishot_enchantment = mcl_enchanting.has_enchantment(player:get_wielded_item(), "multishot")
|
|
local arrow_itemstring = wielditem:get_meta():get("arrow")
|
|
|
|
if not arrow_itemstring then
|
|
return false
|
|
end
|
|
|
|
local playerpos = player:get_pos()
|
|
local dir = player:get_look_dir()
|
|
local yaw = player:get_look_horizontal()
|
|
|
|
if has_multishot_enchantment then
|
|
mcl_bows_s.shoot_arrow_crossbow(arrow_itemstring, {x=playerpos.x,y=playerpos.y+1.5,z=playerpos.z}, {x=dir.x, y=dir.y, z=dir.z + .2}, yaw, player, power, damage, is_critical, player:get_wielded_item(), false)
|
|
mcl_bows_s.shoot_arrow_crossbow(arrow_itemstring, {x=playerpos.x,y=playerpos.y+1.5,z=playerpos.z}, {x=dir.x, y=dir.y, z=dir.z - .2}, yaw, player, power, damage, is_critical, player:get_wielded_item(), false)
|
|
mcl_bows_s.shoot_arrow_crossbow(arrow_itemstring, {x=playerpos.x,y=playerpos.y+1.5,z=playerpos.z}, dir, yaw, player, power, damage, is_critical, player:get_wielded_item(), true)
|
|
else
|
|
mcl_bows_s.shoot_arrow_crossbow(arrow_itemstring, {x=playerpos.x,y=playerpos.y+1.5,z=playerpos.z}, dir, yaw, player, power, damage, is_critical, player:get_wielded_item(), true)
|
|
end
|
|
return true
|
|
end
|
|
|
|
-- Bow item, uncharged state
|
|
minetest.register_tool("mcl_bows:crossbow", {
|
|
description = S("Corssbow"),
|
|
_tt_help = S("Launches arrows"),
|
|
_doc_items_longdesc = S("Bows are ranged weapons to shoot arrows at your foes.").."\n"..
|
|
S("The speed and damage of the arrow increases the longer you charge. The regular damage of the arrow is between 1 and 9. At full charge, there's also a 20% of a critical hit, dealing 10 damage instead."),
|
|
_doc_items_usagehelp = S("To use the bow, you first need to have at least one arrow anywhere in your inventory (unless in Creative Mode). Hold down the right mouse button to charge, release to shoot."),
|
|
_doc_items_durability = BOW_DURABILITY,
|
|
inventory_image = "mcl_bows_crossbow.png",
|
|
wield_scale = mcl_vars.tool_wield_scale,
|
|
stack_max = 1,
|
|
range = 4,
|
|
-- Trick to disable digging as well
|
|
on_use = function() return end,
|
|
on_place = function(itemstack, player, pointed_thing)
|
|
if pointed_thing and pointed_thing.type == "node" then
|
|
-- Call on_rightclick if the pointed node defines it
|
|
local node = minetest.get_node(pointed_thing.under)
|
|
if player and not player:get_player_control().sneak then
|
|
if minetest.registered_nodes[node.name] and minetest.registered_nodes[node.name].on_rightclick then
|
|
return minetest.registered_nodes[node.name].on_rightclick(pointed_thing.under, node, player, itemstack) or itemstack
|
|
end
|
|
end
|
|
end
|
|
|
|
itemstack:get_meta():set_string("active", "true")
|
|
return itemstack
|
|
end,
|
|
on_secondary_use = function(itemstack)
|
|
itemstack:get_meta():set_string("active", "true")
|
|
return itemstack
|
|
end,
|
|
groups = {weapon=1,weapon_ranged=1,crossbow=1,enchantability=1},
|
|
_mcl_uses = 326,
|
|
})
|
|
|
|
minetest.register_tool("mcl_bows:crossbow_loaded", {
|
|
description = S("Corssbow"),
|
|
_tt_help = S("Launches arrows"),
|
|
_doc_items_longdesc = S("Corssbow are ranged weapons to shoot arrows at your foes.").."\n"..
|
|
S("The speed and damage of the arrow increases the longer you charge. The regular damage of the arrow is between 1 and 9. At full charge, there's also a 20% of a critical hit, dealing 10 damage instead."),
|
|
_doc_items_usagehelp = S("To use the corssbow, you first need to have at least one arrow anywhere in your inventory (unless in Creative Mode). Hold down the right mouse button to charge, release to load an arrow into the chamber, then to shoot press left mouse."),
|
|
_doc_items_durability = BOW_DURABILITY,
|
|
inventory_image = "mcl_bows_crossbow_3.png",
|
|
wield_scale = mcl_vars.tool_wield_scale,
|
|
stack_max = 1,
|
|
range = 4,
|
|
-- Trick to disable digging as well
|
|
on_use = function() return end,
|
|
on_place = function(itemstack, player, pointed_thing)
|
|
if pointed_thing and pointed_thing.type == "node" then
|
|
-- Call on_rightclick if the pointed node defines it
|
|
local node = minetest.get_node(pointed_thing.under)
|
|
if player and not player:get_player_control().sneak then
|
|
if minetest.registered_nodes[node.name] and minetest.registered_nodes[node.name].on_rightclick then
|
|
return minetest.registered_nodes[node.name].on_rightclick(pointed_thing.under, node, player, itemstack) or itemstack
|
|
end
|
|
end
|
|
end
|
|
|
|
itemstack:get_meta():set_string("active", "true")
|
|
return itemstack
|
|
end,
|
|
on_secondary_use = function(itemstack)
|
|
itemstack:get_meta():set_string("active", "true")
|
|
return itemstack
|
|
end,
|
|
groups = {weapon=1,weapon_ranged=1,crossbow=1,enchantability=1},
|
|
_mcl_uses = 326,
|
|
})
|
|
|
|
-- Iterates through player inventory and resets all the bows in "charging" state back to their original stage
|
|
local function reset_bows(player)
|
|
local inv = player:get_inventory()
|
|
local list = inv:get_list("main")
|
|
for place, stack in pairs(list) do
|
|
if stack:get_name() == "mcl_bows:crossbow" or stack:get_name() == "mcl_bows:crossbow_enchanted" then
|
|
stack:get_meta():set_string("active", "")
|
|
elseif stack:get_name()=="mcl_bows:crossbow_0" or stack:get_name()=="mcl_bows:crossbow_1" or stack:get_name()=="mcl_bows:crossbow_2" then
|
|
stack:set_name("mcl_bows:crossbow")
|
|
stack:get_meta():set_string("active", "")
|
|
list[place] = stack
|
|
elseif stack:get_name()=="mcl_bows:crossbow_0_enchanted" or stack:get_name()=="mcl_bows:crossbow_1_enchanted" or stack:get_name()=="mcl_bows:crossbow_2_enchanted" then
|
|
stack:set_name("mcl_bows:crossbow_enchanted")
|
|
stack:get_meta():set_string("active", "")
|
|
list[place] = stack
|
|
end
|
|
end
|
|
inv:set_list("main", list)
|
|
end
|
|
|
|
-- Resets the bow charging state and player speed. To be used when the player is no longer charging the bow
|
|
local function reset_bow_state(player, also_reset_bows)
|
|
bow_load[player:get_player_name()] = nil
|
|
bow_index[player:get_player_name()] = nil
|
|
if minetest.get_modpath("playerphysics") then
|
|
playerphysics.remove_physics_factor(player, "speed", "mcl_bows:use_crossbow")
|
|
end
|
|
if also_reset_bows then
|
|
reset_bows(player)
|
|
end
|
|
end
|
|
|
|
-- Bow in charging state
|
|
for level=0, 2 do
|
|
minetest.register_tool("mcl_bows:crossbow_"..level, {
|
|
description = S("Crossbow"),
|
|
_doc_items_create_entry = false,
|
|
inventory_image = "mcl_bows_crossbow_"..level..".png",
|
|
wield_scale = mcl_vars.tool_wield_scale,
|
|
stack_max = 1,
|
|
range = 0, -- Pointing range to 0 to prevent punching with bow :D
|
|
groups = {not_in_creative_inventory=1, not_in_craft_guide=1, bow=1, enchantability=1},
|
|
-- Trick to disable digging as well
|
|
on_use = function() return end,
|
|
on_drop = function(itemstack, dropper, pos)
|
|
reset_bow_state(dropper)
|
|
itemstack:get_meta():set_string("active", "")
|
|
if mcl_enchanting.is_enchanted(itemstack:get_name()) then
|
|
itemstack:set_name("mcl_bows:crossbow_enchanted")
|
|
else
|
|
itemstack:set_name("mcl_bows:crossbow")
|
|
end
|
|
minetest.item_drop(itemstack, dropper, pos)
|
|
itemstack:take_item()
|
|
return itemstack
|
|
end,
|
|
-- Prevent accidental interaction with itemframes and other nodes
|
|
on_place = function(itemstack)
|
|
return itemstack
|
|
end,
|
|
_mcl_uses = 385,
|
|
})
|
|
end
|
|
|
|
|
|
controls.register_on_release(function(player, key, time)
|
|
if key~="RMB" then return end
|
|
--local inv = minetest.get_inventory({type="player", name=player:get_player_name()})
|
|
local wielditem = player:get_wielded_item()
|
|
if wielditem:get_name()=="mcl_bows:crossbow_2" and get_arrow(player) or wielditem:get_name()=="mcl_bows:crossbow_2" and minetest.is_creative_enabled(player:get_player_name()) or wielditem:get_name()=="mcl_bows:crossbow_2_enchanted" and get_arrow(player) or wielditem:get_name()=="mcl_bows:crossbow_2_enchanted" and minetest.is_creative_enabled(player:get_player_name()) then
|
|
local arrow_stack, arrow_stack_id = get_arrow(player)
|
|
local arrow_itemstring
|
|
|
|
if minetest.is_creative_enabled(player:get_player_name()) then
|
|
if arrow_stack then
|
|
arrow_itemstring = arrow_stack:get_name()
|
|
else
|
|
arrow_itemstring = "mcl_bows:arrow"
|
|
end
|
|
else
|
|
arrow_itemstring = arrow_stack:get_name()
|
|
arrow_stack:take_item()
|
|
player:get_inventory():set_stack("main", arrow_stack_id, arrow_stack)
|
|
end
|
|
|
|
wielditem:get_meta():set_string("arrow", arrow_itemstring)
|
|
|
|
if wielditem:get_name()=="mcl_bows:crossbow_2" then
|
|
wielditem:set_name("mcl_bows:crossbow_loaded")
|
|
else
|
|
wielditem:set_name("mcl_bows:crossbow_loaded_enchanted")
|
|
end
|
|
player:set_wielded_item(wielditem)
|
|
minetest.sound_play("mcl_bows_crossbow_load", {pos=player:get_pos(), max_hear_distance=16}, true)
|
|
else
|
|
reset_bow_state(player, true)
|
|
end
|
|
end)
|
|
|
|
controls.register_on_press(function(player, key, time)
|
|
if key~="LMB" then return end
|
|
wielditem = player:get_wielded_item()
|
|
if wielditem:get_name()=="mcl_bows:crossbow_loaded" or wielditem:get_name()=="mcl_bows:crossbow_loaded_enchanted" then
|
|
local enchanted = mcl_enchanting.is_enchanted(wielditem:get_name())
|
|
local speed, damage
|
|
local p_load = bow_load[player:get_player_name()]
|
|
local charge
|
|
-- Type sanity check
|
|
if type(p_load) == "number" then
|
|
charge = minetest.get_us_time() - p_load
|
|
else
|
|
-- In case something goes wrong ...
|
|
-- Just assume minimum charge.
|
|
charge = 0
|
|
minetest.log("warning", "[mcl_bows] Player "..player:get_player_name().." fires arrow with non-numeric bow_load!")
|
|
end
|
|
charge = math.max(math.min(charge, BOW_CHARGE_TIME_FULL), 0)
|
|
|
|
local charge_ratio = charge / BOW_CHARGE_TIME_FULL
|
|
charge_ratio = math.max(math.min(charge_ratio, 1), 0)
|
|
|
|
-- Calculate damage and speed
|
|
-- Fully charged
|
|
local is_critical = false
|
|
speed = BOW_MAX_SPEED
|
|
local r = math.random(1,5)
|
|
if r == 1 then
|
|
-- 20% chance for critical hit
|
|
damage = 10
|
|
is_critical = true
|
|
else
|
|
damage = 9
|
|
end
|
|
|
|
local has_shot = player_shoot_arrow(wielditem, player, speed, damage, is_critical)
|
|
|
|
if enchanted then
|
|
wielditem:set_name("mcl_bows:crossbow_enchanted")
|
|
else
|
|
wielditem:set_name("mcl_bows:crossbow")
|
|
end
|
|
|
|
if has_shot and not minetest.is_creative_enabled(player:get_player_name()) then
|
|
local durability = BOW_DURABILITY
|
|
local unbreaking = mcl_enchanting.get_enchantment(wielditem, "unbreaking")
|
|
local multishot = mcl_enchanting.get_enchantment(wielditem, "multishot")
|
|
if unbreaking > 0 then
|
|
durability = durability * (unbreaking + 1)
|
|
end
|
|
if multishot then
|
|
durability = durability / 3
|
|
end
|
|
wielditem:add_wear(65535/durability)
|
|
end
|
|
player:set_wielded_item(wielditem)
|
|
reset_bow_state(player, true)
|
|
end
|
|
end)
|
|
|
|
controls.register_on_hold(function(player, key, time)
|
|
local name = player:get_player_name()
|
|
local creative = minetest.is_creative_enabled(name)
|
|
if key ~= "RMB" then
|
|
return
|
|
end
|
|
--local inv = minetest.get_inventory({type="player", name=name})
|
|
local wielditem = player:get_wielded_item()
|
|
local enchantments = mcl_enchanting.get_enchantments(wielditem)
|
|
if enchantments.quick_charge then
|
|
BOW_CHARGE_TIME_HALF = _BOW_CHARGE_TIME_HALF - (enchantments.quick_charge * 0.13 * 1000000 * .5)
|
|
BOW_CHARGE_TIME_FULL = _BOW_CHARGE_TIME_FULL - (enchantments.quick_charge * 0.13 * 1000000)
|
|
else
|
|
BOW_CHARGE_TIME_HALF = _BOW_CHARGE_TIME_HALF
|
|
BOW_CHARGE_TIME_FULL = _BOW_CHARGE_TIME_FULL
|
|
end
|
|
|
|
if bow_load[name] == nil and (wielditem:get_name()=="mcl_bows:crossbow" or wielditem:get_name()=="mcl_bows:crossbow_enchanted") and wielditem:get_meta():get("active") and (creative or get_arrow(player)) then
|
|
local enchanted = mcl_enchanting.is_enchanted(wielditem:get_name())
|
|
if enchanted then
|
|
wielditem:set_name("mcl_bows:crossbow_0_enchanted")
|
|
play_load_sound(0, player:get_pos())
|
|
else
|
|
wielditem:set_name("mcl_bows:crossbow_0")
|
|
play_load_sound(0, player:get_pos())
|
|
end
|
|
player:set_wielded_item(wielditem)
|
|
if minetest.get_modpath("playerphysics") then
|
|
-- Slow player down when using bow
|
|
playerphysics.add_physics_factor(player, "speed", "mcl_bows:use_crossbow", PLAYER_USE_CROSSBOW_SPEED)
|
|
end
|
|
bow_load[name] = minetest.get_us_time()
|
|
bow_index[name] = player:get_wield_index()
|
|
else
|
|
if player:get_wield_index() == bow_index[name] then
|
|
if type(bow_load[name]) == "number" then
|
|
if wielditem:get_name() == "mcl_bows:crossbow_0" and minetest.get_us_time() - bow_load[name] >= BOW_CHARGE_TIME_HALF then
|
|
wielditem:set_name("mcl_bows:crossbow_1")
|
|
play_load_sound(1, player:get_pos())
|
|
elseif wielditem:get_name() == "mcl_bows:crossbow_0_enchanted" and minetest.get_us_time() - bow_load[name] >= BOW_CHARGE_TIME_HALF then
|
|
wielditem:set_name("mcl_bows:crossbow_1_enchanted")
|
|
play_load_sound(1, player:get_pos())
|
|
elseif wielditem:get_name() == "mcl_bows:crossbow_1" and minetest.get_us_time() - bow_load[name] >= BOW_CHARGE_TIME_FULL then
|
|
wielditem:set_name("mcl_bows:crossbow_2")
|
|
play_load_sound(2, player:get_pos())
|
|
elseif wielditem:get_name() == "mcl_bows:crossbow_1_enchanted" and minetest.get_us_time() - bow_load[name] >= BOW_CHARGE_TIME_FULL then
|
|
wielditem:set_name("mcl_bows:crossbow_2_enchanted")
|
|
play_load_sound(2, player:get_pos())
|
|
end
|
|
else
|
|
if wielditem:get_name() == "mcl_bows:crossbow_0" or wielditem:get_name() == "mcl_bows:crossbow_1" or wielditem:get_name() == "mcl_bows:crossbow_2" then
|
|
wielditem:set_name("mcl_bows:crossbow")
|
|
play_load_sound(1, player:get_pos())
|
|
elseif wielditem:get_name() == "mcl_bows:crossbow_0_enchanted" or wielditem:get_name() == "mcl_bows:crossbow_1_enchanted" or wielditem:get_name() == "mcl_bows:crossbow_2_enchanted" then
|
|
wielditem:set_name("mcl_bows:crossbow_enchanted")
|
|
play_load_sound(1, player:get_pos())
|
|
end
|
|
end
|
|
player:set_wielded_item(wielditem)
|
|
else
|
|
reset_bow_state(player, true)
|
|
end
|
|
end
|
|
end)
|
|
|
|
minetest.register_globalstep(function(dtime)
|
|
for _, player in pairs(minetest.get_connected_players()) do
|
|
local name = player:get_player_name()
|
|
local wielditem = player:get_wielded_item()
|
|
local wieldindex = player:get_wield_index()
|
|
--local controls = player:get_player_control()
|
|
if type(bow_load[name]) == "number" and ((wielditem:get_name()~="mcl_bows:crossbow_0" and wielditem:get_name()~="mcl_bows:crossbow_1" and wielditem:get_name()~="mcl_bows:crossbow_2" and wielditem:get_name()~="mcl_bows:crossbow_0_enchanted" and wielditem:get_name()~="mcl_bows:crossbow_1_enchanted" and wielditem:get_name()~="mcl_bows:crossbow_2_enchanted") or wieldindex ~= bow_index[name]) then
|
|
reset_bow_state(player, true)
|
|
end
|
|
end
|
|
end)
|
|
|
|
minetest.register_on_joinplayer(function(player)
|
|
reset_bows(player)
|
|
end)
|
|
|
|
minetest.register_on_leaveplayer(function(player)
|
|
reset_bow_state(player, true)
|
|
end)
|
|
|
|
if minetest.get_modpath("mcl_core") and minetest.get_modpath("mcl_mobitems") then
|
|
minetest.register_craft({
|
|
output = "mcl_bows:crossbow",
|
|
recipe = {
|
|
{"mcl_core:stick", "mcl_core:iron_ingot", "mcl_core:stick"},
|
|
{"mcl_mobitems:string", "mcl_bows:arrow", "mcl_mobitems:string"},
|
|
{"", "mcl_core:stick", ""},
|
|
}
|
|
})
|
|
end
|
|
|
|
minetest.register_craft({
|
|
type = "fuel",
|
|
recipe = "group:bow",
|
|
burntime = 15,
|
|
})
|
|
|
|
-- Add entry aliases for the Help
|
|
if minetest.get_modpath("doc") then
|
|
doc.add_entry_alias("tools", "mcl_bows:crossbow", "tools", "mcl_bows:crossbow_0")
|
|
doc.add_entry_alias("tools", "mcl_bows:crossbow", "tools", "mcl_bows:crossbow_1")
|
|
doc.add_entry_alias("tools", "mcl_bows:crossbow", "tools", "mcl_bows:crossbow_2")
|
|
end
|