Merge 0.10->trunk
[prosody.git] / util / termcolours.lua
index ef978364f93ff1ab35c1c348ed0d44d92d82188d..53633b45b1b5744d5fbe94b1d47078e77b6da62b 100644 (file)
@@ -5,6 +5,8 @@
 -- This project is MIT/X11 licensed. Please see the
 -- COPYING file in the source package for more information.
 --
+--
+-- luacheck: ignore 213/i
 
 
 local t_concat, t_insert = table.concat, table.insert;
@@ -12,6 +14,10 @@ local char, format = string.char, string.format;
 local tonumber = tonumber;
 local ipairs = ipairs;
 local io_write = io.write;
+local m_floor = math.floor;
+local type = type;
+local setmetatable = setmetatable;
+local pairs = pairs;
 
 local windows;
 if os.getenv("WINDIR") then
@@ -19,7 +25,7 @@ if os.getenv("WINDIR") then
 end
 local orig_color = windows and windows.get_consolecolor and windows.get_consolecolor();
 
-module "termcolours"
+local _ENV = nil;
 
 local stylemap = {
                        reset = 0; bright = 1, dim = 2, underscore = 4, blink = 5, reverse = 7, hidden = 8;
@@ -45,7 +51,7 @@ local cssmap = {
 };
 
 local fmt_string = char(0x1B).."[%sm%s"..char(0x1B).."[0m";
-function getstring(style, text)
+local function getstring(style, text)
        if style then
                return format(fmt_string, style, text);
        else
@@ -53,7 +59,45 @@ function getstring(style, text)
        end
 end
 
-function getstyle(...)
+local function gray(n)
+       return m_floor(n*3/32)+0xe8;
+end
+local function color(r,g,b)
+       if r == g and g == b then
+               return gray(r);
+       end
+       r = m_floor(r*3/128);
+       g = m_floor(g*3/128);
+       b = m_floor(b*3/128);
+       return 0x10 + ( r * 36 ) + ( g * 6 ) + ( b );
+end
+local function hex2rgb(hex)
+       local r = tonumber(hex:sub(1,2),16);
+       local g = tonumber(hex:sub(3,4),16);
+       local b = tonumber(hex:sub(5,6),16);
+       return r,g,b;
+end
+
+setmetatable(stylemap, { __index = function(_, style)
+       if type(style) == "string" and style:find("%x%x%x%x%x%x") == 1 then
+               local g = style:sub(7) == " background" and "48;5;" or "38;5;";
+               return g .. color(hex2rgb(style));
+       end
+end } );
+
+local csscolors = {
+       red = "ff0000"; fuchsia = "ff00ff"; green = "008000"; white = "ffffff";
+       lime = "00ff00"; yellow = "ffff00"; purple = "800080"; blue = "0000ff";
+       aqua = "00ffff"; olive  = "808000"; black  = "000000"; navy = "000080";
+       teal = "008080"; silver = "c0c0c0"; maroon = "800000"; gray = "808080";
+}
+for colorname, rgb in pairs(csscolors) do
+       stylemap[colorname] = stylemap[colorname] or stylemap[rgb];
+       colorname, rgb = colorname .. " background", rgb .. " background"
+       stylemap[colorname] = stylemap[colorname] or stylemap[rgb];
+end
+
+local function getstyle(...)
        local styles, result = { ... }, {};
        for i, style in ipairs(styles) do
                style = stylemap[style];
@@ -65,7 +109,7 @@ function getstyle(...)
 end
 
 local last = "0";
-function setstyle(style)
+local function setstyle(style)
        style = style or "0";
        if style ~= last then
                io_write("\27["..style.."m");
@@ -82,7 +126,7 @@ if windows then
                end
        end
        if not orig_color then
-               function setstyle(style) end
+               function setstyle() end
        end
 end
 
@@ -95,8 +139,13 @@ local function ansi2css(ansi_codes)
        return "</span><span style='"..t_concat(css, ";").."'>";
 end
 
-function tohtml(input)
+local function tohtml(input)
        return input:gsub("\027%[(.-)m", ansi2css);
 end
 
-return _M;
+return {
+       getstring = getstring;
+       getstyle = getstyle;
+       setstyle = setstyle;
+       tohtml = tohtml;
+};