2019-03-07 21:35:02 +01:00
local S = minetest.get_translator ( " mcl_bows " )
2018-05-07 23:10:49 +02:00
mcl_bows = { }
2018-05-07 15:29:17 +02:00
local arrows = {
2018-05-07 23:10:49 +02:00
[ " mcl_bows:arrow " ] = " mcl_bows:arrow_entity " ,
2018-05-07 15:29:17 +02:00
}
local GRAVITY = 9.81
local BOW_DURABILITY = 385
2018-05-07 16:42:51 +02:00
-- Charging time in microseconds
local BOW_CHARGE_TIME_HALF = 500000 -- bow level 1
local BOW_CHARGE_TIME_FULL = 1000000 -- bow level 2 (full charge)
2018-05-07 15:29:17 +02:00
2018-05-07 20:31:40 +02:00
-- Factor to multiply with player speed while player uses bow
2018-05-07 20:37:45 +02:00
-- This emulates the sneak speed.
local PLAYER_USE_BOW_SPEED = tonumber ( minetest.settings : get ( " movement_speed_crouch " ) ) / tonumber ( minetest.settings : get ( " movement_speed_walk " ) )
2018-05-07 20:31:40 +02:00
2018-05-07 16:25:46 +02:00
-- 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 = 26
2018-05-07 16:42:51 +02:00
--[[ Store the charging state of each player.
keys : player name
value :
2018-05-07 17:43:39 +02:00
nil = not charging or player not existing
2018-05-07 16:42:51 +02:00
number : currently charging , the number is the time from minetest.get_us_time
in which the charging has started
] ]
2018-05-07 15:29:17 +02:00
local bow_load = { }
2018-05-07 17:43:39 +02:00
-- Another player table, this one stores the wield index of the bow being charged
local bow_index = { }
2018-05-07 23:10:49 +02:00
mcl_bows.shoot_arrow = function ( arrow_item , pos , dir , yaw , shooter , power , damage )
2018-05-07 15:29:17 +02:00
local obj = minetest.add_entity ( { x = pos.x , y = pos.y , z = pos.z } , arrows [ arrow_item ] )
if power == nil then
power = 19
end
if damage == nil then
damage = 3
end
2018-05-12 07:00:16 +02:00
obj : set_velocity ( { x = dir.x * power , y = dir.y * power , z = dir.z * power } )
obj : set_acceleration ( { x = 0 , y =- GRAVITY , z = 0 } )
2019-03-06 04:38:57 +01:00
obj : set_yaw ( yaw - math.pi / 2 )
2018-05-07 15:29:17 +02:00
local le = obj : get_luaentity ( )
le._shooter = shooter
le._damage = damage
le._startpos = pos
2020-04-07 00:55:45 +02:00
minetest.sound_play ( " mcl_bows_bow_shoot " , { pos = pos } , true )
2019-12-09 09:29:19 +01:00
if shooter ~= nil and shooter : is_player ( ) then
2018-05-07 15:29:17 +02:00
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 get_arrow = function ( 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_bow " ) ~= 0 then
arrow_stack = it
arrow_stack_id = i
break
end
end
return arrow_stack , arrow_stack_id
end
local player_shoot_arrow = function ( itemstack , player , power , damage )
local arrow_stack , arrow_stack_id = get_arrow ( player )
local arrow_itemstring
if not minetest.settings : get_bool ( " creative_mode " ) then
if not arrow_stack then
return false
end
arrow_itemstring = arrow_stack : get_name ( )
arrow_stack : take_item ( )
local inv = player : get_inventory ( )
inv : set_stack ( " main " , arrow_stack_id , arrow_stack )
end
2019-02-01 06:33:07 +01:00
local playerpos = player : get_pos ( )
2018-05-07 15:29:17 +02:00
local dir = player : get_look_dir ( )
local yaw = player : get_look_horizontal ( )
if not arrow_itemstring then
2018-05-07 23:10:49 +02:00
arrow_itemstring = " mcl_bows:arrow "
2018-05-07 15:29:17 +02:00
end
2018-05-07 23:10:49 +02:00
mcl_bows.shoot_arrow ( arrow_itemstring , { x = playerpos.x , y = playerpos.y + 1.5 , z = playerpos.z } , dir , yaw , player , power , damage )
2018-05-07 15:29:17 +02:00
return true
end
2018-05-07 17:56:17 +02:00
-- Bow item, uncharged state
2018-05-07 23:10:49 +02:00
minetest.register_tool ( " mcl_bows:bow " , {
2019-03-07 21:35:02 +01:00
description = S ( " Bow " ) ,
2020-03-12 01:35:11 +01:00
_tt_help = S ( " Launches arrows " ) ,
2019-03-07 21:35:02 +01:00
_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. " ) ,
2018-05-07 15:29:17 +02:00
_doc_items_durability = BOW_DURABILITY ,
2018-05-07 23:10:49 +02:00
inventory_image = " mcl_bows_bow.png " ,
2020-04-08 04:07:16 +02:00
wield_scale = { x = 1.8 , y = 1.8 , z = 1 } ,
2018-05-07 15:29:17 +02:00
stack_max = 1 ,
-- Trick to disable melee damage to entities.
-- Range not set to 0 (unlike the others) so it can be placed into item frames
range = 1 ,
-- Trick to disable digging as well
on_use = function ( ) end ,
groups = { weapon = 1 , weapon_ranged = 1 } ,
} )
2018-05-07 20:31:40 +02:00
-- Iterates through player inventory and resets all the bows in "charging" state back to their original stage
local reset_bows = function ( player )
local inv = player : get_inventory ( )
local list = inv : get_list ( " main " )
for place , stack in pairs ( list ) do
2018-05-07 23:10:49 +02:00
if stack : get_name ( ) == " mcl_bows:bow_0 " or stack : get_name ( ) == " mcl_bows:bow_1 " or stack : get_name ( ) == " mcl_bows:bow_2 " then
stack : set_name ( " mcl_bows:bow " )
2018-05-07 20:31:40 +02:00
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 reset_bow_state = function ( player , also_reset_bows )
bow_load [ player : get_player_name ( ) ] = nil
bow_index [ player : get_player_name ( ) ] = nil
2018-10-23 18:51:19 +02:00
if minetest.get_modpath ( " playerphysics " ) then
playerphysics.remove_physics_factor ( player , " speed " , " mcl_bows:use_bow " )
2018-05-07 20:31:40 +02:00
end
if also_reset_bows then
reset_bows ( player )
end
end
2018-05-07 17:56:17 +02:00
-- Bow in charging state
for level = 0 , 2 do
2018-05-07 23:10:49 +02:00
minetest.register_tool ( " mcl_bows:bow_ " .. level , {
2019-03-07 21:35:02 +01:00
description = S ( " Bow " ) ,
2018-05-07 17:56:17 +02:00
_doc_items_create_entry = false ,
2018-05-07 23:10:49 +02:00
inventory_image = " mcl_bows_bow_ " .. level .. " .png " ,
2020-04-08 04:07:16 +02:00
wield_scale = { x = 1.8 , y = 1.8 , z = 1 } ,
2018-05-07 17:56:17 +02:00
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 } ,
on_drop = function ( itemstack , dropper , pos )
2018-06-13 17:20:10 +02:00
reset_bow_state ( dropper )
2018-05-07 23:10:49 +02:00
itemstack : set_name ( " mcl_bows:bow " )
2018-05-07 17:56:17 +02:00
minetest.item_drop ( itemstack , dropper , pos )
itemstack : take_item ( )
return itemstack
end ,
2018-05-07 23:22:54 +02:00
-- Prevent accidental interaction with itemframes and other nodes
on_place = function ( itemstack )
return itemstack
end ,
2018-05-07 17:56:17 +02:00
} )
end
2018-05-07 15:29:17 +02:00
2018-05-07 17:46:52 +02:00
2018-05-07 15:29:17 +02:00
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 ( )
2018-05-07 23:10:49 +02:00
if ( wielditem : get_name ( ) == " mcl_bows:bow_0 " or wielditem : get_name ( ) == " mcl_bows:bow_1 " or wielditem : get_name ( ) == " mcl_bows:bow_2 " ) then
2018-05-07 16:25:46 +02:00
local has_shot = false
local speed , damage
2018-05-07 16:42:51 +02:00
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
2018-05-07 23:10:49 +02:00
minetest.log ( " warning " , " [mcl_bows] Player " .. player : get_player_name ( ) .. " fires arrow with non-numeric bow_load! " )
2018-05-07 16:42:51 +02:00
end
2018-05-07 16:25:46 +02:00
charge = math.max ( math.min ( charge , BOW_CHARGE_TIME_FULL ) , 0 )
2018-05-07 16:42:51 +02:00
2018-05-07 16:25:46 +02:00
local charge_ratio = charge / BOW_CHARGE_TIME_FULL
charge_ratio = math.max ( math.min ( charge_ratio , 1 ) , 0 )
-- Calculate damage and speed
-- Fully charged
if charge >= BOW_CHARGE_TIME_FULL then
speed = BOW_MAX_SPEED
2018-05-07 15:29:17 +02:00
local r = math.random ( 1 , 5 )
if r == 1 then
-- 20% chance for critical hit
2018-05-07 16:25:46 +02:00
damage = 10
2018-05-07 15:29:17 +02:00
else
2018-05-07 16:25:46 +02:00
damage = 9
2018-05-07 15:29:17 +02:00
end
2018-05-07 16:25:46 +02:00
-- Partially charged
else
-- Linear speed and damage increase
speed = math.max ( 4 , BOW_MAX_SPEED * charge_ratio )
damage = math.max ( 1 , math.floor ( 9 * charge_ratio ) )
2018-05-07 15:29:17 +02:00
end
2018-05-07 16:25:46 +02:00
has_shot = player_shoot_arrow ( wielditem , player , speed , damage )
2018-05-07 23:10:49 +02:00
wielditem : set_name ( " mcl_bows:bow " )
2018-05-07 16:25:46 +02:00
if has_shot and minetest.settings : get_bool ( " creative_mode " ) == false then
2018-05-07 15:29:17 +02:00
wielditem : add_wear ( 65535 / BOW_DURABILITY )
end
player : set_wielded_item ( wielditem )
2018-05-07 20:31:40 +02:00
reset_bow_state ( player , true )
2018-05-07 15:29:17 +02:00
end
end )
controls.register_on_hold ( function ( player , key , time )
if key ~= " RMB " then
return
end
local name = player : get_player_name ( )
local inv = minetest.get_inventory ( { type = " player " , name = name } )
local wielditem = player : get_wielded_item ( )
2018-05-07 23:10:49 +02:00
if bow_load [ name ] == nil and wielditem : get_name ( ) == " mcl_bows:bow " and ( minetest.settings : get_bool ( " creative_mode " ) or inv : contains_item ( " main " , " mcl_bows:arrow " ) ) then
wielditem : set_name ( " mcl_bows:bow_0 " )
2018-05-07 17:43:39 +02:00
player : set_wielded_item ( wielditem )
2018-10-23 18:51:19 +02:00
if minetest.get_modpath ( " playerphysics " ) then
2018-05-07 20:31:40 +02:00
-- Slow player down when using bow
2018-10-23 18:51:19 +02:00
playerphysics.add_physics_factor ( player , " speed " , " mcl_bows:use_bow " , PLAYER_USE_BOW_SPEED )
2018-05-07 20:31:40 +02:00
end
2018-05-07 15:53:16 +02:00
bow_load [ name ] = minetest.get_us_time ( )
2018-05-07 17:43:39 +02:00
bow_index [ name ] = player : get_wield_index ( )
2018-05-07 15:29:17 +02:00
else
2018-05-07 17:43:39 +02:00
if player : get_wield_index ( ) == bow_index [ name ] then
if type ( bow_load [ name ] ) == " number " then
2018-05-07 23:10:49 +02:00
if wielditem : get_name ( ) == " mcl_bows:bow_0 " and minetest.get_us_time ( ) - bow_load [ name ] >= BOW_CHARGE_TIME_HALF then
wielditem : set_name ( " mcl_bows:bow_1 " )
elseif wielditem : get_name ( ) == " mcl_bows:bow_1 " and minetest.get_us_time ( ) - bow_load [ name ] >= BOW_CHARGE_TIME_FULL then
wielditem : set_name ( " mcl_bows:bow_2 " )
2018-05-07 17:43:39 +02:00
end
else
2018-05-07 23:10:49 +02:00
if wielditem : get_name ( ) == " mcl_bows:bow_0 " or wielditem : get_name ( ) == " mcl_bows:bow_1 " or wielditem : get_name ( ) == " mcl_bows:bow_2 " then
wielditem : set_name ( " mcl_bows:bow " )
2018-05-07 17:43:39 +02:00
end
2018-05-07 15:29:17 +02:00
end
2018-05-07 17:43:39 +02:00
player : set_wielded_item ( wielditem )
2018-05-07 15:29:17 +02:00
else
2018-05-07 20:31:40 +02:00
reset_bow_state ( player , true )
2018-05-07 15:29:17 +02:00
end
end
end )
minetest.register_globalstep ( function ( dtime )
for _ , player in pairs ( minetest.get_connected_players ( ) ) do
2018-05-07 17:43:39 +02:00
local name = player : get_player_name ( )
2018-05-07 15:29:17 +02:00
local wielditem = player : get_wielded_item ( )
2018-05-07 17:43:39 +02:00
local wieldindex = player : get_wield_index ( )
2018-05-07 15:29:17 +02:00
local controls = player : get_player_control ( )
2018-05-07 23:10:49 +02:00
if type ( bow_load [ name ] ) == " number " and ( ( wielditem : get_name ( ) ~= " mcl_bows:bow_0 " and wielditem : get_name ( ) ~= " mcl_bows:bow_1 " and wielditem : get_name ( ) ~= " mcl_bows:bow_2 " ) or wieldindex ~= bow_index [ name ] ) then
2018-05-07 20:31:40 +02:00
reset_bow_state ( player , true )
2018-05-07 15:29:17 +02:00
end
end
end )
2018-05-07 17:43:39 +02:00
minetest.register_on_joinplayer ( function ( player )
reset_bows ( player )
end )
2018-05-07 16:42:51 +02:00
minetest.register_on_leaveplayer ( function ( player )
2018-05-07 20:31:40 +02:00
reset_bow_state ( player , true )
2018-05-07 16:42:51 +02:00
end )
2018-05-07 15:29:17 +02:00
if minetest.get_modpath ( " mcl_core " ) and minetest.get_modpath ( " mcl_mobitems " ) then
minetest.register_craft ( {
2018-05-07 23:10:49 +02:00
output = ' mcl_bows:bow ' ,
2018-05-07 15:29:17 +02:00
recipe = {
{ ' ' , ' mcl_core:stick ' , ' mcl_mobitems:string ' } ,
{ ' mcl_core:stick ' , ' ' , ' mcl_mobitems:string ' } ,
{ ' ' , ' mcl_core:stick ' , ' mcl_mobitems:string ' } ,
}
} )
m inetest.register_craft ( {
2018-05-07 23:10:49 +02:00
output = ' mcl_bows:bow ' ,
2018-05-07 15:29:17 +02:00
recipe = {
{ ' mcl_mobitems:string ' , ' mcl_core:stick ' , ' ' } ,
{ ' mcl_mobitems:string ' , ' ' , ' mcl_core:stick ' } ,
{ ' mcl_mobitems:string ' , ' mcl_core:stick ' , ' ' } ,
}
} )
end
minetest.register_craft ( {
type = " fuel " ,
2018-05-07 23:10:49 +02:00
recipe = " mcl_bows:bow " ,
2018-05-07 15:29:17 +02:00
burntime = 15 ,
} )
-- Add entry aliases for the Help
if minetest.get_modpath ( " doc " ) then
2018-05-07 23:10:49 +02:00
doc.add_entry_alias ( " tools " , " mcl_bows:bow " , " tools " , " mcl_bows:bow_0 " )
doc.add_entry_alias ( " tools " , " mcl_bows:bow " , " tools " , " mcl_bows:bow_1 " )
doc.add_entry_alias ( " tools " , " mcl_bows:bow " , " tools " , " mcl_bows:bow_2 " )
2018-05-07 15:29:17 +02:00
end