2017-01-16 18:40:08 +02:00
|
|
|
|
local S = mobs.intllib
|
|
|
|
|
|
2017-05-25 03:17:24 +03:00
|
|
|
|
mcl_monster_spawner = {}
|
|
|
|
|
|
2017-05-25 05:23:21 +03:00
|
|
|
|
local default_mob = "mobs_mc:pig"
|
2017-01-16 18:40:08 +02:00
|
|
|
|
|
2017-05-25 03:17:24 +03:00
|
|
|
|
-- Monster spawner
|
2017-05-25 05:47:08 +03:00
|
|
|
|
local spawner_default = default_mob.." 0 15 4 15"
|
2017-05-25 01:59:41 +03:00
|
|
|
|
|
|
|
|
|
local function get_mob_textures(mob)
|
|
|
|
|
-- FIXME: Ummm … wtf? Why isn't there a textures attribute?
|
|
|
|
|
return minetest.registered_entities[mob].texture_list[1]
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
local function find_doll(pos)
|
|
|
|
|
for _,obj in ipairs(minetest.env:get_objects_inside_radius(pos, 1)) do
|
|
|
|
|
if not obj:is_player() then
|
2017-05-25 03:06:36 +03:00
|
|
|
|
if obj ~= nil and obj:get_luaentity().name == "mcl_monster_spawner:doll" then
|
2017-05-25 01:59:41 +03:00
|
|
|
|
return obj
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
return nil
|
|
|
|
|
end
|
2017-01-16 18:40:08 +02:00
|
|
|
|
|
2017-05-25 02:14:18 +03:00
|
|
|
|
local function set_doll_properties(doll, mob)
|
|
|
|
|
local mobinfo = minetest.registered_entities[mob]
|
|
|
|
|
local prop = {
|
|
|
|
|
mesh = mobinfo.mesh,
|
|
|
|
|
textures = get_mob_textures(mob),
|
|
|
|
|
visual_size = {
|
2017-05-25 04:14:52 +03:00
|
|
|
|
x = mobinfo.visual_size.x * 0.33333,
|
|
|
|
|
y = mobinfo.visual_size.y * 0.33333,
|
2017-05-25 02:14:18 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
doll:set_properties(prop)
|
2017-05-25 04:09:02 +03:00
|
|
|
|
doll:get_luaentity()._mob = mob
|
2017-05-25 02:14:18 +03:00
|
|
|
|
end
|
|
|
|
|
|
2017-05-25 06:37:13 +03:00
|
|
|
|
|
|
|
|
|
|
2017-05-25 02:59:10 +03:00
|
|
|
|
--[[ Public function: Setup the spawner at pos.
|
|
|
|
|
This function blindly assumes there's actually a spawner at pos.
|
|
|
|
|
If not, then the results are undefined.
|
2017-05-25 06:45:45 +03:00
|
|
|
|
All the arguments are optional!
|
2017-05-25 02:59:10 +03:00
|
|
|
|
|
2017-05-25 06:45:45 +03:00
|
|
|
|
* Mob: ID of mob to spawn (default: mobs_mc:pig)
|
2017-05-25 02:59:10 +03:00
|
|
|
|
* MinLight: Minimum light to spawn (default: 0)
|
|
|
|
|
* MaxLight: Maximum light to spawn (default: 15)
|
2017-05-25 05:47:08 +03:00
|
|
|
|
* MaxMobsInArea: How many mobs are allowed in the area around the spawner (default: 4)
|
|
|
|
|
* PlayerDistance: Spawn mobs only if a player is within this distance; 0 to disable (default: 15)
|
2017-05-25 02:59:10 +03:00
|
|
|
|
* YOffset: Y offset to spawn mobs; 0 to disable (default: 0)
|
|
|
|
|
]]
|
|
|
|
|
|
2017-05-25 03:15:24 +03:00
|
|
|
|
function mcl_monster_spawner.setup_spawner(pos, Mob, MinLight, MaxLight, MaxMobsInArea, PlayerDistance, YOffset)
|
2017-05-25 02:59:10 +03:00
|
|
|
|
-- Activate monster spawner and disable editing functionality
|
2017-05-25 06:45:45 +03:00
|
|
|
|
if Mob == nil then Mob = default_mob end
|
2017-05-25 02:59:10 +03:00
|
|
|
|
if MinLight == nil then MinLight = 0 end
|
2017-05-25 05:47:08 +03:00
|
|
|
|
if MaxLight == nil then MaxLight = 15 end
|
|
|
|
|
if MaxMobsInArea == nil then MaxMobsInArea = 4 end
|
|
|
|
|
if PlayerDistance == nil then PlayerDistance = 15 end
|
2017-05-25 02:59:10 +03:00
|
|
|
|
if YOffset == nil then YOffset = 0 end
|
|
|
|
|
local meta = minetest.get_meta(pos)
|
|
|
|
|
meta:set_string("Mob", Mob)
|
|
|
|
|
meta:set_int("MinLight", MinLight)
|
|
|
|
|
meta:set_int("MaxLight", MaxLight)
|
|
|
|
|
meta:set_int("MaxMobsInArea", MaxMobsInArea)
|
|
|
|
|
meta:set_int("PlayerDistance", PlayerDistance)
|
|
|
|
|
meta:set_int("YOffset", YOffset)
|
|
|
|
|
|
|
|
|
|
-- Create doll
|
2017-05-25 03:06:36 +03:00
|
|
|
|
local doll = minetest.add_entity({x=pos.x, y=pos.y-0.3, z=pos.z}, "mcl_monster_spawner:doll")
|
2017-05-25 02:59:10 +03:00
|
|
|
|
set_doll_properties(doll, Mob)
|
2017-05-25 06:37:13 +03:00
|
|
|
|
|
|
|
|
|
-- Start spawning very soon
|
|
|
|
|
local t = minetest.get_node_timer(pos)
|
|
|
|
|
t:start(2)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- Spawn monsters around pos
|
|
|
|
|
-- NOTE: The node is timer-based, rather than ABM-based.
|
|
|
|
|
local spawn_monsters = function(pos, elapsed)
|
|
|
|
|
|
|
|
|
|
-- get meta
|
|
|
|
|
local meta = minetest.get_meta(pos)
|
|
|
|
|
|
|
|
|
|
-- get settings
|
|
|
|
|
local mob = meta:get_string("Mob")
|
|
|
|
|
local mlig = meta:get_int("MinLight")
|
|
|
|
|
local xlig = meta:get_int("MaxLight")
|
|
|
|
|
local num = meta:get_int("MaxMobsInArea")
|
|
|
|
|
local pla = meta:get_int("PlayerDistance")
|
|
|
|
|
local yof = meta:get_int("YOffset")
|
|
|
|
|
|
|
|
|
|
-- if amount is 0 then do nothing
|
|
|
|
|
if num == 0 then
|
|
|
|
|
return
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- are we spawning a registered mob?
|
|
|
|
|
if not mobs.spawning_mobs[mob] then
|
|
|
|
|
minetest.log("error", "[mobs] Monster Spawner: Mob doesn't exist: "..mob)
|
|
|
|
|
return
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- check objects inside 8×8 area around spawner
|
|
|
|
|
local objs = minetest.get_objects_inside_radius(pos, 8)
|
|
|
|
|
local count = 0
|
|
|
|
|
local ent = nil
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
local timer = minetest.get_node_timer(pos)
|
|
|
|
|
|
|
|
|
|
-- spawn mob if player detected and in range
|
|
|
|
|
if pla > 0 then
|
|
|
|
|
|
|
|
|
|
local in_range = 0
|
|
|
|
|
local objs = minetest.get_objects_inside_radius(pos, pla)
|
|
|
|
|
|
|
|
|
|
for _,oir in pairs(objs) do
|
|
|
|
|
|
|
|
|
|
if oir:is_player() then
|
|
|
|
|
|
|
|
|
|
in_range = 1
|
|
|
|
|
|
|
|
|
|
break
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- player not found
|
|
|
|
|
if in_range == 0 then
|
|
|
|
|
-- Try again quickly
|
|
|
|
|
timer:start(2)
|
|
|
|
|
return
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- count mob objects of same type in area
|
|
|
|
|
for k, obj in ipairs(objs) do
|
|
|
|
|
|
|
|
|
|
ent = obj:get_luaentity()
|
|
|
|
|
|
|
|
|
|
if ent and ent.name and ent.name == mob then
|
|
|
|
|
count = count + 1
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- Are there too many of same type? then fail
|
|
|
|
|
if count >= num then
|
|
|
|
|
timer:start(math.random(5, 20))
|
|
|
|
|
return
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- find air blocks within 8×3×8 nodes of spawner
|
|
|
|
|
local air = minetest.find_nodes_in_area(
|
|
|
|
|
{x = pos.x - 4, y = pos.y - 1 + yof, z = pos.z - 4},
|
|
|
|
|
{x = pos.x + 4, y = pos.y + 1 + yof, z = pos.z + 4},
|
|
|
|
|
{"air"})
|
|
|
|
|
|
|
|
|
|
-- spawn up to 4 mobs in random air blocks
|
|
|
|
|
if air then
|
|
|
|
|
for a=1, 4 do
|
|
|
|
|
if #air <= 0 then
|
|
|
|
|
-- We're out of space! Stop spawning
|
|
|
|
|
break
|
|
|
|
|
end
|
|
|
|
|
local air_index = math.random(#air)
|
|
|
|
|
local pos2 = air[air_index]
|
|
|
|
|
local lig = minetest.get_node_light(pos2) or 0
|
|
|
|
|
|
|
|
|
|
pos2.y = pos2.y + 0.5
|
|
|
|
|
|
|
|
|
|
-- only if light levels are within range
|
|
|
|
|
if lig >= mlig and lig <= xlig then
|
|
|
|
|
minetest.add_entity(pos2, mob)
|
|
|
|
|
end
|
|
|
|
|
table.remove(air, air_index)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- Spawn attempt done. Next spawn attempt much later
|
|
|
|
|
timer:start(math.random(10, 39.95))
|
|
|
|
|
|
2017-05-25 02:59:10 +03:00
|
|
|
|
end
|
|
|
|
|
|
2017-05-25 03:06:36 +03:00
|
|
|
|
minetest.register_node("mcl_monster_spawner:spawner", {
|
2017-01-16 18:40:08 +02:00
|
|
|
|
tiles = {"mob_spawner.png"},
|
|
|
|
|
drawtype = "glasslike",
|
|
|
|
|
paramtype = "light",
|
2017-03-20 21:27:10 +02:00
|
|
|
|
sunlight_propagates = true,
|
2017-01-16 18:40:08 +02:00
|
|
|
|
walkable = true,
|
2017-01-17 00:58:09 +02:00
|
|
|
|
description = S("Monster Spawner"),
|
2017-03-11 20:05:29 +02:00
|
|
|
|
_doc_items_longdesc = S("A monster spawner is a block which regularily causes monsters and animals to appear around it."),
|
2017-03-11 06:34:58 +02:00
|
|
|
|
groups = {pickaxey=1, not_in_creative_inventory = 1, material_stone=1},
|
2017-03-11 17:36:05 +02:00
|
|
|
|
is_ground_content = false,
|
2017-02-01 22:09:24 +02:00
|
|
|
|
drop = "",
|
2017-01-16 18:40:08 +02:00
|
|
|
|
|
2017-05-25 06:45:45 +03:00
|
|
|
|
on_construct = mcl_monster_spawner.setup_spawner,
|
2017-05-25 01:59:41 +03:00
|
|
|
|
|
|
|
|
|
on_destruct = function(pos)
|
2017-05-25 02:59:10 +03:00
|
|
|
|
local meta = minetest.get_meta(pos)
|
2017-05-25 06:45:45 +03:00
|
|
|
|
local obj = find_doll(pos)
|
|
|
|
|
if obj then
|
|
|
|
|
obj:remove()
|
2017-05-25 02:59:10 +03:00
|
|
|
|
end
|
2017-01-16 18:40:08 +02:00
|
|
|
|
end,
|
|
|
|
|
|
2017-05-25 06:37:13 +03:00
|
|
|
|
on_timer = spawn_monsters,
|
|
|
|
|
|
2017-01-16 18:40:08 +02:00
|
|
|
|
on_receive_fields = function(pos, formname, fields, sender)
|
|
|
|
|
|
|
|
|
|
if not fields.text or fields.text == "" then
|
|
|
|
|
return
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
local meta = minetest.get_meta(pos)
|
|
|
|
|
local comm = fields.text:split(" ")
|
|
|
|
|
local name = sender:get_player_name()
|
|
|
|
|
|
|
|
|
|
if minetest.is_protected(pos, name) then
|
|
|
|
|
minetest.record_protection_violation(pos, name)
|
|
|
|
|
return
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
local mob = comm[1] -- mob to spawn
|
|
|
|
|
local mlig = tonumber(comm[2]) -- min light
|
|
|
|
|
local xlig = tonumber(comm[3]) -- max light
|
|
|
|
|
local num = tonumber(comm[4]) -- total mobs in area
|
|
|
|
|
local pla = tonumber(comm[5]) -- player distance (0 to disable)
|
|
|
|
|
local yof = tonumber(comm[6]) or 0 -- Y offset to spawn mob
|
|
|
|
|
|
|
|
|
|
if mob and mob ~= "" and mobs.spawning_mobs[mob] == true
|
|
|
|
|
and num and num >= 0 and num <= 10
|
|
|
|
|
and mlig and mlig >= 0 and mlig <= 15
|
|
|
|
|
and xlig and xlig >= 0 and xlig <= 15
|
|
|
|
|
and pla and pla >=0 and pla <= 20
|
|
|
|
|
and yof and yof > -10 and yof < 10 then
|
|
|
|
|
|
2017-05-25 03:15:24 +03:00
|
|
|
|
mcl_monster_spawner.setup_spawner(pos, mob, mlig, xlig, num, pla, yof)
|
2017-01-16 18:40:08 +02:00
|
|
|
|
else
|
|
|
|
|
minetest.chat_send_player(name, S("Mob Spawner settings failed!"))
|
|
|
|
|
minetest.chat_send_player(name,
|
2017-05-25 04:09:02 +03:00
|
|
|
|
S("Syntax: name min_light[0-14] max_light[0-14] max_mobs_in_area[0 to disable] distance[1-20] y_offset[-10 to 10]"))
|
2017-01-16 18:40:08 +02:00
|
|
|
|
end
|
|
|
|
|
end,
|
2017-02-11 19:46:23 +02:00
|
|
|
|
sounds = mcl_sounds.node_sound_metal_defaults(),
|
2017-02-22 17:22:28 +02:00
|
|
|
|
_mcl_blast_resistance = 25,
|
2017-02-27 19:32:35 +02:00
|
|
|
|
_mcl_hardness = 5,
|
2017-01-16 18:40:08 +02:00
|
|
|
|
})
|
|
|
|
|
|
2017-05-25 01:59:41 +03:00
|
|
|
|
-- Mob spawner doll (rotating icon inside cage)
|
|
|
|
|
|
2017-05-25 03:06:36 +03:00
|
|
|
|
local doll_def = {
|
2017-05-25 01:59:41 +03:00
|
|
|
|
hp_max = 1,
|
|
|
|
|
physical = true,
|
|
|
|
|
collisionbox = {0,0,0,0,0,0},
|
|
|
|
|
visual = "mesh",
|
|
|
|
|
makes_footstep_sound = false,
|
|
|
|
|
timer = 0,
|
|
|
|
|
automatic_rotate = math.pi * 2.9,
|
|
|
|
|
|
|
|
|
|
_mob = default_mob, -- name of the mob this doll represents
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-25 03:06:36 +03:00
|
|
|
|
doll_def.get_staticdata = function(self)
|
2017-05-25 01:59:41 +03:00
|
|
|
|
return self._mob
|
|
|
|
|
end
|
|
|
|
|
|
2017-05-25 03:06:36 +03:00
|
|
|
|
doll_def.on_activate = function(self, staticdata, dtime_s)
|
2017-05-25 01:59:41 +03:00
|
|
|
|
local mob = staticdata
|
2017-05-25 02:14:18 +03:00
|
|
|
|
if mob == "" or mob == nil then
|
|
|
|
|
mob = default_mob
|
2017-05-25 01:59:41 +03:00
|
|
|
|
end
|
2017-05-25 02:14:18 +03:00
|
|
|
|
set_doll_properties(self.object, mob)
|
2017-05-25 01:59:41 +03:00
|
|
|
|
self.object:setvelocity({x=0, y=0, z=0})
|
|
|
|
|
self.object:setacceleration({x=0, y=0, z=0})
|
|
|
|
|
self.object:set_armor_groups({immortal=1})
|
|
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
2017-05-25 03:06:36 +03:00
|
|
|
|
doll_def.on_step = function(self, dtime)
|
2017-05-25 01:59:41 +03:00
|
|
|
|
-- Check if spawner is still present. If not, delete the entity
|
|
|
|
|
self.timer = self.timer + 0.01
|
|
|
|
|
local n = minetest.get_node_or_nil(self.object:getpos())
|
|
|
|
|
if self.timer > 1 then
|
2017-05-25 03:06:36 +03:00
|
|
|
|
if n and n.name and n.name ~= "mcl_monster_spawner:spawner" then
|
2017-05-25 01:59:41 +03:00
|
|
|
|
self.object:remove()
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2017-05-25 03:06:36 +03:00
|
|
|
|
doll_def.on_punch = function(self, hitter) end
|
2017-05-25 01:59:41 +03:00
|
|
|
|
|
2017-05-25 03:06:36 +03:00
|
|
|
|
minetest.register_entity("mcl_monster_spawner:doll", doll_def)
|
2017-05-25 01:59:41 +03:00
|
|
|
|
|
|
|
|
|
|
2017-05-25 00:12:13 +03:00
|
|
|
|
|