Compare commits
14 Commits
Author | SHA1 | Date | |
---|---|---|---|
871b69ab0d | |||
b5ce10d157 | |||
7d2ae76dcb | |||
6966f7f8b6 | |||
07a5f951c0 | |||
fc1fd99541 | |||
119116b89e | |||
bb6c36d7e5 | |||
cdc548ecd5 | |||
7f9973400e | |||
3c174db494 | |||
00d86908d8 | |||
5a214d3b44 | |||
2c0ebde983 |
48
README.txt
@ -1,13 +1,53 @@
|
|||||||
=== SPEARS for MINETEST ===
|
=== SPEARS for MINETEST by Echoes91 ===
|
||||||
|
|
||||||
This mod adds spears to Minetest. It aims at improving the ones introduced within throwing enhanced.
|
This mod adds spears to Minetest.
|
||||||
|
|
||||||
|
How to download:
|
||||||
|
If you have this file, you already know
|
||||||
|
|
||||||
|
Forum topic:
|
||||||
|
https://forum.minetest.net/viewtopic.php?f=11&t=13367
|
||||||
|
|
||||||
How to install:
|
How to install:
|
||||||
http://wiki.minetest.com/wiki/Installing_Mods
|
http://wiki.minetest.com/wiki/Installing_Mods
|
||||||
|
|
||||||
How to use the mod:
|
How to use the mod:
|
||||||
Spear work similarly to other tools such as swords, even if being a little slower. Moreover, a spear can be thrown using the drop key ('Q') or placed on a node using the place key ('right click'); they have also limited digging capabilities.
|
Craft a spear with | stick | stick | (material) |, works also mirrored.
|
||||||
|
Spears can be thrown using right mouse button, they either hurt someone or stick where they land on; damage depends on speed and material, flight trajectory is ballistic with (unrealistic) drag.
|
||||||
|
Spears can be used to fight, but are slower and weaker than swords.
|
||||||
|
Spears can be used to dig, but are slower and weaker and pickaxes.
|
||||||
|
If you miss the target it's your fault, the spear is fine.
|
||||||
|
You can grab a spear on the fly (or maybe wait until it falls?).
|
||||||
|
You can stick a spear into a block, but if you remove the block then the spear falls (obviously).
|
||||||
|
This mod contains the word "spear" more than 100 times, if you want the exact number count them yourself.
|
||||||
|
If you feel clever, throw a spear right above you and look at it to see what happens.
|
||||||
|
You can even throw a spear from above down to your foes, it's even easier to hit.
|
||||||
|
Someone once reported to have hit its own foot but it takes practice to do that.
|
||||||
|
|
||||||
|
|
||||||
|
CHANGELOG
|
||||||
|
|
||||||
|
2.1:
|
||||||
|
- Spears ding hard blocks (cracky < 3)
|
||||||
|
- Fixed stick-into-node position to avoid spears turning black (#2467)
|
||||||
|
- Player velocity added to throw (it's physics baby)
|
||||||
|
|
||||||
|
2.0:
|
||||||
|
- Cleaner, streamlined code without deprecated functions
|
||||||
|
- Spears hurt players
|
||||||
|
- Spears animated in flight
|
||||||
|
- Spears stick into nodes
|
||||||
|
- New textures
|
||||||
|
- New sound when hitting flesh
|
||||||
|
- New receipt to fit with default tools
|
||||||
|
- Drag depends on fluid
|
||||||
|
- Improved instructions
|
||||||
|
- New gold-plated spear to celebrate
|
||||||
|
|
||||||
|
1.1:
|
||||||
|
- Make use of new on_secondary_use API, requires Minetest 5.2.0
|
||||||
|
|
||||||
|
|
||||||
License:
|
License:
|
||||||
Sourcecode: LGPLv2.1 (see http://www.gnu.org/licenses/lgpl-2.1.html)
|
Sourcecode: LGPLv2.1 (see http://www.gnu.org/licenses/lgpl-2.1.html)
|
||||||
Grahpics & sounds: CC-BY 3.0 (see http://creativecommons.org/licenses/by/3.0/legalcode)
|
Grahpics & sounds: CC-BY 3.0 (see http://creativecommons.org/licenses/by/3.0/legalcode)
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
DISABLE_STONE_SPEAR = false
|
DISABLE_STONE_SPEAR = false
|
||||||
DISABLE_STEEL_SPEAR = false
|
DISABLE_STEEL_SPEAR = false
|
||||||
|
DISABLE_COPPER_SPEAR = false
|
||||||
|
DISABLE_BRONZE_SPEAR = false
|
||||||
DISABLE_DIAMOND_SPEAR = false
|
DISABLE_DIAMOND_SPEAR = false
|
||||||
DISABLE_OBSIDIAN_SPEAR = false
|
DISABLE_OBSIDIAN_SPEAR = false
|
||||||
|
180
functions.lua
@ -1,77 +1,133 @@
|
|||||||
function spears_shot (itemstack, player)
|
function spears_throw (itemstack, player, pointed_thing)
|
||||||
local spear = itemstack:get_name() .. '_entity'
|
local spear = itemstack:get_name() .. '_entity'
|
||||||
local playerpos = player:getpos()
|
local player_pos = player:get_pos()
|
||||||
local obj = minetest.add_entity({x=playerpos.x,y=playerpos.y+1.5,z=playerpos.z}, spear)
|
local head_pos = vector.new(player_pos.x, player_pos.y + player:get_properties().eye_height, player_pos.z)
|
||||||
local dir = player:get_look_dir()
|
local direction = player:get_look_dir()
|
||||||
local sp = 14
|
local throw_pos = vector.add(head_pos, vector.multiply(direction,0.5))
|
||||||
local dr = .3
|
local pitch = player:get_look_vertical()
|
||||||
local gravity = 9.8
|
local yaw = player:get_look_horizontal()
|
||||||
obj:setvelocity({x=dir.x*sp, y=dir.y*sp, z=dir.z*sp})
|
local rotation = vector.new(0, yaw + math.pi/2, pitch + math.pi/6)
|
||||||
obj:setacceleration({x=-dir.x*dr, y=-gravity, z=-dir.z*dr})
|
-- Plant into node
|
||||||
obj:setyaw(player:get_look_yaw()+math.pi)
|
if pointed_thing.type == "node" then
|
||||||
minetest.sound_play("spears_sound", {pos=playerpos})
|
local node = minetest.get_node(pointed_thing.under)
|
||||||
obj:get_luaentity().wear = itemstack:get_wear()
|
if minetest.registered_nodes[node.name].walkable and vector.distance(pointed_thing.above, throw_pos) < 1 then
|
||||||
|
local spear_object = minetest.add_entity(vector.divide(vector.add(vector.multiply(pointed_thing.above, 2), pointed_thing.under), 3), spear)
|
||||||
|
spear_object:set_rotation(rotation)
|
||||||
|
spear_object:get_luaentity()._wear = itemstack:get_wear()
|
||||||
|
spear_object:get_luaentity()._stickpos = pointed_thing.under
|
||||||
|
return
|
||||||
|
end
|
||||||
|
end
|
||||||
|
-- Avoid hitting yourself and throw
|
||||||
|
local throw_speed = 12
|
||||||
|
while vector.distance(player_pos, throw_pos) < 1.2 do
|
||||||
|
throw_pos = vector.add(throw_pos, vector.multiply(direction,0.1))
|
||||||
|
end
|
||||||
|
local player_vel = player:get_player_velocity()
|
||||||
|
local spear_object = minetest.add_entity(throw_pos, spear)
|
||||||
|
spear_object:set_velocity(vector.add(player_vel, vector.multiply(direction, throw_speed)))
|
||||||
|
spear_object:set_rotation(rotation)
|
||||||
|
minetest.sound_play("spears_throw", {pos = player_pos}, true)
|
||||||
|
spear_object:get_luaentity()._wear = itemstack:get_wear()
|
||||||
|
spear_object:get_luaentity()._stickpos = nil
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
|
|
||||||
function spears_set_entity(kind, eq, toughness)
|
function spears_set_entity(spear_type, base_damage, toughness)
|
||||||
local SPEAR_ENTITY={
|
local SPEAR_ENTITY={
|
||||||
physical = false,
|
initial_properties = {
|
||||||
timer=0,
|
physical = false,
|
||||||
visual = "wielditem",
|
visual = "item",
|
||||||
visual_size = {x=0.15, y=0.1},
|
visual_size = {x = 0.3, y = 0.3, z = 0.3},
|
||||||
textures = {"spears:spear_" .. kind},
|
wield_item = "spears:spear_" .. spear_type,
|
||||||
lastpos={},
|
collisionbox = {-0.3, -0.3, -0.3, 0.3, 0.3, 0.3},
|
||||||
collisionbox = {0,0,0,0,0,0},
|
},
|
||||||
on_punch = function(self, puncher)
|
|
||||||
if puncher then
|
on_activate = function (self, staticdata, dtime_s)
|
||||||
if puncher:is_player() then
|
self.object:set_armor_groups({immortal = 1})
|
||||||
local stack = {name='spears:spear_' .. kind, wear=self.wear+65535/toughness}
|
end,
|
||||||
local inv = puncher:get_inventory()
|
|
||||||
if inv:room_for_item("main", stack) then
|
on_punch = function (self, puncher)
|
||||||
inv:add_item("main", stack)
|
if puncher:is_player() then -- Grab the spear
|
||||||
self.object:remove()
|
local stack = {name='spears:spear_' .. spear_type, wear = self._wear}
|
||||||
end
|
local inv = puncher:get_inventory()
|
||||||
|
if inv:room_for_item("main", stack) then
|
||||||
|
inv:add_item("main", stack)
|
||||||
|
self.object:remove()
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end,
|
end,
|
||||||
}
|
|
||||||
|
on_step = function(self, dtime)
|
||||||
SPEAR_ENTITY.on_step = function(self, dtime)
|
if not self._wear then
|
||||||
self.timer=self.timer+dtime
|
|
||||||
local pos = self.object:getpos()
|
|
||||||
local node = minetest.get_node(pos)
|
|
||||||
if not self.wear then
|
|
||||||
self.object:remove()
|
|
||||||
return
|
|
||||||
end
|
|
||||||
if self.lastpos.x~=nil then
|
|
||||||
if node.name ~= "air" and not (string.find(node.name, 'grass') and not string.find(node.name, 'dirt')) and not string.find(node.name, 'flowers:') and not string.find(node.name, 'farming:') then
|
|
||||||
self.object:remove()
|
self.object:remove()
|
||||||
if self.wear+65535/toughness < 65535 then
|
return
|
||||||
minetest.add_item(self.lastpos, {name='spears:spear_' .. kind, wear=self.wear+65535/toughness})
|
end
|
||||||
|
local velocity = self.object:get_velocity()
|
||||||
|
local speed = vector.length(velocity)
|
||||||
|
-- Spear is stuck ?
|
||||||
|
if self._stickpos and not self._sticknode then
|
||||||
|
local node = minetest.get_node(self._stickpos)
|
||||||
|
local stick_cracky = minetest.registered_nodes[node.name].groups.cracky
|
||||||
|
if stick_cracky and stick_cracky < 3 then
|
||||||
|
minetest.sound_play("default_metal_footstep", {pos = pos}, true)
|
||||||
|
self.object:remove()
|
||||||
|
minetest.add_item(self.object:get_pos(), {name='spears:spear_' .. spear_type, wear = self._wear})
|
||||||
|
elseif not self._stick_walkable then
|
||||||
|
minetest.sound_play("default_place_node", {pos = throw_pos}, true)
|
||||||
end
|
end
|
||||||
elseif self.timer>0.2 then
|
self._stick_walkable = minetest.registered_nodes[node.name].walkable
|
||||||
local objs = minetest.get_objects_inside_radius({x=pos.x,y=pos.y,z=pos.z}, 1)
|
if not node or not self._stick_walkable then -- Fall when node is removed
|
||||||
for k, obj in pairs(objs) do
|
self.object:remove()
|
||||||
if obj:get_luaentity() ~= nil then
|
minetest.add_item(self.object:get_pos(), {name='spears:spear_' .. spear_type, wear = self._wear})
|
||||||
if obj:get_luaentity().name ~= "spears:spear_" .. kind .. "_entity" and obj:get_luaentity().name ~= "__builtin:item" then
|
return
|
||||||
local speed = vector.length(self.object:getvelocity())
|
end
|
||||||
local damage = (speed + eq)^1.12-20
|
else -- Spear is flying
|
||||||
obj:punch(self.object, 1.0, {
|
local direction = vector.normalize(velocity)
|
||||||
full_punch_interval=1.0,
|
local yaw = minetest.dir_to_yaw(direction)
|
||||||
damage_groups={fleshy=damage},
|
local pitch = math.acos(velocity.y/speed) - math.pi/3
|
||||||
}, nil)
|
local pos = self.object:get_pos()
|
||||||
self.object:remove()
|
local next_pos = vector.add(pos, vector.multiply(velocity, dtime))
|
||||||
if self.wear+65535/toughness < 65535 then
|
local node = minetest.get_node(next_pos)
|
||||||
minetest.add_item(self.lastpos, {name='spears:spear_' .. kind, wear=self.wear+65535/toughness})
|
self.object:set_rotation({x = 0, y = yaw + math.pi/2, z = pitch})
|
||||||
end
|
-- Hit someone?
|
||||||
end
|
local objects_in_radius = minetest.get_objects_inside_radius(next_pos, 0.6)
|
||||||
|
for _,object in ipairs(objects_in_radius) do
|
||||||
|
if object:get_luaentity() ~= self and object:get_armor_groups().fleshy then
|
||||||
|
local damage = (speed + base_damage)^1.15 - 20
|
||||||
|
object:punch(self.object, 1.0, {full_punch_interval = 1.0, damage_groups = {fleshy=damage},}, direction)
|
||||||
|
self.object:remove()
|
||||||
|
minetest.sound_play("spears_hit", {pos = pos}, true)
|
||||||
|
minetest.add_item(pos, {name='spears:spear_' .. spear_type, wear = self._wear + 65535/toughness})
|
||||||
|
return
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
-- Hit a node?
|
||||||
|
if node then
|
||||||
|
if minetest.registered_nodes[node.name].walkable then -- Stick
|
||||||
|
self.object:set_acceleration({x = 0, y = 0, z = 0})
|
||||||
|
self.object:set_velocity({x = 0, y = 0, z = 0})
|
||||||
|
-- Correct position if went too deep to aboid disappearing or turning black
|
||||||
|
while minetest.registered_nodes[minetest.get_node(pos).name].walkable do
|
||||||
|
pos = vector.add(pos, vector.multiply(direction, - 0.002))
|
||||||
|
end
|
||||||
|
self.object:set_pos(pos)
|
||||||
|
self._wear = self._wear + 65535/toughness
|
||||||
|
if self._wear >= 65535 then
|
||||||
|
minetest.sound_play("default_tool_breaks", {pos = pos}, true)
|
||||||
|
self.object:remove()
|
||||||
|
return
|
||||||
|
end
|
||||||
|
self._stickpos = next_pos
|
||||||
|
else -- Get drag
|
||||||
|
local drag = math.max(minetest.registered_nodes[node.name].liquid_viscosity, 0.1)
|
||||||
|
local acceleration = vector.multiply(velocity, -drag)
|
||||||
|
acceleration.y = acceleration.y - 10 * ((7 - drag) / 7)
|
||||||
|
self.object:set_acceleration(acceleration)
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end,
|
||||||
self.lastpos={x=pos.x, y=pos.y, z=pos.z}
|
}
|
||||||
end
|
|
||||||
return SPEAR_ENTITY
|
return SPEAR_ENTITY
|
||||||
end
|
end
|
||||||
|
BIN
screenshot.png
Before Width: | Height: | Size: 293 KiB |
BIN
sounds/spears_hit.ogg
Normal file
BIN
sounds/spears_throw.ogg
Normal file
6
spears.conf.example
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
DISABLE_STONE_SPEAR = true
|
||||||
|
DISABLE_STEEL_SPEAR = true
|
||||||
|
DISABLE_COPPER_SPEAR = true
|
||||||
|
DISABLE_BRONZE_SPEAR = true
|
||||||
|
DISABLE_DIAMOND_SPEAR = true
|
||||||
|
DISABLE_OBSIDIAN_SPEAR = true
|
BIN
textures/spears_spear_bronze.png
Normal file
After Width: | Height: | Size: 234 B |
BIN
textures/spears_spear_copper.png
Normal file
After Width: | Height: | Size: 247 B |
Before Width: | Height: | Size: 272 B After Width: | Height: | Size: 202 B |
BIN
textures/spears_spear_gold.png
Normal file
After Width: | Height: | Size: 209 B |
Before Width: | Height: | Size: 276 B After Width: | Height: | Size: 193 B |
Before Width: | Height: | Size: 273 B After Width: | Height: | Size: 200 B |
Before Width: | Height: | Size: 277 B After Width: | Height: | Size: 226 B |
45
tools.lua
@ -1,18 +1,19 @@
|
|||||||
function spears_register_spear(kind, desc, eq, toughness, material)
|
function spears_register_spear(spear_type, desc, base_damage, toughness, material)
|
||||||
|
|
||||||
minetest.register_tool("spears:spear_" .. kind, {
|
minetest.register_tool("spears:spear_" .. spear_type, {
|
||||||
description = desc .. " spear",
|
description = desc .. " spear",
|
||||||
inventory_image = "spears_spear_" .. kind .. ".png",
|
wield_image = "spears_spear_" .. spear_type .. ".png^[transform4",
|
||||||
wield_scale= {x=2,y=1,z=1},
|
inventory_image = "spears_spear_" .. spear_type .. ".png",
|
||||||
on_drop = function(itemstack, user, pointed_thing)
|
wield_scale= {x = 1.5, y = 1.5, z = 1.5},
|
||||||
spears_shot(itemstack, user)
|
on_secondary_use = function(itemstack, user, pointed_thing)
|
||||||
|
spears_throw(itemstack, user, pointed_thing)
|
||||||
if not minetest.setting_getbool("creative_mode") then
|
if not minetest.setting_getbool("creative_mode") then
|
||||||
itemstack:take_item()
|
itemstack:take_item()
|
||||||
end
|
end
|
||||||
return itemstack
|
return itemstack
|
||||||
end,
|
end,
|
||||||
on_place = function(itemstack, user, pointed_thing)
|
on_place = function(itemstack, user, pointed_thing)
|
||||||
minetest.add_item(pointed_thing.above, itemstack)
|
spears_throw(itemstack, user, pointed_thing)
|
||||||
if not minetest.setting_getbool("creative_mode") then
|
if not minetest.setting_getbool("creative_mode") then
|
||||||
itemstack:take_item()
|
itemstack:take_item()
|
||||||
end
|
end
|
||||||
@ -24,25 +25,27 @@ function spears_register_spear(kind, desc, eq, toughness, material)
|
|||||||
groupcaps={
|
groupcaps={
|
||||||
cracky = {times={[3]=2}, uses=toughness, maxlevel=1},
|
cracky = {times={[3]=2}, uses=toughness, maxlevel=1},
|
||||||
},
|
},
|
||||||
damage_groups = {fleshy=eq},
|
damage_groups = {fleshy=base_damage},
|
||||||
}
|
},
|
||||||
|
sound = {breaks = "default_tool_breaks"},
|
||||||
|
groups = {flammable = 1}
|
||||||
})
|
})
|
||||||
|
|
||||||
SPEAR_ENTITY=spears_set_entity(kind, eq, toughness)
|
local SPEAR_ENTITY = spears_set_entity(spear_type, base_damage, toughness)
|
||||||
|
|
||||||
minetest.register_entity("spears:spear_" .. kind .. "_entity", SPEAR_ENTITY)
|
minetest.register_entity("spears:spear_" .. spear_type .. "_entity", SPEAR_ENTITY)
|
||||||
|
|
||||||
minetest.register_craft({
|
minetest.register_craft({
|
||||||
output = 'spears:spear_' .. kind,
|
output = 'spears:spear_' .. spear_type,
|
||||||
recipe = {
|
recipe = {
|
||||||
{'group:wood', 'group:wood', material},
|
{'group:stick', 'group:stick', material},
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
minetest.register_craft({
|
minetest.register_craft({
|
||||||
output = 'spears:spear_' .. kind,
|
output = 'spears:spear_' .. spear_type,
|
||||||
recipe = {
|
recipe = {
|
||||||
{material, 'group:wood', 'group:wood'},
|
{material, 'group:stick', 'group:stick'},
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
end
|
end
|
||||||
@ -55,6 +58,14 @@ if not DISABLE_STEEL_SPEAR then
|
|||||||
spears_register_spear('steel', 'Steel', 6, 30, 'default:steel_ingot')
|
spears_register_spear('steel', 'Steel', 6, 30, 'default:steel_ingot')
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if not DISABLE_COPPER_SPEAR then
|
||||||
|
spears_register_spear('copper', 'Copper', 5, 35, 'default:copper_ingot')
|
||||||
|
end
|
||||||
|
|
||||||
|
if not DISABLE_BRONZE_SPEAR then
|
||||||
|
spears_register_spear('bronze', 'Bronze', 6, 35, 'default:bronze_ingot')
|
||||||
|
end
|
||||||
|
|
||||||
if not DISABLE_OBSIDIAN_SPEAR then
|
if not DISABLE_OBSIDIAN_SPEAR then
|
||||||
spears_register_spear('obsidian', 'Obsidian', 8, 30, 'default:obsidian')
|
spears_register_spear('obsidian', 'Obsidian', 8, 30, 'default:obsidian')
|
||||||
end
|
end
|
||||||
@ -62,3 +73,7 @@ end
|
|||||||
if not DISABLE_DIAMOND_SPEAR then
|
if not DISABLE_DIAMOND_SPEAR then
|
||||||
spears_register_spear('diamond', 'Diamond', 8, 40, 'default:diamond')
|
spears_register_spear('diamond', 'Diamond', 8, 40, 'default:diamond')
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if not DISABLE_GOLD_SPEAR then
|
||||||
|
spears_register_spear('gold', 'Golden', 5, 40, 'default:gold_ingot')
|
||||||
|
end
|
||||||
|