2016-10-08 21:54:26 +03:00
-- main tables
2016-10-09 22:36:09 +03:00
spawners_mobs = { }
spawners_mobs.mob_tables = { }
2018-01-07 07:31:21 +03:00
local max_obj_per_mapblock = tonumber ( minetest.settings : get ( " max_objects_per_block " ) )
local enable_particles = minetest.settings : get_bool ( " enable_particles " )
2018-01-29 06:41:01 +03:00
local tick_max = 40
local tick_short_max = 40
2016-10-08 21:54:26 +03:00
-- check if mods exists and build tables
for k , mob_mod in ipairs ( ENABLED_MODS ) do
local modpath = minetest.get_modpath ( mob_mod )
-- list of mobs and their info
if ( modpath ) then
for j , mob in ipairs ( MOBS_PROPS [ mob_mod ] ) do
local mob_egg = nil
-- disabled extra check for mobs redo due to incompatibility with Lua 5.1, this method is available from Lua 5.2
-- if mob_mod == "mobs" and not (mobs.mod == "redo") then goto continue end
2017-07-29 17:41:47 +03:00
table.insert ( spawners_mobs.mob_tables ,
{
2018-01-03 22:56:09 +03:00
name = mob.name ,
mod_prefix = mob_mod ,
egg_name_custom = mob.egg_name_custom ,
dummy_size = mob.dummy_size ,
dummy_offset = mob.dummy_offset ,
dummy_mesh = mob.dummy_mesh ,
dummy_texture = mob.dummy_texture ,
night_only = mob.night_only ,
sound_custom = mob.sound_custom
2017-07-29 17:41:47 +03:00
}
)
2016-10-08 21:54:26 +03:00
-- use custom egg or create a default egg
if mob.egg_name_custom ~= " " then
mob_egg = mob.egg_name_custom
else
mob_egg = mob_mod .. " : " .. mob.name
end
-- recipes
minetest.register_craft ( {
output = " spawners_mobs: " .. mob_mod .. " _ " .. mob.name .. " _spawner " ,
recipe = {
{ " default:diamondblock " , " fire:flint_and_steel " , " default:diamondblock " } ,
{ " xpanes:bar_flat " , mob_egg , " xpanes:bar_flat " } ,
{ " default:diamondblock " , " xpanes:bar_flat " , " default:diamondblock " } ,
}
} )
-- ::continue::
end
end
end
2018-01-07 07:31:21 +03:00
--
-- Particles
--
2016-10-12 22:57:06 +03:00
function spawners_mobs . cloud_booom ( pos )
2018-01-07 07:31:21 +03:00
if not enable_particles then return end
2016-10-12 22:57:06 +03:00
minetest.add_particlespawner ( {
amount = 5 ,
time = 2 ,
2017-12-27 19:24:17 +03:00
minpos = vector.subtract ( { x = pos.x - 0.3 , y = pos.y , z = pos.z - 0.3 } , 0.3 ) ,
maxpos = vector.add ( { x = pos.x + 0.3 , y = pos.y , z = pos.z + 0.3 } , 0.3 ) ,
2016-10-12 22:57:06 +03:00
minvel = { x = 0.1 , y = 0.1 , z = 0.1 } ,
maxvel = { x = 0.2 , y = 0.2 , z = 0.2 } ,
2017-12-27 19:24:17 +03:00
minacc = vector.new ( { x =- 0.1 , y = 0.3 , z =- 0.1 } ) ,
maxacc = vector.new ( { x = 0.1 , y = 0.6 , z = 0.1 } ) ,
2016-10-12 22:57:06 +03:00
minexptime = 2 ,
maxexptime = 3 ,
minsize = 4 ,
2017-12-27 19:24:17 +03:00
maxsize = 12 ,
texture = " spawners_mobs_smoke_particle_2.png^[transform " .. math.random ( 0 , 3 ) ,
2016-10-12 22:57:06 +03:00
} )
end
function spawners_mobs . add_flame_effects ( pos )
2018-01-07 07:31:21 +03:00
if not enable_particles then return end
2018-01-09 03:29:50 +03:00
return minetest.add_particlespawner ( {
2016-10-12 22:57:06 +03:00
amount = 6 ,
time = 0 ,
minpos = vector.subtract ( { x = pos.x - 0.001 , y = pos.y - 0.001 , z = pos.z - 0.001 } , 0.5 ) ,
maxpos = vector.add ( { x = pos.x + 0.001 , y = pos.y + 0.001 , z = pos.z + 0.001 } , 0.5 ) ,
minvel = { x =- 0.1 , y =- 0.1 , z =- 0.1 } ,
maxvel = { x = 0.1 , y = 0.1 , z = 0.1 } ,
minacc = vector.new ( ) ,
maxacc = vector.new ( ) ,
minexptime = 1 ,
maxexptime = 5 ,
minsize = .5 ,
maxsize = 2.5 ,
2017-12-27 19:24:17 +03:00
texture = " spawners_mobs_flame_particle_2.png " ,
2016-10-12 22:57:06 +03:00
} )
end
function spawners_mobs . add_smoke_effects ( pos )
2018-01-07 07:31:21 +03:00
if not enable_particles then return end
2018-01-09 03:29:50 +03:00
return minetest.add_particlespawner ( {
2016-10-12 22:57:06 +03:00
amount = 1 ,
time = 0 ,
minpos = vector.subtract ( { x = pos.x - 0.001 , y = pos.y - 0.001 , z = pos.z - 0.001 } , 0.5 ) ,
maxpos = vector.add ( { x = pos.x + 0.001 , y = pos.y + 0.001 , z = pos.z + 0.001 } , 0.5 ) ,
minvel = { x =- 0.5 , y = 0.5 , z =- 0.5 } ,
maxvel = { x = 0.5 , y = 1.5 , z = 0.5 } ,
minacc = vector.new ( { x =- 0.1 , y = 0.1 , z =- 0.1 } ) ,
maxacc = vector.new ( { x = 0.1 , y = 0.3 , z = 0.1 } ) ,
minexptime = .5 ,
2016-10-15 16:41:59 +03:00
maxexptime = 2 ,
2016-10-12 22:57:06 +03:00
minsize = .5 ,
maxsize = 2 ,
2016-10-15 16:41:59 +03:00
texture = " spawners_mobs_smoke_particle.png^[transform " .. math.random ( 0 , 3 ) ,
2016-10-12 22:57:06 +03:00
} )
end
2018-01-07 07:31:21 +03:00
--
-- Timers
--
-- how often node timers for spawners will tick, +/- some random value
function spawners_mobs . tick ( pos )
local meta = minetest.get_meta ( pos )
local tick_counter = meta : get_int ( " tick " )
2018-01-08 06:07:31 +03:00
local owner = meta : get_string ( " owner " )
2018-01-08 22:34:07 +03:00
local privs = minetest.get_player_privs ( owner ) ;
2018-01-07 07:31:21 +03:00
2018-01-08 22:34:07 +03:00
-- not for admin
if not privs.privs then
tick_counter = tick_counter + 1
2018-01-08 06:07:31 +03:00
meta : set_int ( " tick " , tick_counter )
end
2018-01-09 03:29:50 +03:00
-- print("tick_counter: "..tick_counter.." at "..minetest.pos_to_string(pos))
2018-01-08 22:34:07 +03:00
2018-01-07 07:31:21 +03:00
-- rusty spawner
if tick_counter >= tick_max then
spawners_mobs.set_status ( pos , " rusty " )
return
end
2018-01-08 06:07:31 +03:00
minetest.get_node_timer ( pos ) : start ( math.random ( 166 , 286 ) )
2018-01-09 03:29:50 +03:00
-- minetest.get_node_timer(pos):start(math.random(20, 30))
2018-01-07 07:31:21 +03:00
end
-- how often a spawn failure tick is retried (e.g. too dark)
function spawners_mobs . tick_short ( pos )
local meta = minetest.get_meta ( pos )
local tick_short_counter = meta : get_int ( " tick_short " )
if tick_short_counter >= tick_short_max then
spawners_mobs.tick ( pos )
return
else
tick_short_counter = tick_short_counter + 1
meta : set_int ( " tick_short " , tick_short_counter )
2018-01-09 03:29:50 +03:00
-- print("tick_short_counter: "..tick_short_counter.." at "..minetest.pos_to_string(pos))
2018-01-07 07:31:21 +03:00
end
2018-01-08 06:07:31 +03:00
minetest.get_node_timer ( pos ) : start ( math.random ( 40 , 80 ) )
2018-01-09 03:29:50 +03:00
-- minetest.get_node_timer(pos):start(math.random(10, 20))
2018-01-07 07:31:21 +03:00
end
--
-- Core Functions
--
2016-10-08 21:54:26 +03:00
-- start spawning mobs
2018-01-07 07:31:21 +03:00
function spawners_mobs . start_spawning ( spawn_area_random_pos , mob_name , mod_prefix , sound_custom )
if not ( spawn_area_random_pos or how_many or mob_name ) then return end
2016-10-08 21:54:26 +03:00
2018-01-03 22:56:09 +03:00
local sound_name = mod_prefix .. " _ " .. mob_name
2016-10-08 21:54:26 +03:00
-- use custom sounds
if sound_custom ~= " " then
sound_name = sound_custom
end
2018-01-03 22:56:09 +03:00
-- use random colors for sheeps
2016-10-08 21:54:26 +03:00
if mob_name == " sheep_white " then
2018-01-03 22:56:09 +03:00
local sheep_colors = { " black " , " blue " , " brown " , " cyan " , " dark_green " , " dark_grey " , " green " , " grey " , " magenta " , " orange " , " pink " , " red " , " violet " , " white " , " yellow " }
mob_name = " sheep_ " .. sheep_colors [ math.random ( # sheep_colors ) ]
2016-10-08 21:54:26 +03:00
end
2018-01-07 07:31:21 +03:00
for i = 1 , # spawn_area_random_pos do
2018-01-03 22:56:09 +03:00
-- spawn a bit more above the block - prevent spawning inside the block
2018-01-07 07:31:21 +03:00
spawn_area_random_pos [ i ] . y = spawn_area_random_pos [ i ] . y + 0.5
2016-10-12 22:57:06 +03:00
2018-01-07 07:31:21 +03:00
spawners_mobs.cloud_booom ( spawn_area_random_pos [ i ] )
2016-10-12 22:57:06 +03:00
minetest.after ( 1 , function ( )
2018-01-07 07:31:21 +03:00
-- minetest.set_node(spawn_area_random_pos[i], {name = "default:apple"})
local obj = minetest.add_entity ( spawn_area_random_pos [ i ] , mod_prefix .. " : " .. mob_name )
2016-10-12 22:57:06 +03:00
if obj then
if sound_name then
minetest.sound_play ( sound_name , {
2018-01-07 07:31:21 +03:00
pos = spawn_area_random_pos [ i ] ,
max_hear_distance = 8 ,
2018-01-07 22:04:27 +03:00
gain = 0.5
2016-10-12 22:57:06 +03:00
} )
end
2016-10-08 21:54:26 +03:00
end
2016-10-12 22:57:06 +03:00
end )
2016-10-08 21:54:26 +03:00
end
end
2018-01-07 07:31:21 +03:00
function spawners_mobs . on_timer ( pos , elapsed )
local meta = minetest.get_meta ( pos )
local idx = meta : get_int ( " idx " ) or nil
local mob_table = spawners_mobs.mob_tables [ idx ] or false
2016-10-08 21:54:26 +03:00
2018-01-07 07:31:21 +03:00
if not mob_table then return end
2016-10-08 21:54:26 +03:00
2018-01-07 19:48:46 +03:00
local posmin = { x = pos.x - 3 , y = pos.y - 1 , z = pos.z - 3 }
2018-01-03 22:56:09 +03:00
local posmax = { x = pos.x + 4 , y = pos.y + 1 , z = pos.z + 4 }
local player_near = false
local entities_near = 0
local entities_max = 6
local node_light_min = 13
2018-01-07 07:31:21 +03:00
local owner = meta : get_string ( " owner " ) or " "
local mod_prefix = mob_table.mod_prefix
local mob_name = mob_table.name
local sound_custom = mob_table.sound_custom
local night_only = mob_table.night_only
local max_objects = max_obj_per_mapblock / 4
2018-01-03 22:56:09 +03:00
-- check spawner light
local node_light = minetest.get_node_light ( pos )
2018-01-07 07:31:21 +03:00
-- dark
if ( not node_light or node_light < node_light_min ) and not night_only then
-- print("Too dark for mob ( "..mob_name.." ) to spawn. Waiting for day...")
spawners_mobs.set_status ( pos , " waiting " )
-- set infotext
meta : set_string ( " infotext " , mob_name .. " spawner \n owner: " .. owner .. " \n Too dark for mob to spawn. Waiting for day... " )
spawners_mobs.tick_short ( pos )
return
-- light
elseif node_light >= node_light_min and night_only then
-- print("Too much light for mob ( "..mob_name.." ) to spawn. Waiting for night...")
spawners_mobs.set_status ( pos , " waiting " )
-- set infotext
meta : set_string ( " infotext " , mob_name .. " spawner \n owner: " .. owner .. " \n Too much light for mob to spawn. Waiting for night... " )
spawners_mobs.tick_short ( pos )
2018-01-03 22:56:09 +03:00
return
end
-- positions where mobs can spawn
local spawn_area_pos = minetest.find_nodes_in_area ( posmin , posmax , " air " )
2018-01-07 07:31:21 +03:00
2018-01-03 22:56:09 +03:00
-- check if there is enough place to spawn mob
2018-01-07 07:31:21 +03:00
if # spawn_area_pos < 1 then
spawners_mobs.set_status ( pos , " waiting " )
-- set infotext
meta : set_string ( " infotext " , mob_name .. " spawner \n owner: " .. owner .. " \n Not enough place to spawn mob. Find more space! " )
spawners_mobs.tick ( pos )
2018-01-03 22:56:09 +03:00
return
end
2018-01-07 07:31:21 +03:00
-- spawn 2 mobs on 2 different positions by chance
local how_many = math.random ( 2 )
local spawn_area_random_pos = { }
-- get random spawn position from spawn area
for i = 1 , how_many do
while # spawn_area_random_pos < how_many and # spawn_area_pos > 0 do
local random_pos = spawn_area_pos [ math.random ( # spawn_area_pos ) ]
local random_pos_above = minetest.get_node ( { x = random_pos.x , y = random_pos.y + 1 , z = random_pos.z } ) . name
if random_pos_above == " air " and not minetest.is_protected ( random_pos , owner ) then
table.insert ( spawn_area_random_pos , random_pos )
-- print("spawn_area_random_pos: "..#spawn_area_random_pos)
else
table.remove ( spawn_area_pos , i )
-- print("spawn_area_pos: "..#spawn_area_pos)
end
end
end
-- print(dump(spawn_area_random_pos))
-- check if there is still enough place to spawn mob
if # spawn_area_random_pos < 1 then
spawners_mobs.set_status ( pos , " waiting " )
-- set infotext
meta : set_string ( " infotext " , mob_name .. " spawner \n owner: " .. owner .. " \n Not enough place to spawn mob. Searching for new location... " )
spawners_mobs.tick_short ( pos )
2018-01-03 22:56:09 +03:00
return
end
-- area where player and entity count will be detected
local activation_area = minetest.get_objects_inside_radius ( pos , 16 )
2018-01-07 07:31:21 +03:00
-- prevent object clutter on the map
if # activation_area > max_objects then
spawners_mobs.set_status ( pos , " waiting " )
-- set infotext
meta : set_string ( " infotext " , mob_name .. " spawner \n owner: " .. owner .. " \n Too many objects in the area ( " .. # activation_area .. " / " .. max_objects .. " ), clean-up dropped objects first! " )
spawners_mobs.tick_short ( pos )
return
end
2018-01-03 22:56:09 +03:00
for k , object in ipairs ( activation_area ) do
-- find player inside activation area
if object : is_player ( ) then
player_near = true
-- print("found player: "..object:get_player_name())
2016-10-08 21:54:26 +03:00
end
2018-01-03 22:56:09 +03:00
-- find entities inside activation area
if not object : is_player ( ) and
object : get_luaentity ( ) and
object : get_luaentity ( ) . name ~= " __builtin:item " then
local tmp_mob_name = string.split ( object : get_luaentity ( ) . name , " : " ) [ 2 ]
2016-10-08 21:54:26 +03:00
2018-01-08 09:49:54 +03:00
if tmp_mob_name ~= nil then
-- sheeps have colors in names
if string.find ( tmp_mob_name , " sheep " ) and string.find ( mob_name , " sheep " ) and not string.find ( tmp_mob_name , " dummy " ) then
-- print("found entity: "..tmp_mob_name)
entities_near = entities_near + 1
elseif tmp_mob_name == mob_name then
-- print("found entity: "..tmp_mob_name)
entities_near = entities_near + 1
end
else
2018-01-08 19:59:55 +03:00
minetest.log ( " warning " , " [spawners_mobs] tmp_mob_name was nil, luaentity name was: " .. object : get_luaentity ( ) . name .. " at: " .. minetest.pos_to_string ( object : get_pos ( ) ) )
2016-10-08 21:54:26 +03:00
end
end
2018-01-03 22:56:09 +03:00
-- stop looping when met all conditions
if entities_near >= entities_max and player_near then
-- print("max entities reached "..entities_max.." and player_near found, breaking..")
break
2016-10-08 21:54:26 +03:00
end
2018-01-03 22:56:09 +03:00
end
2016-10-08 21:54:26 +03:00
2018-01-03 22:56:09 +03:00
-- don't do anything and try again later when player not near or max entities reached
if entities_near >= entities_max or not player_near then
2018-01-07 07:31:21 +03:00
spawners_mobs.set_status ( pos , " waiting " )
-- sheeps have color in the name
local name = mob_name
if string.find ( mob_name , " sheep " ) then
name = " sheep "
end
meta : set_string ( " infotext " , mob_name .. " spawner \n owner: " .. owner .. " \n max mobs reached: " .. entities_near .. " / " .. entities_max ) -- or player not near
spawners_mobs.tick_short ( pos )
2018-01-03 22:56:09 +03:00
return
2016-10-08 21:54:26 +03:00
end
2018-01-07 07:31:21 +03:00
2018-01-03 22:56:09 +03:00
-- start spawning
2018-01-07 07:31:21 +03:00
spawners_mobs.start_spawning ( spawn_area_random_pos , mob_name , mod_prefix , sound_custom )
spawners_mobs.set_status ( pos , " active " )
meta : set_string ( " infotext " , mob_name .. " spawner \n owner: " .. owner .. " \n spawner is active reached: " .. entities_near .. " / " .. entities_max )
meta : set_int ( " tick " , 0 )
meta : set_int ( " tick_short " , 0 )
spawners_mobs.tick ( pos )
end
--
-- Status Manager
--
function spawners_mobs . set_status ( pos , set_status )
local meta = minetest.get_meta ( pos )
local idx = meta : get_int ( " idx " )
local mob_table = spawners_mobs.mob_tables [ idx ] or false
if not mob_table then return end
local mod_prefix = mob_table.mod_prefix
local mob_name = mob_table.name
local offset = mob_table.dummy_offset
-- get meta
local owner = meta : get_string ( " owner " )
local meta_status = meta : get_string ( " status " )
local id_flame = meta : get_int ( " id_flame " )
local id_smoke = meta : get_int ( " id_smoke " )
--
-- active
--
if set_status == " active " then
-- remove particles and add them again - keeps particles after server restart
-- delete particles
2018-01-09 03:29:50 +03:00
if id_flame ~= - 1 and id_smoke ~= - 1 then
-- print("#1 delete id_flame: "..id_flame.." at "..minetest.pos_to_string(pos))
-- print("#1 delete id_smoke: "..id_smoke.." at "..minetest.pos_to_string(pos))
2018-01-07 07:31:21 +03:00
minetest.delete_particlespawner ( id_flame )
minetest.delete_particlespawner ( id_smoke )
2018-01-09 03:29:50 +03:00
meta : set_int ( " id_flame " , - 1 )
meta : set_int ( " id_smoke " , - 1 )
2018-01-07 07:31:21 +03:00
end
-- add particles
id_flame = spawners_mobs.add_flame_effects ( pos )
id_smoke = spawners_mobs.add_smoke_effects ( pos )
meta : set_int ( " id_flame " , id_flame )
meta : set_int ( " id_smoke " , id_smoke )
2018-01-09 03:29:50 +03:00
-- print("#1 add id_flame: "..id_flame.." at "..minetest.pos_to_string(pos))
-- print("#1 add id_smoke: "..id_smoke.." at "..minetest.pos_to_string(pos))
2018-01-03 22:56:09 +03:00
2018-01-07 07:31:21 +03:00
if meta_status ~= set_status then
-- add dummy entity
minetest.add_entity ( { x = pos.x , y = pos.y + offset , z = pos.z } , " spawners_mobs:dummy_ " .. mod_prefix .. " _ " .. mob_name )
meta : set_string ( " status " , " active " )
minetest.swap_node ( pos , { name = " spawners_mobs: " .. mod_prefix .. " _ " .. mob_name .. " _spawner " } )
end
--
-- waiting
--
elseif set_status == " waiting " and meta_status ~= set_status then
-- delete particles
2018-01-09 03:29:50 +03:00
if id_flame ~= - 1 and id_smoke ~= - 1 then
-- print("#2 delete id_flame: "..id_flame.." at "..minetest.pos_to_string(pos))
-- print("#2 delete id_smoke: "..id_smoke.." at "..minetest.pos_to_string(pos))
2018-01-07 07:31:21 +03:00
minetest.delete_particlespawner ( id_flame )
minetest.delete_particlespawner ( id_smoke )
2018-01-09 03:29:50 +03:00
meta : set_int ( " id_flame " , - 1 )
meta : set_int ( " id_smoke " , - 1 )
2018-01-07 07:31:21 +03:00
end
-- remove dummy
local objs = minetest.get_objects_inside_radius ( pos , 0.5 )
if objs then
for _ , obj in ipairs ( objs ) do
if obj and obj : get_luaentity ( ) and obj : get_luaentity ( ) . name == " spawners_mobs:dummy_ " .. mod_prefix .. " _ " .. mob_name then
obj : remove ( )
end
end
end
meta : set_string ( " status " , " waiting " )
minetest.swap_node ( pos , { name = " spawners_mobs: " .. mod_prefix .. " _ " .. mob_name .. " _spawner_waiting " } )
--
-- rusty
--
elseif set_status == " rusty " and meta_status ~= set_status then
-- delete particles
2018-01-09 03:29:50 +03:00
if id_flame ~= - 1 and id_smoke ~= - 1 then
-- print("#3 delete id_flame: "..id_flame.." at "..minetest.pos_to_string(pos))
-- print("#3 delete id_smoke: "..id_smoke.." at "..minetest.pos_to_string(pos))
2018-01-07 07:31:21 +03:00
minetest.delete_particlespawner ( id_flame )
minetest.delete_particlespawner ( id_smoke )
2018-01-09 03:29:50 +03:00
meta : set_int ( " id_flame " , - 1 )
meta : set_int ( " id_smoke " , - 1 )
2018-01-07 07:31:21 +03:00
end
-- remove dummy
local objs = minetest.get_objects_inside_radius ( pos , 0.5 )
if objs then
for _ , obj in ipairs ( objs ) do
if obj and obj : get_luaentity ( ) and obj : get_luaentity ( ) . name == " spawners_mobs:dummy_ " .. mod_prefix .. " _ " .. mob_name then
obj : remove ( )
end
end
end
meta : set_string ( " status " , " rusty " )
minetest.swap_node ( pos , { name = " spawners_mobs: " .. mod_prefix .. " _ " .. mob_name .. " _spawner_rusty " } )
-- set infotext
meta : set_string ( " infotext " , mob_name .. " spawner \n owner: " .. owner .. " \n Spawner was searching for too long and got rusted! Dig up the spawner and place it again. " )
return
end
2016-10-08 21:54:26 +03:00
end