Home | History | Annotate | Line # | Download | only in dist
colour.c revision 1.4
      1  1.4  christos /* Id */
      2  1.1      jmmv 
      3  1.1      jmmv /*
      4  1.1      jmmv  * Copyright (c) 2008 Nicholas Marriott <nicm (at) users.sourceforge.net>
      5  1.1      jmmv  *
      6  1.1      jmmv  * Permission to use, copy, modify, and distribute this software for any
      7  1.1      jmmv  * purpose with or without fee is hereby granted, provided that the above
      8  1.1      jmmv  * copyright notice and this permission notice appear in all copies.
      9  1.1      jmmv  *
     10  1.1      jmmv  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
     11  1.1      jmmv  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
     12  1.1      jmmv  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
     13  1.1      jmmv  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
     14  1.1      jmmv  * WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
     15  1.1      jmmv  * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
     16  1.1      jmmv  * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
     17  1.1      jmmv  */
     18  1.1      jmmv 
     19  1.1      jmmv #include <sys/types.h>
     20  1.1      jmmv 
     21  1.2        he #include <ctype.h>
     22  1.1      jmmv #include <stdlib.h>
     23  1.1      jmmv #include <string.h>
     24  1.1      jmmv 
     25  1.1      jmmv #include "tmux.h"
     26  1.1      jmmv 
     27  1.1      jmmv /*
     28  1.1      jmmv  * Colour to string conversion functions. Bit 8 of the colour means it is one
     29  1.1      jmmv  * of the 256 colour palette.
     30  1.1      jmmv  */
     31  1.1      jmmv 
     32  1.2        he /* An RGB colour. */
     33  1.2        he struct colour_rgb {
     34  1.2        he 	u_char	r;
     35  1.2        he 	u_char	g;
     36  1.2        he 	u_char	b;
     37  1.2        he };
     38  1.2        he 
     39  1.2        he /* 256 colour RGB table, generated on first use. */
     40  1.2        he struct colour_rgb *colour_rgb_256;
     41  1.2        he 
     42  1.2        he void	colour_rgb_generate256(void);
     43  1.2        he u_int	colour_rgb_distance(struct colour_rgb *, struct colour_rgb *);
     44  1.2        he int	colour_rgb_find(struct colour_rgb *);
     45  1.2        he 
     46  1.2        he /* Generate 256 colour RGB table. */
     47  1.2        he void
     48  1.2        he colour_rgb_generate256(void)
     49  1.2        he {
     50  1.2        he 	struct colour_rgb	*rgb;
     51  1.2        he 	u_int			 i, r, g, b;
     52  1.2        he 
     53  1.2        he 	/*
     54  1.2        he 	 * Allocate the table. The first 16 colours are often changed by users
     55  1.2        he 	 * and terminals so don't include them.
     56  1.2        he 	 */
     57  1.2        he 	colour_rgb_256 = xcalloc(240, sizeof *colour_rgb_256);
     58  1.2        he 
     59  1.2        he 	/* Add the colours first. */
     60  1.2        he 	r = g = b = 0;
     61  1.2        he 	for (i = 240; i > 24; i--) {
     62  1.2        he 		rgb = &colour_rgb_256[240 - i];
     63  1.2        he 
     64  1.2        he 		if (r != 0)
     65  1.2        he 			rgb->r = (r * 40) + 55;
     66  1.2        he 		if (g != 0)
     67  1.2        he 			rgb->g = (g * 40) + 55;
     68  1.2        he 		if (b != 0)
     69  1.2        he 			rgb->b = (b * 40) + 55;
     70  1.2        he 
     71  1.2        he 		b++;
     72  1.2        he 		if (b > 5) {
     73  1.2        he 			b = 0;
     74  1.2        he 			g++;
     75  1.2        he 		}
     76  1.2        he 		if (g > 5) {
     77  1.2        he 			g = 0;
     78  1.2        he 			r++;
     79  1.2        he 		}
     80  1.2        he 	}
     81  1.2        he 
     82  1.2        he 	/* Then add the greys. */
     83  1.2        he 	for (i = 24; i > 0; i--) {
     84  1.2        he 		rgb = &colour_rgb_256[240 - i];
     85  1.2        he 
     86  1.2        he 		rgb->r = 8 + (24 - i) * 10;
     87  1.2        he 		rgb->g = 8 + (24 - i) * 10;
     88  1.2        he 		rgb->b = 8 + (24 - i) * 10;
     89  1.2        he 	}
     90  1.2        he }
     91  1.2        he 
     92  1.4  christos /* Get colour RGB distance. */
     93  1.2        he u_int
     94  1.2        he colour_rgb_distance(struct colour_rgb *rgb1, struct colour_rgb *rgb2)
     95  1.2        he {
     96  1.2        he 	int	r, g, b;
     97  1.2        he 
     98  1.2        he 	r = rgb1->r - rgb2->r;
     99  1.2        he 	g = rgb1->g - rgb2->g;
    100  1.2        he 	b = rgb1->b - rgb2->b;
    101  1.2        he 	return (r * r + g * g + b * b);
    102  1.2        he }
    103  1.2        he 
    104  1.2        he /* Work out the nearest colour from the 256 colour set. */
    105  1.2        he int
    106  1.2        he colour_rgb_find(struct colour_rgb *rgb)
    107  1.2        he {
    108  1.4  christos 	u_int	distance, lowest, colour, i;
    109  1.2        he 
    110  1.2        he 	if (colour_rgb_256 == NULL)
    111  1.2        he 		colour_rgb_generate256();
    112  1.2        he 
    113  1.2        he 	colour = 16;
    114  1.2        he 	lowest = UINT_MAX;
    115  1.3        he 	for (i = 0; i < 240; i++) {
    116  1.2        he 		distance = colour_rgb_distance(&colour_rgb_256[i], rgb);
    117  1.2        he 		if (distance < lowest) {
    118  1.2        he 			lowest = distance;
    119  1.2        he 			colour = 16 + i;
    120  1.2        he 		}
    121  1.2        he 	}
    122  1.2        he 	return (colour);
    123  1.2        he }
    124  1.2        he 
    125  1.2        he /* Set grid cell foreground colour. */
    126  1.1      jmmv void
    127  1.1      jmmv colour_set_fg(struct grid_cell *gc, int c)
    128  1.1      jmmv {
    129  1.1      jmmv 	if (c & 0x100)
    130  1.1      jmmv 		gc->flags |= GRID_FLAG_FG256;
    131  1.1      jmmv 	gc->fg = c;
    132  1.1      jmmv }
    133  1.1      jmmv 
    134  1.2        he /* Set grid cell background colour. */
    135  1.1      jmmv void
    136  1.1      jmmv colour_set_bg(struct grid_cell *gc, int c)
    137  1.1      jmmv {
    138  1.1      jmmv 	if (c & 0x100)
    139  1.1      jmmv 		gc->flags |= GRID_FLAG_BG256;
    140  1.1      jmmv 	gc->bg = c;
    141  1.1      jmmv }
    142  1.1      jmmv 
    143  1.2        he /* Convert colour to a string. */
    144  1.1      jmmv const char *
    145  1.1      jmmv colour_tostring(int c)
    146  1.1      jmmv {
    147  1.1      jmmv 	static char	s[32];
    148  1.1      jmmv 
    149  1.1      jmmv 	if (c & 0x100) {
    150  1.1      jmmv 		xsnprintf(s, sizeof s, "colour%u", c & ~0x100);
    151  1.1      jmmv 		return (s);
    152  1.1      jmmv 	}
    153  1.1      jmmv 
    154  1.1      jmmv 	switch (c) {
    155  1.1      jmmv 	case 0:
    156  1.1      jmmv 		return ("black");
    157  1.1      jmmv 	case 1:
    158  1.1      jmmv 		return ("red");
    159  1.1      jmmv 	case 2:
    160  1.1      jmmv 		return ("green");
    161  1.1      jmmv 	case 3:
    162  1.1      jmmv 		return ("yellow");
    163  1.1      jmmv 	case 4:
    164  1.1      jmmv 		return ("blue");
    165  1.1      jmmv 	case 5:
    166  1.1      jmmv 		return ("magenta");
    167  1.1      jmmv 	case 6:
    168  1.1      jmmv 		return ("cyan");
    169  1.1      jmmv 	case 7:
    170  1.1      jmmv 		return ("white");
    171  1.1      jmmv 	case 8:
    172  1.1      jmmv 		return ("default");
    173  1.4  christos 	case 90:
    174  1.4  christos 		return ("brightblack");
    175  1.4  christos 	case 91:
    176  1.4  christos 		return ("brightred");
    177  1.4  christos 	case 92:
    178  1.4  christos 		return ("brightgreen");
    179  1.4  christos 	case 93:
    180  1.4  christos 		return ("brightyellow");
    181  1.4  christos 	case 94:
    182  1.4  christos 		return ("brightblue");
    183  1.4  christos 	case 95:
    184  1.4  christos 		return ("brightmagenta");
    185  1.4  christos 	case 96:
    186  1.4  christos 		return ("brightcyan");
    187  1.4  christos 	case 97:
    188  1.4  christos 		return ("brightwhite");
    189  1.1      jmmv 	}
    190  1.1      jmmv 	return (NULL);
    191  1.1      jmmv }
    192  1.1      jmmv 
    193  1.2        he /* Convert colour from string. */
    194  1.1      jmmv int
    195  1.1      jmmv colour_fromstring(const char *s)
    196  1.1      jmmv {
    197  1.2        he 	const char		*errstr;
    198  1.2        he 	const char		*cp;
    199  1.2        he 	struct colour_rgb	 rgb;
    200  1.2        he 	int			 n;
    201  1.2        he 
    202  1.2        he 	if (*s == '#' && strlen(s) == 7) {
    203  1.2        he 		for (cp = s + 1; isxdigit((u_char) *cp); cp++)
    204  1.2        he 			;
    205  1.2        he 		if (*cp != '\0')
    206  1.2        he 			return (-1);
    207  1.2        he 		n = sscanf(s + 1, "%2hhx%2hhx%2hhx", &rgb.r, &rgb.g, &rgb.b);
    208  1.2        he 		if (n != 3)
    209  1.2        he 			return (-1);
    210  1.2        he 		return (colour_rgb_find(&rgb) | 0x100);
    211  1.2        he 	}
    212  1.1      jmmv 
    213  1.1      jmmv 	if (strncasecmp(s, "colour", (sizeof "colour") - 1) == 0) {
    214  1.1      jmmv 		n = strtonum(s + (sizeof "colour") - 1, 0, 255, &errstr);
    215  1.1      jmmv 		if (errstr != NULL)
    216  1.1      jmmv 			return (-1);
    217  1.1      jmmv 		return (n | 0x100);
    218  1.1      jmmv 	}
    219  1.1      jmmv 
    220  1.1      jmmv 	if (strcasecmp(s, "black") == 0 || (s[0] == '0' && s[1] == '\0'))
    221  1.1      jmmv 		return (0);
    222  1.1      jmmv 	if (strcasecmp(s, "red") == 0 || (s[0] == '1' && s[1] == '\0'))
    223  1.1      jmmv 		return (1);
    224  1.1      jmmv 	if (strcasecmp(s, "green") == 0 || (s[0] == '2' && s[1] == '\0'))
    225  1.1      jmmv 		return (2);
    226  1.1      jmmv 	if (strcasecmp(s, "yellow") == 0 || (s[0] == '3' && s[1] == '\0'))
    227  1.1      jmmv 		return (3);
    228  1.1      jmmv 	if (strcasecmp(s, "blue") == 0 || (s[0] == '4' && s[1] == '\0'))
    229  1.1      jmmv 		return (4);
    230  1.1      jmmv 	if (strcasecmp(s, "magenta") == 0 || (s[0] == '5' && s[1] == '\0'))
    231  1.1      jmmv 		return (5);
    232  1.1      jmmv 	if (strcasecmp(s, "cyan") == 0 || (s[0] == '6' && s[1] == '\0'))
    233  1.1      jmmv 		return (6);
    234  1.1      jmmv 	if (strcasecmp(s, "white") == 0 || (s[0] == '7' && s[1] == '\0'))
    235  1.1      jmmv 		return (7);
    236  1.1      jmmv 	if (strcasecmp(s, "default") == 0 || (s[0] == '8' && s[1] == '\0'))
    237  1.1      jmmv 		return (8);
    238  1.4  christos 	if (strcasecmp(s, "brightblack") == 0 ||
    239  1.4  christos 	    (s[0] == '9' && s[1] == '0' && s[1] == '\0'))
    240  1.4  christos 		return (90);
    241  1.4  christos 	if (strcasecmp(s, "brightred") == 0 ||
    242  1.4  christos 	    (s[0] == '9' && s[1] == '1' && s[1] == '\0'))
    243  1.4  christos 		return (91);
    244  1.4  christos 	if (strcasecmp(s, "brightgreen") == 0 ||
    245  1.4  christos 	    (s[0] == '9' && s[1] == '2' && s[1] == '\0'))
    246  1.4  christos 		return (92);
    247  1.4  christos 	if (strcasecmp(s, "brightyellow") == 0 ||
    248  1.4  christos 	    (s[0] == '9' && s[1] == '3' && s[1] == '\0'))
    249  1.4  christos 		return (93);
    250  1.4  christos 	if (strcasecmp(s, "brightblue") == 0 ||
    251  1.4  christos 	    (s[0] == '9' && s[1] == '4' && s[1] == '\0'))
    252  1.4  christos 		return (94);
    253  1.4  christos 	if (strcasecmp(s, "brightmagenta") == 0 ||
    254  1.4  christos 	    (s[0] == '9' && s[1] == '5' && s[1] == '\0'))
    255  1.4  christos 		return (95);
    256  1.4  christos 	if (strcasecmp(s, "brightcyan") == 0 ||
    257  1.4  christos 	    (s[0] == '9' && s[1] == '6' && s[1] == '\0'))
    258  1.4  christos 		return (96);
    259  1.4  christos 	if (strcasecmp(s, "brightwhite") == 0 ||
    260  1.4  christos 	    (s[0] == '9' && s[1] == '7' && s[1] == '\0'))
    261  1.4  christos 		return (97);
    262  1.1      jmmv 	return (-1);
    263  1.1      jmmv }
    264  1.1      jmmv 
    265  1.2        he /* Convert 256 colour palette to 16. */
    266  1.1      jmmv u_char
    267  1.1      jmmv colour_256to16(u_char c)
    268  1.1      jmmv {
    269  1.1      jmmv 	static const u_char table[256] = {
    270  1.1      jmmv 		 0,  1,  2,  3,  4,  5,  6,  7,  8,  9, 10, 11, 12, 13, 14, 15,
    271  1.1      jmmv 		 0,  4,  4,  4, 12, 12,  2,  6,  4,  4, 12, 12,  2,  2,  6,  4,
    272  1.1      jmmv 		12, 12,  2,  2,  2,  6, 12, 12, 10, 10, 10, 10, 14, 12, 10, 10,
    273  1.1      jmmv 		10, 10, 10, 14,  1,  5,  4,  4, 12, 12,  3,  8,  4,  4, 12, 12,
    274  1.1      jmmv 		 2,  2,  6,  4, 12, 12,  2,  2,  2,  6, 12, 12, 10, 10, 10, 10,
    275  1.1      jmmv 		14, 12, 10, 10, 10, 10, 10, 14,  1,  1,  5,  4, 12, 12,  1,  1,
    276  1.1      jmmv 		 5,  4, 12, 12,  3,  3,  8,  4, 12, 12,  2,  2,  2,  6, 12, 12,
    277  1.1      jmmv 		10, 10, 10, 10, 14, 12, 10, 10, 10, 10, 10, 14,  1,  1,  1,  5,
    278  1.1      jmmv 		12, 12,  1,  1,  1,  5, 12, 12,  1,  1,  1,  5, 12, 12,  3,  3,
    279  1.1      jmmv 		 3,  7, 12, 12, 10, 10, 10, 10, 14, 12, 10, 10, 10, 10, 10, 14,
    280  1.1      jmmv 		 9,  9,  9,  9, 13, 12,  9,  9,  9,  9, 13, 12,  9,  9,  9,  9,
    281  1.1      jmmv 		13, 12,  9,  9,  9,  9, 13, 12, 11, 11, 11, 11,  7, 12, 10, 10,
    282  1.1      jmmv 		10, 10, 10, 14,  9,  9,  9,  9,  9, 13,  9,  9,  9,  9,  9, 13,
    283  1.1      jmmv 		 9,  9,  9,  9,  9, 13,  9,  9,  9,  9,  9, 13,  9,  9,  9,  9,
    284  1.1      jmmv 		 9, 13, 11, 11, 11, 11, 11, 15,  0,  0,  0,  0,  0,  0,  8,  8,
    285  1.1      jmmv 		 8,  8,  8,  8,  7,  7,  7,  7,  7,  7, 15, 15, 15, 15, 15, 15
    286  1.1      jmmv 	};
    287  1.1      jmmv 
    288  1.1      jmmv 	return (table[c]);
    289  1.1      jmmv }
    290