2019-03-07 22:35:02 +02:00
local S = minetest.get_translator ( " mcl_bows " )
2018-05-08 01:38:00 +03:00
-- Time in seconds after which a stuck arrow is deleted
local ARROW_TIMEOUT = 60
2018-05-09 16:10:56 +03:00
-- Time after which stuck arrow is rechecked for being stuck
local STUCK_RECHECK_TIME = 5
local GRAVITY = 9.81
local YAW_OFFSET = - math.pi / 2
2018-05-08 01:38:00 +03:00
2017-07-26 20:35:56 +03:00
local mod_awards = minetest.get_modpath ( " awards " ) and minetest.get_modpath ( " mcl_achievements " )
2018-05-09 16:10:56 +03:00
local mod_button = minetest.get_modpath ( " mesecons_button " )
2017-07-26 20:35:56 +03:00
2018-05-08 00:10:49 +03:00
minetest.register_craftitem ( " mcl_bows:arrow " , {
2019-03-07 22:35:02 +02:00
description = S ( " Arrow " ) ,
_doc_items_longdesc = S ( " Arrows are ammunition for bows and dispensers. " ) .. " \n " ..
S ( " An arrow fired from a bow has a regular damage of 1-9. At full charge, there's a 20% chance of a critical hit dealing 10 damage instead. An arrow fired from a dispenser always deals 3 damage. " ) .. " \n " ..
S ( " Arrows might get stuck on solid blocks and can be retrieved again. They are also capable of pushing wooden buttons. " ) ,
_doc_items_usagehelp = S ( " To use arrows as ammunition for a bow, just put them anywhere in your inventory, they will be used up automatically. To use arrows as ammunition for a dispenser, place them in the dispenser's inventory. To retrieve an arrow that sticks in a block, simply walk close to it. " ) ,
2018-05-08 00:10:49 +03:00
inventory_image = " mcl_bows_arrow_inv.png " ,
2017-01-16 15:29:41 +02:00
groups = { ammo = 1 , ammo_bow = 1 } ,
2018-02-01 23:45:19 +02:00
_on_dispense = function ( itemstack , dispenserpos , droppos , dropnode , dropdir )
-- Shoot arrow
local shootpos = vector.add ( dispenserpos , vector.multiply ( dropdir , 0.51 ) )
2018-05-09 16:10:56 +03:00
local yaw = math.atan2 ( dropdir.z , dropdir.x ) + YAW_OFFSET
2018-05-08 00:10:49 +03:00
mcl_bows.shoot_arrow ( itemstack : get_name ( ) , shootpos , dropdir , yaw , nil , 19 , 3 )
2018-02-01 23:45:19 +02:00
end ,
2015-06-29 20:55:56 +03:00
} )
2018-05-09 16:56:52 +03:00
-- This is a fake node, used as model for the arrow entity.
-- It's not supposed to be usable as item or real node.
-- TODO: Use a proper mesh for the arrow entity
2018-05-08 00:10:49 +03:00
minetest.register_node ( " mcl_bows:arrow_box " , {
2015-06-29 20:55:56 +03:00
drawtype = " nodebox " ,
2017-01-04 23:36:51 +02:00
is_ground_content = false ,
2015-06-29 20:55:56 +03:00
node_box = {
type = " fixed " ,
fixed = {
-- Shaft
2018-05-09 16:56:52 +03:00
{ - 6.5 / 17 , - 1.5 / 17 , - 1.5 / 17 , - 4.5 / 17 , 1.5 / 17 , 1.5 / 17 } ,
{ - 4.5 / 17 , - 0.5 / 17 , - 0.5 / 17 , 5.5 / 17 , 0.5 / 17 , 0.5 / 17 } ,
{ 5.5 / 17 , - 1.5 / 17 , - 1.5 / 17 , 6.5 / 17 , 1.5 / 17 , 1.5 / 17 } ,
-- Tip
2015-06-29 20:55:56 +03:00
{ - 4.5 / 17 , 2.5 / 17 , 2.5 / 17 , - 3.5 / 17 , - 2.5 / 17 , - 2.5 / 17 } ,
{ - 8.5 / 17 , 0.5 / 17 , 0.5 / 17 , - 6.5 / 17 , - 0.5 / 17 , - 0.5 / 17 } ,
2018-05-09 16:56:52 +03:00
-- Fletching
2015-06-29 20:55:56 +03:00
{ 6.5 / 17 , 1.5 / 17 , 1.5 / 17 , 7.5 / 17 , 2.5 / 17 , 2.5 / 17 } ,
{ 7.5 / 17 , - 2.5 / 17 , 2.5 / 17 , 6.5 / 17 , - 1.5 / 17 , 1.5 / 17 } ,
{ 7.5 / 17 , 2.5 / 17 , - 2.5 / 17 , 6.5 / 17 , 1.5 / 17 , - 1.5 / 17 } ,
{ 6.5 / 17 , - 1.5 / 17 , - 1.5 / 17 , 7.5 / 17 , - 2.5 / 17 , - 2.5 / 17 } ,
2018-05-09 18:31:04 +03:00
2015-06-29 20:55:56 +03:00
{ 7.5 / 17 , 2.5 / 17 , 2.5 / 17 , 8.5 / 17 , 3.5 / 17 , 3.5 / 17 } ,
{ 8.5 / 17 , - 3.5 / 17 , 3.5 / 17 , 7.5 / 17 , - 2.5 / 17 , 2.5 / 17 } ,
{ 8.5 / 17 , 3.5 / 17 , - 3.5 / 17 , 7.5 / 17 , 2.5 / 17 , - 2.5 / 17 } ,
{ 7.5 / 17 , - 2.5 / 17 , - 2.5 / 17 , 8.5 / 17 , - 3.5 / 17 , - 3.5 / 17 } ,
}
} ,
2018-05-08 00:10:49 +03:00
tiles = { " mcl_bows_arrow.png^[transformFX " , " mcl_bows_arrow.png^[transformFX " , " mcl_bows_arrow_back.png " , " mcl_bows_arrow_front.png " , " mcl_bows_arrow.png " , " mcl_bows_arrow.png^[transformFX " } ,
2018-05-09 16:56:52 +03:00
paramtype = " light " ,
paramtype2 = " facedir " ,
sunlight_propagates = true ,
groups = { not_in_creative_inventory = 1 , dig_immediate = 3 } ,
2015-06-29 20:55:56 +03:00
} )
2018-05-08 15:35:15 +03:00
local ARROW_ENTITY = {
physical = true ,
2015-06-29 20:55:56 +03:00
visual = " wielditem " ,
visual_size = { x = 0.4 , y = 0.4 } ,
2018-05-08 00:10:49 +03:00
textures = { " mcl_bows:arrow_box " } ,
2018-05-09 17:37:13 +03:00
collisionbox = { - 0.19 , - 0.125 , - 0.19 , 0.19 , 0.125 , 0.19 } ,
2018-05-08 15:35:15 +03:00
collide_with_objects = false ,
2017-02-21 23:51:07 +02:00
_lastpos = { } ,
2017-03-06 02:02:30 +02:00
_startpos = nil ,
2017-02-21 23:51:07 +02:00
_damage = 1 , -- Damage on impact
2018-05-08 01:38:00 +03:00
_stuck = false , -- Whether arrow is stuck
_stucktimer = nil , -- Amount of time (in seconds) the arrow has been stuck so far
2018-05-09 16:10:56 +03:00
_stuckrechecktimer = nil , -- An additional timer for periodically re-checking the stuck status of an arrow
_stuckin = nil , --Position of node in which arow is stuck.
2017-02-21 23:51:07 +02:00
_shooter = nil , -- ObjectRef of player or mob who shot it
2018-05-12 08:00:16 +03:00
_viscosity = 0 , -- Viscosity of node the arrow is currently in
_deflection_cooloff = 0 , -- Cooloff timer after an arrow deflection, to prevent many deflections in quick succession
2015-06-29 20:55:56 +03:00
}
2018-05-09 17:49:37 +03:00
-- Destroy arrow entity self at pos and drops it as an item
2018-05-09 16:10:56 +03:00
local spawn_item = function ( self , pos )
if not minetest.settings : get_bool ( " creative_mode " ) then
local item = minetest.add_item ( pos , " mcl_bows:arrow " )
item : set_velocity ( { x = 0 , y = 0 , z = 0 } )
item : set_yaw ( self.object : get_yaw ( ) )
end
self.object : remove ( )
end
2018-05-08 15:35:15 +03:00
ARROW_ENTITY.on_step = function ( self , dtime )
2019-02-01 07:33:07 +02:00
local pos = self.object : get_pos ( )
2018-05-08 19:30:57 +03:00
local dpos = table.copy ( pos ) -- digital pos
dpos = vector.round ( dpos )
local node = minetest.get_node ( dpos )
2015-06-29 20:55:56 +03:00
2018-05-08 01:38:00 +03:00
if self._stuck then
self._stucktimer = self._stucktimer + dtime
2018-05-09 16:10:56 +03:00
self._stuckrechecktimer = self._stuckrechecktimer + dtime
2018-05-08 01:38:00 +03:00
if self._stucktimer > ARROW_TIMEOUT then
self.object : remove ( )
return
end
2018-05-09 16:10:56 +03:00
-- Drop arrow as item when it is no longer stuck
2018-05-09 18:04:13 +03:00
-- FIXME: Arrows are a bit slot to react and continue to float in mid air for a few seconds.
2018-05-09 16:10:56 +03:00
if self._stuckrechecktimer > STUCK_RECHECK_TIME then
local stuckin_def
if self._stuckin then
stuckin_def = minetest.registered_nodes [ minetest.get_node ( self._stuckin ) . name ]
end
-- TODO: In MC, arrow just falls down without turning into an item
if stuckin_def and stuckin_def.walkable == false then
spawn_item ( self , pos )
return
end
self._stuckrechecktimer = 0
end
2018-05-09 19:19:42 +03:00
-- Pickup arrow if player is nearby (not in Creative Mode)
local objects = minetest.get_objects_inside_radius ( pos , 1 )
2018-05-08 01:38:00 +03:00
for _ , obj in ipairs ( objects ) do
if obj : is_player ( ) then
if not minetest.settings : get_bool ( " creative_mode " ) then
if obj : get_inventory ( ) : room_for_item ( " main " , " mcl_bows:arrow " ) then
obj : get_inventory ( ) : add_item ( " main " , " mcl_bows:arrow " )
minetest.sound_play ( " item_drop_pickup " , {
pos = pos ,
max_hear_distance = 16 ,
gain = 1.0 ,
} )
end
end
2018-05-09 19:19:42 +03:00
self.object : remove ( )
return
2018-05-08 01:38:00 +03:00
end
end
2018-05-09 18:31:04 +03:00
2018-05-12 20:52:40 +03:00
-- Check for object "collision". Done every tick (hopefully this is not too stressing)
2018-05-08 01:38:00 +03:00
else
2018-05-12 20:52:40 +03:00
-- We just check for any hurtable objects nearby.
-- The radius of 3 is fairly liberal, but anything lower than than will cause
-- arrow to hilariously go through mobs often.
-- TODO: Implement an ACTUAL collision detection (engine support needed).
local objs = minetest.get_objects_inside_radius ( pos , 3 )
2017-05-27 00:10:18 +03:00
local closest_object
local closest_distance
2018-05-12 08:00:16 +03:00
if self._deflection_cooloff > 0 then
self._deflection_cooloff = self._deflection_cooloff - dtime
end
2017-05-27 00:10:18 +03:00
-- Iterate through all objects and remember the closest attackable object
2015-06-29 20:55:56 +03:00
for k , obj in pairs ( objs ) do
2018-05-27 11:33:49 +03:00
local ok = false
2017-07-25 02:14:32 +03:00
-- Arrows can only damage players and mobs
if obj ~= self._shooter and obj : is_player ( ) then
ok = true
elseif obj : get_luaentity ( ) ~= nil then
if obj ~= self._shooter and obj : get_luaentity ( ) . _cmi_is_mob then
2017-05-27 00:10:18 +03:00
ok = true
end
end
if ok then
2019-02-01 07:33:07 +02:00
local dist = vector.distance ( pos , obj : get_pos ( ) )
2017-05-27 00:10:18 +03:00
if not closest_object or not closest_distance then
closest_object = obj
closest_distance = dist
elseif dist < closest_distance then
closest_object = obj
closest_distance = dist
end
end
end
-- If an attackable object was found, we will damage the closest one only
if closest_object ~= nil then
local obj = closest_object
2017-07-25 02:14:32 +03:00
local is_player = obj : is_player ( )
local lua = obj : get_luaentity ( )
if obj ~= self._shooter and ( is_player or ( lua and lua._cmi_is_mob ) ) then
obj : punch ( self.object , 1.0 , {
full_punch_interval = 1.0 ,
damage_groups = { fleshy = self._damage } ,
} , nil )
2018-05-07 23:23:15 +03:00
if is_player then
if self._shooter and self._shooter : is_player ( ) then
-- “Ding” sound for hitting another player
2018-05-08 00:10:49 +03:00
minetest.sound_play ( { name = " mcl_bows_hit_player " , gain = 0.1 } , { to_player = self._shooter } )
2018-05-07 23:23:15 +03:00
end
2017-07-25 02:14:32 +03:00
end
2017-03-06 02:02:30 +02:00
2017-07-25 02:14:32 +03:00
if lua then
local entity_name = lua.name
2017-03-06 02:02:30 +02:00
-- Achievement for hitting skeleton, wither skeleton or stray (TODO) with an arrow at least 50 meters away
2017-05-27 01:37:25 +03:00
-- NOTE: Range has been reduced because mobs unload much earlier than that ... >_>
2017-03-06 02:02:30 +02:00
-- TODO: This achievement should be given for the kill, not just a hit
2017-05-27 01:37:25 +03:00
if self._shooter and self._shooter : is_player ( ) and vector.distance ( pos , self._startpos ) >= 20 then
2017-07-26 20:35:56 +03:00
if mod_awards and ( entity_name == " mobs_mc:skeleton " or entity_name == " mobs_mc:stray " or entity_name == " mobs_mc:witherskeleton " ) then
2017-03-06 02:02:30 +02:00
awards.unlock ( self._shooter : get_player_name ( ) , " mcl:snipeSkeleton " )
end
end
2017-05-21 06:55:55 +03:00
end
2015-06-29 20:55:56 +03:00
self.object : remove ( )
2018-05-08 15:35:15 +03:00
return
2015-06-29 20:55:56 +03:00
end
end
end
2017-05-27 01:27:56 +03:00
-- Check for node collision
2018-05-08 01:38:00 +03:00
if self._lastpos . x ~= nil and not self._stuck then
2017-01-26 20:05:25 +02:00
local def = minetest.registered_nodes [ node.name ]
2018-05-08 15:35:15 +03:00
local vel = self.object : get_velocity ( )
2018-05-12 08:00:16 +03:00
-- Arrow has stopped in one axis, so it probably hit something.
-- This detection is a bit clunky, but sadly, MT does not offer a direct collision detection for us. :-(
2018-05-08 15:35:15 +03:00
if ( math.abs ( vel.x ) < 0.0001 ) or ( math.abs ( vel.z ) < 0.0001 ) or ( math.abs ( vel.y ) < 0.00001 ) then
2018-05-09 16:10:56 +03:00
-- Check for the node to which the arrow is pointing
local dir
2018-05-09 16:17:19 +03:00
if math.abs ( vel.y ) < 0.00001 then
if self._lastpos . y < pos.y then
dir = { x = 0 , y = 1 , z = 0 }
else
dir = { x = 0 , y =- 1 , z = 0 }
end
2018-05-09 16:10:56 +03:00
else
dir = minetest.facedir_to_dir ( minetest.dir_to_facedir ( minetest.yaw_to_dir ( self.object : get_yaw ( ) - YAW_OFFSET ) ) )
end
2018-05-09 19:04:34 +03:00
self._stuckin = vector.add ( dpos , dir )
2018-05-09 16:10:56 +03:00
local snode = minetest.get_node ( self._stuckin )
local sdef = minetest.registered_nodes [ snode.name ]
-- If node is non-walkable, unknown or ignore, don't make arrow stuck.
-- This causes a deflection in the engine.
if not sdef or sdef.walkable == false or snode.name == " ignore " then
self._stuckin = nil
2018-05-12 08:00:16 +03:00
if self._deflection_cooloff <= 0 then
-- Lose 1/3 of velocity on deflection
2018-05-12 16:10:34 +03:00
local newvel = vector.multiply ( vel , 0.6667 )
2018-05-09 16:10:56 +03:00
2018-05-12 16:10:34 +03:00
self.object : set_velocity ( newvel )
2018-05-12 08:00:16 +03:00
-- Reset deflection cooloff timer to prevent many deflections happening in quick succession
2018-05-12 16:10:34 +03:00
self._deflection_cooloff = 1.0
2018-05-12 08:00:16 +03:00
end
else
2018-05-09 16:10:56 +03:00
2018-05-12 08:00:16 +03:00
-- Node was walkable, make arrow stuck
self._stuck = true
self._stucktimer = 0
self._stuckrechecktimer = 0
2018-05-09 16:10:56 +03:00
2018-05-12 08:00:16 +03:00
self.object : set_velocity ( { x = 0 , y = 0 , z = 0 } )
self.object : set_acceleration ( { x = 0 , y = 0 , z = 0 } )
-- Push the button! Push, push, push the button!
if mod_button and minetest.get_item_group ( node.name , " button " ) > 0 and minetest.get_item_group ( node.name , " button_push_by_arrow " ) == 1 then
local bdir = minetest.wallmounted_to_dir ( node.param2 )
-- Check the button orientation
if vector.equals ( vector.add ( dpos , bdir ) , self._stuckin ) then
mesecon.push_button ( dpos , node )
end
2018-05-09 19:04:34 +03:00
end
2018-05-08 15:35:15 +03:00
end
2018-05-08 01:02:07 +03:00
elseif ( def and def.liquidtype ~= " none " ) then
-- Slow down arrow in liquids
local v = def.liquid_viscosity
if not v then
v = 0
end
2018-05-12 08:00:16 +03:00
local old_v = self._viscosity
self._viscosity = v
2018-05-08 01:02:07 +03:00
local vpenalty = math.max ( 0.1 , 0.98 - 0.1 * v )
if math.abs ( vel.x ) > 0.001 then
vel.x = vel.x * vpenalty
end
if math.abs ( vel.z ) > 0.001 then
vel.z = vel.z * vpenalty
end
self.object : set_velocity ( vel )
2015-06-29 20:55:56 +03:00
end
end
2017-05-27 01:27:56 +03:00
2018-05-09 16:10:56 +03:00
-- Update yaw
if not self._stuck then
local vel = self.object : get_velocity ( )
self.object : set_yaw ( minetest.dir_to_yaw ( vel ) + YAW_OFFSET )
end
2017-05-27 01:27:56 +03:00
-- Update internal variable
2017-02-21 23:51:07 +02:00
self._lastpos = { x = pos.x , y = pos.y , z = pos.z }
2015-06-29 20:55:56 +03:00
end
2018-05-09 17:49:37 +03:00
-- Force recheck of stuck arrows when punched.
-- Otherwise, punching has no effect.
ARROW_ENTITY.on_punch = function ( self )
if self._stuck then
self._stuckrechecktimer = STUCK_RECHECK_TIME
end
end
2018-05-08 18:16:11 +03:00
ARROW_ENTITY.get_staticdata = function ( self )
local out = {
lastpos = self._lastpos ,
startpos = self._startpos ,
damage = self._damage ,
stuck = self._stuck ,
2018-05-09 16:10:56 +03:00
stuckin = self._stuckin ,
2018-05-08 18:16:11 +03:00
}
2018-05-09 18:31:04 +03:00
if self._stuck then
-- If _stucktimer is missing for some reason, assume the maximum
if not self._stucktimer then
self._stucktimer = ARROW_TIMEOUT
end
out.stuckstarttime = minetest.get_gametime ( ) - self._stucktimer
end
2018-05-08 18:16:11 +03:00
if self._shooter and self._shooter : is_player ( ) then
out.shootername = self._shooter : get_player_name ( )
end
return minetest.serialize ( out )
end
ARROW_ENTITY.on_activate = function ( self , staticdata , dtime_s )
local data = minetest.deserialize ( staticdata )
if data then
self._stuck = data.stuck
2018-05-09 18:31:04 +03:00
if data.stuck then
if data.stuckstarttime then
-- First, check if the stuck arrow is aleady past its life timer.
-- If yes, delete it.
self._stucktimer = minetest.get_gametime ( ) - data.stuckstarttime
if self._stucktimer > ARROW_TIMEOUT then
self.object : remove ( )
return
end
end
-- Perform a stuck recheck on the next step.
2018-05-09 16:10:56 +03:00
self._stuckrechecktimer = STUCK_RECHECK_TIME
2018-05-09 18:31:04 +03:00
self._stuckin = data.stuckin
2018-05-09 16:10:56 +03:00
end
2018-05-09 18:31:04 +03:00
-- Get the remaining arrow state
self._lastpos = data.lastpos
self._startpos = data.startpos
self._damage = data.damage
2018-05-08 18:16:11 +03:00
if data.shootername then
local shooter = minetest.get_player_by_name ( data.shootername )
if shooter and shooter : is_player ( ) then
self._shooter = shooter
end
end
end
2018-05-28 16:49:04 +03:00
self.object : set_armor_groups ( { immortal = 1 } )
2018-05-08 18:16:11 +03:00
end
2018-05-08 15:35:15 +03:00
minetest.register_entity ( " mcl_bows:arrow_entity " , ARROW_ENTITY )
2015-06-29 20:55:56 +03:00
2017-07-26 20:35:56 +03:00
if minetest.get_modpath ( " mcl_core " ) and minetest.get_modpath ( " mcl_mobitems " ) then
minetest.register_craft ( {
2018-05-08 00:10:49 +03:00
output = ' mcl_bows:arrow 4 ' ,
2017-07-26 20:35:56 +03:00
recipe = {
{ ' mcl_core:flint ' } ,
{ ' mcl_core:stick ' } ,
{ ' mcl_mobitems:feather ' }
}
} )
end
2018-05-09 18:04:13 +03:00
if minetest.get_modpath ( " doc_identifier " ) ~= nil then
doc.sub . identifier.register_object ( " mcl_bows:arrow_entity " , " craftitems " , " mcl_bows:arrow " )
end