/************************************************************************** * color.c -- This file is part of GNU nano. * * * * Copyright (C) 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009, * * 2010, 2011, 2013, 2014, 2015 Free Software Foundation, Inc. * * Copyright (C) 2014, 2015, 2016 Benno Schulenberg * * * * GNU nano 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 3 of the License, * * or (at your option) any later version. * * * * GNU nano 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, see http://www.gnu.org/licenses/. * * * **************************************************************************/ #include "proto.h" #include #include #include #include #include #ifdef HAVE_MAGIC_H #include #endif #ifndef DISABLE_COLOR /* Initialize the colors for nano's interface, and assign pair numbers * for the colors in each syntax. */ void set_colorpairs(void) { const syntaxtype *sint; bool using_defaults = FALSE; short foreground, background; size_t i; /* Tell ncurses to enable colors. */ start_color(); #ifdef HAVE_USE_DEFAULT_COLORS /* Allow using the default colors, if available. */ using_defaults = (use_default_colors() != ERR); #endif /* Initialize the color pairs for nano's interface elements. */ for (i = 0; i < NUMBER_OF_ELEMENTS; i++) { bool bright = FALSE; if (parse_color_names(specified_color_combo[i], &foreground, &background, &bright)) { if (foreground == -1 && !using_defaults) foreground = COLOR_WHITE; if (background == -1 && !using_defaults) background = COLOR_BLACK; init_pair(i + 1, foreground, background); interface_color_pair[i] = COLOR_PAIR(i + 1) | (bright ? A_BOLD : A_NORMAL); } else { if (i != FUNCTION_TAG) interface_color_pair[i] = hilite_attribute; else interface_color_pair[i] = A_NORMAL; } free(specified_color_combo[i]); specified_color_combo[i] = NULL; } /* For each syntax, go through its list of colors and assign each * its pair number, giving identical color pairs the same number. */ for (sint = syntaxes; sint != NULL; sint = sint->next) { colortype *ink; int new_number = NUMBER_OF_ELEMENTS + 1; for (ink = sint->color; ink != NULL; ink = ink->next) { const colortype *beforenow = sint->color; while (beforenow != ink && (beforenow->fg != ink->fg || beforenow->bg != ink->bg || beforenow->bright != ink->bright)) beforenow = beforenow->next; if (beforenow != ink) ink->pairnum = beforenow->pairnum; else ink->pairnum = new_number++; ink->attributes = COLOR_PAIR(ink->pairnum) | (ink->bright ? A_BOLD : A_NORMAL); } } } /* Initialize the color information. */ void color_init(void) { const colortype *ink; bool using_defaults = FALSE; short foreground, background; assert(openfile != NULL); /* If the terminal is not capable of colors, forget it. */ if (!has_colors()) return; #ifdef HAVE_USE_DEFAULT_COLORS /* Allow using the default colors, if available. */ using_defaults = (use_default_colors() != ERR); #endif /* For each coloring expression, initialize the color pair. */ for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) { foreground = ink->fg; background = ink->bg; if (foreground == -1 && !using_defaults) foreground = COLOR_WHITE; if (background == -1 && !using_defaults) background = COLOR_BLACK; init_pair(ink->pairnum, foreground, background); #ifdef DEBUG fprintf(stderr, "init_pair(): fg = %hd, bg = %hd\n", foreground, background); #endif } } /* Try to match the given shibboleth string with one of the regexes in * the list starting at head. Return TRUE upon success. */ bool found_in_list(regexlisttype *head, const char *shibboleth) { regexlisttype *item; regex_t rgx; for (item = head; item != NULL; item = item->next) { regcomp(&rgx, fixbounds(item->full_regex), NANO_REG_EXTENDED); if (regexec(&rgx, shibboleth, 0, NULL, 0) == 0) { regfree(&rgx); return TRUE; } regfree(&rgx); } return FALSE; } /* Update the color information based on the current filename and content. */ void color_update(void) { syntaxtype *sint = NULL; colortype *ink; assert(openfile != NULL); /* If the rcfiles were not read, or contained no syntaxes, get out. */ if (syntaxes == NULL) return; /* If we specified a syntax-override string, use it. */ if (syntaxstr != NULL) { /* An override of "none" is like having no syntax at all. */ if (strcmp(syntaxstr, "none") == 0) return; for (sint = syntaxes; sint != NULL; sint = sint->next) { if (strcmp(sint->name, syntaxstr) == 0) break; } if (sint == NULL) statusline(ALERT, _("Unknown syntax name: %s"), syntaxstr); } /* If no syntax-override string was specified, or it didn't match, * try finding a syntax based on the filename (extension). */ if (sint == NULL) { char *reserved = charalloc(PATH_MAX + 1); char *currentdir = getcwd(reserved, PATH_MAX + 1); char *joinednames = charalloc(PATH_MAX + 1); char *fullname = NULL; if (currentdir == NULL) free(reserved); else { /* Concatenate the current working directory with the * specified filename, and canonicalize the result. */ sprintf(joinednames, "%s/%s", currentdir, openfile->filename); fullname = get_full_path(joinednames); free(currentdir); } if (fullname == NULL) fullname = mallocstrcpy(fullname, openfile->filename); for (sint = syntaxes; sint != NULL; sint = sint->next) { if (found_in_list(sint->extensions, fullname)) break; } free(joinednames); free(fullname); } /* If the filename didn't match anything, try the first line. */ if (sint == NULL) { #ifdef DEBUG fprintf(stderr, "No result from file extension, trying headerline...\n"); #endif for (sint = syntaxes; sint != NULL; sint = sint->next) { if (found_in_list(sint->headers, openfile->fileage->data)) break; } } #ifdef HAVE_LIBMAGIC /* If we still don't have an answer, try using magic. */ if (sint == NULL) { struct stat fileinfo; magic_t cookie = NULL; const char *magicstring = NULL; #ifdef DEBUG fprintf(stderr, "No result from headerline either, trying libmagic...\n"); #endif if (stat(openfile->filename, &fileinfo) == 0) { /* Open the magic database and get a diagnosis of the file. */ cookie = magic_open(MAGIC_SYMLINK | #ifdef DEBUG MAGIC_DEBUG | MAGIC_CHECK | #endif MAGIC_ERROR); if (cookie == NULL || magic_load(cookie, NULL) < 0) statusline(ALERT, _("magic_load() failed: %s"), strerror(errno)); else { magicstring = magic_file(cookie, openfile->filename); if (magicstring == NULL) statusline(ALERT, _("magic_file(%s) failed: %s"), openfile->filename, magic_error(cookie)); #ifdef DEBUG fprintf(stderr, "Returned magic string is: %s\n", magicstring); #endif } } /* Now try and find a syntax that matches the magic string. */ if (magicstring != NULL) { for (sint = syntaxes; sint != NULL; sint = sint->next) { if (found_in_list(sint->magics, magicstring)) break; } } if (stat(openfile->filename, &fileinfo) == 0) magic_close(cookie); } #endif /* HAVE_LIBMAGIC */ /* If nothing at all matched, see if there is a default syntax. */ if (sint == NULL) { for (sint = syntaxes; sint != NULL; sint = sint->next) { if (strcmp(sint->name, "default") == 0) break; } } openfile->syntax = sint; openfile->colorstrings = (sint == NULL ? NULL : sint->color); /* If a syntax was found, compile its specified regexes (which have * already been checked for validity when they were read in). */ for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) { if (ink->start == NULL) { ink->start = (regex_t *)nmalloc(sizeof(regex_t)); regcomp(ink->start, fixbounds(ink->start_regex), ink->rex_flags); } if (ink->end_regex != NULL && ink->end == NULL) { ink->end = (regex_t *)nmalloc(sizeof(regex_t)); regcomp(ink->end, fixbounds(ink->end_regex), ink->rex_flags); } } } /* Reset the multiline coloring cache for one specific regex (given by * index) for lines that need reevaluation. */ void reset_multis_for_id(filestruct *fileptr, int index) { filestruct *row; /* Reset the cache of earlier lines, as far back as needed. */ for (row = fileptr->prev; row != NULL; row = row->prev) { alloc_multidata_if_needed(row); if (row->multidata[index] == CNONE) break; row->multidata[index] = -1; } for (; row != NULL; row = row->prev) { alloc_multidata_if_needed(row); if (row->multidata[index] != CNONE) break; row->multidata[index] = -1; } /* Reset the cache of the current line. */ fileptr->multidata[index] = -1; /* Reset the cache of later lines, as far ahead as needed. */ for (row = fileptr->next; row != NULL; row = row->next) { alloc_multidata_if_needed(row); if (row->multidata[index] == CNONE) break; row->multidata[index] = -1; } for (; row != NULL; row = row->next) { alloc_multidata_if_needed(row); if (row->multidata[index] != CNONE) break; row->multidata[index] = -1; } refresh_needed = TRUE; } /* Reset multi-line strings around the filestruct fileptr, trying to be * smart about stopping. Bool force means: reset everything regardless, * useful when we don't know how much screen state has changed. */ void reset_multis(filestruct *fileptr, bool force) { const colortype *ink; int nobegin = 0, noend = 0; regmatch_t startmatch, endmatch; /* If there is no syntax or no multiline regex, there is nothing to do. */ if (openfile->syntax == NULL || openfile->syntax->nmultis == 0) return; for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) { /* If it's not a multi-line regex, amscray. */ if (ink->end == NULL) continue; alloc_multidata_if_needed(fileptr); if (force == FALSE) { /* Check whether the multidata still matches the current situation. */ nobegin = regexec(ink->start, fileptr->data, 1, &startmatch, 0); noend = regexec(ink->end, fileptr->data, 1, &endmatch, 0); if ((fileptr->multidata[ink->id] == CWHOLELINE || fileptr->multidata[ink->id] == CNONE) && nobegin && noend) continue; else if (fileptr->multidata[ink->id] == CSTARTENDHERE && !nobegin && !noend && startmatch.rm_so < endmatch.rm_so) continue; else if (fileptr->multidata[ink->id] == CBEGINBEFORE && nobegin && !noend) continue; else if (fileptr->multidata[ink->id] == CENDAFTER && !nobegin && noend) continue; } /* If we got here, things have changed. */ reset_multis_for_id(fileptr, ink->id); /* If start and end are the same, push the resets further. */ if (force == FALSE && !nobegin && !noend && startmatch.rm_so == endmatch.rm_so) reset_multis_for_id(fileptr, ink->id); } } /* Allocate (for one line) the cache space for multiline color regexes. */ void alloc_multidata_if_needed(filestruct *fileptr) { int i; if (fileptr->multidata == NULL) { fileptr->multidata = (short *)nmalloc(openfile->syntax->nmultis * sizeof(short)); for (i = 0; i < openfile->syntax->nmultis; i++) fileptr->multidata[i] = -1; } } /* Precalculate the multi-line start and end regex info so we can * speed up rendering (with any hope at all...). */ void precalc_multicolorinfo(void) { const colortype *ink; regmatch_t startmatch, endmatch; filestruct *fileptr, *endptr; if (openfile->colorstrings == NULL || ISSET(NO_COLOR_SYNTAX)) return; #ifdef DEBUG fprintf(stderr, "Entering precalculation of multiline color info\n"); #endif for (ink = openfile->colorstrings; ink != NULL; ink = ink->next) { /* If this is not a multi-line regex, skip it. */ if (ink->end == NULL) continue; #ifdef DEBUG fprintf(stderr, "Starting work on color id %d\n", ink->id); #endif for (fileptr = openfile->fileage; fileptr != NULL; fileptr = fileptr->next) { int startx = 0, nostart = 0; #ifdef DEBUG fprintf(stderr, "working on lineno %ld... ", (long)fileptr->lineno); #endif alloc_multidata_if_needed(fileptr); while ((nostart = regexec(ink->start, &fileptr->data[startx], 1, &startmatch, (startx == 0) ? 0 : REG_NOTBOL)) == 0) { /* Look for an end, and start marking how many lines are * encompassed, which should speed up rendering later. */ startx += startmatch.rm_eo; #ifdef DEBUG fprintf(stderr, "start found at pos %lu... ", (unsigned long)startx); #endif /* Look first on this line for an end. */ if (regexec(ink->end, &fileptr->data[startx], 1, &endmatch, (startx == 0) ? 0 : REG_NOTBOL) == 0) { startx += endmatch.rm_eo; /* Step ahead when both start and end are mere anchors. */ if (startmatch.rm_so == startmatch.rm_eo && endmatch.rm_so == endmatch.rm_eo) startx += 1; fileptr->multidata[ink->id] = CSTARTENDHERE; #ifdef DEBUG fprintf(stderr, "end found on this line\n"); #endif continue; } /* Nice, we didn't find the end regex on this line. Let's start looking for it. */ for (endptr = fileptr->next; endptr != NULL; endptr = endptr->next) { #ifdef DEBUG fprintf(stderr, "\nadvancing to line %ld to find end... ", (long)endptr->lineno); #endif if (regexec(ink->end, endptr->data, 1, &endmatch, 0) == 0) break; } if (endptr == NULL) { #ifdef DEBUG fprintf(stderr, "no end found, breaking out\n"); #endif break; } #ifdef DEBUG fprintf(stderr, "end found\n"); #endif /* We found it, we found it, la la la la la. Mark all * the lines in between and the end properly. */ fileptr->multidata[ink->id] = CENDAFTER; #ifdef DEBUG fprintf(stderr, "marking line %ld as CENDAFTER\n", (long)fileptr->lineno); #endif for (fileptr = fileptr->next; fileptr != endptr; fileptr = fileptr->next) { alloc_multidata_if_needed(fileptr); fileptr->multidata[ink->id] = CWHOLELINE; #ifdef DEBUG fprintf(stderr, "marking intermediary line %ld as CWHOLELINE\n", (long)fileptr->lineno); #endif } alloc_multidata_if_needed(endptr); fileptr->multidata[ink->id] = CBEGINBEFORE; #ifdef DEBUG fprintf(stderr, "marking line %ld as CBEGINBEFORE\n", (long)fileptr->lineno); #endif /* Skip to the end point of the match. */ startx = endmatch.rm_eo; #ifdef DEBUG fprintf(stderr, "jumping to line %ld pos %lu to continue\n", (long)fileptr->lineno, (unsigned long)startx); #endif } if (nostart && startx == 0) { #ifdef DEBUG fprintf(stderr, "no match\n"); #endif fileptr->multidata[ink->id] = CNONE; continue; } } } } #endif /* !DISABLE_COLOR */