diff options
author | Avi Halachmi (:avih) <avihpit@yahoo.com> | 2016-10-22 16:48:30 +0300 |
---|---|---|
committer | James Ross-Gowan <rossymiles@gmail.com> | 2016-10-25 00:03:01 +1100 |
commit | 363982c77445e02a169d4fcbd013e2814bab90d6 (patch) | |
tree | 3dbfc425fe2de41efe9c990c3e8916043d8f941a | |
parent | 02d2c2cc9722dbda832917118bd4190a38e70f5a (diff) |
vo_tct: support also 256 colors output
-rw-r--r-- | DOCS/man/vo.rst | 3 | ||||
-rw-r--r-- | video/out/vo_tct.c | 64 |
2 files changed, 58 insertions, 9 deletions
diff --git a/DOCS/man/vo.rst b/DOCS/man/vo.rst index 9575948eea..392e6ad41c 100644 --- a/DOCS/man/vo.rst +++ b/DOCS/man/vo.rst @@ -406,6 +406,9 @@ Available video output drivers are: Assume the terminal has the specified character width and/or height. These default to 80x25 if the terminal size cannot be determined. + ``--vo-tct-256=<yes|no>`` (default: no) + Use 256 colors - for terminals which don't support true color. + ``image`` Output each frame into an image file in the current directory. Each file takes the frame number padded with leading zeros as name. diff --git a/video/out/vo_tct.c b/video/out/vo_tct.c index c72a66e940..68b29e9bc8 100644 --- a/video/out/vo_tct.c +++ b/video/out/vo_tct.c @@ -41,8 +41,10 @@ #define ESC_CLEAR_SCREEN "\e[2J" #define ESC_CLEAR_COLORS "\e[0m" #define ESC_GOTOXY "\e[%d;%df" -#define ESC_COLOR_BACKGROUND "\e[48;2;%d;%d;%dm" -#define ESC_COLOR_FOREGROUND "\e[38;2;%d;%d;%dm" +#define ESC_COLOR_BG "\e[48;2;%d;%d;%dm" +#define ESC_COLOR_FG "\e[38;2;%d;%d;%dm" +#define ESC_COLOR256_BG "\e[48;5;%dm" +#define ESC_COLOR256_FG "\e[38;5;%dm" #define DEFAULT_WIDTH 80 #define DEFAULT_HEIGHT 25 @@ -50,6 +52,7 @@ struct vo_tct_opts { int algo; int width; // 0 -> default int height; // 0 -> default + int term256; // 0 -> true color }; #define OPT_BASE_STRUCT struct vo_tct_opts @@ -60,6 +63,7 @@ static const struct m_sub_options vo_tct_conf = { {"half-blocks", ALGO_HALF_BLOCKS})), OPT_INT("vo-tct-width", width, 0), OPT_INT("vo-tct-height", height, 0), + OPT_FLAG("vo-tct-256", term256, 0), {0} }, .defaults = &(const struct vo_tct_opts) { @@ -80,10 +84,40 @@ struct priv { struct mp_sws_context *sws; }; +// Convert RGB24 to xterm-256 8-bit value +// For simplicity, assume RGB space is perceptually uniform. +// There are 5 places where one of two outputs needs to be chosen when the +// input is the exact middle: +// - The r/g/b channels and the gray value: the higher value output is chosen. +// - If the gray and color have same distance from the input - color is chosen. +static int rgb_to_x256(uint8_t r, uint8_t g, uint8_t b) +{ + // Calculate the nearest 0-based color index at 16 .. 231 +# define v2ci(v) (v < 48 ? 0 : v < 115 ? 1 : (v - 35) / 40) + int ir = v2ci(r), ig = v2ci(g), ib = v2ci(b); // 0..5 each +# define color_index() (36 * ir + 6 * ig + ib) /* 0..215, lazy evaluation */ + + // Calculate the nearest 0-based gray index at 232 .. 255 + int average = (r + g + b) / 3; + int gray_index = average > 238 ? 23 : (average - 3) / 10; // 0..23 + + // Calculate the represented colors back from the index + static const int i2cv[6] = {0, 0x5f, 0x87, 0xaf, 0xd7, 0xff}; + int cr = i2cv[ir], cg = i2cv[ig], cb = i2cv[ib]; // r/g/b, 0..255 each + int gv = 8 + 10 * gray_index; // same value for r/g/b, 0..255 + + // Return the one which is nearer to the original input rgb value +# define dist_square(A,B,C, a,b,c) ((A-a)*(A-a) + (B-b)*(B-b) + (C-c)*(C-c)) + int color_err = dist_square(cr, cg, cb, r, g, b); + int gray_err = dist_square(gv, gv, gv, r, g, b); + return color_err <= gray_err ? 16 + color_index() : 232 + gray_index; +} + static void write_plain( const int dwidth, const int dheight, const int swidth, const int sheight, - const unsigned char *source, const int source_stride) + const unsigned char *source, const int source_stride, + bool term256) { assert(source); const int tx = (dwidth - swidth) / 2; @@ -95,7 +129,11 @@ static void write_plain( unsigned char b = *row++; unsigned char g = *row++; unsigned char r = *row++; - printf(ESC_COLOR_BACKGROUND, r, g, b); + if (term256) { + printf(ESC_COLOR256_BG, rgb_to_x256(r, g, b)); + } else { + printf(ESC_COLOR_BG, r, g, b); + } printf(" "); } printf(ESC_CLEAR_COLORS); @@ -106,7 +144,8 @@ static void write_plain( static void write_half_blocks( const int dwidth, const int dheight, const int swidth, const int sheight, - unsigned char *source, int source_stride) + unsigned char *source, int source_stride, + bool term256) { assert(source); const int tx = (dwidth - swidth) / 2; @@ -122,8 +161,13 @@ static void write_half_blocks( unsigned char b_down = *row_down++; unsigned char g_down = *row_down++; unsigned char r_down = *row_down++; - printf(ESC_COLOR_BACKGROUND, r_up, g_up, b_up); - printf(ESC_COLOR_FOREGROUND, r_down, g_down, b_down); + if (term256) { + printf(ESC_COLOR256_BG, rgb_to_x256(r_up, g_up, b_up)); + printf(ESC_COLOR256_FG, rgb_to_x256(r_down, g_down, b_down)); + } else { + printf(ESC_COLOR_BG, r_up, g_up, b_up); + printf(ESC_COLOR_FG, r_down, g_down, b_down); + } printf("\xe2\x96\x84"); // UTF8 bytes of U+2584 (lower half block) } printf(ESC_CLEAR_COLORS); @@ -202,11 +246,13 @@ static void flip_page(struct vo *vo) if (p->opts->algo == ALGO_PLAIN) { write_plain( vo->dwidth, vo->dheight, p->swidth, p->sheight, - p->frame->planes[0], p->frame->stride[0]); + p->frame->planes[0], p->frame->stride[0], + p->opts->term256); } else { write_half_blocks( vo->dwidth, vo->dheight, p->swidth, p->sheight, - p->frame->planes[0], p->frame->stride[0]); + p->frame->planes[0], p->frame->stride[0], + p->opts->term256); } fflush(stdout); } |