/* ----------------------------------------------------------------------- * * * Copyright 2004-2008 H. Peter Anvin - All Rights Reserved * * 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, Inc., 51 Franklin St, Fifth Floor, * Boston MA 02110-1301, USA; either version 2 of the License, or * (at your option) any later version; incorporated herein by reference. * * ----------------------------------------------------------------------- */ #include <stdlib.h> #include <string.h> #include <stdio.h> #include <colortbl.h> #include "menu.h" /* * The color/attribute indexes (\1#X, \2#XX, \3#XXX) are as follows * * 00 - screen Rest of the screen * 01 - border Border area * 02 - title Title bar * 03 - unsel Unselected menu item * 04 - hotkey Unselected hotkey * 05 - sel Selection bar * 06 - hotsel Selected hotkey * 07 - scrollbar Scroll bar * 08 - tabmsg Press [Tab] message * 09 - cmdmark Command line marker * 10 - cmdline Command line * 11 - pwdborder Password box border * 12 - pwdheader Password box header * 13 - pwdentry Password box contents * 14 - timeout_msg Timeout message * 15 - timeout Timeout counter * 16 - help Current entry help text * 17 - disabled Disabled menu item */ static const struct color_table default_colors[] = { {"screen", "37;40", 0x80ffffff, 0x00000000, SHADOW_NORMAL}, {"border", "30;44", 0x40000000, 0x00000000, SHADOW_NORMAL}, {"title", "1;36;44", 0xc00090f0, 0x00000000, SHADOW_NORMAL}, {"unsel", "37;44", 0x90ffffff, 0x00000000, SHADOW_NORMAL}, {"hotkey", "1;37;44", 0xffffffff, 0x00000000, SHADOW_NORMAL}, {"sel", "7;37;40", 0xe0000000, 0x20ff8000, SHADOW_ALL}, {"hotsel", "1;7;37;40", 0xe0400000, 0x20ff8000, SHADOW_ALL}, {"scrollbar", "30;44", 0x40000000, 0x00000000, SHADOW_NORMAL}, {"tabmsg", "31;40", 0x90ffff00, 0x00000000, SHADOW_NORMAL}, {"cmdmark", "1;36;40", 0xc000ffff, 0x00000000, SHADOW_NORMAL}, {"cmdline", "37;40", 0xc0ffffff, 0x00000000, SHADOW_NORMAL}, {"pwdborder", "30;47", 0x80ffffff, 0x20ffffff, SHADOW_NORMAL}, {"pwdheader", "31;47", 0x80ff8080, 0x20ffffff, SHADOW_NORMAL}, {"pwdentry", "30;47", 0x80ffffff, 0x20ffffff, SHADOW_NORMAL}, {"timeout_msg", "37;40", 0x80ffffff, 0x00000000, SHADOW_NORMAL}, {"timeout", "1;37;40", 0xc0ffffff, 0x00000000, SHADOW_NORMAL}, {"help", "37;40", 0xc0ffffff, 0x00000000, SHADOW_NORMAL}, {"disabled", "1;30;44", 0x60cccccc, 0x00000000, SHADOW_NORMAL}, }; #define NCOLORS (sizeof default_colors/sizeof default_colors[0]) const int message_base_color = NCOLORS; const int menu_color_table_size = NCOLORS + 256; /* Algorithmically generate the msgXX colors */ void set_msg_colors_global(struct color_table *tbl, unsigned int fg, unsigned int bg, enum color_table_shadow shadow) { struct color_table *cp = tbl + message_base_color; unsigned int i; unsigned int fga, bga; unsigned int fgh, bgh; unsigned int fg_idx, bg_idx; unsigned int fg_rgb, bg_rgb; static const unsigned int pc2rgb[8] = { 0x000000, 0x0000ff, 0x00ff00, 0x00ffff, 0xff0000, 0xff00ff, 0xffff00, 0xffffff }; /* Converting PC RGBI to sensible RGBA values is an "interesting" proposition. This algorithm may need plenty of tweaking. */ fga = fg & 0xff000000; fgh = ((fg >> 1) & 0xff000000) | 0x80000000; bga = bg & 0xff000000; bgh = ((bg >> 1) & 0xff000000) | 0x80000000; for (i = 0; i < 256; i++) { fg_idx = i & 15; bg_idx = i >> 4; fg_rgb = pc2rgb[fg_idx & 7] & fg; bg_rgb = pc2rgb[bg_idx & 7] & bg; if (fg_idx & 8) { /* High intensity foreground */ fg_rgb |= fgh; } else { fg_rgb |= fga; } if (bg_idx == 0) { /* Default black background, assume transparent */ bg_rgb = 0; } else if (bg_idx & 8) { bg_rgb |= bgh; } else { bg_rgb |= bga; } cp->argb_fg = fg_rgb; cp->argb_bg = bg_rgb; cp->shadow = shadow; cp++; } } struct color_table *default_color_table(void) { unsigned int i; const struct color_table *dp; struct color_table *cp; struct color_table *color_table; static const int pc2ansi[8] = { 0, 4, 2, 6, 1, 5, 3, 7 }; static char msg_names[6 * 256]; char *mp; color_table = calloc(NCOLORS + 256, sizeof(struct color_table)); dp = default_colors; cp = color_table; for (i = 0; i < NCOLORS; i++) { *cp = *dp; cp->ansi = refstrdup(dp->ansi); cp++; dp++; } mp = msg_names; for (i = 0; i < 256; i++) { cp->name = mp; mp += sprintf(mp, "msg%02x", i) + 1; rsprintf(&cp->ansi, "%s3%d;4%d", (i & 8) ? "1;" : "", pc2ansi[i & 7], pc2ansi[(i >> 4) & 7]); cp++; } /*** XXX: This needs to move to run_menu() ***/ console_color_table = color_table; console_color_table_size = NCOLORS + 256; set_msg_colors_global(color_table, MSG_COLORS_DEF_FG, MSG_COLORS_DEF_BG, MSG_COLORS_DEF_SHADOW); return color_table; } struct color_table *copy_color_table(const struct color_table *master) { const struct color_table *dp; struct color_table *color_table, *cp; unsigned int i; color_table = calloc(NCOLORS + 256, sizeof(struct color_table)); dp = master; cp = color_table; for (i = 0; i < NCOLORS + 256; i++) { *cp = *dp; cp->ansi = refstr_get(dp->ansi); cp++; dp++; } return color_table; }