16 Commits

Author SHA1 Message Date
Jordan Irwin
0536311f0e Call sound groups directly 2021-08-06 23:27:02 -07:00
Jordan Irwin
b6561dc8f4 Use sounds mod for "hiss" & delete sounds from unknown source 2021-08-06 15:33:23 -07:00
Jordan Irwin
0adc6635c4 Add to TODO list 2021-08-06 14:59:46 -07:00
Jordan Irwin
4143bc9543 Use sounds mod for explosion 2021-08-06 14:59:30 -07:00
Jordan Irwin
5f9e529caf Add to TODO list 2021-08-03 03:06:27 -07:00
Jordan Irwin
02d797fce2 Add minimum required Minetest version to mod.conf 2021-07-20 18:00:24 -07:00
Jordan Irwin
6d9cc8f80b Add to TODO list 2021-07-20 17:56:28 -07:00
Jordan Irwin
fbf7df4cfa Delete unused spawn.lua 2021-07-20 17:56:28 -07:00
Jordan Irwin
0e5ef436d0 Add to TODO list 2021-07-20 17:56:28 -07:00
Jordan Irwin
150190c2cd Update README 2021-07-20 17:56:28 -07:00
Jordan Irwin
9b80373efc Add to TODO list 2021-07-20 17:56:27 -07:00
Jordan Irwin
e59081ea02 Fix chasing player when close 2021-07-20 17:56:27 -07:00
Jordan Irwin
af218cc5ec Add to TODO list 2021-07-20 17:56:27 -07:00
Jordan Irwin
00a7f5a550 Fix tnt drop 2021-07-20 17:56:27 -07:00
Jordan Irwin
2b275380df Fix entity vertical positioning 2021-07-20 17:56:27 -07:00
Jordan Irwin
86392efe85 Begin converting to cmer API 2021-07-20 17:56:27 -07:00
10 changed files with 224 additions and 84 deletions

View File

@@ -18,7 +18,7 @@ Settings documented in [reference][].
### Requirements: ### Requirements:
- Depends: default, tnt - Depends: cmer, default, tnt
- Optional depends: nether, sounds, simple_protection - Optional depends: nether, sounds, simple_protection
### Links: ### Links:

View File

@@ -3,3 +3,13 @@ TODO:
- add version using mobs_redo API - add version using mobs_redo API
- add version using cmer API - add version using cmer API
- add griefing option - add griefing option
- add wear to armor when exploding
- disable attacking if damage disabled
- make powered versions run faster when chasing
- fix knockback when chasing
- fix hit sound when chasing
- make sneeker stop when within 2 of player
- use asm_spawneggs for egg
- add localization support
- add Spanish translation
- fix removing single node if protected

View File

@@ -1,4 +1,9 @@
next
----
- uses "sounds" mod
v1.1 v1.1
---- ----
- added sound when hit - added sound when hit
@@ -7,6 +12,7 @@ v1.1
- added setting to customize spawn nodes - added setting to customize spawn nodes
- added simple_protection support - added simple_protection support
v1.0 v1.0
---- ----
- forked & renamed from Rui's original "creeper" mod - forked & renamed from Rui's original "creeper" mod

View File

