2023-05-24 19:39:20 +02:00
|
|
|
local texmod = {}
|
|
|
|
local metatable = {__index = texmod}
|
|
|
|
|
|
|
|
local function new(self)
|
|
|
|
return setmetatable(self, metatable)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- `texmod{...}` may be used to create texture modifiers, bypassing the checks
|
|
|
|
setmetatable(texmod, {__call = new})
|
|
|
|
|
|
|
|
-- Constructors / "generators"
|
|
|
|
|
|
|
|
function texmod.file(filename)
|
2023-05-27 14:27:49 +02:00
|
|
|
-- See `TEXTURENAME_ALLOWED_CHARS` in Minetest (`src/network/networkprotocol.h`)
|
|
|
|
assert(not filename:find"[^%w_.-]", "invalid characters in file name")
|
2023-05-24 19:39:20 +02:00
|
|
|
return new{
|
2023-05-24 21:09:24 +02:00
|
|
|
type = "file",
|
2023-05-24 19:39:20 +02:00
|
|
|
filename = filename
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod.png(data)
|
|
|
|
assert(type(data) == "string")
|
|
|
|
return new{
|
|
|
|
type = "png",
|
|
|
|
data = data
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod.combine(w, h, blits)
|
|
|
|
assert(w % 1 == 0 and w > 0)
|
|
|
|
assert(h % 1 == 0 and h > 0)
|
|
|
|
for _, blit in ipairs(blits) do
|
|
|
|
assert(blit.x % 1 == 0)
|
|
|
|
assert(blit.y % 1 == 0)
|
|
|
|
assert(blit.texture)
|
|
|
|
end
|
|
|
|
return new{
|
|
|
|
type = "combine",
|
|
|
|
w = w,
|
|
|
|
h = h,
|
|
|
|
blits = blits
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod.inventorycube(top, left, right)
|
|
|
|
return new{
|
|
|
|
type = "inventorycube",
|
|
|
|
top = top,
|
|
|
|
left = left,
|
|
|
|
right = right
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Methods / "modifiers"
|
|
|
|
|
|
|
|
function texmod:overlay(overlay)
|
|
|
|
return new{
|
|
|
|
type = "overlay",
|
|
|
|
base = self,
|
|
|
|
over = overlay
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:brighten()
|
|
|
|
return new{
|
|
|
|
type = "brighten",
|
|
|
|
base = self,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:noalpha()
|
|
|
|
return new{
|
|
|
|
type = "noalpha",
|
|
|
|
base = self
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:resize(w, h)
|
|
|
|
assert(w % 1 == 0 and w > 0)
|
|
|
|
assert(h % 1 == 0 and h > 0)
|
|
|
|
return new{
|
|
|
|
type = "resize",
|
|
|
|
base = self,
|
|
|
|
w = w,
|
|
|
|
h = h,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
local function assert_uint8(num)
|
|
|
|
assert(num % 1 == 0 and num >= 0 and num <= 0xFF)
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:makealpha(r, g, b)
|
|
|
|
assert_uint8(r); assert_uint8(g); assert_uint8(b)
|
|
|
|
return new{
|
|
|
|
type = "makealpha",
|
|
|
|
base = self,
|
|
|
|
r = r, g = g, b = b
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:opacity(ratio)
|
|
|
|
assert_uint8(ratio)
|
|
|
|
return new{
|
|
|
|
type = "opacity",
|
|
|
|
base = self,
|
|
|
|
ratio = ratio
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
local function tobool(val)
|
|
|
|
return not not val
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:invert(channels --[[set with keys "r", "g", "b", "a"]])
|
|
|
|
return new{
|
|
|
|
type = "invert",
|
|
|
|
base = self,
|
|
|
|
r = tobool(channels.r),
|
|
|
|
g = tobool(channels.g),
|
|
|
|
b = tobool(channels.b),
|
|
|
|
a = tobool(channels.a)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:flip(flip_axis --[["x" or "y"]])
|
2023-05-27 14:24:54 +02:00
|
|
|
return self:transform(assert(
|
|
|
|
(flip_axis == "x" and "fx")
|
|
|
|
or (flip_axis == "y" and "fy")
|
|
|
|
or (not flip_axis and "i")))
|
2023-05-24 19:39:20 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:rotate(deg)
|
2023-05-27 14:24:54 +02:00
|
|
|
assert(deg % 90 == 0)
|
2023-05-24 19:39:20 +02:00
|
|
|
deg = deg % 360
|
2023-05-27 14:24:54 +02:00
|
|
|
return self:transform(("r%d"):format(deg))
|
2023-05-24 19:39:20 +02:00
|
|
|
end
|
|
|
|
|
2023-05-27 14:24:54 +02:00
|
|
|
-- D4 group transformations (see https://proofwiki.org/wiki/Definition:Dihedral_Group_D4),
|
|
|
|
-- represented using indices into a table of matrices
|
|
|
|
-- TODO (...) try to come up with a more elegant solution
|
|
|
|
do
|
|
|
|
-- Matrix multiplication for composition: First applies a, then b <=> b * a
|
|
|
|
local function mat_2x2_compose(a, b)
|
|
|
|
local a_1_1, a_1_2, a_2_1, a_2_2 = unpack(a)
|
|
|
|
local b_1_1, b_1_2, b_2_1, b_2_2 = unpack(b)
|
|
|
|
return {
|
|
|
|
a_1_1 * b_1_1 + a_2_1 * b_1_2, a_1_2 * b_1_1 + a_2_2 * b_1_2;
|
|
|
|
a_1_1 * b_2_1 + a_2_1 * b_2_2, a_1_2 * b_2_1 + a_2_2 * b_2_2
|
|
|
|
}
|
|
|
|
end
|
|
|
|
local r90 ={
|
|
|
|
0, -1;
|
|
|
|
1, 0
|
|
|
|
}
|
|
|
|
local fx = {
|
|
|
|
-1, 0;
|
|
|
|
0, 1
|
|
|
|
}
|
|
|
|
local fy = {
|
|
|
|
1, 0;
|
|
|
|
0, -1
|
2023-05-24 19:39:20 +02:00
|
|
|
}
|
2023-05-27 14:24:54 +02:00
|
|
|
local r180 = mat_2x2_compose(r90, r90)
|
|
|
|
local r270 = mat_2x2_compose(r180, r90)
|
|
|
|
local fxr90 = mat_2x2_compose(fx, r90)
|
|
|
|
local fyr90 = mat_2x2_compose(fy, r90)
|
|
|
|
local transform_mats = {[0] = {1, 0; 0, 1}, r90, r180, r270, fx, fxr90, fy, fyr90}
|
|
|
|
local transform_idx_by_name = {i = 0, r90 = 1, r180 = 2, r270 = 3, fx = 4, fxr90 = 5, fy = 6, fyr90 = 7}
|
|
|
|
-- Lookup tables for getting the flipped axis / rotation angle
|
|
|
|
local flip_by_idx = {
|
|
|
|
[4] = "x",
|
|
|
|
[5] = "x",
|
|
|
|
[6] = "y",
|
|
|
|
[7] = "y",
|
|
|
|
}
|
|
|
|
local rot_by_idx = {
|
|
|
|
[1] = 90,
|
|
|
|
[2] = 180,
|
|
|
|
[3] = 270,
|
|
|
|
[5] = 90,
|
|
|
|
[7] = 90,
|
|
|
|
}
|
|
|
|
local idx_by_mat_2x2 = {}
|
|
|
|
local function transform_idx(mat)
|
|
|
|
-- note: assumes mat[i] in {-1, 0, 1}
|
|
|
|
return mat[1] + 3*(mat[2] + 3*(mat[3] + 3*mat[4]))
|
|
|
|
end
|
|
|
|
for i = 0, 7 do
|
|
|
|
idx_by_mat_2x2[transform_idx(transform_mats[i])] = i
|
|
|
|
end
|
|
|
|
-- Compute a multiplication table
|
|
|
|
local composition_idx = {}
|
|
|
|
local function ij_idx(i, j)
|
|
|
|
return i*8 + j
|
|
|
|
end
|
|
|
|
for i = 0, 7 do
|
|
|
|
for j = 0, 7 do
|
|
|
|
composition_idx[ij_idx(i, j)] = assert(idx_by_mat_2x2[
|
|
|
|
transform_idx(mat_2x2_compose(transform_mats[i], transform_mats[j]))])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
function texmod:transform(...)
|
|
|
|
if select("#", ...) == 0 then return self end
|
|
|
|
local idx = ...
|
|
|
|
if type(idx) == "string" then
|
|
|
|
idx = assert(transform_idx_by_name[idx:lower()])
|
|
|
|
end
|
|
|
|
local base = self
|
|
|
|
if self.type == "transform" then
|
|
|
|
-- Merge with a `^[transform` base image
|
|
|
|
assert(transform_mats[idx])
|
|
|
|
base = self.base
|
|
|
|
idx = composition_idx[ij_idx(self.idx, idx)]
|
|
|
|
end
|
|
|
|
assert(transform_mats[idx])
|
|
|
|
if idx == 0 then return base end -- identity
|
|
|
|
return new{
|
|
|
|
type = "transform",
|
|
|
|
base = base,
|
|
|
|
idx = idx,
|
|
|
|
-- Redundantly store this information for convenience. Do not modify!
|
|
|
|
flip_axis = flip_by_idx[idx],
|
|
|
|
rotation_deg = rot_by_idx[idx] or 0,
|
|
|
|
}:transform(select(2, ...))
|
|
|
|
end
|
2023-05-24 19:39:20 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:verticalframe(framecount, frame)
|
|
|
|
assert(framecount >= 1)
|
|
|
|
assert(frame >= 0)
|
|
|
|
return new{
|
|
|
|
type = "verticalframe",
|
|
|
|
base = self,
|
|
|
|
framecount = framecount,
|
|
|
|
frame = frame
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
local function crack(self, name, ...)
|
|
|
|
local tilecount, framecount, frame
|
|
|
|
if select("#", ...) == 2 then
|
|
|
|
tilecount, framecount, frame = 1, ...
|
|
|
|
else
|
|
|
|
assert(select("#", ...) == 3, "invalid number of arguments")
|
|
|
|
tilecount, framecount, frame = ...
|
|
|
|
end
|
|
|
|
assert(tilecount >= 1)
|
|
|
|
assert(framecount >= 1)
|
|
|
|
assert(frame >= 0)
|
|
|
|
return new{
|
|
|
|
type = name,
|
|
|
|
base = self,
|
|
|
|
tilecount = tilecount,
|
|
|
|
framecount = framecount,
|
|
|
|
frame = frame
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:crack(...)
|
|
|
|
return crack(self, "crack", ...)
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:cracko(...)
|
|
|
|
return crack(self, "cracko", ...)
|
|
|
|
end
|
|
|
|
texmod.crack_with_opacity = texmod.cracko
|
|
|
|
|
|
|
|
function texmod:sheet(w, h, x, y)
|
|
|
|
assert(w % 1 == 0 and w >= 1)
|
|
|
|
assert(h % 1 == 0 and h >= 1)
|
2023-05-28 12:30:13 +02:00
|
|
|
assert(x % 1 == 0 and x >= 0)
|
|
|
|
assert(y % 1 == 0 and y >= 0)
|
2023-05-24 19:39:20 +02:00
|
|
|
return new{
|
|
|
|
type = "sheet",
|
|
|
|
base = self,
|
|
|
|
w = w,
|
|
|
|
h = h,
|
|
|
|
x = x,
|
|
|
|
y = y
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
local colorspec = modlib.minetest.colorspec
|
|
|
|
|
|
|
|
function texmod:multiply(color)
|
|
|
|
return new{
|
|
|
|
type = "multiply",
|
|
|
|
base = self,
|
|
|
|
color = colorspec.from_any(color) -- copies a given colorspec
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:colorize(color, ratio)
|
|
|
|
color = colorspec.from_any(color) -- copies a given colorspec
|
|
|
|
if ratio == "alpha" then
|
|
|
|
assert(color.alpha or 0xFF == 0xFF)
|
|
|
|
else
|
2023-05-27 17:21:41 +02:00
|
|
|
ratio = ratio or color.alpha or 0xFF
|
2023-05-24 19:39:20 +02:00
|
|
|
assert_uint8(ratio)
|
|
|
|
if color.alpha == ratio then
|
|
|
|
ratio = nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return new{
|
|
|
|
type = "colorize",
|
|
|
|
base = self,
|
|
|
|
color = color,
|
|
|
|
ratio = ratio
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2023-05-28 12:56:19 +02:00
|
|
|
function texmod:mask(_mask)
|
2023-05-24 19:39:20 +02:00
|
|
|
return new{
|
|
|
|
type = "mask",
|
|
|
|
base = self,
|
2023-05-28 12:56:19 +02:00
|
|
|
_mask = _mask
|
2023-05-24 19:39:20 +02:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
function texmod:lowpart(percent, overlay)
|
|
|
|
assert(percent % 1 == 0 and percent >= 0 and percent <= 100)
|
|
|
|
return new{
|
|
|
|
type = "lowpart",
|
|
|
|
base = self,
|
|
|
|
percent = percent,
|
|
|
|
over = overlay
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2023-05-27 14:24:54 +02:00
|
|
|
return texmod, metatable
|