mirror of
https://github.com/sbrl/Minetest-WorldEditAdditions.git
synced 2024-11-09 00:43:48 +01:00
db5d25d1de
A lot of the maths remains in the old style, but at least it doesn't take a zero-indexed table
86 lines
3.4 KiB
Lua
86 lines
3.4 KiB
Lua
local wea = worldeditadditions
|
|
local Vector3 = wea.Vector3
|
|
|
|
-- ██████ ██████ ███ ██ ██ ██ ██████ ██ ██ ██ ███████
|
|
-- ██ ██ ██ ████ ██ ██ ██ ██ ██ ██ ██ ██ ██
|
|
-- ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ █████
|
|
-- ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██
|
|
-- ██████ ██████ ██ ████ ████ ██████ ███████ ████ ███████
|
|
worldedit.register_command("convolve", {
|
|
params = "<kernel> [<width>[,<height>]] [<sigma>]",
|
|
description = "Advanced version of //smooth from we_env. Convolves over the defined region with the given kernel. Possible kernels: box, pascal, gaussian. The width & height (if specified) must be odd integers. If the height is not specified, it defaults to the width. gaussian should give the smoothest result, but the width & height must be identical. The sigma value is only applicable to gaussian kernels, and can be thought of as the 'smoothness' to apply.",
|
|
privs = { worldedit = true },
|
|
require_pos = 2,
|
|
parse = function(params_text)
|
|
if not params_text then params_text = "" end
|
|
|
|
-- local parts = wea.split(params_text, "%s+", false)
|
|
local parts = wea.split_shell(params_text)
|
|
|
|
local kernel_name = "gaussian"
|
|
local width = 5
|
|
local height = 5
|
|
local sigma = nil
|
|
|
|
if #parts >= 1 and #parts[1] > 0 then
|
|
kernel_name = parts[1]
|
|
end
|
|
if #parts >= 2 then
|
|
local parts_dimension = wea.split(parts[2], ",%s*", false)
|
|
width = tonumber(parts_dimension[1])
|
|
if not width then
|
|
return false, "Error: Invalid width (it must be a positive odd integer)."
|
|
end
|
|
if #parts_dimension >= 2 then
|
|
height = tonumber(parts_dimension[2])
|
|
if not height then
|
|
return false, "Error: Invalid height (it must be a positive odd integer)."
|
|
end
|
|
else
|
|
height = width
|
|
end
|
|
end
|
|
if #parts >= 3 then
|
|
sigma = tonumber(parts[3])
|
|
if not sigma then
|
|
return false, "Error: Invalid sigma value (it must be a valid number - floating point numbers are allowed)"
|
|
end
|
|
end
|
|
|
|
return true, kernel_name, math.floor(width), math.floor(height), sigma
|
|
end,
|
|
nodes_needed = function(name)
|
|
return worldedit.volume(worldedit.pos1[name], worldedit.pos2[name])
|
|
end,
|
|
func = function(name, kernel_name, kernel_width, kernel_height, sigma)
|
|
local start_time = wea.get_ms_time()
|
|
|
|
local success, kernel = wea.get_conv_kernel(kernel_name, kernel_width, kernel_height, sigma)
|
|
if not success then return success, kernel end
|
|
|
|
local kernel_size = Vector3.new(
|
|
kernel_height,
|
|
0,
|
|
kernel_width
|
|
)
|
|
|
|
local pos1, pos2 = Vector3.sort(
|
|
worldedit.pos1[name],
|
|
worldedit.pos2[name]
|
|
)
|
|
|
|
local stats
|
|
success, stats = wea.convolve(
|
|
pos1, pos2,
|
|
kernel, kernel_size
|
|
)
|
|
if not success then return success, stats end
|
|
|
|
local time_taken = wea.get_ms_time() - start_time
|
|
|
|
|
|
minetest.log("action", name.." used //convolve at "..pos1.." - "..pos2..", adding "..stats.added.." nodes and removing "..stats.removed.." nodes in "..time_taken.."s")
|
|
return true, "Added "..stats.added.." and removed "..stats.removed.." nodes in " .. wea.format.human_time(time_taken)
|
|
end
|
|
})
|