@@ -7,7 +7,7 @@ elseif core.get_modpath("sounds") then
hit_sound = "sounds_entity_hit" hit_sound = "sounds_entity_hit"
end end
--[[
local function jump(self, pos, direction) local function jump(self, pos, direction)
local velocity = self.object:get_velocity() local velocity = self.object:get_velocity()
if core.registered_nodes[core.get_node(pos).name].climbable then if core.registered_nodes[core.get_node(pos).name].climbable then
@@ -75,24 +75,105 @@ local def = {
animation_speed = 30, animation_speed = 30,
knockback_level = 2 knockback_level = 2
} }
]]
def.on_activate = function(self, staticdata) local spawn_nodes = {}
if sneeker.spawn_nodes ~= "" then
if not sneeker.spawn_nodes:find(",") then
table.insert(spawn_nodes, sneeker.spawn_nodes)
else
for _, node in ipairs(sneeker.spawn_nodes:split(",")) do
local node = node:trim()
if node ~= "" then
table.insert(spawn_nodes, node)
end
end
end
end
if #spawn_nodes == 0 then
sneeker.log("warning", "no spawning nodes set, cannot spawn")
end
local item_drops = {}
if core.registered_items["tnt:gunpowder"] then
table.insert(item_drops, {"tnt:gunpowder", {min=1, max=2}, chance=0.66})
end
local def = {
name = "sneeker:sneeker",
nametag = "Sneeker",
stats = {
hp = 20,
hostile_2 = true,
lifetime = sneeker.lifetime,
can_jump = 5,
can_swim = true,
has_knockback = true,
sneaky = true,
},
modes = {
idle = {chance=0.3},
walk = {chance=0.7, moving_speed=1.5},
--chase = {moving_speed=1.5},
--death = {},
},
model = {
mesh = "character.b3d",
textures = {"sneeker.png"},
collisionbox = {-0.25, 0.3, -0.25, 0.25, 1.8, 0.25},
rotation = -90,
animations = {
idle = {start=0, stop=79, speed=30},
walk = {start=168, stop=187, speed=30},
--chase = {start=168, stop=187, speed=30},
--death = {},
},
},
--sounds = {},
drops = item_drops,
--[[
combat = {
search_enemy = true,
},
]]
spawning = {
abm_nodes = {
spawn_on = spawn_nodes,
},
abm_interval = sneeker.spawn_interval,
abm_chance = sneeker.spawn_chance,
number = 1,
light = {min=sneeker.spawn_minlight, max=sneeker.spawn_maxlight},
height_limit = {min=sneeker.spawn_minheight, max=sneeker.spawn_maxheight},
},
}
def.on_activate = function(self, staticdata, dtime_s)
--[[
self.yaw = 0 self.yaw = 0
self.anim = 1 self.anim = 1
self.timer = 0 self.timer = 0
]]
self.visualx = 1 self.visualx = 1
--[[
self.jump_timer = 0 self.jump_timer = 0
self.turn_timer = 0 self.turn_timer = 0
self.turn_speed = 0 self.turn_speed = 0
]]
self.powered = false self.powered = false
--[[
self.knockback = false self.knockback = false
self.state = math.random(1, 2) self.state = math.random(1, 2)
self.old_y = self.object:get_pos().y self.old_y = self.object:get_pos().y
]]
--[[
-- despawning -- despawning
self.lifetime = sneeker.lifetime self.lifetime = sneeker.lifetime
self.lifetimer = 0 self.lifetimer = 0
self.check_despawn_player_distance = true self.check_despawn_player_distance = true
]]
local data = core.deserialize(staticdata) local data = core.deserialize(staticdata)
if data and type(data) == "table" then if data and type(data) == "table" then
@@ -114,9 +195,9 @@ local function isnan(n)
end end
local function expand(self) local function expand(self)
if self.chase and self.visualx < 2 then if self.expanding and self.visualx < 2 then
if self.hiss == false then if self.hiss == false and sounds then
core.sound_play("sneeker_hiss", {object=self.object, gain=1.5, max_hear_distance=2*64}) sounds.fuse(1, {object=self.object, gain=1.5, max_hear_distance=2*64})
end end
self.visualx = self.visualx+0.05 self.visualx = self.visualx+0.05
self.object:set_properties({ self.object:set_properties({
@@ -135,9 +216,9 @@ end
local function explode(self, pos) local function explode(self, pos)
self.object:remove() self.object:remove()
sneeker.boom(pos, self.powered) sneeker.boom(pos, self.powered)
core.sound_play("sneeker_explode", {object=self.object, gain=sneeker.boom_gain, max_hear_distance=2*64})
end end
--[[
local function h_collides(pos, collision_info, touching_ground) local function h_collides(pos, collision_info, touching_ground)
if not touching_ground or type(collision_info) ~= "table" or #collision_info == 0 then if not touching_ground or type(collision_info) ~= "table" or #collision_info == 0 then
return false return false
@@ -166,11 +247,53 @@ local function h_collides(pos, collision_info, touching_ground)
return h_vel.x < walk_speed and h_vel.z < walk_speed, h_col.node_pos return h_vel.x < walk_speed and h_vel.z < walk_speed, h_col.node_pos
end end
]]
local function chase(self, target)
-- FIXME: why does setting mode = "chase" not work?
if not self.chase_anim then
self.object:set_animation({x=168, y=187}, 30, 0)
self.chase_anim = true
end
local pos = self.object:get_pos()
local tpos = target:get_pos()
local vec = {x=tpos.x-pos.x, y=tpos.y-pos.y, z=tpos.z-pos.z}
local yaw = math.atan(vec.z/vec.x)+math.pi^2
if tpos.x > pos.x then
yaw = yaw+math.pi
end
yaw = yaw-2
self.object:set_yaw(yaw)
local direction = {x=math.sin(yaw)*-1, y=0, z=math.cos(yaw)}
-- FIXME: hack
local can_set = true
for _, c in ipairs({direction.x*2.5, direction.z*2.5}) do
if isnan(c) then
can_set = false
break
end
end
if can_set then
self.object:set_velocity({x=direction.x*2.5, y=self.object:get_velocity().y, z=direction.z*2.5})
end
end
local function end_chase(self)
self.chasing = nil
self.chase_anim = false
self.mode = "idle"
self.expanding = false
end
def.on_step = function(self, dtime, moveresult) def.on_step = function(self, dtime, moveresult)
--[[
-- update lifetime timer -- update lifetime timer
-- FIXME: this is longer than realtime -- FIXME: this is longer than realtime
self.lifetimer = self.lifetimer + dtime self.lifetimer = self.lifetimer + dtime_s
if self.lifetimer >= self.lifetime then if self.lifetimer >= self.lifetime then
-- TODO: should have a death animation -- TODO: should have a death animation
self.object:remove() self.object:remove()
@@ -183,9 +306,11 @@ def.on_step = function(self, dtime, moveresult)
local ANIM_STAND = 1 local ANIM_STAND = 1
local ANIM_WALK = 2 local ANIM_WALK = 2
]]
local pos = self.object:get_pos() local pos = self.object:get_pos()
--[[
if sneeker.despawn_player_far then if sneeker.despawn_player_far then
-- run check about once per 60 seconds -- run check about once per 60 seconds
local interval_reached = math.floor(self.lifetimer % 60) == 0 local interval_reached = math.floor(self.lifetimer % 60) == 0
@@ -246,9 +371,22 @@ def.on_step = function(self, dtime, moveresult)
self.yaw = self.yaw-self.turn_speed self.yaw = self.yaw-self.turn_speed
self.object:set_yaw(self.yaw) self.object:set_yaw(self.yaw)
end end
]]
expand(self) expand(self)
local players_within_10 = {}
for _, object in ipairs(core.get_objects_inside_radius(pos, 10)) do
if object:is_player() then
table.insert(players_within_10, object)
end
end
if not self.chasing then
self.chasing = players_within_10[1]
end
--[[
self.chase = false self.chase = false
for _, object in ipairs(inside) do for _, object in ipairs(inside) do
@@ -302,8 +440,49 @@ def.on_step = function(self, dtime, moveresult)
end end
end end
end end
]]
if self.state == "chase" then if self.chasing then
if self.chasing:get_hp() > 0 then
self.mode = "chase"
local within_2 = false
for _, object in ipairs(core.get_objects_inside_radius(pos, 2)) do
if object == self.chasing then
within_2 = true
break
end
end
if within_2 then
if self.visualx >= 2 then
explode(self, pos)
self.chasing = nil
return true
end
self.expanding = true
else
self.expanding = false
end
local within_10 = false
for _, object in ipairs(players_within_10) do
if object == self.chasing then
within_10 = true
break
end
end
if within_10 then
-- follow player
chase(self, self.chasing)
return true
end
end
end_chase(self)
--[[
if self.anim ~= ANIM_WALK then if self.anim ~= ANIM_WALK then
self.object:set_animation({x=animation.walk_START, y=animation.walk_END}, anim_speed, 0) self.object:set_animation({x=animation.walk_START, y=animation.walk_END}, anim_speed, 0)
self.anim = ANIM_WALK self.anim = ANIM_WALK
@@ -325,7 +504,9 @@ def.on_step = function(self, dtime, moveresult)
end end
end end
end end
]]
--[[
if #inside ~= 0 then if #inside ~= 0 then
for _, object in ipairs(inside) do for _, object in ipairs(inside) do
if object:is_player() and object:get_hp() ~= 0 then if object:is_player() and object:get_hp() ~= 0 then
@@ -372,8 +553,12 @@ def.on_step = function(self, dtime, moveresult)
else else
self.state = "stand" self.state = "stand"
end end
]]
elseif self.mode == "chase" then
self.mode = "idle"
end end
--[[
-- Swim -- Swim
local node = core.get_node(pos) local node = core.get_node(pos)
if core.get_item_group(node.name, "water") ~= 0 then if core.get_item_group(node.name, "water") ~= 0 then
@@ -389,8 +574,10 @@ def.on_step = function(self, dtime, moveresult)
end end
return true return true
]]
end end
--[[
def.on_punch = function(self, puncher, time_from_last_punch, tool_capabilities, dir) def.on_punch = function(self, puncher, time_from_last_punch, tool_capabilities, dir)
if hit_sound then if hit_sound then
core.sound_play(hit_sound, {object=self.object}, parameters, true) core.sound_play(hit_sound, {object=self.object}, parameters, true)
@@ -416,6 +603,7 @@ def.on_punch = function(self, puncher, time_from_last_punch, tool_capabilities,
local obj = core.add_item(p, {name="tnt:gunpowder", count=math.random(0, 2)}) local obj = core.add_item(p, {name="tnt:gunpowder", count=math.random(0, 2)})
end end
end end
]]
def.get_staticdata = function(self) def.get_staticdata = function(self)
return core.serialize({ return core.serialize({
@@ -423,7 +611,11 @@ def.get_staticdata = function(self)
}) })
end end
--[[
core.register_entity("sneeker:sneeker", def) core.register_entity("sneeker:sneeker", def)
]]
cmer.register_mob(def)
core.register_craftitem("sneeker:spawnegg", { core.register_craftitem("sneeker:spawnegg", {
description = "Sneeker Spawn Egg", description = "Sneeker Spawn Egg",

View File

@@ -28,7 +28,6 @@ end
local scripts = { local scripts = {
"settings", "settings",
"tnt_function", "tnt_function",
"spawn",
"entity", "entity",
} }

View File

@@ -1,7 +1,9 @@
name = sneeker name = sneeker
title = Sneeker title = Sneeker
description = An explosive nuisance. description = An explosive nuisance.
license = MIT
version = 1.1 version = 1.1
author = Rui author = Rui, Jordan Irwin (AntumDeluge)
depends = default, tnt min_minetest_version = 5.0
depends = cmer, default, tnt
optional_depends = nether, sounds, simple_protection optional_depends = nether, sounds, simple_protection

Binary file not shown.

Binary file not shown.

View File

@@ -1,71 +0,0 @@
local spawn_nodes = {}
if sneeker.spawn_nodes ~= "" then
if not sneeker.spawn_nodes:find(",") then
table.insert(spawn_nodes, sneeker.spawn_nodes)
else
for _, node in ipairs(sneeker.spawn_nodes:split(",")) do
local node = node:trim()
if node ~= "" then
table.insert(spawn_nodes, node)
end
end
end
end
if #spawn_nodes == 0 then
sneeker.log("warning", "no spawning nodes set, cannot spawn")
end
core.register_abm({
nodenames = spawn_nodes,
neighbors = {"air"},
interval = sneeker.spawn_interval,
chance = sneeker.spawn_chance,
action = function(pos, node, aoc, aocw)
if aoc >= sneeker.spawn_mapblock_limit then return end
-- check above target node
pos.y = pos.y+1
local pos_string = tostring(math.floor(pos.x))
.. "," .. tostring(math.floor(pos.y))
.. "," .. tostring(math.floor(pos.z))
if sneeker.spawn_require_player_nearby then
local player_nearby = false
for _, entity in ipairs(core.get_objects_inside_radius(pos, sneeker.spawn_player_radius)) do
if entity:is_player() then
player_nearby = true
break
end
end
if not player_nearby then return end
end
if pos.y > sneeker.spawn_maxheight or pos.y < sneeker.spawn_minheight then
return
end
-- needs two vertical air nodes to spawn
for _, node_pos in ipairs({pos, {x=pos.x, y=pos.y+1, z=pos.z}}) do
if core.get_node(node_pos).name ~= "air" then
return
end
end
local llevel = core.get_node_light(pos)
if not llevel or llevel > sneeker.spawn_maxlight or llevel < sneeker.spawn_minlight then
return
end
local spawned = core.add_entity(pos, "sneeker:sneeker")
if not spawned then
sneeker.log("warning", "Failed to spawn at: " .. pos_string)
else
sneeker.log("debug", "Spawned at: " .. pos_string)
end
end
})

View File

@@ -307,7 +307,9 @@ function sneeker.boom(pos, large)
if large then if large then
radius = large_radius radius = large_radius
end end
core.sound_play("sneeker_explode", {pos=pos, gain=1.5, max_hear_distance=2*64}) if sounds then
sounds.explosion(1, {pos=pos, gain=sneeker.boom_gain, max_hear_distance=2*64})
end
core.set_node(pos, {name="tnt:boom"}) core.set_node(pos, {name="tnt:boom"})
core.get_node_timer(pos):start(0.5) core.get_node_timer(pos):start(0.5)
local drops = explode(pos, radius) local drops = explode(pos, radius)