1
0
Fork 0

[mcl_mobs, mobs_mc] TEMP! Remove `goto` to run on RasbPI4, Oil_boi free to revert

formspec-v4
kay27 2021-04-10 21:15:04 +04:00 committed by Elias Fleckenstein
parent 2116b2b9d0
commit ea41c82834
2 changed files with 104 additions and 130 deletions

View File

@ -3,6 +3,7 @@ local get_node = minetest.get_node
local get_item_group = minetest.get_item_group local get_item_group = minetest.get_item_group
local get_node_light = minetest.get_node_light local get_node_light = minetest.get_node_light
local find_nodes_in_area_under_air = minetest.find_nodes_in_area_under_air local find_nodes_in_area_under_air = minetest.find_nodes_in_area_under_air
local get_biome_data = minetest.get_biome_data
local new_vector = vector.new local new_vector = vector.new
local math_random = math.random local math_random = math.random
local get_biome_name = minetest.get_biome_name local get_biome_name = minetest.get_biome_name
@ -541,16 +542,14 @@ if mobs_spawn then
timer = timer + dtime timer = timer + dtime
if timer >= 8 then if timer >= 8 then
timer = 0 timer = 0
for _,player in ipairs(minetest.get_connected_players()) do for _,player in pairs(minetest.get_connected_players()) do
for i = 1,math.random(3,8) do for i = 1,math_random(3,8) do
local player_pos = player:get_pos() local player_pos = player:get_pos()
local _,dimension = mcl_worlds.y_to_layer(player_pos.y) local _,dimension = mcl_worlds.y_to_layer(player_pos.y)
if dimension == "void" or dimension == "default" then if dimension ~= "void" and dimension ~= "default" then
goto continue -- ignore void and unloaded area
end
local min,max = decypher_limits(player_pos.y) local min,max = decypher_limits(player_pos.y)
@ -558,79 +557,47 @@ if mobs_spawn then
local spawning_position_list = find_nodes_in_area_under_air(new_vector(goal_pos.x,min,goal_pos.z), vector.new(goal_pos.x,max,goal_pos.z), {"group:solid", "group:water", "group:lava"}) local spawning_position_list = find_nodes_in_area_under_air(new_vector(goal_pos.x,min,goal_pos.z), vector.new(goal_pos.x,max,goal_pos.z), {"group:solid", "group:water", "group:lava"})
--couldn't find node if #spawning_position_list > 0 then --couldn't find node
if #spawning_position_list <= 0 then
goto continue
end
local spawning_position = spawning_position_list[math_random(1,#spawning_position_list)] local spawning_position = spawning_position_list[math_random(1,#spawning_position_list)]
--Prevent strange behavior/too close to player --Prevent strange behavior/too close to player
if not spawning_position or vector_distance(player_pos, spawning_position) < 15 then if spawning_position and vector_distance(player_pos, spawning_position) > 15 then
goto continue
end
local gotten_node = get_node(spawning_position).name local gotten_node = get_node(spawning_position).name
if not gotten_node or gotten_node == "air" then --skip air nodes if gotten_node and gotten_node ~= "air" then --skip air nodes
goto continue
end
local gotten_biome = minetest.get_biome_data(spawning_position) local gotten_biome = get_biome_data(spawning_position)
if not gotten_biome then if gotten_biome then --skip if in unloaded area
goto continue --skip if in unloaded area
end
gotten_biome = get_biome_name(gotten_biome.biome) --makes it easier to work with gotten_biome = get_biome_name(gotten_biome.biome) --makes it easier to work with
--grab random mob --grab random mob
local mob_def = spawn_dictionary[math.random(1,#spawn_dictionary)] local mob_def = spawn_dictionary[math.random(1,#spawn_dictionary)]
if not mob_def then if mob_def --skip if something ridiculous happens (nil mob def)
goto continue --skip if something ridiculous happens (nil mob def) and mob_def.dimension == dimension --skip if not correct dimension
end and biome_check(mob_def.biomes, gotten_biome) then --skip if not in correct biome
--skip if not correct dimension
if mob_def.dimension ~= dimension then
goto continue
end
--skip if not in correct biome
if not biome_check(mob_def.biomes, gotten_biome) then
goto continue
end
--add this so mobs don't spawn inside nodes --add this so mobs don't spawn inside nodes
spawning_position.y = spawning_position.y + 1 spawning_position.y = spawning_position.y + 1
if spawning_position.y < mob_def.min_height or spawning_position.y > mob_def.max_height then if spawning_position.y >= mob_def.min_height and spawning_position.y <= mob_def.max_height then
goto continue
end
--only need to poll for node light if everything else worked --only need to poll for node light if everything else worked
local gotten_light = get_node_light(spawning_position) local gotten_light = get_node_light(spawning_position)
--don't spawn if not in light limits --don't spawn if not in light limits
if gotten_light < mob_def.min_light or gotten_light > mob_def.max_light then if gotten_light >= mob_def.min_light and gotten_light <= mob_def.max_light then
goto continue
end
local is_water = get_item_group(gotten_node, "water") ~= 0 local is_water = get_item_group(gotten_node, "water") ~= 0
local is_lava = get_item_group(gotten_node, "lava") ~= 0 local is_lava = get_item_group(gotten_node, "lava") ~= 0
if mob_def.type_of_spawning == "ground" and is_water then if mob_def.type_of_spawning ~= "ground" or not (is_water or is_lava) then
goto continue
end
if mob_def.type_of_spawning == "ground" and is_lava then
goto continue
end
--finally do the heavy check (for now) of mobs in area --finally do the heavy check (for now) of mobs in area
if count_mobs(spawning_position, mob_def.spawn_class) >= mob_def.aoc then if count_mobs(spawning_position, mob_def.spawn_class) < mob_def.aoc then
goto continue
end
--adjust the position for water and lava mobs --adjust the position for water and lava mobs
if mob_def.type_of_spawning == "water" or mob_def.type_of_spawning == "lava" then if mob_def.type_of_spawning == "water" or mob_def.type_of_spawning == "lava" then
@ -639,8 +606,16 @@ if mobs_spawn then
--everything is correct, spawn mob --everything is correct, spawn mob
minetest.add_entity(spawning_position, mob_def.name) minetest.add_entity(spawning_position, mob_def.name)
end
::continue:: --this is a safety catch end
end
end
end
end
end
end
end
end
end end
end end
end end

View File

@ -346,8 +346,7 @@ mobs:register_mob("mobs_mc:enderman", {
--skip player if they have no data - log it --skip player if they have no data - log it
if not player_eye_height then if not player_eye_height then
minetest.log("error", "Enderman at location: ".. dump(enderpos).." has indexed a null player!") minetest.log("error", "Enderman at location: ".. dump(enderpos).." has indexed a null player!")
goto continue else
end
--calculate very quickly the exact location the player is looking --calculate very quickly the exact location the player is looking
--within the distance between the two "heads" (player and enderman) --within the distance between the two "heads" (player and enderman)
@ -367,8 +366,8 @@ mobs:register_mob("mobs_mc:enderman", {
self.provoked = "broke_contact" self.provoked = "broke_contact"
end end
end end
end
::continue:: -- this is a sweep over statement, this can be used to continue even when errors occurred
end end
end end
end end