mirror of
https://github.com/radareorg/radare2.git
synced 2024-12-02 18:27:18 +00:00
218 lines
5.1 KiB
C
218 lines
5.1 KiB
C
/* tiv - terminal image viewer - MIT 2013-2022 - pancake */
|
|
|
|
#include <r_cons.h>
|
|
#include <r_th.h>
|
|
|
|
#define XY(b,x,y) ( b+((y)*(w*3))+(x*3) )
|
|
#define ABS(x) (((x)<0)?-(x):(x))
|
|
#define POND(x,y) (ABS((x)) * (y))
|
|
|
|
typedef void (*Renderer)(PrintfCallback cb_printf, const ut8*, const ut8 *);
|
|
|
|
static int reduce8(int r, int g, int b) {
|
|
int colors_len = 8;
|
|
int select = 0;
|
|
int odistance = -1;
|
|
int i, k = 1;
|
|
const int colors[][3] = {
|
|
{ 0x00,0x00,0x00 }, // black
|
|
{ 0xd0,0x10,0x10 }, // red
|
|
{ 0x10,0xe0,0x10 }, // green
|
|
{ 0xf7,0xf5,0x3a }, // yellow
|
|
{ 0x10,0x10,0xf0 }, // blue // XXX
|
|
{ 0xfb,0x3d,0xf8 }, // pink
|
|
{ 0x10,0xf0,0xf0 }, // turqoise
|
|
{ 0xf0,0xf0,0xf0 }, // white
|
|
};
|
|
|
|
r /= k; r *= k;
|
|
g /= k; g *= k;
|
|
b /= k; b *= k;
|
|
// B&W
|
|
if (r < 30 && g < 30 && b < 30) {
|
|
return 0;
|
|
}
|
|
if (r > 200 && g > 200 && b > 200) {
|
|
return 7;
|
|
}
|
|
odistance = -1;
|
|
for (i = 0; i < colors_len; i++) {
|
|
int distance =
|
|
POND (colors[i][0]-r, r)
|
|
+ POND (colors[i][1]-g, g)
|
|
+ POND (colors[i][2]-b, b);
|
|
if (odistance == -1 || distance < odistance) {
|
|
odistance = distance;
|
|
select = i;
|
|
}
|
|
}
|
|
return select;
|
|
}
|
|
|
|
static void render_ansi(PrintfCallback cb_printf, const ut8 *c, const ut8 *d) {
|
|
int fg = 0;
|
|
int color = reduce8 (c[0], c[1], c[2]);
|
|
if (color == -1) {
|
|
return;
|
|
}
|
|
//if (c[0]<30 && c[1]<30 && c[2]<30) fg = 1;
|
|
cb_printf ("\x1b[%dm", color+(fg?30:40));
|
|
}
|
|
|
|
static int rgb(int r, int g, int b) {
|
|
r = R_DIM (r, 0, 255);
|
|
g = R_DIM (g, 0, 255);
|
|
b = R_DIM (b, 0, 255);
|
|
r = (int)(r / 50.6);
|
|
g = (int)(g / 50.6);
|
|
b = (int)(b / 50.6);
|
|
return 16 + (r * 36) + (g * 6) + b;
|
|
}
|
|
|
|
static void render_sixel(PrintfCallback cb_printf, const ut8 *c, const ut8 *d) {
|
|
int r = (c[0] * 100 / 255);
|
|
int g = (c[1] * 100 / 255);
|
|
int b = (c[2] * 100 / 255);
|
|
cb_printf ("#0;2;%d;%d;%d", r, g, b); // bg is black
|
|
r = (d[1] * 100 / 255);
|
|
g = (d[2] * 100 / 255);
|
|
b = (d[3] * 100 / 255);
|
|
cb_printf ("#1;2;%d;%d;%d", r, g, b); // fg is color
|
|
// cb_printf ("#%d!6%c", 1, 0x3f); // 6x6 pixel block
|
|
cb_printf ("#%d~~~~~~", 1); // 6x6 pixel block
|
|
}
|
|
|
|
static void render_256(PrintfCallback cb_printf, const ut8 *c, const ut8 *d) {
|
|
cb_printf ("\x1b[%d;5;%dm", 38, rgb (c[0], c[1], c[2]));
|
|
cb_printf ("\x1b[%d;5;%dm", 48, rgb (d[0], d[1], d[2]));
|
|
}
|
|
|
|
static void render_rgb(PrintfCallback cb_printf, const ut8 *c, const ut8 *d) {
|
|
cb_printf ("\x1b[38;2;%d;%d;%dm", c[0], c[1], c[2]);
|
|
cb_printf ("\x1b[48;2;%d;%d;%dm", d[0], d[1], d[2]);
|
|
}
|
|
|
|
static void render_greyscale(PrintfCallback cb_printf, const ut8 *c, const ut8 *d) {
|
|
int color1, color2, k;
|
|
color1 = (c[0] + c[1] + c[2]) / 3;
|
|
color2 = (d[0] + d[1] + d[2]) / 3;
|
|
k = 231 + ((int)((float)color1 / 10.3));
|
|
if (k < 232) {
|
|
k = 232;
|
|
}
|
|
cb_printf ("\x1b[%d;5;%dm", 48, k); // bg
|
|
k = 231 + ((int)((float)color2 / 10.3));
|
|
if (k < 232) {
|
|
k = 232;
|
|
}
|
|
cb_printf ("\x1b[%d;5;%dm", 38, k); // fg
|
|
}
|
|
|
|
static void render_ascii(PrintfCallback cb_printf, const ut8 *c, const ut8 *d) {
|
|
const char * const pal = " `.,-:+*%$#";
|
|
int idx, pal_len = strlen (pal);
|
|
float p = ((double)(c[0] + c[1] + c[2]) / 3);
|
|
float q = ((double)(d[0] + d[1] + d[2]) / 3);
|
|
idx = (int)(((p + q)/2) / (255 / pal_len));
|
|
if (idx >= pal_len) {
|
|
idx = pal_len - 1;
|
|
}
|
|
cb_printf ("%c", pal[idx]);
|
|
}
|
|
|
|
static void do_render(Renderer renderer, PrintfCallback cb_printf, const ut8 *buf, int len, int w, int h) {
|
|
const ut8 *c, *d;
|
|
int x, y;
|
|
if (renderer == render_sixel) {
|
|
cb_printf ("\x1bPq");
|
|
for (y = 0; y < h; y += 6) {
|
|
for (x = 0; x < w; x+=3) {
|
|
c = XY (buf, x, y);
|
|
d = XY (buf, x, y + 1);
|
|
if (d + 3 > (buf + len)) {
|
|
break;
|
|
}
|
|
render_sixel (cb_printf, c, d);
|
|
}
|
|
cb_printf ("$-");
|
|
}
|
|
cb_printf ("\x1b\\\n");
|
|
return;
|
|
}
|
|
for (y = 0; y < h; y += 2) {
|
|
for (x = 0; x < w; x++) {
|
|
c = XY (buf, x, y);
|
|
d = XY (buf, x, y + 1);
|
|
if (d + 3 > (buf + len)) {
|
|
break;
|
|
}
|
|
renderer (cb_printf, c, d);
|
|
if (renderer != render_ascii) {
|
|
render_ascii (cb_printf, c, d);
|
|
}
|
|
}
|
|
cb_printf ((renderer == render_ascii)? "\n": "\x1b[0m\n");
|
|
}
|
|
}
|
|
|
|
static Renderer select_renderer(int mode) {
|
|
switch (mode) {
|
|
case 'a':
|
|
return render_ascii;
|
|
case 'A':
|
|
return render_ansi;
|
|
case 'g':
|
|
return render_greyscale;
|
|
case 's':
|
|
return render_sixel;
|
|
case '2':
|
|
return render_256;
|
|
default:
|
|
return render_rgb;
|
|
}
|
|
}
|
|
|
|
R_API void r_cons_image(const ut8 *buf, int bufsz, int width, int mode) {
|
|
const int height = (bufsz / width) / 3;
|
|
Renderer renderer = select_renderer (mode);
|
|
do_render (renderer, r_cons_printf, buf, bufsz, width, height);
|
|
}
|
|
|
|
#if 0
|
|
int
|
|
main(int argc, const char **argv) {
|
|
ut8 *buf, *c, *d;
|
|
int n, x, y, w, h, imgsz, readsz;
|
|
if (argc<3) {
|
|
printf ("stiv . suckless terminal image viewer\n");
|
|
printf ("Usage: stiv [width] [height] [ascii|ansi|grey|256|rgb] < rgb24\n");
|
|
return 1;
|
|
}
|
|
w = atoi (argv[1]);
|
|
h = atoi (argv[2]);
|
|
Renderer renderer;
|
|
if (argc>3) {
|
|
renderer = select_renderer (argv[3]);
|
|
} else {
|
|
renderer = render_rgb
|
|
};
|
|
if (w<1 || h<1) {
|
|
printf ("Invalid arguments\n");
|
|
return 1;
|
|
}
|
|
imgsz = w * h * 3;
|
|
buf = malloc (imgsz);
|
|
readsz = 0;
|
|
do {
|
|
n = read(0, buf+readsz, imgsz);
|
|
if (n<1) break;
|
|
readsz += n;
|
|
} while (readsz < imgsz);
|
|
|
|
do_render (render, buf, readsz, w, h);
|
|
|
|
free (buf);
|
|
return 0;
|
|
}
|
|
#endif
|