186 lines
5.9 KiB
C
186 lines
5.9 KiB
C
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
│ Copyright 2020 Justine Alexandra Roberts Tunney │
|
|
│ │
|
|
│ This program is free software; you can redistribute it and/or modify │
|
|
│ it under the terms of the GNU General Public License as published by │
|
|
│ the Free Software Foundation; version 2 of the License. │
|
|
│ │
|
|
│ This program is distributed in the hope that it will be useful, but │
|
|
│ WITHOUT ANY WARRANTY; without even the implied warranty of │
|
|
│ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU │
|
|
│ General Public License for more details. │
|
|
│ │
|
|
│ You should have received a copy of the GNU General Public License │
|
|
│ along with this program; if not, write to the Free Software │
|
|
│ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA │
|
|
│ 02110-1301 USA │
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
#include "dsp/tty/quant.h"
|
|
#include "libc/fmt/fmt.h"
|
|
#include "libc/log/check.h"
|
|
#include "libc/macros.h"
|
|
#include "libc/math.h"
|
|
#include "libc/mem/mem.h"
|
|
#include "libc/runtime/runtime.h"
|
|
#include "libc/stdio/stdio.h"
|
|
#include "libc/str/str.h"
|
|
#include "libc/sysv/consts/ex.h"
|
|
#include "libc/sysv/consts/exit.h"
|
|
#include "third_party/getopt/getopt.h"
|
|
|
|
#define USAGE \
|
|
" [FLAGS] [PATH...]\n\
|
|
\n\
|
|
Flags:\n\
|
|
-o PATH output path\n\
|
|
-h shows this information\n\
|
|
\n"
|
|
|
|
static size_t linecap_;
|
|
static FILE *in_, *out_;
|
|
static char *inpath_, *outpath_, *line_;
|
|
|
|
void PrintUsage(int rc, FILE *f) {
|
|
fputs("Usage: ", f);
|
|
fputs(program_invocation_name, f);
|
|
fputs(USAGE, f);
|
|
exit(rc);
|
|
}
|
|
|
|
void GetOpts(int *argc, char *argv[]) {
|
|
int opt;
|
|
outpath_ = "-";
|
|
while ((opt = getopt(*argc, argv, "?ho:")) != -1) {
|
|
switch (opt) {
|
|
case 'o':
|
|
outpath_ = optarg;
|
|
break;
|
|
case '?':
|
|
case 'h':
|
|
PrintUsage(EXIT_SUCCESS, stdout);
|
|
default:
|
|
PrintUsage(EX_USAGE, stderr);
|
|
}
|
|
}
|
|
if (optind == *argc) {
|
|
argv[(*argc)++] = "-";
|
|
}
|
|
}
|
|
|
|
#define U256F1(X) ((float)((X)&0xffu) * 256.0f)
|
|
#define F1U256(X) MAX(MIN((int)rintl(roundl(256.0f * (X))), 255), 0)
|
|
|
|
forceinline struct TtyRgb getquant(unsigned xt) {
|
|
return g_ansi2rgb_[xt];
|
|
}
|
|
|
|
forceinline unsigned dist(int x, int y) {
|
|
return x - y;
|
|
}
|
|
|
|
forceinline unsigned sqr(int x) {
|
|
return x * x;
|
|
}
|
|
|
|
static unsigned rgb2hsl(unsigned rgba) {
|
|
/* this is broken */
|
|
unsigned h8, s8, l8;
|
|
float r, g, b, h, s, d, l, cmax, cmin;
|
|
r = U256F1(rgba);
|
|
g = U256F1(rgba >> 010);
|
|
b = U256F1(rgba >> 020);
|
|
cmax = MAX(MAX(r, g), b);
|
|
cmin = MIN(MIN(r, g), b);
|
|
h = 0.0f;
|
|
s = 0.0f;
|
|
d = cmax - cmin;
|
|
l = (cmax + cmin) / 2.0f;
|
|
if (cmax != cmin) {
|
|
s = l > 0.5L ? d / (2.0f - cmax - cmin) : d / (cmax + cmin);
|
|
if (cmax == r) {
|
|
h = (g - b) / d + (g < b ? 6.0f : 0.0f);
|
|
} else if (cmax == g) {
|
|
h = (b - r) / d + 2.0f;
|
|
} else if (cmax == b) {
|
|
h = (r - g) / d + 4.0f;
|
|
}
|
|
h /= 6.0f;
|
|
}
|
|
h8 = F1U256(h);
|
|
s8 = F1U256(s);
|
|
l8 = F1U256(l);
|
|
return ((rgba >> 030) & 255) << 030 | l8 << 020 | s8 << 010 | h8;
|
|
}
|
|
static struct TtyRgb rgb2hsl2(struct TtyRgb rgb) {
|
|
unsigned x =
|
|
(unsigned)rgb.b << 020 | (unsigned)rgb.g << 010 | (unsigned)rgb.r;
|
|
unsigned y = rgb2hsl(x);
|
|
return (struct TtyRgb){
|
|
.r = y & 0xff, .g = (y >> 010) & 0xff, .b = (y >> 020) & 0xff};
|
|
}
|
|
static unsigned rgbdist(struct TtyRgb x, struct TtyRgb y) {
|
|
x = rgb2hsl2(x);
|
|
y = rgb2hsl2(y);
|
|
return sqrt(sqr(dist(x.r, y.r)) + sqr(dist(x.g, y.g)) + sqr(dist(x.b, y.b)));
|
|
}
|
|
|
|
static unsigned xtdist(unsigned x, unsigned y) {
|
|
return rgbdist(getquant(x), getquant(y));
|
|
}
|
|
|
|
void Show(unsigned color, unsigned bg, unsigned fg, unsigned glyph) {
|
|
uint8_t r, g, b;
|
|
b = (color >> 020) & 0xff;
|
|
g = (color >> 010) & 0xff;
|
|
r = color & 0xff;
|
|
printf("\tmix\t0x%04x,%3d,%3d,%3d,%3d,%3d,%3d\t# \e[48;2;%d;%d;%dm \e[0m\n",
|
|
rgbdist((struct TtyRgb){r, g, b, 0}, (struct TtyRgb){0, 0, 0, 0}), r,
|
|
g, b, bg, fg, glyph, r, g, b);
|
|
}
|
|
|
|
void ProcessFile(void) {
|
|
char *p;
|
|
unsigned color1, bg1, fg1, glyph1;
|
|
unsigned color, bg, fg, glyph;
|
|
color1 = -1u;
|
|
bg1 = -1u;
|
|
fg1 = -1u;
|
|
glyph1 = -1u;
|
|
while ((getline(&line_, &linecap_, in_)) != -1) {
|
|
p = chomp(line_);
|
|
sscanf(p, "%x, %u,%u,%u", &color, &bg, &fg, &glyph);
|
|
if (color != color1) {
|
|
if (color1 != -1u) {
|
|
Show(color1, bg1, fg1, glyph1);
|
|
}
|
|
color1 = color;
|
|
bg1 = bg;
|
|
fg1 = fg;
|
|
glyph1 = glyph;
|
|
}
|
|
if ((fg1 && !fg) || (fg && fg1 && xtdist(fg, bg) < xtdist(fg1, bg1))) {
|
|
color1 = color;
|
|
bg1 = bg;
|
|
fg1 = fg;
|
|
glyph1 = glyph;
|
|
}
|
|
}
|
|
Show(color1, bg1, fg1, glyph1);
|
|
}
|
|
|
|
int main(int argc, char *argv[]) {
|
|
size_t i;
|
|
GetOpts(&argc, argv);
|
|
CHECK_NOTNULL((out_ = fopen(outpath_, "w")));
|
|
for (i = optind; i < argc; ++i) {
|
|
CHECK_NOTNULL((in_ = fopen((inpath_ = argv[i]), "r")));
|
|
ProcessFile();
|
|
CHECK_NE(-1, fclose_s(&in_));
|
|
}
|
|
CHECK_NE(-1, fclose_s(&out_));
|
|
free(line_);
|
|
return 0;
|
|
}
|