functions.lua 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. -- Function is badly named! Should be 'entity_ignores_arrow'.
  2. -- Return 'true' if the entity cannot be hit, otherwise return 'false' if the entity should be punched.
  3. -- Note: 'entity_name' is the registered name of the entity to be checked for hit.
  4. function throwing.entity_blocks_arrow(entity_name)
  5. -- Dropped itemstacks don't take damage.
  6. if entity_name == "__builtin:item" then
  7. return true
  8. end
  9. -- Ignore other arrows/fireballs in flight.
  10. local is_arrow = (string.find(entity_name, "arrow") or string.find(entity_name, "fireball"))
  11. if is_arrow then
  12. return true
  13. end
  14. -- Ignore health gauges above players.
  15. if entity_name == "gauges:hp_bar" then
  16. return true
  17. end
  18. -- Ignore sound beacones.
  19. if entity_name:find("^soundbeacon:") then
  20. return true
  21. end
  22. -- Entity is unknown, so punch it for damage!
  23. return false
  24. end
  25. --~
  26. --~ Shot and reload system
  27. --~
  28. local players = {}
  29. minetest.register_on_joinplayer(function(player)
  30. local playerName = player:get_player_name()
  31. players[playerName] = {
  32. reloading=false,
  33. }
  34. end)
  35. minetest.register_on_leaveplayer(function(player)
  36. local playerName = player:get_player_name()
  37. players[playerName] = nil
  38. end)
  39. function throwing_shoot_arrow (itemstack, player, stiffness, is_cross)
  40. if not player or not player:is_player() then return end
  41. local arrow = itemstack:get_metadata()
  42. local imeta = itemstack:get_meta()
  43. if arrow == "" then
  44. arrow = imeta:get_string("arrow")
  45. end
  46. if arrow == "" then return end
  47. local playerpos = utility.get_foot_pos(player:get_pos())
  48. local obj = minetest.add_entity({x=playerpos.x, y=playerpos.y+1.4, z=playerpos.z}, arrow)
  49. if not obj then return end
  50. if not obj:get_luaentity() then return end
  51. itemstack:set_metadata("")
  52. imeta:set_string("arrow", nil)
  53. imeta:set_string("ar_desc", nil)
  54. toolranks.apply_description(imeta, itemstack:get_definition())
  55. local dir = player:get_look_dir()
  56. obj:setvelocity({x=dir.x*stiffness, y=dir.y*stiffness, z=dir.z*stiffness})
  57. obj:setacceleration({x=dir.x*-3, y=-8.5, z=dir.z*-3})
  58. obj:setyaw(player:get_look_yaw()+math.pi)
  59. if is_cross then
  60. minetest.sound_play("throwing_crossbow_sound", {pos=playerpos})
  61. else
  62. minetest.sound_play("throwing_bow_sound", {pos=playerpos})
  63. end
  64. obj:get_luaentity().player = player
  65. obj:get_luaentity().player_name = player:get_player_name()
  66. obj:get_luaentity().inventory = player:get_inventory()
  67. obj:get_luaentity().stack = player:get_inventory():get_stack("main", player:get_wield_index()-1)
  68. -- Return the modified itemstack.
  69. return itemstack
  70. end
  71. function throwing_unload (itemstack, player, unloaded, wear)
  72. if itemstack:get_metadata() then
  73. for _,arrow in ipairs(throwing_arrows) do
  74. local arw = itemstack:get_metadata()
  75. if arw == "" then
  76. local imeta = itemstack:get_meta()
  77. arw = imeta:get_string("arrow")
  78. end
  79. if arw ~= "" then
  80. if arw == arrow[2] then
  81. local leftover = player:get_inventory():add_item("main", arrow[1])
  82. minetest.item_drop(leftover, player, player:get_pos())
  83. end
  84. end
  85. end
  86. end
  87. if wear >= 65535 then
  88. ambiance.sound_play("default_tool_breaks", player:get_pos(), 1.0, 20)
  89. itemstack:take_item(itemstack:get_count())
  90. return itemstack
  91. else
  92. local newstack = ItemStack(unloaded)
  93. newstack:set_wear(wear)
  94. local imeta = newstack:get_meta()
  95. local ometa = itemstack:get_meta()
  96. imeta:set_string("en_desc", ometa:get_string("en_desc"))
  97. toolranks.apply_description(imeta, newstack:get_definition())
  98. return newstack
  99. end
  100. end
  101. function throwing_arrow_punch_entity (target, self, damage)
  102. -- Get tool capabilities from the tool-data API.
  103. local toolcaps = td_api.arrow_toolcaps(self._name or "", damage)
  104. local player = minetest.get_player_by_name(self.player_name or "")
  105. if player and player:is_player() then
  106. if target:is_player() then
  107. -- If target is a player, and not a mob, we can't use the shooter as the
  108. -- attacker. This would only actually apply damage if the shooter was a
  109. -- short distance from the target. So for this case, we have to "fake" the
  110. -- target punching themselves.
  111. target:punch(self.object, 1.0, toolcaps, nil)
  112. else
  113. -- The target of the arrow (a mob) sees the shooter as the attacker,
  114. -- *not* the arrow entity itself. If this were not so, players
  115. -- could shoot mobs with arrows without retaliation.
  116. target:punch(player, 1.0, toolcaps, nil)
  117. end
  118. else
  119. -- Shooter logged off game after firing arrow. Use basic fallback.
  120. target:punch(self.object, 1.0, toolcaps, nil)
  121. end
  122. end
  123. function throwing_reload (index, indexname, pname, pos, is_cross, loaded)
  124. -- This function is called after some delay.
  125. local player = minetest.get_player_by_name(pname)
  126. -- Check for nil. Can happen if player leaves game right after reloading.
  127. if not player or not players[pname] then
  128. return
  129. end
  130. players[pname].reloading = false
  131. local playerinv = player:get_inventory()
  132. local itemstack = playerinv:get_stack("main", index)
  133. if not itemstack or itemstack:get_count() ~= 1 then
  134. return
  135. end
  136. -- Check if the player is still wielding the same object.
  137. -- This check isn't very secure, but we don't care too much.
  138. local same_selected = false
  139. if index == player:get_wield_index() then
  140. if indexname == itemstack:get_name() then
  141. same_selected = true
  142. end
  143. end
  144. if same_selected then
  145. if (pos.x == player:getpos().x and pos.y == player:getpos().y and pos.z == player:getpos().z) or not is_cross then
  146. local wear = itemstack:get_wear()
  147. local bowdef = minetest.registered_items[itemstack:get_name()]
  148. local bowname = bowdef.description
  149. local arrow_stack = playerinv:get_stack("main", index + 1)
  150. for _, arrow in ipairs(throwing_arrows) do
  151. if arrow_stack:get_name() == arrow[1] then
  152. -- Remove arrow from beside bow.
  153. arrow_stack:take_item()
  154. playerinv:set_stack("main", index + 1, arrow_stack)
  155. local name = arrow[1]
  156. local arrowdesc = utility.get_short_desc(minetest.registered_items[name].description or "")
  157. local entity = arrow[2]
  158. -- Replace with loaded bow item.
  159. local newstack = ItemStack(loaded)
  160. newstack:set_wear(wear)
  161. local imeta = newstack:get_meta()
  162. -- Preserve name of bow (if named).
  163. local ometa = itemstack:get_meta()
  164. imeta:set_string("en_desc", ometa:get_string("en_desc"))
  165. imeta:set_string("arrow", entity)
  166. imeta:set_string("ar_desc", arrowdesc)
  167. toolranks.apply_description(imeta, bowdef)
  168. -- Don't need to iterate through remaining arrow types.
  169. playerinv:set_stack("main", index, newstack)
  170. return
  171. end
  172. end
  173. end
  174. end
  175. end
  176. -- Bows and crossbows
  177. function throwing_register_bow (name, desc, scale, stiffness, reload_time, toughness, is_cross, craft)
  178. minetest.register_tool("throwing:" .. name, {
  179. description = desc,
  180. inventory_image = "throwing_" .. name .. ".png",
  181. wield_scale = scale,
  182. stack_max = 1,
  183. groups = {not_repaired_by_anvil=1},
  184. on_use = function(itemstack, user, pt)
  185. if not user or not user:is_player() then
  186. return
  187. end
  188. local pos = user:get_pos()
  189. local pname = user:get_player_name()
  190. local index = user:get_wield_index()
  191. local inv = user:get_inventory()
  192. local stack = inv:get_stack("main", index)
  193. local indexname = ""
  194. if stack and stack:get_count() == 1 then
  195. indexname = stack:get_name()
  196. end
  197. -- Reload bow after some delay.
  198. if not players[pname].reloading then
  199. players[pname].reloading = true
  200. minetest.after(reload_time, throwing_reload, index, indexname, pname, pos, is_cross, "throwing:" .. name .. "_loaded")
  201. end
  202. end,
  203. })
  204. minetest.register_tool("throwing:" .. name .. "_loaded", {
  205. description = desc,
  206. inventory_image = "throwing_" .. name .. "_loaded.png",
  207. wield_scale = scale,
  208. stack_max = 1,
  209. groups = {not_in_creative_inventory=1, not_repaired_by_anvil=1},
  210. on_use = function(itemstack, user, pt)
  211. if not user or not user:is_player() then
  212. return
  213. end
  214. local control = user:get_player_control()
  215. local unloaded = "throwing:" .. name
  216. local wear = itemstack:get_wear()
  217. -- Unload the bow.
  218. if control.sneak then
  219. local newstack = throwing_unload(itemstack, user, unloaded, wear)
  220. if newstack then
  221. return newstack
  222. end
  223. return itemstack
  224. end
  225. -- Fire the bow.
  226. local newstack = throwing_shoot_arrow(itemstack, user, stiffness, is_cross)
  227. if newstack then
  228. wear = wear + (65535 / toughness)
  229. newstack = throwing_unload(newstack, user, unloaded, wear)
  230. end
  231. if newstack then
  232. return newstack
  233. end
  234. return itemstack
  235. end,
  236. })
  237. minetest.register_craft({
  238. output = 'throwing:' .. name,
  239. recipe = craft
  240. })
  241. minetest.register_craft({
  242. output = 'throwing:' .. name,
  243. recipe = {
  244. {craft[1][3], craft[1][2], craft[1][1]},
  245. {craft[2][3], craft[2][2], craft[2][1]},
  246. {craft[3][3], craft[3][2], craft[3][1]},
  247. }
  248. })
  249. end
  250. -- Determine if a node should block an arrow.
  251. -- Cheapest checks should come first.
  252. function throwing_node_should_block_arrow (nn)
  253. if nn == "air" then return false end
  254. if snow.is_snow(nn) then return false end
  255. --if nn == "ignore" then return true end
  256. if string.find(nn, "^throwing:") or
  257. string.find(nn, "^fire:") or
  258. string.find(nn, "^default:fence") or
  259. string.find(nn, "ladder") then
  260. return false
  261. end
  262. local def = minetest.reg_ns_nodes[nn]
  263. if def then
  264. local dt = def.drawtype
  265. local pt2 = def.paramtype2
  266. if dt == "airlike" or
  267. dt == "signlike" or
  268. dt == "torchlike" or
  269. dt == "raillike" or
  270. dt == "plantlike" or
  271. (dt == "nodebox" and pt2 == "wallmounted") then
  272. return false
  273. end
  274. end
  275. return true
  276. end
  277. throwing.node_blocks_arrow = throwing_node_should_block_arrow