/* Color setup Copyright (C) 1994 Miguel de Icaza. 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; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include #include #include #include "global.h" #include "tty.h" #include "setup.h" /* For the externs */ #include "color.h" /* "$Id$" */ /* Set to force black and white display at program startup */ int disable_colors = 0; /* Set if we are actually using colors */ int use_colors = 0; /* Color styles for normal and error dialogs */ int dialog_colors [4]; int alarm_colors [4]; #define ELEMENTS(arr) ( sizeof(arr) / sizeof((arr)[0]) ) #ifdef HAVE_SLANG # define color_map_fg(n) color_map[n].fg # define color_map_bg(n) color_map[n].bg #else # define color_map_fg(n) (color_map[n].fg & COLOR_WHITE) # define color_map_bg(n) (color_map[n].bg & COLOR_WHITE) #endif struct colorpair { char *name; /* Name of the entry */ CTYPE fg; /* foreground color */ CTYPE bg; /* background color */ }; struct colorpair color_map [] = { { "normal=", 0, 0 }, /* normal */ /* 1 */ { "selected=", 0, 0 }, /* selected */ { "marked=", 0, 0 }, /* marked */ { "markselect=", 0, 0 }, /* marked/selected */ { "errors=", 0, 0 }, /* errors */ { "menu=", 0, 0 }, /* menu entry */ { "reverse=", 0, 0 }, /* reverse */ /* Dialog colors */ { "dnormal=", 0, 0 }, /* Dialog normal */ /* 8 */ { "dfocus=", 0, 0 }, /* Dialog focused */ { "dhotnormal=", 0, 0 }, /* Dialog normal/hot */ { "dhotfocus=", 0, 0 }, /* Dialog focused/hot */ { "viewunderline=", 0, 0 }, /* _\b? sequence in view, underline in editor */ { "menusel=", 0, 0 }, /* Menu selected color */ /* 13 */ { "menuhot=", 0, 0 }, /* Color for menu hotkeys */ { "menuhotsel=", 0, 0 }, /* Menu hotkeys/selected entry */ { "helpnormal=", 0, 0 }, /* Help normal */ /* 16 */ { "helpitalic=", 0, 0 }, /* Italic in help */ { "helpbold=", 0, 0 }, /* Bold in help */ { "helplink=", 0, 0 }, /* Not selected hyperlink */ { "helpslink=", 0, 0 }, /* Selected hyperlink */ { "gauge=", 0, 0 }, /* Color of the progress bar (percentage) *//* 21 */ { "input=", 0, 0 }, /* Per file types colors */ { "directory=", 0, 0 }, /* 23 */ { "executable=", 0, 0 }, { "link=", 0, 0 }, /* symbolic link (neither stalled nor link to directory) */ { "stalledlink=",0, 0 }, /* stalled symbolic link */ { "device=", 0, 0 }, { "special=", 0, 0 }, /* sockets, fifo */ { "core=", 0, 0 }, /* core files */ /* 29 */ { 0, 0, 0 }, /* not usable (DEFAULT_COLOR_INDEX) *//* 30 */ { 0, 0, 0 }, /* unused */ { 0, 0, 0 }, /* not usable (A_REVERSE) */ { 0, 0, 0 }, /* not usable (A_REVERSE_BOLD) */ /* editor colors start at 34 */ { "editnormal=", 0, 0 }, /* normal */ /* 34 */ { "editbold=", 0, 0 }, /* search->found */ { "editmarked=", 0, 0 }, /* marked/selected */ }; struct color_table_s { char *name; int value; }; static struct color_table_s const color_table [] = { { "black", COLOR_BLACK }, { "gray", COLOR_BLACK | A_BOLD }, { "red", COLOR_RED }, { "brightred", COLOR_RED | A_BOLD }, { "green", COLOR_GREEN }, { "brightgreen", COLOR_GREEN | A_BOLD }, { "brown", COLOR_YELLOW }, { "yellow", COLOR_YELLOW | A_BOLD }, { "blue", COLOR_BLUE }, { "brightblue", COLOR_BLUE | A_BOLD }, { "magenta", COLOR_MAGENTA }, { "brightmagenta", COLOR_MAGENTA | A_BOLD }, { "cyan", COLOR_CYAN }, { "brightcyan", COLOR_CYAN | A_BOLD }, { "lightgray", COLOR_WHITE }, { "white", COLOR_WHITE | A_BOLD }, { "default", 0 } /* hack for transparent background */ }; #ifdef HAVE_SLANG # define color_value(i) color_table [i].name # define color_name(i) color_table [i].name #else # define color_value(i) color_table [i].value # define color_name(i) color_table [i].name #endif static void get_color (char *cpp, CTYPE *colp) { int i; for (i = 0; i < ELEMENTS(color_table); i++){ if (strcmp (cpp, color_name (i)) == 0){ *colp = color_value (i); return; } } } static void get_two_colors (char **cpp, struct colorpair *colorpairp) { char *p = *cpp; int state; state = 0; for (; *p; p++){ if (*p == ':'){ *p = 0; get_color (*cpp, state ? &colorpairp->bg : &colorpairp->fg); *p = ':'; *cpp = p + 1; return; } if (*p == ','){ state = 1; *p = 0; get_color (*cpp, &colorpairp->fg); *p = ','; *cpp = p + 1; } } get_color (*cpp, state ? &colorpairp->bg : &colorpairp->fg); } static void configure_colors_string (char *the_color_string) { char *color_string, *p; int i, found; if (!the_color_string) return; p = color_string = g_strdup (the_color_string); while (color_string && *color_string){ while (*color_string == ' ' || *color_string == '\t') color_string++; found = 0; for (i = 0; i < ELEMENTS(color_map); i++){ int klen; if (!color_map [i].name) continue; klen = strlen (color_map [i].name); if (strncmp (color_string, color_map [i].name, klen) == 0){ color_string += klen; get_two_colors (&color_string, &color_map [i]); found = 1; } } if (!found){ while (*color_string && *color_string != ':') color_string++; if (*color_string) color_string++; } } g_free (p); } static void configure_colors (void) { extern char *command_line_colors; extern char *default_edition_colors; configure_colors_string (default_edition_colors); configure_colors_string (setup_color_string); configure_colors_string (term_color_string); configure_colors_string (getenv ("MC_COLOR_TABLE")); configure_colors_string (command_line_colors); } #ifndef HAVE_SLANG #define MAX_PAIRS 64 int attr_pairs [MAX_PAIRS]; #endif static void load_dialog_colors (void) { dialog_colors [0] = COLOR_NORMAL; dialog_colors [1] = COLOR_FOCUS; dialog_colors [2] = COLOR_HOT_NORMAL; dialog_colors [3] = COLOR_HOT_FOCUS; alarm_colors [0] = ERROR_COLOR; alarm_colors [1] = REVERSE_COLOR; alarm_colors [2] = ERROR_COLOR; alarm_colors [3] = COLOR_HOT_NORMAL; } void init_colors (void) { int i; int hascolors; /* FIXME: if S-Lang is used, this function must be called regardless of whether we are interested in its result */ hascolors = has_colors (); if (!disable_colors && hascolors){ use_colors = 1; } if (use_colors){ start_color (); configure_colors (); #ifndef HAVE_SLANG if (ELEMENTS (color_map) > MAX_PAIRS){ /* This message should only be seen by the developers */ fprintf (stderr, "Too many defined colors, resize MAX_PAIRS on color.c"); exit (1); } #endif /* !HAVE_SLANG */ if (use_colors) { #ifdef HAVE_SLANG /* * We are relying on undocumented feature of * S-Lang to make COLOR_PAIR(DEFAULT_COLOR_INDEX) * the default fg/bg of the terminal. * Hopefully, future versions of S-Lang will * document this feature. */ SLtt_set_color (DEFAULT_COLOR_INDEX, NULL, "default", "default"); #else /* Always white on black */ mc_init_pair(DEFAULT_COLOR_INDEX, COLOR_WHITE, COLOR_BLACK); #endif /* !HAVE_SLANG */ } for (i = 0; i < ELEMENTS (color_map); i++){ if (!color_map [i].name) continue; mc_init_pair (i+1, color_map_fg(i), color_map_bg(i)); #ifndef HAVE_SLANG /* * ncurses doesn't remember bold attribute in the color pairs, * so we should keep track of it in a separate array. */ attr_pairs [i+1] = color_map [i].fg & A_BOLD; #endif /* !HAVE_SLANG */ } } load_dialog_colors (); } /* Functions necessary to implement syntax highlighting */ static int max_index = 0; static int alloc_color_pair (CTYPE foreground, CTYPE background) { mc_init_pair (++max_index, foreground, background); return max_index; } static struct colors_avail { struct colors_avail *next; char *fg, *bg; int index; } c = { 0, 0, 0, 0 }; #ifdef HAVE_SLANG void mc_init_pair (int index, CTYPE foreground, CTYPE background) { /* hack for transparent background for Eterm, rxvt or else */ if (background && !strcmp (background, "default")) background = NULL; /* if foreground is default, I guess we should use normal fore-color. */ SLtt_set_color (index, "", foreground, background); if (index > max_index) max_index = index; } int try_alloc_color_pair (char *fg, char *bg) { struct colors_avail *p = &c; c.index = EDITOR_NORMAL_COLOR_INDEX; for (;;) { if (((fg && p->fg) ? !strcmp (fg, p->fg) : fg == p->fg) != 0 && ((bg && p->bg) ? !strcmp (bg, p->bg) : bg == p->bg) != 0) return p->index; if (!p->next) break; p = p->next; } p->next = g_new (struct colors_avail, 1); p = p->next; p->next = 0; p->fg = fg ? g_strdup (fg) : 0; p->bg = bg ? g_strdup (bg) : 0; if (!fg) /* Index in color_map array = COLOR_INDEX - 1 */ fg = color_map[EDITOR_NORMAL_COLOR_INDEX - 1].fg; if (!bg) bg = color_map[EDITOR_NORMAL_COLOR_INDEX - 1].bg; p->index = alloc_color_pair (fg, bg); return p->index; } #else /* !HAVE_SLANG */ void mc_init_pair (int index, CTYPE foreground, CTYPE background) { init_pair (index, foreground, background); if (index > max_index) max_index = index; } int try_alloc_color_pair (char *fg, char *bg) { int fg_index, bg_index; int bold_attr; struct colors_avail *p = &c; c.index = EDITOR_NORMAL_COLOR_INDEX; for (;;) { if (((fg && p->fg) ? !strcmp (fg, p->fg) : fg == p->fg) != 0 && ((bg && p->bg) ? !strcmp (bg, p->bg) : bg == p->bg) != 0) return p->index; if (!p->next) break; p = p->next; } p->next = g_new (struct colors_avail, 1); p = p->next; p->next = 0; p->fg = fg ? g_strdup (fg) : 0; p->bg = bg ? g_strdup (bg) : 0; if (!fg) /* Index in color_map array = COLOR_INDEX - 1 */ fg_index = color_map[EDITOR_NORMAL_COLOR_INDEX - 1].fg; else get_color (fg, &fg_index); if (!bg) bg_index = color_map[EDITOR_NORMAL_COLOR_INDEX - 1].bg; else get_color (bg, &bg_index); bold_attr = fg_index & A_BOLD; fg_index = fg_index & COLOR_WHITE; bg_index = bg_index & COLOR_WHITE; p->index = alloc_color_pair (fg_index, bg_index); attr_pairs [p->index] = bold_attr; return p->index; } #endif /* !HAVE_SLANG */ void dealloc_color_pairs (void) { struct colors_avail *p, *next; for (p = c.next; p; p = next) { next = p->next; if (p->fg) g_free (p->fg); if (p->bg) g_free (p->bg); g_free (p); } c.next = NULL; }