/* ----------------------------------------------------------------------- *
*
* 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;
}