2016-10-08 21:54:26 +03:00
|
|
|
-- main tables
|
2016-10-09 22:36:09 +03:00
|
|
|
spawners_mobs = {}
|
|
|
|
spawners_mobs.mob_tables = {}
|
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
|
|
|
|
|
2016-10-09 22:36:09 +03:00
|
|
|
table.insert(spawners_mobs.mob_tables, {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})
|
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
|
|
|
|
else
|
|
|
|
-- print something ?
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-10-15 16:41:59 +03:00
|
|
|
function spawners_mobs.meta_get_int(key, pos)
|
|
|
|
local meta = minetest.get_meta(pos)
|
|
|
|
return meta:get_int(key)
|
|
|
|
end
|
|
|
|
|
|
|
|
function spawners_mobs.meta_set_int(key, value, pos)
|
|
|
|
local meta = minetest.get_meta(pos)
|
|
|
|
meta:set_int(key, value)
|
|
|
|
end
|
|
|
|
|
|
|
|
function spawners_mobs.meta_set_str(key, value, pos)
|
|
|
|
local meta = minetest.get_meta(pos)
|
|
|
|
meta:set_string(key, value)
|
|
|
|
end
|
|
|
|
|
2016-10-12 22:57:06 +03:00
|
|
|
-- particles
|
|
|
|
function spawners_mobs.cloud_booom(pos)
|
|
|
|
minetest.add_particlespawner({
|
|
|
|
amount = 5,
|
|
|
|
time = 2,
|
2016-11-13 14:44:52 +03:00
|
|
|
minpos = vector.subtract({x=pos.x-0.6, y=pos.y, z=pos.z-0.6}, 0.6),
|
|
|
|
maxpos = vector.add({x=pos.x+0.6, y=pos.y, z=pos.z+0.6}, 0.6),
|
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},
|
|
|
|
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 = 2,
|
|
|
|
maxexptime = 3,
|
|
|
|
minsize = 4,
|
|
|
|
maxsize = 8,
|
2016-10-15 16:41:59 +03:00
|
|
|
texture = "spawners_mobs_smoke2_particle.png^[transform"..math.random(0,3),
|
2016-10-12 22:57:06 +03:00
|
|
|
})
|
|
|
|
end
|
|
|
|
|
|
|
|
function spawners_mobs.add_flame_effects(pos)
|
|
|
|
local id = minetest.add_particlespawner({
|
|
|
|
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,
|
|
|
|
texture = "spawners_mobs_flame_particle.png",
|
|
|
|
})
|
|
|
|
|
|
|
|
return id
|
|
|
|
end
|
|
|
|
|
|
|
|
function spawners_mobs.add_smoke_effects(pos)
|
|
|
|
local id = minetest.add_particlespawner({
|
|
|
|
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
|
|
|
})
|
|
|
|
|
|
|
|
return id
|
|
|
|
end
|
|
|
|
|
2016-10-08 21:54:26 +03:00
|
|
|
-- start spawning mobs
|
2016-10-15 16:41:59 +03:00
|
|
|
function spawners_mobs.start_spawning(random_pos, how_many, mob_name, mod_prefix, sound_custom)
|
2016-10-12 22:57:06 +03:00
|
|
|
if not (random_pos or how_many or mob_name) then return end
|
2016-10-08 21:54:26 +03:00
|
|
|
|
|
|
|
local sound_name
|
|
|
|
-- remove 'spawners_mobs:' from the string
|
|
|
|
local mob_name = string.sub(mob_name,15)
|
|
|
|
|
|
|
|
-- use custom sounds
|
|
|
|
if sound_custom ~= "" then
|
|
|
|
sound_name = sound_custom
|
|
|
|
else
|
|
|
|
sound_name = mod_prefix.."_"..mob_name
|
|
|
|
end
|
|
|
|
|
|
|
|
-- use random colours for sheeps
|
|
|
|
if mob_name == "sheep_white" then
|
|
|
|
local mob_name1 = ""
|
|
|
|
local sheep_colours = {"black", "blue", "brown", "cyan", "dark_green", "dark_grey", "green", "grey", "magenta", "orange", "pink", "red", "violet", "white", "yellow"}
|
|
|
|
local random_colour = math.random(1, #sheep_colours)
|
|
|
|
mob_name1 = string.split(mob_name, "_")
|
|
|
|
mob_name1 = mob_name1[1]
|
|
|
|
mob_name = mob_name1.."_"..sheep_colours[random_colour]
|
|
|
|
end
|
|
|
|
|
|
|
|
for i=1,how_many do
|
2016-10-15 16:41:59 +03:00
|
|
|
random_pos.y = random_pos.y+0.5
|
2016-10-12 22:57:06 +03:00
|
|
|
|
|
|
|
spawners_mobs.cloud_booom(random_pos)
|
|
|
|
|
|
|
|
local obj
|
|
|
|
|
|
|
|
minetest.after(1, function()
|
|
|
|
obj = minetest.add_entity(random_pos, mod_prefix..":"..mob_name)
|
|
|
|
|
|
|
|
if obj then
|
|
|
|
if sound_name then
|
|
|
|
minetest.sound_play(sound_name, {
|
|
|
|
random_pos = random_pos,
|
|
|
|
max_hear_distance = 32,
|
|
|
|
gain = 5,
|
|
|
|
})
|
|
|
|
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
|
|
|
|
|
2016-10-09 22:36:09 +03:00
|
|
|
function spawners_mobs.check_around_radius(pos)
|
2016-10-08 21:54:26 +03:00
|
|
|
local player_near = false
|
|
|
|
local radius = 21
|
|
|
|
|
|
|
|
for _,obj in ipairs(minetest.get_objects_inside_radius(pos, radius)) do
|
|
|
|
if obj:is_player() then
|
|
|
|
player_near = true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return player_near
|
|
|
|
end
|
|
|
|
|
2016-10-09 22:36:09 +03:00
|
|
|
function spawners_mobs.check_node_status(pos, mob, night_only)
|
|
|
|
local player_near = spawners_mobs.check_around_radius(pos)
|
2016-10-08 21:54:26 +03:00
|
|
|
|
|
|
|
if player_near then
|
|
|
|
local random_pos = false
|
|
|
|
local min_node_light = 10
|
|
|
|
local tod = minetest.get_timeofday() * 24000
|
|
|
|
local node_light = minetest.get_node_light(pos)
|
|
|
|
|
|
|
|
if not node_light then
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
local spawn_positions = {}
|
|
|
|
local right = minetest.get_node({x=pos.x+1, y=pos.y, z=pos.z})
|
|
|
|
local front = minetest.get_node({x=pos.x, y=pos.y, z=pos.z+1})
|
|
|
|
local left = minetest.get_node({x=pos.x-1, y=pos.y, z=pos.z})
|
|
|
|
local back = minetest.get_node({x=pos.x, y=pos.y, z=pos.z-1})
|
|
|
|
local top = minetest.get_node({x=pos.x, y=pos.y+1, z=pos.z})
|
|
|
|
local bottom = minetest.get_node({x=pos.x, y=pos.y-1, z=pos.z})
|
|
|
|
|
|
|
|
-- make sure that at least one side of the spawner is open
|
|
|
|
if right.name == "air" then
|
|
|
|
table.insert(spawn_positions, {x=pos.x+1.5, y=pos.y, z=pos.z})
|
|
|
|
end
|
|
|
|
if front.name == "air" then
|
|
|
|
table.insert(spawn_positions, {x=pos.x, y=pos.y, z=pos.z+1.5})
|
|
|
|
end
|
|
|
|
if left.name == "air" then
|
|
|
|
table.insert(spawn_positions, {x=pos.x-1.5, y=pos.y, z=pos.z})
|
|
|
|
end
|
|
|
|
if back.name == "air" then
|
|
|
|
table.insert(spawn_positions, {x=pos.x, y=pos.y, z=pos.z-1.5})
|
|
|
|
end
|
|
|
|
if top.name == "air" then
|
|
|
|
table.insert(spawn_positions, {x=pos.x, y=pos.y+1.5, z=pos.z})
|
|
|
|
end
|
|
|
|
if bottom.name == "air" then
|
|
|
|
table.insert(spawn_positions, {x=pos.x, y=pos.y-1.5, z=pos.z})
|
|
|
|
end
|
|
|
|
|
|
|
|
if #spawn_positions < 1 then
|
2016-10-15 16:41:59 +03:00
|
|
|
-- spawner is closed from all sides
|
2016-10-08 21:54:26 +03:00
|
|
|
return false
|
|
|
|
else
|
|
|
|
-- pick random from the open sides
|
|
|
|
local pick_random
|
|
|
|
|
|
|
|
if #spawn_positions == 1 then
|
|
|
|
pick_random = #spawn_positions
|
|
|
|
else
|
|
|
|
pick_random = math.random(1,#spawn_positions)
|
|
|
|
end
|
|
|
|
|
|
|
|
for k, v in pairs (spawn_positions) do
|
|
|
|
if k == pick_random then
|
|
|
|
random_pos = v
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- check the node above and below the found air node
|
|
|
|
local node_above = minetest.get_node({x=random_pos.x, y=random_pos.y+1, z=random_pos.z}).name
|
|
|
|
local node_below = minetest.get_node({x=random_pos.x, y=random_pos.y-1, z=random_pos.z}).name
|
|
|
|
|
|
|
|
if not (node_above == "air" or node_below == "air") then
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
if night_only ~= "disable" then
|
|
|
|
-- spawn only at day
|
|
|
|
if not night_only and node_light < min_node_light then
|
|
|
|
return false, true
|
|
|
|
end
|
|
|
|
|
|
|
|
-- spawn only at night
|
|
|
|
if night_only then
|
|
|
|
if not (19359 > tod and tod > 5200) or node_light < min_node_light then
|
|
|
|
return random_pos
|
|
|
|
else
|
|
|
|
return false, true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-10-15 16:41:59 +03:00
|
|
|
-- random_pos, waiting
|
2016-10-08 21:54:26 +03:00
|
|
|
return random_pos, false
|
|
|
|
else
|
2016-10-15 16:41:59 +03:00
|
|
|
-- random_pos, waiting
|
2016-10-08 21:54:26 +03:00
|
|
|
return false, true
|
|
|
|
end
|
|
|
|
end
|