1
0
mirror of https://codeberg.org/tenplus1/mobs_redo.git synced 2025-01-09 17:30:21 +01:00
mobs_redo/spawner.lua

217 lines
5.1 KiB
Lua
Raw Normal View History

2016-04-15 16:51:04 +02:00
2023-08-13 11:07:09 +02:00
local S = mobs.translate
2016-06-11 10:47:43 +02:00
2023-07-26 18:43:40 +02:00
-- are we a real player ?
local function is_player(player)
if player and type(player) == "userdata" and minetest.is_player(player) then
return true
end
end
2024-03-08 16:04:58 +01:00
local square = math.sqrt
2023-07-26 18:43:40 +02:00
local get_distance = function(a, b)
if not a or not b then return 50 end -- nil check and default distance
local x, y, z = a.x - b.x, a.y - b.y, a.z - b.z
return square(x * x + y * y + z * z)
end
2023-07-26 18:43:40 +02:00
-- mob spawner
2020-07-01 16:14:31 +02:00
local spawner_default = "mobs_animal:pumba 10 15 0 0 0"
minetest.register_node("mobs:spawner", {
tiles = {"mob_spawner.png"},
drawtype = "glasslike",
paramtype = "light",
walkable = true,
2016-06-11 10:47:43 +02:00
description = S("Mob Spawner"),
2024-01-29 17:01:59 +01:00
groups = {cracky = 1, pickaxey = 3},
is_ground_content = false,
2024-01-29 17:01:59 +01:00
_mcl_hardness = 1,
_mcl_blast_resistance = 5,
on_construct = function(pos)
local meta = minetest.get_meta(pos)
2020-07-01 16:14:31 +02:00
-- setup formspec
local head = S("(mob name) (min light) (max light) (amount)"
.. " (player distance) (Y offset)")
-- text entry formspec
2023-08-13 11:07:09 +02:00
meta:set_string("formspec", "size[10,3.5]"
2020-07-01 16:14:31 +02:00
.. "label[0.15,0.5;" .. minetest.formspec_escape(head) .. "]"
.. "field[1,2.5;8.5,0.8;text;" .. S("Command:")
2020-05-21 21:38:08 +02:00
.. ";${command}]")
2020-07-01 16:14:31 +02:00
2016-06-11 10:47:43 +02:00
meta:set_string("infotext", S("Spawner Not Active (enter settings)"))
meta:set_string("command", spawner_default)
end,
on_right_click = function(pos, placer)
if minetest.is_protected(pos, placer:get_player_name()) then
return
end
end,
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)
2016-05-21 20:35:24 +02:00
local yof = tonumber(comm[6]) or 0 -- Y offset to spawn mob
2020-07-01 16:14:31 +02:00
if mob and mob ~= "" and mobs.spawning_mobs[mob]
and num and num >= 0 and num <= 10
and mlig and mlig >= 0 and mlig <= 15
and xlig and xlig >= 0 and xlig <= 15
2020-07-01 16:14:31 +02:00
and pla and pla >= 0 and pla <= 20
2016-05-21 20:35:24 +02:00
and yof and yof > -10 and yof < 10 then
meta:set_string("command", fields.text)
2016-06-11 10:47:43 +02:00
meta:set_string("infotext", S("Spawner Active (@1)", mob))
else
2016-06-11 10:47:43 +02:00
minetest.chat_send_player(name, S("Mob Spawner settings failed!"))
minetest.chat_send_player(name,
2020-07-01 16:14:31 +02:00
S("Syntax: “name min_light[0-14] max_light[0-14] max_mobs_in_area[0 to disable] player_distance[1-20] y_offset[-10 to 10]”"))
end
end
})
2017-02-10 13:04:57 +01:00
2017-10-09 12:59:01 +02:00
local max_per_block = tonumber(minetest.settings:get("max_objects_per_block") or 99)
2017-02-10 13:04:57 +01:00
-- spawner abm
minetest.register_abm({
2016-12-29 20:34:10 +01:00
label = "Mob spawner node",
nodenames = {"mobs:spawner"},
interval = 10,
2024-02-23 18:12:14 +01:00
chance = 4,
catch_up = false,
action = function(pos, node, active_object_count, active_object_count_wider)
2017-02-10 13:04:57 +01:00
-- return if too many entities already
if active_object_count_wider >= max_per_block then
return
end
-- get meta and command
local meta = minetest.get_meta(pos)
local comm = meta:get_string("command"):split(" ")
-- get settings from command
local mob = comm[1]
local mlig = tonumber(comm[2])
local xlig = tonumber(comm[3])
local num = tonumber(comm[4])
local pla = tonumber(comm[5]) or 0
local yof = tonumber(comm[6]) or 0
-- if amount is 0 then do nothing
if num == 0 then
return
end
2017-02-10 13:04:57 +01:00
-- are we spawning a registered mob?
if not mobs.spawning_mobs[mob] then
--print ("--- mob doesn't exist", mob)
return
end
2016-05-21 20:35:24 +02:00
-- check objects inside 9x9 area around spawner
local objs = minetest.get_objects_inside_radius(pos, 9)
local count = 0
2020-05-21 21:38:08 +02:00
local ent
-- count mob objects of same type in area
2020-05-21 21:38:08 +02:00
for _, obj in ipairs(objs) do
ent = obj:get_luaentity()
2017-02-10 13:04:57 +01:00
if ent and ent.name and ent.name == mob then
count = count + 1
end
end
-- is there too many of same type?
if count >= num then
return
end
2024-02-23 18:09:42 +01:00
-- when player distance above 0, spawn mob if player detected and in range
if pla > 0 then
2024-02-23 18:09:42 +01:00
local in_range, player
local players = minetest.get_connected_players()
2024-02-23 18:09:42 +01:00
for i = 1, #players do
2024-02-23 18:09:42 +01:00
player = players[i]
if get_distance(player:get_pos(), pos) <= pla then
2024-02-23 18:09:42 +01:00
in_range = true
break
end
end
-- player not found
2024-02-23 18:09:42 +01:00
if not in_range then
return
end
end
-- set medium mob usually spawns in (defaults to air)
local reg = minetest.registered_entities[mob].fly_in
if not reg or type(reg) == "string" then
reg = {(reg or "air")}
end
-- find air blocks within 5 nodes of spawner
local air = minetest.find_nodes_in_area(
{x = pos.x - 5, y = pos.y + yof, z = pos.z - 5},
{x = pos.x + 5, y = pos.y + yof, z = pos.z + 5}, reg)
-- spawn in random air block
if air and #air > 0 then
local pos2 = air[math.random(#air)]
local lig = minetest.get_node_light(pos2) or 0
pos2.y = pos2.y + 0.5
-- only if light levels are within range
2017-09-08 10:41:36 +02:00
if lig >= mlig and lig <= xlig
and minetest.registered_entities[mob] then
minetest.add_entity(pos2, mob)
end
end
end
})