Minetest-WorldEditAdditions/worldeditadditions_commands/commands/selectors/sfactor.lua

88 lines
3.7 KiB
Lua
Raw Permalink Normal View History

2022-09-18 21:20:04 +00:00
local wea = worldeditadditions
local wea_c = worldeditadditions_core
local Vector3 = wea_c.Vector3
2021-06-27 21:41:16 +00:00
-- ███████ ███████ █████ ██████ ████████ ██████ ██████
-- ██ ██ ██ ██ ██ ██ ██ ██ ██ ██
-- ███████ █████ ███████ ██ ██ ██ ██ ██████
-- ██ ██ ██ ██ ██ ██ ██ ██ ██ ██
-- ███████ ██ ██ ██ ██████ ██ ██████ ██ ██
worldeditadditions_core.register_command("sfactor", {
2021-06-27 21:41:16 +00:00
params = "<mode> <factor> [<target=xz>]",
description = "Make the length of one or more target axes of the current selection to be multiple(s) of <factor>.",
privs = { worldedit = true },
require_pos = 2,
parse = function(params_text)
2022-09-18 21:20:04 +00:00
local parts = wea_c.split(params_text, "%s+", false)
2021-06-27 21:41:16 +00:00
if #parts < 2 then
return false, "Error: Not enough arguments. Expected \"<mode> <factor> [<target>]\"."
end
2022-09-18 21:20:04 +00:00
local mode, fac, targ = wea_c.table.unpack(parts)
local modeSet = wea_c.table.makeset {"grow", "shrink", "avg"}
2021-06-27 21:41:16 +00:00
-- Mode parsing
if mode == "average" then -- If mode is average set to avg
mode = "avg"
elseif not modeSet[mode] then -- If mode is invalid throw error
return false, "Error: Invalid <mode> \""..mode.."\". Expected \"grow\", \"shrink\", or \"average\"/\"avg\"."
end
-- Factor parsing
local factest = tonumber(fac)
if not factest then
return false, "Error: Invalid <factor> \""..fac.."\". Expected a number."
elseif factest < 2 then
return false, "Error: <factor> is too low. Expected a number equal to or greater than 2."
else
fac = math.floor(factest+0.5)
end
-- Target parsing
if not targ then -- If no target set to default (xz)
targ = "xz"
elseif targ:match("[xyz]+") then -- ensure correct target syntax
2022-09-18 21:20:04 +00:00
targ = table.concat(wea_c.tochars(targ:match("[xyz]+"),true,true))
2021-06-27 21:41:16 +00:00
else
return false, "Error: Invalid <target> \""..targ.."\". Expected \"x\" and or \"y\" and or \"z\"."
end
return true, mode, fac, targ
end,
func = function(name, mode, fac, targ)
2022-09-18 21:20:04 +00:00
local pos1, pos2 = Vector3.clone(worldedit.pos1[name]), Vector3.clone(worldedit.pos2[name])
local delta = pos2 - pos1 -- local delta equation: Vd(a) = V2(a) - V1(a)
2021-06-27 21:41:16 +00:00
local _tl = #targ -- Get targ length as a variable incase mode is "average"/"avg"
2022-09-18 21:20:04 +00:00
local targ = wea_c.tocharset(targ) -- Break up targ string into set table
2021-06-27 21:41:16 +00:00
local _m = 0 -- _m is the container to hold the average of the axes in targ
-- set _m to the max, min or mean of the target axes depending on mode (_tl is the length of targ) or base if it exists
if mode == "avg" then
for k,v in pairs(targ) do _m = _m + math.abs(delta[k]) end
_m = _m / _tl
end
2022-09-18 21:20:04 +00:00
-- Equation: round(delta[<axis>] / factor) * factor
local eval = function(int,fac_inner)
local tmp, abs, neg = int / fac_inner, math.abs(int), int < 0
2021-06-27 21:41:16 +00:00
if mode == "avg" then
if int > _m then int = math.floor(abs / fac_inner) * fac_inner
else int = math.ceil(abs / fac_inner) * fac_inner end
elseif mode == "shrink" then int = math.floor(abs / fac_inner) * fac_inner
else int = math.ceil(abs / fac_inner) * fac_inner end
2021-06-27 21:41:16 +00:00
if int < fac_inner then int = fac_inner end -- Ensure selection doesn't collapse to 0
2021-06-27 22:11:00 +00:00
if neg then int = int * -1 end -- Ensure correct facing direction
2021-06-27 21:41:16 +00:00
return int
end
for k,v in pairs(targ) do delta[k] = eval(delta[k],fac) end
2022-09-18 21:20:04 +00:00
worldedit.pos2[name] = pos1 + delta
2021-06-27 21:41:16 +00:00
worldedit.mark_pos2(name)
2022-09-18 21:20:04 +00:00
return true, "position 2 set to "..pos2
2021-06-27 21:41:16 +00:00
end
})