2020-09-20 17:03:29 +00:00
|
|
|
-- ███████ ██ ██ ██ ██████ ███████ ███████
|
|
|
|
-- ██ ██ ██ ██ ██ ██ ██ ██
|
|
|
|
-- █████ ██ ██ ██ ██████ ███████ █████
|
|
|
|
-- ██ ██ ██ ██ ██ ██ ██
|
|
|
|
-- ███████ ███████ ███████ ██ ██ ███████ ███████
|
2018-06-09 12:05:09 +00:00
|
|
|
|
2020-05-10 22:29:49 +00:00
|
|
|
function worldeditadditions.ellipsoid(position, radius, target_node, hollow)
|
2018-06-09 12:05:09 +00:00
|
|
|
-- position = { x, y, z }
|
2018-06-09 19:02:30 +00:00
|
|
|
local hollow_inner_radius = {
|
|
|
|
x = radius.x - 1,
|
|
|
|
y = radius.y - 1,
|
|
|
|
z = radius.z - 1
|
|
|
|
}
|
2018-06-09 12:05:09 +00:00
|
|
|
|
|
|
|
-- Fetch the nodes in the specified area
|
|
|
|
-- OPTIMIZE: We should be able to calculate a more efficient box-area here
|
2018-06-09 12:10:42 +00:00
|
|
|
local manip, area = worldedit.manip_helpers.init_radius(position, math.max(radius.x, radius.y, radius.z))
|
2018-06-09 12:05:09 +00:00
|
|
|
local data = manip:get_data()
|
|
|
|
|
|
|
|
local node_id = minetest.get_content_id(target_node)
|
|
|
|
local node_id_air = minetest.get_content_id("air")
|
|
|
|
|
|
|
|
local stride_z, stride_y = area.zstride, area.ystride
|
|
|
|
|
2018-06-09 12:10:42 +00:00
|
|
|
local count = 0 -- The number of nodes replaced
|
|
|
|
|
2018-06-09 18:38:23 +00:00
|
|
|
local idx_z_base = area:index(position.x - radius.x, position.y - radius.y, position.z - radius.z) -- initial z offset
|
2018-06-09 12:05:09 +00:00
|
|
|
for z = -radius.z, radius.z do
|
|
|
|
|
|
|
|
local idx_y_base = idx_z_base
|
2018-06-09 12:36:25 +00:00
|
|
|
for y = -radius.y, radius.y do
|
2018-06-09 12:05:09 +00:00
|
|
|
|
|
|
|
local i = idx_y_base
|
2018-06-09 12:36:25 +00:00
|
|
|
for x = -radius.x, radius.x do
|
2018-06-09 12:05:09 +00:00
|
|
|
|
|
|
|
-- If we're inside the ellipse, then fill it in
|
2018-06-09 18:35:32 +00:00
|
|
|
local x_comp, y_comp, z_comp = x/radius.x, y/radius.y, z/radius.z
|
2018-06-09 19:02:30 +00:00
|
|
|
local ellipsoid_dist = x_comp*x_comp + y_comp*y_comp + z_comp*z_comp
|
|
|
|
if ellipsoid_dist <= 1 then
|
|
|
|
local place_ok = not hollow;
|
2018-06-09 18:35:32 +00:00
|
|
|
|
2018-06-09 19:02:30 +00:00
|
|
|
if not place_ok then
|
|
|
|
-- It must be hollow! Do some additional calculations.
|
|
|
|
local hx_comp = x/hollow_inner_radius.x
|
|
|
|
local hy_comp = y/hollow_inner_radius.y
|
|
|
|
local hz_comp = z/hollow_inner_radius.z
|
|
|
|
|
|
|
|
-- It's only ok to place it if it's outside our inner ellipse
|
|
|
|
place_ok = hx_comp*hx_comp + hy_comp*hy_comp + hz_comp*hz_comp >= 1
|
|
|
|
end
|
2018-06-09 18:35:32 +00:00
|
|
|
|
2018-06-09 19:02:30 +00:00
|
|
|
if place_ok then
|
|
|
|
data[i] = node_id
|
|
|
|
count = count + 1
|
|
|
|
end
|
2018-06-09 12:05:09 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
i = i + 1
|
|
|
|
end
|
2018-06-09 12:36:25 +00:00
|
|
|
idx_y_base = idx_y_base + stride_y
|
2018-06-09 12:05:09 +00:00
|
|
|
|
|
|
|
end
|
2018-06-09 12:36:25 +00:00
|
|
|
idx_z_base = idx_z_base + stride_z
|
2018-06-09 12:05:09 +00:00
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
-- Save the modified nodes back to disk & return
|
|
|
|
worldedit.manip_helpers.finish(manip, data)
|
|
|
|
|
2018-06-09 12:10:42 +00:00
|
|
|
return count
|
2018-06-09 12:05:09 +00:00
|
|
|
end
|