local p = {}
local function format_it(tbl, name)
for i, v in ipairs(tbl) do
tbl[i] = string.format("%06x", tbl[i])
end
p[name] = tbl
end
local function mod(a, b)
return a - math.floor(a/b) * b
end
-- takes hex color as string, gives back table
local function colorTable(color)
local clen = string.len(color)
assert(clen >= 6 and clen <= 8)
assert(string.gmatch("^"..("[0-9a-fA-F]"):rep(6)..("[0-9a-fA-F]?"):rep(2).."$", color))
local ret = {}
ret["R"] = tonumber(color:sub(1,2), 16)
ret["G"] = tonumber(color:sub(3,4), 16)
ret["B"] = tonumber(color:sub(5,6), 16)
ret["A"] = tonumber(color:sub(7,8), 16) or 255
return ret
end
local black = colorTable("000000")
local white = colorTable("FFFFFF")
local function colorAsRGBA(colorTbl)
return string.format("rgba(%.2f,%.2f,%.2f,%.2f)",
colorTbl["R"], colorTbl["G"],
colorTbl["B"], colorTbl["A"])
end
-- simple mix of two colors. t_ means target. percent is as float (.4=40%)
local function colorMix(colorTbl, t_colorTbl, percent)
local ret = {}
local diff = {}
diff["R"] = t_colorTbl["R"] - colorTbl["R"]
diff["G"] = t_colorTbl["G"] - colorTbl["G"]
diff["B"] = t_colorTbl["B"] - colorTbl["B"]
diff["A"] = t_colorTbl["A"] - colorTbl["A"]
ret["R"] = colorTbl["R"] + (diff["R"] * percent)
ret["G"] = colorTbl["G"] + (diff["G"] * percent)
ret["B"] = colorTbl["B"] + (diff["B"] * percent)
ret["A"] = colorTbl["A"] + (diff["A"] * percent)
return ret
end
p.HTML = function(frame, page)
local args = require('Module:Arguments').getArgs(frame)
p.args = args
local ret = ''
local inp = args[1]
if not inp then return nil end
local steps = mw.ustring.len(inp)
local repeat_ = (args["repeat"] and tonumber(args["repeat"])) or 1
local newcolors = {}
if args.colors then
for k = 1, repeat_, 1 do
nc = mw.text.gsplit(args.colors, ",", true)
for c in nc do
newcolors[#newcolors+1] = tonumber(c:sub(2, -1), 16)
end
end
end
format_it(#newcolors > 0 and newcolors or
-- red orange yellow green blue indigo-violet
{0xFF0000, 0xFF7F00, 0xFFFF00, 0x00FF00, 0x0000FF, 0x8B00FF}, "roygbiv")
-- lua pattern from https://stackoverflow.com/questions/13235091/extract-the-first-letter-of-a-utf-8-string-with-lua
local i = 0
for c in inp:gmatch("[%z\1-\127\194-\244][\128-\191]*") do
local color = colorTable(p.roygbiv[mod(i, #p.roygbiv)+1])
local s_per = steps / #p.roygbiv
local j = (i / s_per)
local jf = math.floor(j)
if args.gradient then
local cidx = mod(jf, #p.roygbiv)+1
if mod(cidx, 2) == 0 then cidx = cidx + 1 end
if cidx > #p.roygbiv then cidx = #p.roygbiv end
color = colorTable(p.roygbiv[cidx])
local tidx = mod(jf+1, #p.roygbiv)+1
if mod(tidx, 2) == 1 then tidx = tidx + 1 end
if tidx > #p.roygbiv then tidx = cidx end
local tcolor = colorTable(p.roygbiv[tidx])
color = colorMix(color, tcolor, mod(j, 1))
end
local subdued = nil
if args.theme then
subdued = args.theme:match("^subdued(%d+)%%$")
end
if subdued then
color = colorMix(color, black, tonumber(subdued) / 100)
end
assert(color)
if args.bg ~= "y" then
local rgba = colorAsRGBA(color)
ret = (ret .. "<span style='" .. (args.bg == "black" and "background-color:black;" or "")
.. "color:" .. rgba .. ";'>" .. c .."</span>")
else
local rgba = colorAsRGBA(args.fgcolor and colorTable(args.fgcolor:sub(2, -1)) or black)
ret = (ret .. "<span style='background-color:" .. colorAsRGBA(color) .. ";"
.. "color:" .. rgba .. "'>" .. c .."</span>")
end
i = i + 1
end
return ret
end
return p