2021-04-11 16:52:31 +00:00
-- API for Mobs Redo: MineClone 2 Delux 2.0 DRM Free Early Access Super Extreme Edition
2021-04-13 12:07:32 +00:00
-- current state of things: Why?
2021-04-11 16:52:31 +00:00
-- lua locals
2021-04-12 12:47:07 +00:00
--localize minetest functions
local minetest_settings = minetest.settings
local minetest_get_objects_inside_radius = minetest.get_objects_inside_radius
local minetest_get_modpath = minetest.get_modpath
local minetest_registered_nodes = minetest.registered_nodes
local minetest_get_node = minetest.get_node
local minetest_get_item_group = minetest.get_item_group
local minetest_registered_entities = minetest.registered_entities
local minetest_line_of_sight = minetest.line_of_sight
local minetest_after = minetest.after
local minetest_sound_play = minetest.sound_play
local minetest_add_particlespawner = minetest.add_particlespawner
local minetest_registered_items = minetest.registered_items
local minetest_set_node = minetest.set_node
local minetest_add_item = minetest.add_item
local minetest_get_craft_result = minetest.get_craft_result
local minetest_find_path = minetest.find_path
local minetest_is_protected = minetest.is_protected
local minetest_is_creative_enabled = minetest.is_creative_enabled
local minetest_find_node_near = minetest.find_node_near
local minetest_find_nodes_in_area_under_air = minetest.find_nodes_in_area_under_air
local minetest_raycast = minetest.raycast
local minetest_get_us_time = minetest.get_us_time
local minetest_add_entity = minetest.add_entity
local minetest_get_natural_light = minetest.get_natural_light
local minetest_get_node_or_nil = minetest.get_node_or_nil
2021-04-11 16:52:31 +00:00
-- localize math functions
local math_pi = math.pi
local math_sin = math.sin
local math_cos = math.cos
local math_abs = math.abs
local math_min = math.min
local math_max = math.max
local math_atan = math.atan
local math_random = math.random
local math_floor = math.floor
2021-04-13 11:39:57 +00:00
-- localize vector functions
local vector_new = vector.new
2015-06-29 17:55:56 +00:00
mobs = { }
2021-04-12 12:47:07 +00:00
-- mob constants
2018-02-04 06:11:44 +00:00
local MAX_MOB_NAME_LENGTH = 30
2021-04-12 12:47:07 +00:00
local BREED_TIME = 30
local BREED_TIME_AGAIN = 300
local CHILD_GROW_TIME = 60 * 20
local DEATH_DELAY = 0.5
local DEFAULT_FALL_SPEED = - 10
local FLOP_HEIGHT = 5.0
local FLOP_HOR_SPEED = 1.5
local MOB_CAP = { }
2020-04-11 00:46:03 +00:00
MOB_CAP.hostile = 70
MOB_CAP.passive = 10
MOB_CAP.ambient = 15
2021-04-12 12:47:07 +00:00
MOB_CAP.water = 15
2020-04-11 00:46:03 +00:00
2021-04-13 12:20:11 +00:00
-- Load main settings
2021-04-13 12:08:29 +00:00
local damage_enabled = minetest_settings : get_bool ( " enable_damage " )
local disable_blood = minetest_settings : get_bool ( " mobs_disable_blood " )
local mobs_drop_items = minetest_settings : get_bool ( " mobs_drop_items " ) ~= false
local mobs_griefing = minetest_settings : get_bool ( " mobs_griefing " ) ~= false
local spawn_protected = minetest_settings : get_bool ( " mobs_spawn_protected " ) ~= false
local remove_far = true
local difficulty = tonumber ( minetest_settings : get ( " mob_difficulty " ) ) or 1.0
local show_health = false
local max_per_block = tonumber ( minetest_settings : get ( " max_objects_per_block " ) or 64 )
local mobs_spawn_chance = tonumber ( minetest_settings : get ( " mobs_spawn_chance " ) or 2.5 )
2021-04-13 12:20:11 +00:00
-- pathfinding settings
local enable_pathfinding = true
local stuck_timeout = 3 -- how long before mob gets stuck in place and starts searching
local stuck_path_timeout = 10 -- how long will mob follow path before giving up
-- default nodes
local node_ice = " mcl_core:ice "
local node_snowblock = " mcl_core:snowblock "
local node_snow = " mcl_core:snow "
mobs.fallback_node = minetest.registered_aliases [ " mapgen_dirt " ] or " mcl_core:dirt "
local mod_weather = minetest_get_modpath ( " mcl_weather " ) ~= nil
local mod_explosions = minetest_get_modpath ( " mcl_explosions " ) ~= nil
local mod_mobspawners = minetest_get_modpath ( " mcl_mobspawners " ) ~= nil
local mod_hunger = minetest_get_modpath ( " mcl_hunger " ) ~= nil
local mod_worlds = minetest_get_modpath ( " mcl_worlds " ) ~= nil
local mod_armor = minetest_get_modpath ( " mcl_armor " ) ~= nil
local mod_experience = minetest_get_modpath ( " mcl_experience " ) ~= nil
2021-04-13 12:07:32 +00:00
-- random locals I found
2021-04-13 12:20:11 +00:00
local los_switcher = false
2021-04-13 12:07:32 +00:00
local height_switcher = false
2021-04-13 12:20:11 +00:00
-- Get translator
2019-03-07 19:43:39 +00:00
local S = minetest.get_translator ( " mcl_mobs " )
2017-05-25 08:33:19 +00:00
2017-07-04 23:52:39 +00:00
-- CMI support check
local use_cmi = minetest.global_exists ( " cmi " )
2017-11-03 23:22:43 +00:00
2017-01-16 16:40:08 +00:00
-- Invisibility mod check
mobs.invis = { }
2017-11-03 23:22:43 +00:00
if minetest.global_exists ( " invisibility " ) then
2017-01-16 16:40:08 +00:00
mobs.invis = invisibility
end
2017-11-03 23:22:43 +00:00
-- creative check
function mobs . is_creative ( name )
2020-07-10 14:08:40 +00:00
return minetest.is_creative_enabled ( name )
2017-11-03 23:22:43 +00:00
end
2017-01-16 16:40:08 +00:00
local atan = function ( x )
2017-05-25 08:33:19 +00:00
if not x or x ~= x then
2017-01-16 16:40:08 +00:00
return 0
else
2021-04-11 16:52:31 +00:00
return math_atan ( x )
2017-01-16 16:40:08 +00:00
end
end
2021-04-13 12:08:29 +00:00
2017-11-03 23:22:43 +00:00
2020-12-05 00:30:16 +00:00
-- Shows helpful debug info above each mob
2021-04-12 12:47:07 +00:00
local mobs_debug = minetest_settings : get_bool ( " mobs_debug " , false )
2020-12-05 00:30:16 +00:00
2017-11-03 23:22:43 +00:00
-- Peaceful mode message so players will know there are no monsters
2021-04-12 12:47:07 +00:00
if minetest_settings : get_bool ( " only_peaceful_mobs " , false ) then
2017-11-03 23:22:43 +00:00
minetest.register_on_joinplayer ( function ( player )
minetest.chat_send_player ( player : get_player_name ( ) ,
2019-03-07 19:43:39 +00:00
S ( " Peaceful mode active! No monsters will spawn. " ) )
2017-11-03 23:22:43 +00:00
end )
end
2017-01-16 16:40:08 +00:00
2021-04-11 15:58:33 +00:00
local collision = function ( self )
pos = self.object : get_pos ( )
--do collision detection from the base of the mob
collisionbox = self.object : get_properties ( ) . collisionbox
pos.y = pos.y + collisionbox [ 2 ]
collision_boundary = collisionbox [ 4 ]
radius = collision_boundary
if collisionbox [ 5 ] > collision_boundary then
radius = collisionbox [ 5 ]
end
collision_count = 0
2021-04-12 12:47:07 +00:00
for _ , object in ipairs ( minetest_get_objects_inside_radius ( pos , radius * 1.25 ) ) do
2021-04-11 15:58:33 +00:00
if object ~= self.object and ( object : is_player ( ) or object : get_luaentity ( ) . _cmi_is_mob == true ) and
--don't collide with rider, rider don't collide with thing
( not object : get_attach ( ) or ( object : get_attach ( ) and object : get_attach ( ) ~= self.object ) ) and
( not self.object : get_attach ( ) or ( self.object : get_attach ( ) and self.object : get_attach ( ) ~= object ) ) then
--stop infinite loop
collision_count = collision_count + 1
if collision_count > 100 then
break
end
pos2 = object : get_pos ( )
object_collisionbox = object : get_properties ( ) . collisionbox
pos2.y = pos2.y + object_collisionbox [ 2 ]
object_collision_boundary = object_collisionbox [ 4 ]
--this is checking the difference of the object collided with's possision
--if positive top of other object is inside (y axis) of current object
y_base_diff = ( pos2.y + object_collisionbox [ 5 ] ) - pos.y
y_top_diff = ( pos.y + collisionbox [ 5 ] ) - pos2.y
distance = vector.distance ( vector.new ( pos.x , 0 , pos.z ) , vector.new ( pos2.x , 0 , pos2.z ) )
if distance <= collision_boundary + object_collision_boundary and y_base_diff >= 0 and y_top_diff >= 0 then
dir = vector.direction ( pos , pos2 )
dir.y = 0
--eliminate mob being stuck in corners
if dir.x == 0 and dir.z == 0 then
2021-04-11 16:52:31 +00:00
dir = vector.new ( math_random ( - 1 , 1 ) * math_random ( ) , 0 , math_random ( - 1 , 1 ) * math_random ( ) )
2021-04-11 15:58:33 +00:00
end
local velocity = vector.multiply ( dir , 1.1 )
--local velocity = vector.normalize(dir)
vel1 = vector.multiply ( velocity , - 1 )
vel2 = velocity
self.object : add_velocity ( vel1 )
2021-04-11 18:47:56 +00:00
--reenable fire spreading eventually
2021-04-11 15:58:33 +00:00
if object : is_player ( ) then
object : add_player_velocity ( vel2 )
--if self.on_fire then
-- start_fire(object)
--end
--if is_player_on_fire(object) then
-- start_fire(self.object)
--end
else
object : add_velocity ( vel2 )
--if self.on_fire then
-- start_fire(object)
--end
--if object:get_luaentity().on_fire then
-- start_fire(self.object)
--end
end
end
end
end
end
2017-05-25 08:33:19 +00:00
-- move mob in facing direction
2017-07-04 23:52:39 +00:00
local set_velocity = function ( self , v )
2020-12-03 16:37:44 +00:00
local c_x , c_y = 0 , 0
-- halt mob if it has been ordered to stay
2018-05-29 15:00:30 +00:00
if self.order == " stand " then
2019-03-06 03:38:57 +00:00
self.object : set_velocity ( { x = 0 , y = 0 , z = 0 } )
2018-05-29 15:00:30 +00:00
return
end
2017-11-03 23:22:43 +00:00
local yaw = ( self.object : get_yaw ( ) or 0 ) + self.rotate
2020-12-03 16:37:44 +00:00
2021-04-11 15:58:33 +00:00
self.object : add_velocity ( {
2021-04-11 16:52:31 +00:00
x = ( math_sin ( yaw ) * - v ) + c_x ,
2020-12-03 16:37:44 +00:00
y = self.object : get_velocity ( ) . y ,
2021-04-11 16:52:31 +00:00
z = ( math_cos ( yaw ) * v ) + c_y ,
2017-01-16 16:40:08 +00:00
} )
end
2017-05-25 08:33:19 +00:00
2020-12-03 16:37:44 +00:00
2017-11-03 23:22:43 +00:00
-- calculate mob velocity
2017-07-04 23:52:39 +00:00
local get_velocity = function ( self )
2017-01-16 16:40:08 +00:00
2019-03-06 03:38:57 +00:00
local v = self.object : get_velocity ( )
2021-01-05 19:30:59 +00:00
if v then
return ( v.x * v.x + v.z * v.z ) ^ 0.5
end
2017-01-16 16:40:08 +00:00
2021-01-05 19:30:59 +00:00
return 0
2017-01-16 16:40:08 +00:00
end
2017-11-03 23:22:43 +00:00
-- set and return valid yaw
2021-01-23 14:40:12 +00:00
local set_yaw = function ( self , yaw , delay , dtime )
2017-05-25 08:33:19 +00:00
if not yaw or yaw ~= yaw then
yaw = 0
2017-01-16 16:40:08 +00:00
end
2018-05-29 15:00:30 +00:00
delay = delay or 0
if delay == 0 then
2021-01-23 14:40:12 +00:00
if self.shaking and dtime then
2021-04-11 16:52:31 +00:00
yaw = yaw + ( math_random ( ) * 2 - 1 ) * 5 * dtime
2021-01-23 14:40:12 +00:00
end
2018-05-29 15:00:30 +00:00
self.object : set_yaw ( yaw )
2021-03-25 08:24:38 +00:00
update_roll ( self )
2018-05-29 15:00:30 +00:00
return yaw
end
self.target_yaw = yaw
self.delay = delay
2017-05-25 08:33:19 +00:00
2018-05-29 15:00:30 +00:00
return self.target_yaw
end
-- global function to set mob yaw
2021-01-23 14:40:12 +00:00
function mobs : yaw ( self , yaw , delay , dtime )
set_yaw ( self , yaw , delay , dtime )
2017-05-25 08:33:19 +00:00
end
-- set defined animation
2020-12-05 03:33:23 +00:00
local set_animation = function ( self , anim , fixed_frame )
2020-12-05 11:59:12 +00:00
if not self.animation or not anim then
return
end
if self.state == " die " and anim ~= " die " and anim ~= " stand " then
return
end
2017-05-25 08:33:19 +00:00
2017-01-16 16:40:08 +00:00
self.animation . current = self.animation . current or " "
2020-12-05 11:59:12 +00:00
if ( anim == self.animation . current
2017-05-25 08:33:19 +00:00
or not self.animation [ anim .. " _start " ]
2020-12-05 11:59:12 +00:00
or not self.animation [ anim .. " _end " ] ) and self.state ~= " die " then
2017-05-25 08:33:19 +00:00
return
end
self.animation . current = anim
2017-01-16 16:40:08 +00:00
2020-12-05 03:33:23 +00:00
local a_start = self.animation [ anim .. " _start " ]
local a_end
if fixed_frame then
a_end = a_start
else
a_end = self.animation [ anim .. " _end " ]
end
2017-05-25 08:33:19 +00:00
self.object : set_animation ( {
2020-12-05 03:33:23 +00:00
x = a_start ,
y = a_end } ,
2017-07-04 23:52:39 +00:00
self.animation [ anim .. " _speed " ] or self.animation . speed_normal or 15 ,
0 , self.animation [ anim .. " _loop " ] ~= false )
end
2017-01-16 16:40:08 +00:00
2017-07-04 23:52:39 +00:00
-- above function exported for mount.lua
2017-07-26 14:55:36 +00:00
function mobs : set_animation ( self , anim )
2017-07-04 23:52:39 +00:00
set_animation ( self , anim )
2017-05-25 08:33:19 +00:00
end
2017-01-16 16:40:08 +00:00
2020-12-05 13:42:03 +00:00
mobs.death_effect = function ( pos , yaw , collisionbox , rotate )
2020-08-19 16:31:45 +00:00
local min , max
if collisionbox then
min = { x = collisionbox [ 1 ] , y = collisionbox [ 2 ] , z = collisionbox [ 3 ] }
max = { x = collisionbox [ 4 ] , y = collisionbox [ 5 ] , z = collisionbox [ 6 ] }
else
min = { x = - 0.5 , y = 0 , z = - 0.5 }
max = { x = 0.5 , y = 0.5 , z = 0.5 }
end
2020-12-05 13:42:03 +00:00
if rotate then
2021-04-11 16:52:31 +00:00
min = vector.rotate ( min , { x = 0 , y = yaw , z = math_pi / 2 } )
max = vector.rotate ( max , { x = 0 , y = yaw , z = math_pi / 2 } )
2020-12-05 13:42:03 +00:00
min , max = vector.sort ( min , max )
min = vector.multiply ( min , 0.5 )
max = vector.multiply ( max , 0.5 )
end
2020-08-19 16:31:45 +00:00
2021-04-12 12:47:07 +00:00
minetest_add_particlespawner ( {
2020-12-05 04:11:06 +00:00
amount = 50 ,
time = 0.001 ,
2020-08-19 16:31:45 +00:00
minpos = vector.add ( pos , min ) ,
maxpos = vector.add ( pos , max ) ,
2020-12-05 04:11:06 +00:00
minvel = vector.new ( - 5 , - 5 , - 5 ) ,
maxvel = vector.new ( 5 , 5 , 5 ) ,
minexptime = 1.1 ,
2020-08-19 16:31:45 +00:00
maxexptime = 1.5 ,
2020-12-05 04:11:06 +00:00
minsize = 1 ,
maxsize = 2 ,
collisiondetection = false ,
vertical = false ,
texture = " mcl_particles_mob_death.png^[colorize:#000000:255 " ,
2020-08-19 16:31:45 +00:00
} )
2020-12-05 04:11:06 +00:00
2021-04-12 12:47:07 +00:00
minetest_sound_play ( " mcl_mobs_mob_poof " , {
2020-12-05 04:11:06 +00:00
pos = pos ,
gain = 1.0 ,
max_hear_distance = 8 ,
} , true )
2020-08-19 16:31:45 +00:00
end
2017-05-25 08:33:19 +00:00
2017-01-16 16:40:08 +00:00
-- execute current state (stand, walk, run, attacks)
2020-04-08 13:03:03 +00:00
-- returns true if mob has died
2017-01-16 16:40:08 +00:00
local do_states = function ( self , dtime )
2017-08-06 10:49:13 +00:00
local yaw = self.object : get_yaw ( ) or 0
2017-01-16 16:40:08 +00:00
2021-04-13 12:20:11 +00:00
--[[
2017-01-16 16:40:08 +00:00
if self.state == " stand " then
2021-04-11 16:52:31 +00:00
if math_random ( 1 , 4 ) == 1 then
2017-01-16 16:40:08 +00:00
local lp = nil
2017-11-03 23:22:43 +00:00
local s = self.object : get_pos ( )
2021-04-12 12:47:07 +00:00
local objs = minetest_get_objects_inside_radius ( s , 3 )
2017-01-16 16:40:08 +00:00
2017-05-25 08:33:19 +00:00
for n = 1 , # objs do
2017-01-16 16:40:08 +00:00
2017-05-25 08:33:19 +00:00
if objs [ n ] : is_player ( ) then
2017-11-03 23:22:43 +00:00
lp = objs [ n ] : get_pos ( )
2017-05-25 08:33:19 +00:00
break
2017-01-16 16:40:08 +00:00
end
end
-- look at any players nearby, otherwise turn randomly
if lp then
local vec = {
x = lp.x - s.x ,
z = lp.z - s.z
}
2021-04-11 16:52:31 +00:00
yaw = ( atan ( vec.z / vec.x ) + math_pi / 2 ) - self.rotate
2017-01-16 16:40:08 +00:00
2021-04-11 16:52:31 +00:00
if lp.x > s.x then yaw = yaw + math_pi end
2017-01-16 16:40:08 +00:00
else
2021-04-11 16:52:31 +00:00
yaw = yaw + math_random ( - 0.5 , 0.5 )
2017-01-16 16:40:08 +00:00
end
2018-05-29 15:00:30 +00:00
yaw = set_yaw ( self , yaw , 8 )
2017-01-16 16:40:08 +00:00
end
set_velocity ( self , 0 )
set_animation ( self , " stand " )
-- npc's ordered to stand stay standing
if self.type ~= " npc "
or self.order ~= " stand " then
if self.walk_chance ~= 0
2017-11-03 23:22:43 +00:00
and self.facing_fence ~= true
2021-04-11 16:52:31 +00:00
and math_random ( 1 , 100 ) <= self.walk_chance
2020-01-30 17:04:50 +00:00
and is_at_cliff_or_danger ( self ) == false then
2017-01-16 16:40:08 +00:00
set_velocity ( self , self.walk_velocity )
self.state = " walk "
set_animation ( self , " walk " )
end
end
elseif self.state == " walk " then
2017-11-03 23:22:43 +00:00
local s = self.object : get_pos ( )
2017-01-16 16:40:08 +00:00
local lp = nil
-- is there something I need to avoid?
2020-01-29 22:11:20 +00:00
if ( self.water_damage > 0
and self.lava_damage > 0 )
or self.breath_max ~= - 1 then
2017-01-16 16:40:08 +00:00
2021-04-12 12:47:07 +00:00
lp = minetest_find_node_near ( s , 1 , { " group:water " , " group:lava " } )
2017-01-16 16:40:08 +00:00
elseif self.water_damage > 0 then
2021-04-12 12:47:07 +00:00
lp = minetest_find_node_near ( s , 1 , { " group:water " } )
2017-01-16 16:40:08 +00:00
elseif self.lava_damage > 0 then
2021-04-12 12:47:07 +00:00
lp = minetest_find_node_near ( s , 1 , { " group:lava " } )
2020-01-29 22:11:20 +00:00
elseif self.fire_damage > 0 then
2021-04-12 12:47:07 +00:00
lp = minetest_find_node_near ( s , 1 , { " group:fire " } )
2020-01-29 22:11:20 +00:00
2017-01-16 16:40:08 +00:00
end
2020-01-30 17:04:50 +00:00
local is_in_danger = false
2017-01-16 16:40:08 +00:00
if lp then
2020-07-31 13:35:40 +00:00
-- If mob in or on dangerous block, look for land
if ( is_node_dangerous ( self , self.standing_in ) or
is_node_dangerous ( self , self.standing_on ) ) or ( is_node_waterhazard ( self , self.standing_in ) or is_node_waterhazard ( self , self.standing_on ) ) and ( not self.fly ) then
is_in_danger = true
2020-07-12 20:56:41 +00:00
2020-07-31 13:35:40 +00:00
-- If mob in or on dangerous block, look for land
if is_in_danger then
-- Better way to find shore - copied from upstream
2021-04-12 12:47:07 +00:00
lp = minetest_find_nodes_in_area_under_air (
2020-07-31 13:35:40 +00:00
{ x = s.x - 5 , y = s.y - 0.5 , z = s.z - 5 } ,
{ x = s.x + 5 , y = s.y + 1 , z = s.z + 5 } ,
2021-01-03 14:10:50 +00:00
{ " group:solid " } )
2020-05-03 15:25:12 +00:00
2021-04-11 16:52:31 +00:00
lp = # lp > 0 and lp [ math_random ( # lp ) ]
2021-01-03 14:10:50 +00:00
2020-07-31 13:35:40 +00:00
-- did we find land?
if lp then
2017-05-25 08:33:19 +00:00
2020-07-31 13:35:40 +00:00
local vec = {
x = lp.x - s.x ,
z = lp.z - s.z
}
2017-05-25 08:33:19 +00:00
2021-04-11 16:52:31 +00:00
yaw = ( atan ( vec.z / vec.x ) + math_pi / 2 ) - self.rotate
2017-05-25 08:33:19 +00:00
2021-04-11 16:52:31 +00:00
if lp.x > s.x then yaw = yaw + math_pi end
2017-05-25 08:33:19 +00:00
2020-07-31 13:35:40 +00:00
-- look towards land and move in that direction
yaw = set_yaw ( self , yaw , 6 )
set_velocity ( self , self.walk_velocity )
2017-05-25 08:33:19 +00:00
2020-07-31 13:35:40 +00:00
end
end
2017-05-25 08:33:19 +00:00
2020-01-29 22:11:20 +00:00
-- A danger is near but mob is not inside
2017-05-25 08:33:19 +00:00
else
2020-01-29 22:11:20 +00:00
-- Randomly turn
2021-04-11 16:52:31 +00:00
if math_random ( 1 , 100 ) <= 30 then
yaw = yaw + math_random ( - 0.5 , 0.5 )
2020-01-29 22:11:20 +00:00
yaw = set_yaw ( self , yaw , 8 )
end
2017-01-16 16:40:08 +00:00
end
2018-05-29 15:00:30 +00:00
yaw = set_yaw ( self , yaw , 8 )
2017-01-16 16:40:08 +00:00
-- otherwise randomly turn
2021-04-11 16:52:31 +00:00
elseif math_random ( 1 , 100 ) <= 30 then
2017-01-16 16:40:08 +00:00
2021-04-11 16:52:31 +00:00
yaw = yaw + math_random ( - 0.5 , 0.5 )
2018-05-29 15:00:30 +00:00
yaw = set_yaw ( self , yaw , 8 )
2017-01-16 16:40:08 +00:00
end
2020-01-30 17:04:50 +00:00
-- stand for great fall or danger or fence in front
local cliff_or_danger = false
if is_in_danger then
cliff_or_danger = is_at_cliff_or_danger ( self )
end
2017-11-03 23:22:43 +00:00
if self.facing_fence == true
2020-01-30 17:04:50 +00:00
or cliff_or_danger
2021-04-11 16:52:31 +00:00
or math_random ( 1 , 100 ) <= 30 then
2017-01-16 16:40:08 +00:00
set_velocity ( self , 0 )
self.state = " stand "
set_animation ( self , " stand " )
2020-12-22 14:29:24 +00:00
local yaw = self.object : get_yaw ( ) or 0
yaw = set_yaw ( self , yaw + 0.78 , 8 )
2017-01-16 16:40:08 +00:00
else
2020-01-29 22:11:20 +00:00
2017-01-16 16:40:08 +00:00
set_velocity ( self , self.walk_velocity )
2017-05-25 08:33:19 +00:00
if flight_check ( self )
and self.animation
and self.animation . fly_start
and self.animation . fly_end then
set_animation ( self , " fly " )
else
set_animation ( self , " walk " )
end
2017-01-16 16:40:08 +00:00
end
-- runaway when punched
elseif self.state == " runaway " then
self.runaway_timer = self.runaway_timer + 1
-- stop after 5 seconds or when at cliff
if self.runaway_timer > 5
2020-01-30 17:04:50 +00:00
or is_at_cliff_or_danger ( self ) then
2017-01-16 16:40:08 +00:00
self.runaway_timer = 0
set_velocity ( self , 0 )
self.state = " stand "
set_animation ( self , " stand " )
2020-12-22 14:29:24 +00:00
local yaw = self.object : get_yaw ( ) or 0
yaw = set_yaw ( self , yaw + 0.78 , 8 )
2017-01-16 16:40:08 +00:00
else
set_velocity ( self , self.run_velocity )
2020-08-03 13:37:58 +00:00
set_animation ( self , " run " )
2017-01-16 16:40:08 +00:00
end
-- attack routines (explode, dogfight, shoot, dogshoot)
elseif self.state == " attack " then
2017-11-03 23:22:43 +00:00
local s = self.object : get_pos ( )
local p = self.attack : get_pos ( ) or s
2017-01-16 16:40:08 +00:00
2018-03-30 22:18:40 +00:00
-- stop attacking if player invisible or out of range
2020-02-25 15:09:26 +00:00
if not self.attack
2017-11-03 23:22:43 +00:00
or not self.attack : get_pos ( )
2020-02-25 15:09:26 +00:00
or not object_in_range ( self , self.attack )
2017-01-16 16:40:08 +00:00
or self.attack : get_hp ( ) <= 0
or ( self.attack : is_player ( ) and mobs.invis [ self.attack : get_player_name ( ) ] ) then
self.state = " stand "
set_velocity ( self , 0 )
set_animation ( self , " stand " )
self.attack = nil
self.v_start = false
self.timer = 0
self.blinktimer = 0
2018-05-29 15:00:30 +00:00
self.path . way = nil
2017-01-16 16:40:08 +00:00
return
end
2020-12-01 16:10:37 +00:00
-- calculate distance from mob and enemy
local dist = vector.distance ( p , s )
2017-01-16 16:40:08 +00:00
if self.attack_type == " explode " then
local vec = {
x = p.x - s.x ,
z = p.z - s.z
}
2021-04-11 16:52:31 +00:00
yaw = ( atan ( vec.z / vec.x ) + math_pi / 2 ) - self.rotate
2017-01-16 16:40:08 +00:00
2021-04-11 16:52:31 +00:00
if p.x > s.x then yaw = yaw + math_pi end
2017-01-16 16:40:08 +00:00
2021-01-23 14:40:12 +00:00
yaw = set_yaw ( self , yaw , 0 , dtime )
2017-01-16 16:40:08 +00:00
2018-03-30 22:18:40 +00:00
local node_break_radius = self.explosion_radius or 1
local entity_damage_radius = self.explosion_damage_radius
or ( node_break_radius * 2 )
-- start timer when in reach and line of sight
if not self.v_start
and dist <= self.reach
and line_of_sight ( self , s , p , 2 ) then
2017-11-03 23:22:43 +00:00
self.v_start = true
self.timer = 0
self.blinktimer = 0
2019-12-09 11:17:51 +00:00
mob_sound ( self , " fuse " , nil , false )
2018-03-30 22:18:40 +00:00
2018-03-31 01:51:49 +00:00
-- stop timer if out of reach or direct line of sight
2018-03-30 22:18:40 +00:00
elseif self.allow_fuse_reset
and self.v_start
2021-02-11 23:27:55 +00:00
and ( dist >= self.explosiontimer_reset_radius
2018-03-30 22:18:40 +00:00
or not line_of_sight ( self , s , p , 2 ) ) then
self.v_start = false
self.timer = 0
self.blinktimer = 0
self.blinkstatus = false
2019-09-10 14:00:41 +00:00
remove_texture_mod ( self , " ^[brighten " )
2017-11-03 23:22:43 +00:00
end
2017-01-16 16:40:08 +00:00
2018-03-30 22:18:40 +00:00
-- walk right up to player unless the timer is active
2021-02-11 23:27:55 +00:00
if self.v_start and ( self.stop_to_explode or dist < self.reach ) then
2017-11-03 23:22:43 +00:00
set_velocity ( self , 0 )
else
set_velocity ( self , self.run_velocity )
end
2017-01-16 16:40:08 +00:00
2017-11-03 23:22:43 +00:00
if self.animation and self.animation . run_start then
set_animation ( self , " run " )
2017-01-16 16:40:08 +00:00
else
2017-11-03 23:22:43 +00:00
set_animation ( self , " walk " )
end
if self.v_start then
2017-01-16 16:40:08 +00:00
self.timer = self.timer + dtime
self.blinktimer = ( self.blinktimer or 0 ) + dtime
if self.blinktimer > 0.2 then
self.blinktimer = 0
if self.blinkstatus then
2019-09-10 14:00:41 +00:00
remove_texture_mod ( self , " ^[brighten " )
2015-06-29 17:55:56 +00:00
else
2019-09-10 14:00:41 +00:00
add_texture_mod ( self , " ^[brighten " )
2015-06-29 17:55:56 +00:00
end
2017-01-16 16:40:08 +00:00
self.blinkstatus = not self.blinkstatus
2015-06-29 17:55:56 +00:00
end
2017-01-16 16:40:08 +00:00
2017-11-03 23:22:43 +00:00
if self.timer > self.explosion_timer then
2017-01-16 16:40:08 +00:00
2017-11-03 23:22:43 +00:00
local pos = self.object : get_pos ( )
2017-01-16 16:40:08 +00:00
2020-05-02 16:50:25 +00:00
if mod_explosions then
2021-04-12 12:47:07 +00:00
if mobs_griefing and not minetest_is_protected ( pos , " " ) then
2021-01-26 13:13:21 +00:00
mcl_explosions.explode ( mcl_util.get_object_center ( self.object ) , self.explosion_strength , { drop_chance = 1.0 } , self.object )
2017-07-25 02:30:23 +00:00
else
2021-04-12 12:47:07 +00:00
minetest_sound_play ( self.sounds . explode , {
2017-07-25 02:30:23 +00:00
pos = pos ,
gain = 1.0 ,
max_hear_distance = self.sounds . distance or 32
2020-04-06 22:55:45 +00:00
} , true )
2017-07-25 02:30:23 +00:00
2018-03-30 22:18:40 +00:00
entity_physics ( pos , entity_damage_radius )
2020-08-19 16:47:58 +00:00
effect ( pos , 32 , " mcl_particles_smoke.png " , nil , nil , node_break_radius , 1 , 0 )
2017-07-25 02:30:23 +00:00
end
2020-05-02 16:50:25 +00:00
end
2021-01-02 09:56:40 +00:00
mcl_burning.extinguish ( self.object )
2020-05-02 16:50:25 +00:00
self.object : remove ( )
2017-05-25 08:33:19 +00:00
2020-04-08 13:03:03 +00:00
return true
2015-06-29 17:55:56 +00:00
end
end
2017-01-16 16:40:08 +00:00
elseif self.attack_type == " dogfight "
or ( self.attack_type == " dogshoot " and dogswitch ( self , dtime ) == 2 )
or ( self.attack_type == " dogshoot " and dist <= self.reach and dogswitch ( self ) == 0 ) then
if self.fly
and dist > self.reach then
local p1 = s
2021-04-11 16:52:31 +00:00
local me_y = math_floor ( p1.y )
2017-01-16 16:40:08 +00:00
local p2 = p
2021-04-11 16:52:31 +00:00
local p_y = math_floor ( p2.y + 1 )
2019-03-06 03:38:57 +00:00
local v = self.object : get_velocity ( )
2017-01-16 16:40:08 +00:00
2017-05-25 08:33:19 +00:00
if flight_check ( self , s ) then
2017-01-16 16:40:08 +00:00
if me_y < p_y then
2019-03-06 03:38:57 +00:00
self.object : set_velocity ( {
2017-01-16 16:40:08 +00:00
x = v.x ,
y = 1 * self.walk_velocity ,
z = v.z
} )
elseif me_y > p_y then
2019-03-06 03:38:57 +00:00
self.object : set_velocity ( {
2017-01-16 16:40:08 +00:00
x = v.x ,
y = - 1 * self.walk_velocity ,
z = v.z
} )
end
2015-06-29 17:55:56 +00:00
else
2017-01-16 16:40:08 +00:00
if me_y < p_y then
2019-03-06 03:38:57 +00:00
self.object : set_velocity ( {
2017-01-16 16:40:08 +00:00
x = v.x ,
y = 0.01 ,
z = v.z
} )
elseif me_y > p_y then
2019-03-06 03:38:57 +00:00
self.object : set_velocity ( {
2017-01-16 16:40:08 +00:00
x = v.x ,
y = - 0.01 ,
z = v.z
} )
end
2015-06-29 17:55:56 +00:00
end
2017-01-16 16:40:08 +00:00
2015-06-29 17:55:56 +00:00
end
2017-01-16 16:40:08 +00:00
-- rnd: new movement direction
if self.path . following
and self.path . way
and self.attack_type ~= " dogshoot " then
-- no paths longer than 50
if # self.path . way > 50
or dist < self.reach then
self.path . following = false
return
end
local p1 = self.path . way [ 1 ]
if not p1 then
self.path . following = false
return
end
2021-04-11 16:52:31 +00:00
if math_abs ( p1.x - s.x ) + math_abs ( p1.z - s.z ) < 0.6 then
2017-01-16 16:40:08 +00:00
-- reached waypoint, remove it from queue
table.remove ( self.path . way , 1 )
end
-- set new temporary target
p = { x = p1.x , y = p1.y , z = p1.z }
2015-06-29 17:55:56 +00:00
end
2017-01-16 16:40:08 +00:00
local vec = {
x = p.x - s.x ,
z = p.z - s.z
}
2021-04-11 16:52:31 +00:00
yaw = ( atan ( vec.z / vec.x ) + math_pi / 2 ) - self.rotate
2017-01-16 16:40:08 +00:00
2021-04-11 16:52:31 +00:00
if p.x > s.x then yaw = yaw + math_pi end
2017-01-16 16:40:08 +00:00
2021-01-23 14:40:12 +00:00
yaw = set_yaw ( self , yaw , 0 , dtime )
2017-01-16 16:40:08 +00:00
-- move towards enemy if beyond mob reach
if dist > self.reach then
-- path finding by rnd
if self.pathfinding -- only if mob has pathfinding enabled
and enable_pathfinding then
smart_mobs ( self , s , p , dist , dtime )
2015-06-29 17:55:56 +00:00
end
2017-01-16 16:40:08 +00:00
2020-01-30 17:04:50 +00:00
if is_at_cliff_or_danger ( self ) then
2017-01-16 16:40:08 +00:00
set_velocity ( self , 0 )
set_animation ( self , " stand " )
2020-12-22 14:29:24 +00:00
local yaw = self.object : get_yaw ( ) or 0
yaw = set_yaw ( self , yaw + 0.78 , 8 )
2015-06-29 17:55:56 +00:00
else
2017-01-16 16:40:08 +00:00
if self.path . stuck then
set_velocity ( self , self.walk_velocity )
else
set_velocity ( self , self.run_velocity )
end
2017-07-29 11:34:27 +00:00
if self.animation and self.animation . run_start then
set_animation ( self , " run " )
else
set_animation ( self , " walk " )
end
2015-06-29 17:55:56 +00:00
end
2017-01-16 16:40:08 +00:00
else -- rnd: if inside reach range
self.path . stuck = false
self.path . stuck_timer = 0
self.path . following = false -- not stuck anymore
set_velocity ( self , 0 )
if not self.custom_attack then
if self.timer > 1 then
self.timer = 0
if self.double_melee_attack
2021-04-11 16:52:31 +00:00
and math_random ( 1 , 2 ) == 1 then
2017-01-16 16:40:08 +00:00
set_animation ( self , " punch2 " )
2015-06-29 17:55:56 +00:00
else
2017-01-16 16:40:08 +00:00
set_animation ( self , " punch " )
2015-06-29 17:55:56 +00:00
end
2017-01-16 16:40:08 +00:00
local p2 = p
local s2 = s
2017-05-25 08:33:19 +00:00
p2.y = p2.y + .5
s2.y = s2.y + .5
2017-01-16 16:40:08 +00:00
2017-05-25 08:33:19 +00:00
if line_of_sight ( self , p2 , s2 ) == true then
2017-01-16 16:40:08 +00:00
-- play attack sound
2019-12-09 11:17:51 +00:00
mob_sound ( self , " attack " )
2017-01-16 16:40:08 +00:00
2017-05-25 08:33:19 +00:00
-- punch player (or what player is attached to)
local attached = self.attack : get_attach ( )
if attached then
self.attack = attached
2015-06-29 17:55:56 +00:00
end
2017-01-16 16:40:08 +00:00
self.attack : punch ( self.object , 1.0 , {
full_punch_interval = 1.0 ,
damage_groups = { fleshy = self.damage }
} , nil )
2015-06-29 17:55:56 +00:00
end
end
2017-01-16 16:40:08 +00:00
else -- call custom attack every second
if self.custom_attack
and self.timer > 1 then
self.timer = 0
self.custom_attack ( self , p )
2015-06-29 17:55:56 +00:00
end
end
end
2017-01-16 16:40:08 +00:00
elseif self.attack_type == " shoot "
or ( self.attack_type == " dogshoot " and dogswitch ( self , dtime ) == 1 )
or ( self.attack_type == " dogshoot " and dist > self.reach and dogswitch ( self ) == 0 ) then
p.y = p.y - .5
s.y = s.y + .5
2019-12-08 17:48:49 +00:00
local dist = vector.distance ( p , s )
2017-01-16 16:40:08 +00:00
local vec = {
x = p.x - s.x ,
y = p.y - s.y ,
z = p.z - s.z
}
2021-04-11 16:52:31 +00:00
yaw = ( atan ( vec.z / vec.x ) + math_pi / 2 ) - self.rotate
2017-01-16 16:40:08 +00:00
2021-04-11 16:52:31 +00:00
if p.x > s.x then yaw = yaw + math_pi end
2017-01-16 16:40:08 +00:00
2021-01-23 14:40:12 +00:00
yaw = set_yaw ( self , yaw , 0 , dtime )
2017-01-16 16:40:08 +00:00
set_velocity ( self , 0 )
2021-01-06 10:18:18 +00:00
local p = self.object : get_pos ( )
p.y = p.y + ( self.collisionbox [ 2 ] + self.collisionbox [ 5 ] ) / 2
2017-01-16 16:40:08 +00:00
if self.shoot_interval
and self.timer > self.shoot_interval
2021-04-12 12:47:07 +00:00
and not minetest_raycast ( p , self.attack : get_pos ( ) , false , false ) : next ( )
2021-04-11 16:52:31 +00:00
and math_random ( 1 , 100 ) <= 60 then
2017-01-16 16:40:08 +00:00
2015-06-29 17:55:56 +00:00
self.timer = 0
2017-01-16 16:40:08 +00:00
set_animation ( self , " shoot " )
-- play shoot attack sound
2019-12-09 11:17:51 +00:00
mob_sound ( self , " shoot_attack " )
2017-01-16 16:40:08 +00:00
2019-12-09 08:29:19 +00:00
-- Shoot arrow
2021-04-12 12:47:07 +00:00
if minetest_registered_entities [ self.arrow ] then
2017-05-25 08:33:19 +00:00
2019-12-09 08:29:19 +00:00
local arrow , ent
local v = 1
if not self.shoot_arrow then
2021-04-04 01:39:20 +00:00
self.firing = true
2021-04-12 12:47:07 +00:00
minetest_after ( 1 , function ( )
2021-04-04 01:39:20 +00:00
self.firing = false
end )
2021-04-12 12:47:07 +00:00
arrow = minetest_add_entity ( p , self.arrow )
2019-12-09 08:29:19 +00:00
ent = arrow : get_luaentity ( )
if ent.velocity then
v = ent.velocity
end
ent.switch = 1
ent.owner_id = tostring ( self.object ) -- add unique owner id to arrow
end
2017-05-25 08:33:19 +00:00
2019-12-09 08:29:19 +00:00
local amount = ( vec.x * vec.x + vec.y * vec.y + vec.z * vec.z ) ^ 0.5
-- offset makes shoot aim accurate
2017-05-25 08:33:19 +00:00
vec.y = vec.y + self.shoot_offset
vec.x = vec.x * ( v / amount )
vec.y = vec.y * ( v / amount )
vec.z = vec.z * ( v / amount )
2019-12-09 08:29:19 +00:00
if self.shoot_arrow then
vec = vector.normalize ( vec )
self : shoot_arrow ( p , vec )
else
arrow : set_velocity ( vec )
end
2017-05-25 08:33:19 +00:00
end
2015-06-29 17:55:56 +00:00
end
2017-01-16 16:40:08 +00:00
end
end
2021-04-13 12:20:11 +00:00
] ] --
2017-01-16 16:40:08 +00:00
end
2020-02-22 19:47:25 +00:00
2017-05-25 08:33:19 +00:00
-- get entity staticdata
local mob_staticdata = function ( self )
2020-12-17 15:22:34 +00:00
--[[
2017-05-25 08:33:19 +00:00
-- remove mob when out of range unless tamed
if remove_far
2018-09-14 12:48:48 +00:00
and self.can_despawn
2017-05-25 08:33:19 +00:00
and self.remove_ok
2018-09-14 12:48:48 +00:00
and ( ( not self.nametag ) or ( self.nametag == " " ) )
and self.lifetimer <= 20 then
2017-05-25 08:33:19 +00:00
2021-01-06 09:36:57 +00:00
minetest.log ( " action " , " Mob " .. name .. " despawns in mob_staticdata at " .. minetest.pos_to_string ( self.object . get_pos ( ) , 1 ) )
2021-01-02 09:56:40 +00:00
mcl_burning.extinguish ( self.object )
2017-05-25 08:33:19 +00:00
self.object : remove ( )
return " " -- nil
end
2020-12-17 15:22:34 +00:00
--]]
2017-05-25 08:33:19 +00:00
self.remove_ok = true
self.attack = nil
self.following = nil
self.state = " stand "
2017-07-04 23:52:39 +00:00
if use_cmi then
self.serialized_cmi_components = cmi.serialize_components ( self._cmi_components )
end
2017-05-25 08:33:19 +00:00
local tmp = { }
for _ , stat in pairs ( self ) do
local t = type ( stat )
if t ~= " function "
and t ~= " nil "
2017-07-04 23:52:39 +00:00
and t ~= " userdata "
and _ ~= " _cmi_components " then
2017-05-25 08:33:19 +00:00
tmp [ _ ] = self [ _ ]
end
end
return minetest.serialize ( tmp )
end
-- activate mob and reload settings
2017-07-04 23:52:39 +00:00
local mob_activate = function ( self , staticdata , def , dtime )
2017-01-16 16:40:08 +00:00
2017-11-03 23:22:43 +00:00
-- remove monsters in peaceful mode
if self.type == " monster "
2021-04-12 12:47:07 +00:00
and minetest_settings : get_bool ( " only_peaceful_mobs " , false ) then
2021-01-02 09:56:40 +00:00
mcl_burning.extinguish ( self.object )
2017-01-16 16:40:08 +00:00
self.object : remove ( )
return
end
-- load entity variables
local tmp = minetest.deserialize ( staticdata )
if tmp then
for _ , stat in pairs ( tmp ) do
self [ _ ] = stat
end
end
-- select random texture, set model and size
if not self.base_texture then
2017-05-25 08:33:19 +00:00
-- compatiblity with old simple mobs textures
if type ( def.textures [ 1 ] ) == " string " then
def.textures = { def.textures }
end
2021-04-11 16:52:31 +00:00
self.base_texture = def.textures [ math_random ( 1 , # def.textures ) ]
2017-01-16 16:40:08 +00:00
self.base_mesh = def.mesh
self.base_size = self.visual_size
self.base_colbox = self.collisionbox
2018-01-08 01:03:31 +00:00
self.base_selbox = self.selectionbox
2017-01-16 16:40:08 +00:00
end
2018-01-26 17:06:32 +00:00
-- for current mobs that dont have this set
if not self.base_selbox then
self.base_selbox = self.selectionbox or self.base_colbox
end
2017-01-16 16:40:08 +00:00
-- set texture, model and size
local textures = self.base_texture
local mesh = self.base_mesh
local vis_size = self.base_size
local colbox = self.base_colbox
2018-01-08 01:03:31 +00:00
local selbox = self.base_selbox
2017-01-16 16:40:08 +00:00
-- specific texture if gotten
if self.gotten == true
and def.gotten_texture then
textures = def.gotten_texture
end
-- specific mesh if gotten
if self.gotten == true
and def.gotten_mesh then
mesh = def.gotten_mesh
end
-- set child objects to half size
if self.child == true then
vis_size = {
x = self.base_size . x * .5 ,
y = self.base_size . y * .5 ,
}
if def.child_texture then
textures = def.child_texture [ 1 ]
end
colbox = {
self.base_colbox [ 1 ] * .5 ,
self.base_colbox [ 2 ] * .5 ,
self.base_colbox [ 3 ] * .5 ,
self.base_colbox [ 4 ] * .5 ,
self.base_colbox [ 5 ] * .5 ,
self.base_colbox [ 6 ] * .5
}
2018-01-08 01:03:31 +00:00
selbox = {
self.base_selbox [ 1 ] * .5 ,
self.base_selbox [ 2 ] * .5 ,
self.base_selbox [ 3 ] * .5 ,
self.base_selbox [ 4 ] * .5 ,
self.base_selbox [ 5 ] * .5 ,
self.base_selbox [ 6 ] * .5
}
2017-01-16 16:40:08 +00:00
end
if self.health == 0 then
2021-04-11 16:52:31 +00:00
self.health = math_random ( self.hp_min , self.hp_max )
2017-01-16 16:40:08 +00:00
end
2019-10-02 16:28:28 +00:00
if self.breath == nil then
self.breath = self.breath_max
end
2017-01-16 16:40:08 +00:00
2017-11-03 23:22:43 +00:00
-- pathfinding init
2017-01-16 16:40:08 +00:00
self.path = { }
self.path . way = { } -- path to follow, table of positions
self.path . lastpos = { x = 0 , y = 0 , z = 0 }
self.path . stuck = false
self.path . following = false -- currently following path?
self.path . stuck_timer = 0 -- if stuck for too long search for path
2020-02-19 15:47:57 +00:00
-- Armor groups
-- immortal=1 because we use custom health
-- handling (using "health" property)
2020-01-30 15:12:25 +00:00
local armor
if type ( self.armor ) == " table " then
armor = table.copy ( self.armor )
armor.immortal = 1
else
armor = { immortal = 1 , fleshy = self.armor }
end
self.object : set_armor_groups ( armor )
2017-11-03 23:22:43 +00:00
self.old_y = self.object : get_pos ( ) . y
2017-01-16 16:40:08 +00:00
self.old_health = self.health
self.sounds . distance = self.sounds . distance or 10
self.textures = textures
self.mesh = mesh
self.collisionbox = colbox
2018-01-08 01:03:31 +00:00
self.selectionbox = selbox
2017-01-16 16:40:08 +00:00
self.visual_size = vis_size
2020-01-29 22:11:20 +00:00
self.standing_in = " ignore "
self.standing_on = " ignore "
2019-01-31 01:44:05 +00:00
self.jump_sound_cooloff = 0 -- used to prevent jump sound from being played too often in short time
2019-01-31 05:31:04 +00:00
self.opinion_sound_cooloff = 0 -- used to prevent sound spam of particular sound types
2017-01-16 16:40:08 +00:00
2019-09-10 14:00:41 +00:00
self.texture_mods = { }
2020-01-06 16:28:08 +00:00
self.object : set_texture_mod ( " " )
self.v_start = false
self.timer = 0
self.blinktimer = 0
self.blinkstatus = false
2019-09-10 14:00:41 +00:00
2017-11-03 23:22:43 +00:00
-- check existing nametag
if not self.nametag then
self.nametag = def.nametag
end
2017-01-16 16:40:08 +00:00
-- set anything changed above
self.object : set_properties ( self )
2021-04-11 16:52:31 +00:00
set_yaw ( self , ( math_random ( 0 , 360 ) - 180 ) / 180 * math_pi , 6 )
2021-04-13 12:20:11 +00:00
--update_tag(self)
--set_animation(self, "stand")
2017-07-04 23:52:39 +00:00
2017-11-03 23:22:43 +00:00
-- run on_spawn function if found
if self.on_spawn and not self.on_spawn_run then
if self.on_spawn ( self ) then
self.on_spawn_run = true -- if true, set flag to run once only
end
end
-- run after_activate
if def.after_activate then
def.after_activate ( self , staticdata , def , dtime )
end
2017-07-04 23:52:39 +00:00
if use_cmi then
self._cmi_components = cmi.activate_components ( self.serialized_cmi_components )
cmi.notify_activate ( self.object , dtime )
end
2017-01-16 16:40:08 +00:00
end
2017-05-25 08:33:19 +00:00
2017-01-16 16:40:08 +00:00
local mob_step = function ( self , dtime )
2021-01-02 20:42:07 +00:00
2021-04-11 18:47:56 +00:00
if not self or not self.object or not self.object : get_luaentity ( ) then
return false
end
2021-04-13 11:39:57 +00:00
if self.state == " die " then
print ( " need custom die stop moving thing " )
return
end
2021-04-11 18:47:56 +00:00
-- can mob be pushed, if so calculate direction -- do this first to prevent issues
if self.pushable then
collision ( self )
end
2021-01-02 20:42:07 +00:00
2021-04-13 11:39:57 +00:00
--if not self.fire_resistant then
-- mcl_burning.tick(self.object, dtime)
--end
--if use_cmi then
--cmi.notify_step(self.object, dtime)
--end
2017-07-04 23:52:39 +00:00
2017-11-03 23:22:43 +00:00
local pos = self.object : get_pos ( )
2017-05-25 08:33:19 +00:00
local yaw = 0
2017-01-16 16:40:08 +00:00
2021-04-13 11:39:57 +00:00
--if mobs_debug then
--update_tag(self)
--end
2020-12-05 12:51:29 +00:00
2020-12-05 00:30:16 +00:00
2021-04-11 18:47:56 +00:00
2021-04-13 11:39:57 +00:00
--if self.jump_sound_cooloff > 0 then
-- self.jump_sound_cooloff = self.jump_sound_cooloff - dtime
--end
2021-04-11 18:47:56 +00:00
2021-04-13 11:39:57 +00:00
--if self.opinion_sound_cooloff > 0 then
-- self.opinion_sound_cooloff = self.opinion_sound_cooloff - dtime
--end
--if falling(self, pos) then
2020-04-12 21:11:18 +00:00
-- Return if mob died after falling
2021-04-13 11:39:57 +00:00
-- return
--end
2017-01-16 16:40:08 +00:00
2018-05-29 15:00:30 +00:00
-- smooth rotation by ThomasMonroe314
if self.delay and self.delay > 0 then
2020-04-05 19:09:27 +00:00
local yaw = self.object : get_yaw ( ) or 0
2018-05-29 15:00:30 +00:00
if self.delay == 1 then
yaw = self.target_yaw
else
2021-04-11 16:52:31 +00:00
local dif = math_abs ( yaw - self.target_yaw )
2018-05-29 15:00:30 +00:00
if yaw > self.target_yaw then
2021-04-11 16:52:31 +00:00
if dif > math_pi then
dif = 2 * math_pi - dif -- need to add
2018-05-29 15:00:30 +00:00
yaw = yaw + dif / self.delay
else
yaw = yaw - dif / self.delay -- need to subtract
end
elseif yaw < self.target_yaw then
2021-04-11 16:52:31 +00:00
if dif > math_pi then
dif = 2 * math_pi - dif
2018-05-29 15:00:30 +00:00
yaw = yaw - dif / self.delay -- need to subtract
else
yaw = yaw + dif / self.delay -- need to add
end
end
2021-04-11 16:52:31 +00:00
if yaw > ( math_pi * 2 ) then yaw = yaw - ( math_pi * 2 ) end
if yaw < 0 then yaw = yaw + ( math_pi * 2 ) end
2018-05-29 15:00:30 +00:00
end
self.delay = self.delay - 1
2021-01-23 14:40:12 +00:00
if self.shaking then
2021-04-11 16:52:31 +00:00
yaw = yaw + ( math_random ( ) * 2 - 1 ) * 5 * dtime
2021-01-23 14:40:12 +00:00
end
2018-05-29 15:00:30 +00:00
self.object : set_yaw ( yaw )
2021-04-13 12:20:11 +00:00
--update_roll(self)
2018-05-29 15:00:30 +00:00
end
-- end rotation
2017-01-16 16:40:08 +00:00
-- run custom function (defined in mob lua file)
2021-04-13 11:39:57 +00:00
--if self.do_custom then
2017-01-16 16:40:08 +00:00
-- when false skip going any further
2021-04-13 11:39:57 +00:00
--if self.do_custom(self, dtime) == false then
-- return
--end
--end
2017-01-16 16:40:08 +00:00
2021-04-06 12:50:34 +00:00
-- knockback timer
2021-04-13 11:39:57 +00:00
--if self.pause_timer > 0 then
2021-04-06 12:50:34 +00:00
2021-04-13 11:39:57 +00:00
-- self.pause_timer = self.pause_timer - dtime
2021-04-06 12:50:34 +00:00
2021-04-13 11:39:57 +00:00
-- return
--end
2021-04-06 12:50:34 +00:00
2017-01-16 16:40:08 +00:00
-- attack timer
2021-04-13 11:39:57 +00:00
--self.timer = self.timer + dtime
2017-01-16 16:40:08 +00:00
2021-04-11 22:29:32 +00:00
--[[
2017-01-16 16:40:08 +00:00
if self.state ~= " attack " then
if self.timer < 1 then
2021-04-11 22:29:32 +00:00
print ( " returning>>error code 1 " )
2017-01-16 16:40:08 +00:00
return
end
self.timer = 0
end
2021-04-11 22:29:32 +00:00
] ] --
2017-01-16 16:40:08 +00:00
-- never go over 100
2021-04-13 11:39:57 +00:00
--if self.timer > 100 then
-- self.timer = 1
--end
2017-01-16 16:40:08 +00:00
-- mob plays random sound at times
2021-04-13 11:39:57 +00:00
--if math_random(1, 70) == 1 then
-- mob_sound(self, "random", true)
--end
2017-01-16 16:40:08 +00:00
-- environmental damage timer (every 1 second)
2021-04-13 11:39:57 +00:00
--self.env_damage_timer = self.env_damage_timer + dtime
--if (self.state == "attack" and self.env_damage_timer > 1)
--or self.state ~= "attack" then
--
-- self.env_damage_timer = 0
--
-- -- check for environmental damage (water, fire, lava etc.)
-- if do_env_damage(self) then
-- return
-- end
--
2018-05-29 15:00:30 +00:00
-- node replace check (cow eats grass etc.)
2021-04-13 11:39:57 +00:00
-- replace(self, pos)
--end
2017-01-16 16:40:08 +00:00
2021-04-13 11:39:57 +00:00
--monster_attack(self)
2017-01-16 16:40:08 +00:00
2021-04-13 11:39:57 +00:00
--npc_attack(self)
2017-01-16 16:40:08 +00:00
2021-04-13 11:39:57 +00:00
--breed(self)
2017-01-16 16:40:08 +00:00
2021-04-13 11:39:57 +00:00
--if do_states(self, dtime) then
-- return
--end
2017-01-16 16:40:08 +00:00
2021-04-13 11:39:57 +00:00
--do_jump(self)
2017-05-25 08:33:19 +00:00
2021-04-13 11:39:57 +00:00
--runaway_from(self)
2020-07-31 13:35:40 +00:00
2021-04-13 11:39:57 +00:00
--if is_at_water_danger(self) and self.state ~= "attack" then
-- if math_random(1, 10) <= 6 then
-- set_velocity(self, 0)
-- self.state = "stand"
-- set_animation(self, "stand")
-- yaw = yaw + math_random(-0.5, 0.5)
-- yaw = set_yaw(self, yaw, 8)
-- end
--end
2020-07-31 13:35:40 +00:00
2021-04-11 18:21:19 +00:00
2020-12-03 16:37:44 +00:00
-- Add water flowing for mobs from mcl_item_entity
2021-04-13 11:39:57 +00:00
--[[
local p , node , nn , def
p = self.object : get_pos ( )
node = minetest_get_node_or_nil ( p )
if node then
nn = node.name
2021-04-13 12:07:32 +00:00
def = minetest_registered_nodes [ nnenable_physicss if not on / in flowing liquid
2021-04-13 11:39:57 +00:00
self._flowing = false
enable_physics ( self.object , self , true )
return
end
2020-12-03 16:37:44 +00:00
--Mob following code.
2020-07-31 13:35:40 +00:00
follow_flop ( self )
2018-01-26 17:06:32 +00:00
2021-04-13 11:39:57 +00:00
2020-07-25 02:15:40 +00:00
if is_at_cliff_or_danger ( self ) then
set_velocity ( self , 0 )
self.state = " stand "
set_animation ( self , " stand " )
2020-12-22 14:29:24 +00:00
local yaw = self.object : get_yaw ( ) or 0
yaw = set_yaw ( self , yaw + 0.78 , 8 )
2020-07-25 02:15:40 +00:00
end
2020-12-17 15:22:34 +00:00
-- Despawning: when lifetimer expires, remove mob
if remove_far
and self.can_despawn == true
2021-01-24 23:47:28 +00:00
and ( ( not self.nametag ) or ( self.nametag == " " ) )
and self.state ~= " attack "
and self.following == nil then
2020-12-17 15:22:34 +00:00
self.lifetimer = self.lifetimer - dtime
2021-01-06 01:09:05 +00:00
if self.despawn_immediately or self.lifetimer <= 0 then
minetest.log ( " action " , " Mob " .. self.name .. " despawns in mob_step at " .. minetest.pos_to_string ( pos , 1 ) )
mcl_burning.extinguish ( self.object )
self.object : remove ( )
elseif self.lifetimer <= 10 then
2021-04-11 16:52:31 +00:00
if math_random ( 10 ) < 4 then
2021-01-06 01:09:05 +00:00
self.despawn_immediately = true
2021-01-04 15:40:18 +00:00
else
self.lifetimer = 20
2020-12-17 15:22:34 +00:00
end
end
end
2021-04-13 11:39:57 +00:00
] ] --
2017-01-16 16:40:08 +00:00
end
2017-05-25 08:33:19 +00:00
2017-01-16 16:40:08 +00:00
-- default function when mobs are blown up with TNT
local do_tnt = function ( obj , damage )
obj.object : punch ( obj.object , 1.0 , {
full_punch_interval = 1.0 ,
damage_groups = { fleshy = damage } ,
} , nil )
return false , true , { }
end
2017-05-25 08:33:19 +00:00
2017-01-16 16:40:08 +00:00
mobs.spawning_mobs = { }
2018-05-30 09:34:17 +00:00
-- Code to execute before custom on_rightclick handling
local on_rightclick_prefix = function ( self , clicker )
local item = clicker : get_wielded_item ( )
-- Name mob with nametag
2018-05-31 16:32:26 +00:00
if not self.ignores_nametag and item : get_name ( ) == " mcl_mobs:nametag " then
2018-05-30 09:34:17 +00:00
local tag = item : get_meta ( ) : get_string ( " name " )
if tag ~= " " then
if string.len ( tag ) > MAX_MOB_NAME_LENGTH then
tag = string.sub ( tag , 1 , MAX_MOB_NAME_LENGTH )
end
self.nametag = tag
update_tag ( self )
2018-07-02 21:27:11 +00:00
if not mobs.is_creative ( clicker : get_player_name ( ) ) then
2018-05-30 09:34:17 +00:00
item : take_item ( )
2018-07-03 23:53:08 +00:00
clicker : set_wielded_item ( item )
2018-05-30 09:34:17 +00:00
end
return true
end
end
return false
end
local create_mob_on_rightclick = function ( on_rightclick )
return function ( self , clicker )
local stop = on_rightclick_prefix ( self , clicker )
if ( not stop ) and ( on_rightclick ) then
on_rightclick ( self , clicker )
end
end
end
2017-05-25 08:33:19 +00:00
-- register mob entity
2017-01-16 16:40:08 +00:00
function mobs : register_mob ( name , def )
mobs.spawning_mobs [ name ] = true
2018-09-14 12:48:48 +00:00
local can_despawn
if def.can_despawn ~= nil then
can_despawn = def.can_despawn
2021-04-03 17:12:24 +00:00
elseif def.spawn_class == " passive " then
can_despawn = false
2018-09-14 12:48:48 +00:00
else
2019-01-31 21:00:43 +00:00
can_despawn = true
2018-09-14 12:48:48 +00:00
end
2019-10-02 16:28:28 +00:00
local function scale_difficulty ( value , default , min , special )
if ( not value ) or ( value == default ) or ( value == special ) then
return default
else
2021-04-11 16:52:31 +00:00
return math_max ( min , value * difficulty )
2019-10-02 16:28:28 +00:00
end
end
2020-01-06 13:46:10 +00:00
local collisionbox = def.collisionbox or { - 0.25 , - 0.25 , - 0.25 , 0.25 , 0.25 , 0.25 }
-- Workaround for <https://github.com/minetest/minetest/issues/5966>:
-- Increase upper Y limit to avoid mobs glitching through solid nodes.
-- FIXME: Remove workaround if it's no longer needed.
if collisionbox [ 5 ] < 0.79 then
collisionbox [ 5 ] = 0.79
end
2021-04-13 12:07:32 +00:00
2017-01-16 16:40:08 +00:00
minetest.register_entity ( name , {
2021-02-22 01:10:04 +00:00
use_texture_alpha = def.use_texture_alpha ,
2020-01-06 13:46:10 +00:00
stepheight = def.stepheight or 0.6 ,
2017-01-16 16:40:08 +00:00
name = name ,
type = def.type ,
attack_type = def.attack_type ,
fly = def.fly ,
2020-01-30 15:52:07 +00:00
fly_in = def.fly_in or { " air " , " __airlike " } ,
2017-01-16 16:40:08 +00:00
owner = def.owner or " " ,
order = def.order or " " ,
on_die = def.on_die ,
2019-02-05 18:12:28 +00:00
spawn_small_alternative = def.spawn_small_alternative ,
2017-01-16 16:40:08 +00:00
do_custom = def.do_custom ,
2017-05-25 08:33:19 +00:00
jump_height = def.jump_height or 4 , -- was 6
2017-01-16 16:40:08 +00:00
rotate = math.rad ( def.rotate or 0 ) , -- 0=front, 90=side, 180=back, 270=side2
2018-09-14 12:48:48 +00:00
lifetimer = def.lifetimer or 57.73 ,
2019-10-02 16:28:28 +00:00
hp_min = scale_difficulty ( def.hp_min , 5 , 1 ) ,
hp_max = scale_difficulty ( def.hp_max , 10 , 1 ) ,
2020-12-06 14:46:42 +00:00
xp_min = def.xp_min or 0 ,
xp_max = def.xp_max or 0 ,
2021-02-05 12:34:49 +00:00
xp_timestamp = 0 ,
2019-10-03 10:12:50 +00:00
breath_max = def.breath_max or 15 ,
2019-10-02 16:28:28 +00:00
breathes_in_water = def.breathes_in_water or false ,
2017-01-16 16:40:08 +00:00
physical = true ,
2020-01-06 13:46:10 +00:00
collisionbox = collisionbox ,
2018-01-08 01:03:31 +00:00
selectionbox = def.selectionbox or def.collisionbox ,
2017-01-16 16:40:08 +00:00
visual = def.visual ,
visual_size = def.visual_size or { x = 1 , y = 1 } ,
mesh = def.mesh ,
makes_footstep_sound = def.makes_footstep_sound or false ,
2019-03-09 00:04:18 +00:00
view_range = def.view_range or 16 ,
2017-01-16 16:40:08 +00:00
walk_velocity = def.walk_velocity or 1 ,
run_velocity = def.run_velocity or 2 ,
2019-10-02 16:28:28 +00:00
damage = scale_difficulty ( def.damage , 0 , 0 ) ,
2017-01-16 16:40:08 +00:00
light_damage = def.light_damage or 0 ,
2018-05-31 01:09:27 +00:00
sunlight_damage = def.sunlight_damage or 0 ,
2017-01-16 16:40:08 +00:00
water_damage = def.water_damage or 0 ,
2019-10-02 16:28:28 +00:00
lava_damage = def.lava_damage or 8 ,
2019-10-02 16:43:48 +00:00
fire_damage = def.fire_damage or 1 ,
2019-01-31 06:23:35 +00:00
suffocation = def.suffocation or true ,
2017-01-16 16:40:08 +00:00
fall_damage = def.fall_damage or 1 ,
2020-12-05 11:59:12 +00:00
fall_speed = def.fall_speed or DEFAULT_FALL_SPEED , -- must be lower than -2
2017-01-16 16:40:08 +00:00
drops = def.drops or { } ,
armor = def.armor or 100 ,
2018-05-30 09:34:17 +00:00
on_rightclick = create_mob_on_rightclick ( def.on_rightclick ) ,
2017-01-16 16:40:08 +00:00
arrow = def.arrow ,
shoot_interval = def.shoot_interval ,
sounds = def.sounds or { } ,
animation = def.animation ,
follow = def.follow ,
2017-05-25 08:33:19 +00:00
jump = def.jump ~= false ,
2017-01-16 16:40:08 +00:00
walk_chance = def.walk_chance or 50 ,
attacks_monsters = def.attacks_monsters or false ,
group_attack = def.group_attack or false ,
passive = def.passive or false ,
2018-05-29 15:00:30 +00:00
knock_back = def.knock_back ~= false ,
2017-01-16 16:40:08 +00:00
shoot_offset = def.shoot_offset or 0 ,
floats = def.floats or 1 , -- floats in water by default
2021-03-22 00:53:57 +00:00
floats_on_lava = def.floats_on_lava or 0 ,
2017-01-16 16:40:08 +00:00
replace_rate = def.replace_rate ,
replace_what = def.replace_what ,
replace_with = def.replace_with ,
replace_offset = def.replace_offset or 0 ,
2017-07-04 23:52:39 +00:00
on_replace = def.on_replace ,
2017-01-16 16:40:08 +00:00
timer = 0 ,
2020-05-13 20:15:46 +00:00
env_damage_timer = 0 ,
2017-01-16 16:40:08 +00:00
tamed = false ,
pause_timer = 0 ,
horny = false ,
hornytimer = 0 ,
gotten = false ,
health = 0 ,
reach = def.reach or 3 ,
htimer = 0 ,
texture_list = def.textures ,
child_texture = def.child_texture ,
docile_by_day = def.docile_by_day or false ,
time_of_day = 0.5 ,
fear_height = def.fear_height or 0 ,
runaway = def.runaway ,
runaway_timer = 0 ,
pathfinding = def.pathfinding ,
immune_to = def.immune_to or { } ,
2020-05-02 16:50:25 +00:00
explosion_radius = def.explosion_radius , -- LEGACY
explosion_damage_radius = def.explosion_damage_radius , -- LEGACY
2021-02-11 23:27:55 +00:00
explosiontimer_reset_radius = def.explosiontimer_reset_radius ,
2017-11-03 23:22:43 +00:00
explosion_timer = def.explosion_timer or 3 ,
2018-03-30 22:18:40 +00:00
allow_fuse_reset = def.allow_fuse_reset ~= false ,
stop_to_explode = def.stop_to_explode ~= false ,
2017-01-16 16:40:08 +00:00
custom_attack = def.custom_attack ,
double_melee_attack = def.double_melee_attack ,
dogshoot_switch = def.dogshoot_switch ,
dogshoot_count = 0 ,
dogshoot_count_max = def.dogshoot_count_max or 5 ,
2017-05-25 08:33:19 +00:00
dogshoot_count2_max = def.dogshoot_count2_max or ( def.dogshoot_count_max or 5 ) ,
2017-01-16 16:40:08 +00:00
attack_animals = def.attack_animals or false ,
specific_attack = def.specific_attack ,
2018-01-26 17:06:32 +00:00
runaway_from = def.runaway_from ,
2017-05-25 08:33:19 +00:00
owner_loyal = def.owner_loyal ,
2017-11-03 23:22:43 +00:00
facing_fence = false ,
2017-07-04 23:52:39 +00:00
_cmi_is_mob = true ,
2020-12-03 16:37:44 +00:00
pushable = def.pushable or true ,
2017-01-16 16:40:08 +00:00
2018-05-30 09:34:17 +00:00
-- MCL2 extensions
2020-06-08 05:51:48 +00:00
teleport = teleport ,
do_teleport = def.do_teleport ,
2020-04-11 00:46:03 +00:00
spawn_class = def.spawn_class ,
2018-05-30 09:34:17 +00:00
ignores_nametag = def.ignores_nametag or false ,
2018-05-30 10:01:53 +00:00
rain_damage = def.rain_damage or 0 ,
2019-03-08 02:40:46 +00:00
glow = def.glow ,
2018-09-14 12:48:48 +00:00
can_despawn = can_despawn ,
2019-03-08 22:52:41 +00:00
child = def.child or false ,
2019-09-10 14:00:41 +00:00
texture_mods = { } ,
2019-12-09 08:29:19 +00:00
shoot_arrow = def.shoot_arrow ,
2019-12-09 11:17:51 +00:00
sounds_child = def.sounds_child ,
2020-05-02 16:50:25 +00:00
explosion_strength = def.explosion_strength ,
2020-05-13 20:15:46 +00:00
suffocation_timer = 0 ,
2020-08-03 13:37:58 +00:00
follow_velocity = def.follow_velocity or 2.4 ,
2020-12-05 13:42:03 +00:00
instant_death = def.instant_death or false ,
2020-12-24 16:48:40 +00:00
fire_resistant = def.fire_resistant or false ,
2020-12-29 21:08:38 +00:00
fire_damage_resistant = def.fire_damage_resistant or false ,
2021-01-02 11:43:50 +00:00
ignited_by_sunlight = def.ignited_by_sunlight or false ,
2019-03-08 22:52:41 +00:00
-- End of MCL2 extensions
2018-05-30 09:34:17 +00:00
2017-11-03 23:22:43 +00:00
on_spawn = def.on_spawn ,
2017-01-16 16:40:08 +00:00
on_blast = def.on_blast or do_tnt ,
on_step = mob_step ,
2017-11-03 23:22:43 +00:00
do_punch = def.do_punch ,
2017-01-16 16:40:08 +00:00
on_punch = mob_punch ,
2017-11-03 23:22:43 +00:00
on_breed = def.on_breed ,
on_grown = def.on_grown ,
2020-02-22 19:47:25 +00:00
on_detach_child = mob_detach_child ,
2017-07-04 23:52:39 +00:00
on_activate = function ( self , staticdata , dtime )
2021-04-08 03:07:04 +00:00
--this is a temporary hack so mobs stop
--glitching and acting really weird with the
--default built in engine collision detection
self.object : set_properties ( {
collide_with_objects = false ,
2021-04-13 11:39:57 +00:00
} )
self.object : set_acceleration ( vector_new ( 0 , - 9.81 , 0 ) )
2017-07-04 23:52:39 +00:00
return mob_activate ( self , staticdata , def , dtime )
2017-01-16 16:40:08 +00:00
end ,
get_staticdata = function ( self )
2017-05-25 08:33:19 +00:00
return mob_staticdata ( self )
2017-01-16 16:40:08 +00:00
end ,
2021-01-03 14:10:50 +00:00
2020-07-12 20:56:41 +00:00
harmed_by_heal = def.harmed_by_heal ,
2017-01-16 16:40:08 +00:00
} )
2021-04-12 12:47:07 +00:00
if minetest_get_modpath ( " doc_identifier " ) ~= nil then
2019-01-27 23:04:12 +00:00
doc.sub . identifier.register_object ( name , " basics " , " mobs " )
end
2017-01-16 16:40:08 +00:00
end -- END mobs:register_mob function
2017-05-25 08:33:19 +00:00
2017-01-16 16:40:08 +00:00
-- register arrow for shoot attack
2015-06-29 17:55:56 +00:00
function mobs : register_arrow ( name , def )
2017-01-16 16:40:08 +00:00
if not name or not def then return end -- errorcheck
2015-06-29 17:55:56 +00:00
minetest.register_entity ( name , {
2017-01-16 16:40:08 +00:00
2015-06-29 17:55:56 +00:00
physical = false ,
visual = def.visual ,
visual_size = def.visual_size ,
textures = def.textures ,
velocity = def.velocity ,
hit_player = def.hit_player ,
hit_node = def.hit_node ,
2017-01-16 16:40:08 +00:00
hit_mob = def.hit_mob ,
2020-01-30 22:11:16 +00:00
hit_object = def.hit_object ,
2017-01-16 16:40:08 +00:00
drop = def.drop or false , -- drops arrow as registered item when true
collisionbox = { 0 , 0 , 0 , 0 , 0 , 0 } , -- remove box around arrows
timer = 0 ,
switch = 0 ,
owner_id = def.owner_id ,
2017-05-25 08:33:19 +00:00
rotate = def.rotate ,
2021-04-04 01:07:51 +00:00
on_punch = function ( self )
local vel = self.object : get_velocity ( )
self.object : set_velocity ( { x = vel.x * - 1 , y = vel.y * - 1 , z = vel.z * - 1 } )
2017-11-03 23:22:43 +00:00
local pos = self.object : get_pos ( )
2017-01-16 16:40:08 +00:00
if self.switch == 0
or self.timer > 150
or not within_limits ( pos , 0 ) then
2021-01-02 09:56:40 +00:00
mcl_burning.extinguish ( self.object )
2018-06-02 22:56:29 +00:00
self.object : remove ( ) ;
2017-01-16 16:40:08 +00:00
2015-06-29 17:55:56 +00:00
return
end
2017-01-16 16:40:08 +00:00
-- does arrow have a tail (fireball)
if def.tail
and def.tail == 1
and def.tail_texture then
2017-05-25 08:33:19 +00:00
minetest.add_particle ( {
pos = pos ,
velocity = { x = 0 , y = 0 , z = 0 } ,
acceleration = { x = 0 , y = 0 , z = 0 } ,
expirationtime = def.expire or 0.25 ,
collisiondetection = false ,
2017-01-16 16:40:08 +00:00
texture = def.tail_texture ,
2017-05-25 08:33:19 +00:00
size = def.tail_size or 5 ,
glow = def.glow or 0 ,
2017-01-16 16:40:08 +00:00
} )
end
if self.hit_node then
local node = node_ok ( pos ) . name
2021-04-12 12:47:07 +00:00
if minetest_registered_nodes [ node ] . walkable then
2017-01-16 16:40:08 +00:00
self.hit_node ( self , pos , node )
if self.drop == true then
pos.y = pos.y + 1
self.lastpos = ( self.lastpos or pos )
2021-04-12 12:47:07 +00:00
minetest_add_item ( self.lastpos , self.object : get_luaentity ( ) . name )
2017-01-16 16:40:08 +00:00
end
2018-06-02 22:56:29 +00:00
self.object : remove ( ) ;
2017-01-16 16:40:08 +00:00
2015-06-29 17:55:56 +00:00
return
end
end
2017-01-16 16:40:08 +00:00
2020-01-30 22:11:16 +00:00
if self.hit_player or self.hit_mob or self.hit_object then
2017-01-16 16:40:08 +00:00
2021-04-12 12:47:07 +00:00
for _ , player in pairs ( minetest_get_objects_inside_radius ( pos , 1.5 ) ) do
2017-01-16 16:40:08 +00:00
if self.hit_player
and player : is_player ( ) then
self.hit_player ( self , player )
2018-06-02 22:56:29 +00:00
self.object : remove ( ) ;
2017-01-16 16:40:08 +00:00
return
end
local entity = player : get_luaentity ( )
2017-07-25 02:30:23 +00:00
if entity
and self.hit_mob
and entity._cmi_is_mob == true
2017-01-16 16:40:08 +00:00
and tostring ( player ) ~= self.owner_id
2017-07-25 02:30:23 +00:00
and entity.name ~= self.object : get_luaentity ( ) . name then
2017-01-16 16:40:08 +00:00
self.hit_mob ( self , player )
2018-06-02 22:56:29 +00:00
self.object : remove ( ) ;
2020-01-30 22:11:16 +00:00
return
end
2017-01-16 16:40:08 +00:00
2020-01-30 22:11:16 +00:00
if entity
and self.hit_object
and ( not entity._cmi_is_mob )
and tostring ( player ) ~= self.owner_id
and entity.name ~= self.object : get_luaentity ( ) . name then
self.hit_object ( self , player )
self.object : remove ( ) ;
2017-01-16 16:40:08 +00:00
return
end
end
end
self.lastpos = pos
2015-06-29 17:55:56 +00:00
end
} )
end
2017-07-04 23:52:39 +00:00
-- Register spawn eggs
-- Note: This also introduces the “spawn_egg” group:
-- * spawn_egg=1: Spawn egg (generic mob, no metadata)
-- * spawn_egg=2: Spawn egg (captured/tamed mob, metadata)
2017-01-16 16:40:08 +00:00
function mobs : register_egg ( mob , desc , background , addegg , no_creative )
2017-07-04 23:52:39 +00:00
local grp = { spawn_egg = 1 }
2017-01-16 16:40:08 +00:00
-- do NOT add this egg to creative inventory (e.g. dungeon master)
2020-07-10 14:46:47 +00:00
if no_creative == true then
2017-07-04 23:52:39 +00:00
grp.not_in_creative_inventory = 1
2017-01-16 16:40:08 +00:00
end
local invimg = background
if addegg == 1 then
invimg = " mobs_chicken_egg.png^( " .. invimg ..
" ^[mask:mobs_chicken_egg_overlay.png) "
end
2017-05-25 08:33:19 +00:00
-- register old stackable mob egg
minetest.register_craftitem ( mob , {
description = desc ,
inventory_image = invimg ,
groups = grp ,
2018-03-30 22:18:40 +00:00
2019-03-07 19:43:39 +00:00
_doc_items_longdesc = S ( " This allows you to place a single mob. " ) ,
_doc_items_usagehelp = S ( " Just place it where you want the mob to appear. Animals will spawn tamed, unless you hold down the sneak key while placing. If you place this on a mob spawner, you change the mob it spawns. " ) ,
2018-01-07 15:53:25 +00:00
2017-05-25 08:33:19 +00:00
on_place = function ( itemstack , placer , pointed_thing )
2017-01-16 16:40:08 +00:00
local pos = pointed_thing.above
2017-05-25 08:33:19 +00:00
-- am I clicking on something with existing on_rightclick function?
2021-04-12 12:47:07 +00:00
local under = minetest_get_node ( pointed_thing.under )
local def = minetest_registered_nodes [ under.name ]
2017-08-06 10:49:13 +00:00
if def and def.on_rightclick then
2017-05-25 08:33:19 +00:00
return def.on_rightclick ( pointed_thing.under , under , placer , itemstack )
end
2017-01-16 16:40:08 +00:00
if pos
2021-04-13 12:20:11 +00:00
--and within_limits(pos, 0)
2021-04-12 12:47:07 +00:00
and not minetest_is_protected ( pos , placer : get_player_name ( ) ) then
2017-01-16 16:40:08 +00:00
2018-01-07 15:53:25 +00:00
local name = placer : get_player_name ( )
local privs = minetest.get_player_privs ( name )
2018-05-31 00:47:37 +00:00
if mod_mobspawners and under.name == " mcl_mobspawners:spawner " then
2021-04-12 12:47:07 +00:00
if minetest_is_protected ( pointed_thing.under , name ) then
2019-02-08 21:17:51 +00:00
minetest.record_protection_violation ( pointed_thing.under , name )
return itemstack
end
2019-02-08 16:55:14 +00:00
if not privs.maphack then
2019-03-07 19:43:39 +00:00
minetest.chat_send_player ( name , S ( " You need the “maphack” privilege to change the mob spawner. " ) )
2019-02-08 16:55:14 +00:00
return itemstack
end
2018-01-07 15:53:25 +00:00
mcl_mobspawners.setup_spawner ( pointed_thing.under , itemstack : get_name ( ) )
2020-07-10 14:08:40 +00:00
if not mobs.is_creative ( name ) then
2018-01-07 15:53:25 +00:00
itemstack : take_item ( )
end
return itemstack
end
2021-04-12 12:47:07 +00:00
if not minetest_registered_entities [ mob ] then
2018-05-29 15:00:30 +00:00
return itemstack
2018-01-26 17:06:32 +00:00
end
2021-04-12 12:47:07 +00:00
if minetest_settings : get_bool ( " only_peaceful_mobs " , false )
and minetest_registered_entities [ mob ] . type == " monster " then
2020-01-06 12:46:43 +00:00
minetest.chat_send_player ( name , S ( " Only peaceful mobs allowed! " ) )
return itemstack
end
2020-12-08 21:42:01 +00:00
pos.y = pos.y - 0.5
2017-01-16 16:40:08 +00:00
2021-04-12 12:47:07 +00:00
local mob = minetest_add_entity ( pos , mob )
2020-12-08 21:42:01 +00:00
minetest.log ( " action " , " Mob spawned: " .. name .. " at " .. minetest.pos_to_string ( pos ) )
2017-01-16 16:40:08 +00:00
local ent = mob : get_luaentity ( )
2017-11-03 23:22:43 +00:00
-- don't set owner if monster or sneak pressed
2017-07-04 23:52:39 +00:00
if ent.type ~= " monster "
2017-08-06 10:49:13 +00:00
and not placer : get_player_control ( ) . sneak then
2017-05-25 08:33:19 +00:00
ent.owner = placer : get_player_name ( )
ent.tamed = true
2017-01-16 16:40:08 +00:00
end
2018-05-29 15:00:30 +00:00
2018-02-04 06:11:44 +00:00
-- set nametag
local nametag = itemstack : get_meta ( ) : get_string ( " name " )
if nametag ~= " " then
if string.len ( nametag ) > MAX_MOB_NAME_LENGTH then
nametag = string.sub ( nametag , 1 , MAX_MOB_NAME_LENGTH )
end
ent.nametag = nametag
update_tag ( ent )
end
2017-01-16 16:40:08 +00:00
-- if not in creative then take item
2017-11-03 23:22:43 +00:00
if not mobs.is_creative ( placer : get_player_name ( ) ) then
2017-01-16 16:40:08 +00:00
itemstack : take_item ( )
end
end
return itemstack
end ,
} )
2017-05-25 08:33:19 +00:00
2017-01-16 16:40:08 +00:00
end
2017-05-25 08:33:19 +00:00