xref: /freebsd/contrib/dialog/rc.c (revision a96ef4501919d7ac08e94e98dc34b0bdd744802b)
14c8945a0SNathan Whitehorn /*
2*a96ef450SBaptiste Daroussin  *  $Id: rc.c,v 1.60 2020/11/25 00:06:40 tom Exp $
34c8945a0SNathan Whitehorn  *
44c8945a0SNathan Whitehorn  *  rc.c -- routines for processing the configuration file
54c8945a0SNathan Whitehorn  *
6*a96ef450SBaptiste Daroussin  *  Copyright 2000-2019,2020	Thomas E. Dickey
74c8945a0SNathan Whitehorn  *
84c8945a0SNathan Whitehorn  *  This program is free software; you can redistribute it and/or modify
94c8945a0SNathan Whitehorn  *  it under the terms of the GNU Lesser General Public License, version 2.1
104c8945a0SNathan Whitehorn  *  as published by the Free Software Foundation.
114c8945a0SNathan Whitehorn  *
124c8945a0SNathan Whitehorn  *  This program is distributed in the hope that it will be useful, but
134c8945a0SNathan Whitehorn  *  WITHOUT ANY WARRANTY; without even the implied warranty of
144c8945a0SNathan Whitehorn  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
154c8945a0SNathan Whitehorn  *  Lesser General Public License for more details.
164c8945a0SNathan Whitehorn  *
174c8945a0SNathan Whitehorn  *  You should have received a copy of the GNU Lesser General Public
184c8945a0SNathan Whitehorn  *  License along with this program; if not, write to
194c8945a0SNathan Whitehorn  *	Free Software Foundation, Inc.
204c8945a0SNathan Whitehorn  *	51 Franklin St., Fifth Floor
214c8945a0SNathan Whitehorn  *	Boston, MA 02110, USA.
224c8945a0SNathan Whitehorn  *
234c8945a0SNathan Whitehorn  *  An earlier version of this program lists as authors
244c8945a0SNathan Whitehorn  *	Savio Lam (lam836@cs.cuhk.hk)
254c8945a0SNathan Whitehorn  */
264c8945a0SNathan Whitehorn 
274c8945a0SNathan Whitehorn #include <dialog.h>
284c8945a0SNathan Whitehorn 
294c8945a0SNathan Whitehorn #include <dlg_keys.h>
304c8945a0SNathan Whitehorn 
314c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
324c8945a0SNathan Whitehorn #include <dlg_colors.h>
33*a96ef450SBaptiste Daroussin #include <dlg_internals.h>
34*a96ef450SBaptiste Daroussin 
35*a96ef450SBaptiste Daroussin #define L_PAREN '('
36*a96ef450SBaptiste Daroussin #define R_PAREN ')'
37*a96ef450SBaptiste Daroussin 
38*a96ef450SBaptiste Daroussin #define MIN_TOKEN 3
39*a96ef450SBaptiste Daroussin #ifdef HAVE_RC_FILE2
40*a96ef450SBaptiste Daroussin #define MAX_TOKEN 5
41*a96ef450SBaptiste Daroussin #else
42*a96ef450SBaptiste Daroussin #define MAX_TOKEN MIN_TOKEN
43*a96ef450SBaptiste Daroussin #endif
44*a96ef450SBaptiste Daroussin 
45*a96ef450SBaptiste Daroussin #define UNKNOWN_COLOR -2
464c8945a0SNathan Whitehorn 
474c8945a0SNathan Whitehorn /*
484c8945a0SNathan Whitehorn  * For matching color names with color values
494c8945a0SNathan Whitehorn  */
504c8945a0SNathan Whitehorn static const color_names_st color_names[] =
514c8945a0SNathan Whitehorn {
524c8945a0SNathan Whitehorn #ifdef HAVE_USE_DEFAULT_COLORS
534c8945a0SNathan Whitehorn     {"DEFAULT", -1},
544c8945a0SNathan Whitehorn #endif
554c8945a0SNathan Whitehorn     {"BLACK", COLOR_BLACK},
564c8945a0SNathan Whitehorn     {"RED", COLOR_RED},
574c8945a0SNathan Whitehorn     {"GREEN", COLOR_GREEN},
584c8945a0SNathan Whitehorn     {"YELLOW", COLOR_YELLOW},
594c8945a0SNathan Whitehorn     {"BLUE", COLOR_BLUE},
604c8945a0SNathan Whitehorn     {"MAGENTA", COLOR_MAGENTA},
614c8945a0SNathan Whitehorn     {"CYAN", COLOR_CYAN},
624c8945a0SNathan Whitehorn     {"WHITE", COLOR_WHITE},
634c8945a0SNathan Whitehorn };				/* color names */
64*a96ef450SBaptiste Daroussin #define COLOR_COUNT     TableSize(color_names)
654c8945a0SNathan Whitehorn #endif /* HAVE_COLOR */
664c8945a0SNathan Whitehorn 
674c8945a0SNathan Whitehorn #define GLOBALRC "/etc/dialogrc"
684c8945a0SNathan Whitehorn #define DIALOGRC ".dialogrc"
694c8945a0SNathan Whitehorn 
704c8945a0SNathan Whitehorn /* Types of values */
714c8945a0SNathan Whitehorn #define VAL_INT  0
724c8945a0SNathan Whitehorn #define VAL_STR  1
734c8945a0SNathan Whitehorn #define VAL_BOOL 2
744c8945a0SNathan Whitehorn 
754c8945a0SNathan Whitehorn /* Type of line in configuration file */
764c8945a0SNathan Whitehorn typedef enum {
774c8945a0SNathan Whitehorn     LINE_ERROR = -1,
784c8945a0SNathan Whitehorn     LINE_EQUALS,
794c8945a0SNathan Whitehorn     LINE_EMPTY
804c8945a0SNathan Whitehorn } PARSE_LINE;
814c8945a0SNathan Whitehorn 
824c8945a0SNathan Whitehorn /* number of configuration variables */
83*a96ef450SBaptiste Daroussin #define VAR_COUNT        TableSize(vars)
844c8945a0SNathan Whitehorn 
854c8945a0SNathan Whitehorn /* check if character is string quoting characters */
86f4f33ea0SBaptiste Daroussin #define isquote(c)       ((c) == '"' || (c) == '\'')
874c8945a0SNathan Whitehorn 
884c8945a0SNathan Whitehorn /* get last character of string */
894c8945a0SNathan Whitehorn #define lastch(str)      str[strlen(str)-1]
904c8945a0SNathan Whitehorn 
914c8945a0SNathan Whitehorn /*
924c8945a0SNathan Whitehorn  * Configuration variables
934c8945a0SNathan Whitehorn  */
944c8945a0SNathan Whitehorn typedef struct {
954c8945a0SNathan Whitehorn     const char *name;		/* name of configuration variable as in DIALOGRC */
964c8945a0SNathan Whitehorn     void *var;			/* address of actual variable to change */
974c8945a0SNathan Whitehorn     int type;			/* type of value */
984c8945a0SNathan Whitehorn     const char *comment;	/* comment to put in "rc" file */
994c8945a0SNathan Whitehorn } vars_st;
1004c8945a0SNathan Whitehorn 
1014c8945a0SNathan Whitehorn /*
1024c8945a0SNathan Whitehorn  * This table should contain only references to dialog_state, since dialog_vars
1034c8945a0SNathan Whitehorn  * is reset specially in dialog.c before each widget.
1044c8945a0SNathan Whitehorn  */
1054c8945a0SNathan Whitehorn static const vars_st vars[] =
1064c8945a0SNathan Whitehorn {
1074c8945a0SNathan Whitehorn     {"aspect",
1084c8945a0SNathan Whitehorn      &dialog_state.aspect_ratio,
1094c8945a0SNathan Whitehorn      VAL_INT,
1104c8945a0SNathan Whitehorn      "Set aspect-ration."},
1114c8945a0SNathan Whitehorn 
1124c8945a0SNathan Whitehorn     {"separate_widget",
1134c8945a0SNathan Whitehorn      &dialog_state.separate_str,
1144c8945a0SNathan Whitehorn      VAL_STR,
1154c8945a0SNathan Whitehorn      "Set separator (for multiple widgets output)."},
1164c8945a0SNathan Whitehorn 
1174c8945a0SNathan Whitehorn     {"tab_len",
1184c8945a0SNathan Whitehorn      &dialog_state.tab_len,
1194c8945a0SNathan Whitehorn      VAL_INT,
1204c8945a0SNathan Whitehorn      "Set tab-length (for textbox tab-conversion)."},
1214c8945a0SNathan Whitehorn 
1224c8945a0SNathan Whitehorn     {"visit_items",
1234c8945a0SNathan Whitehorn      &dialog_state.visit_items,
1244c8945a0SNathan Whitehorn      VAL_BOOL,
1254c8945a0SNathan Whitehorn      "Make tab-traversal for checklist, etc., include the list."},
1264c8945a0SNathan Whitehorn 
1274c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
1284c8945a0SNathan Whitehorn     {"use_shadow",
1294c8945a0SNathan Whitehorn      &dialog_state.use_shadow,
1304c8945a0SNathan Whitehorn      VAL_BOOL,
1314c8945a0SNathan Whitehorn      "Shadow dialog boxes? This also turns on color."},
1324c8945a0SNathan Whitehorn 
1334c8945a0SNathan Whitehorn     {"use_colors",
1344c8945a0SNathan Whitehorn      &dialog_state.use_colors,
1354c8945a0SNathan Whitehorn      VAL_BOOL,
1364c8945a0SNathan Whitehorn      "Turn color support ON or OFF"},
1374c8945a0SNathan Whitehorn #endif				/* HAVE_COLOR */
1384c8945a0SNathan Whitehorn };				/* vars */
1394c8945a0SNathan Whitehorn 
1404c8945a0SNathan Whitehorn static int
skip_whitespace(char * str,int n)1414c8945a0SNathan Whitehorn skip_whitespace(char *str, int n)
1424c8945a0SNathan Whitehorn {
143f4f33ea0SBaptiste Daroussin     while (isblank(UCH(str[n])) && str[n] != '\0')
1444c8945a0SNathan Whitehorn 	n++;
1454c8945a0SNathan Whitehorn     return n;
1464c8945a0SNathan Whitehorn }
1474c8945a0SNathan Whitehorn 
1484c8945a0SNathan Whitehorn static int
skip_keyword(char * str,int n)1494c8945a0SNathan Whitehorn skip_keyword(char *str, int n)
1504c8945a0SNathan Whitehorn {
1514c8945a0SNathan Whitehorn     while (isalnum(UCH(str[n])) && str[n] != '\0')
1524c8945a0SNathan Whitehorn 	n++;
1534c8945a0SNathan Whitehorn     return n;
1544c8945a0SNathan Whitehorn }
1554c8945a0SNathan Whitehorn 
156*a96ef450SBaptiste Daroussin static void
trim_token(char ** tok)157*a96ef450SBaptiste Daroussin trim_token(char **tok)
158*a96ef450SBaptiste Daroussin {
159*a96ef450SBaptiste Daroussin     char *tmp = *tok + skip_whitespace(*tok, 0);
160*a96ef450SBaptiste Daroussin 
161*a96ef450SBaptiste Daroussin     *tok = tmp;
162*a96ef450SBaptiste Daroussin 
163*a96ef450SBaptiste Daroussin     while (*tmp != '\0' && !isblank(UCH(*tmp)))
164*a96ef450SBaptiste Daroussin 	tmp++;
165*a96ef450SBaptiste Daroussin 
166*a96ef450SBaptiste Daroussin     *tmp = '\0';
167*a96ef450SBaptiste Daroussin }
168*a96ef450SBaptiste Daroussin 
169*a96ef450SBaptiste Daroussin static int
from_boolean(const char * str)170*a96ef450SBaptiste Daroussin from_boolean(const char *str)
171*a96ef450SBaptiste Daroussin {
172*a96ef450SBaptiste Daroussin     int code = -1;
173*a96ef450SBaptiste Daroussin 
174*a96ef450SBaptiste Daroussin     if (str != NULL && *str != '\0') {
175*a96ef450SBaptiste Daroussin 	if (!dlg_strcmp(str, "ON")) {
176*a96ef450SBaptiste Daroussin 	    code = 1;
177*a96ef450SBaptiste Daroussin 	} else if (!dlg_strcmp(str, "OFF")) {
178*a96ef450SBaptiste Daroussin 	    code = 0;
179*a96ef450SBaptiste Daroussin 	}
180*a96ef450SBaptiste Daroussin     }
181*a96ef450SBaptiste Daroussin     return code;
182*a96ef450SBaptiste Daroussin }
183*a96ef450SBaptiste Daroussin 
184*a96ef450SBaptiste Daroussin static int
from_color_name(const char * str)185*a96ef450SBaptiste Daroussin from_color_name(const char *str)
186*a96ef450SBaptiste Daroussin {
187*a96ef450SBaptiste Daroussin     int code = UNKNOWN_COLOR;
188*a96ef450SBaptiste Daroussin 
189*a96ef450SBaptiste Daroussin     if (str != NULL && *str != '\0') {
190*a96ef450SBaptiste Daroussin 	size_t i;
191*a96ef450SBaptiste Daroussin 
192*a96ef450SBaptiste Daroussin 	for (i = 0; i < COLOR_COUNT; ++i) {
193*a96ef450SBaptiste Daroussin 	    if (!dlg_strcmp(str, color_names[i].name)) {
194*a96ef450SBaptiste Daroussin 		code = color_names[i].value;
195*a96ef450SBaptiste Daroussin 		break;
196*a96ef450SBaptiste Daroussin 	    }
197*a96ef450SBaptiste Daroussin 	}
198*a96ef450SBaptiste Daroussin     }
199*a96ef450SBaptiste Daroussin     return code;
200*a96ef450SBaptiste Daroussin }
201*a96ef450SBaptiste Daroussin 
2024c8945a0SNathan Whitehorn static int
find_vars(char * name)2034c8945a0SNathan Whitehorn find_vars(char *name)
2044c8945a0SNathan Whitehorn {
2054c8945a0SNathan Whitehorn     int result = -1;
2064c8945a0SNathan Whitehorn     unsigned i;
2074c8945a0SNathan Whitehorn 
2084c8945a0SNathan Whitehorn     for (i = 0; i < VAR_COUNT; i++) {
2094c8945a0SNathan Whitehorn 	if (dlg_strcmp(vars[i].name, name) == 0) {
2104c8945a0SNathan Whitehorn 	    result = (int) i;
2114c8945a0SNathan Whitehorn 	    break;
2124c8945a0SNathan Whitehorn 	}
2134c8945a0SNathan Whitehorn     }
2144c8945a0SNathan Whitehorn     return result;
2154c8945a0SNathan Whitehorn }
2164c8945a0SNathan Whitehorn 
2174c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
2184c8945a0SNathan Whitehorn static int
find_color(char * name)2194c8945a0SNathan Whitehorn find_color(char *name)
2204c8945a0SNathan Whitehorn {
2214c8945a0SNathan Whitehorn     int result = -1;
2224c8945a0SNathan Whitehorn     int i;
2234c8945a0SNathan Whitehorn     int limit = dlg_color_count();
2244c8945a0SNathan Whitehorn 
2254c8945a0SNathan Whitehorn     for (i = 0; i < limit; i++) {
2264c8945a0SNathan Whitehorn 	if (dlg_strcmp(dlg_color_table[i].name, name) == 0) {
2274c8945a0SNathan Whitehorn 	    result = i;
2284c8945a0SNathan Whitehorn 	    break;
2294c8945a0SNathan Whitehorn 	}
2304c8945a0SNathan Whitehorn     }
2314c8945a0SNathan Whitehorn     return result;
2324c8945a0SNathan Whitehorn }
2334c8945a0SNathan Whitehorn 
234*a96ef450SBaptiste Daroussin static const char *
to_color_name(int code)235*a96ef450SBaptiste Daroussin to_color_name(int code)
2364c8945a0SNathan Whitehorn {
237*a96ef450SBaptiste Daroussin     const char *result = "?";
238*a96ef450SBaptiste Daroussin     size_t n;
239*a96ef450SBaptiste Daroussin     for (n = 0; n < TableSize(color_names); ++n) {
240*a96ef450SBaptiste Daroussin 	if (code == color_names[n].value) {
241*a96ef450SBaptiste Daroussin 	    result = color_names[n].name;
242*a96ef450SBaptiste Daroussin 	    break;
243*a96ef450SBaptiste Daroussin 	}
244*a96ef450SBaptiste Daroussin     }
245*a96ef450SBaptiste Daroussin     return result;
246*a96ef450SBaptiste Daroussin }
2474c8945a0SNathan Whitehorn 
248*a96ef450SBaptiste Daroussin static const char *
to_boolean(int code)249*a96ef450SBaptiste Daroussin to_boolean(int code)
250*a96ef450SBaptiste Daroussin {
251*a96ef450SBaptiste Daroussin     return code ? "ON" : "OFF";
2524c8945a0SNathan Whitehorn }
2534c8945a0SNathan Whitehorn 
2544c8945a0SNathan Whitehorn /*
2554c8945a0SNathan Whitehorn  * Extract the foreground, background and highlight values from an attribute
256*a96ef450SBaptiste Daroussin  * represented as a string in one of these forms:
2574c8945a0SNathan Whitehorn  *
258*a96ef450SBaptiste Daroussin  * "(foreground,background,highlight,underline,reverse)"
259*a96ef450SBaptiste Daroussin  * "(foreground,background,highlight,underline)"
2604c8945a0SNathan Whitehorn  * "(foreground,background,highlight)"
261*a96ef450SBaptiste Daroussin  * "xxxx_color"
2624c8945a0SNathan Whitehorn  */
2634c8945a0SNathan Whitehorn static int
str_to_attr(char * str,DIALOG_COLORS * result)264*a96ef450SBaptiste Daroussin str_to_attr(char *str, DIALOG_COLORS * result)
2654c8945a0SNathan Whitehorn {
266*a96ef450SBaptiste Daroussin     char *tokens[MAX_TOKEN + 1];
267*a96ef450SBaptiste Daroussin     char tempstr[MAX_LEN + 1];
2682a3e3873SBaptiste Daroussin     size_t have;
269*a96ef450SBaptiste Daroussin     size_t i = 0;
270*a96ef450SBaptiste Daroussin     size_t tok_count = 0;
2714c8945a0SNathan Whitehorn 
272*a96ef450SBaptiste Daroussin     memset(result, 0, sizeof(*result));
273*a96ef450SBaptiste Daroussin     result->fg = -1;
274*a96ef450SBaptiste Daroussin     result->bg = -1;
275*a96ef450SBaptiste Daroussin     result->hilite = -1;
276*a96ef450SBaptiste Daroussin 
277*a96ef450SBaptiste Daroussin     if (str[0] != L_PAREN || lastch(str) != R_PAREN) {
278*a96ef450SBaptiste Daroussin 	int ret;
279*a96ef450SBaptiste Daroussin 
280*a96ef450SBaptiste Daroussin 	if ((ret = find_color(str)) >= 0) {
281*a96ef450SBaptiste Daroussin 	    *result = dlg_color_table[ret];
2822a3e3873SBaptiste Daroussin 	    return 0;
2832a3e3873SBaptiste Daroussin 	}
284*a96ef450SBaptiste Daroussin 	/* invalid representation */
285*a96ef450SBaptiste Daroussin 	return -1;
2862a3e3873SBaptiste Daroussin     }
2874c8945a0SNathan Whitehorn 
2884c8945a0SNathan Whitehorn     /* remove the parenthesis */
2892a3e3873SBaptiste Daroussin     have = strlen(str);
2902a3e3873SBaptiste Daroussin     if (have > MAX_LEN) {
2912a3e3873SBaptiste Daroussin 	have = MAX_LEN - 1;
2922a3e3873SBaptiste Daroussin     } else {
2932a3e3873SBaptiste Daroussin 	have -= 2;
2942a3e3873SBaptiste Daroussin     }
2952a3e3873SBaptiste Daroussin     memcpy(tempstr, str + 1, have);
2962a3e3873SBaptiste Daroussin     tempstr[have] = '\0';
2974c8945a0SNathan Whitehorn 
298*a96ef450SBaptiste Daroussin     /* parse comma-separated tokens, allow up to
299*a96ef450SBaptiste Daroussin      * one more than max tokens to detect extras */
300*a96ef450SBaptiste Daroussin     while (tok_count < TableSize(tokens)) {
3014c8945a0SNathan Whitehorn 
302*a96ef450SBaptiste Daroussin 	tokens[tok_count++] = &tempstr[i];
3034c8945a0SNathan Whitehorn 
304*a96ef450SBaptiste Daroussin 	while (tempstr[i] != '\0' && tempstr[i] != ',')
3054c8945a0SNathan Whitehorn 	    i++;
3064c8945a0SNathan Whitehorn 
3074c8945a0SNathan Whitehorn 	if (tempstr[i] == '\0')
3084c8945a0SNathan Whitehorn 	    break;
309*a96ef450SBaptiste Daroussin 
310*a96ef450SBaptiste Daroussin 	tempstr[i++] = '\0';
3114c8945a0SNathan Whitehorn     }
3124c8945a0SNathan Whitehorn 
313*a96ef450SBaptiste Daroussin     if (tok_count < MIN_TOKEN || tok_count > MAX_TOKEN) {
314*a96ef450SBaptiste Daroussin 	/* invalid representation */
315*a96ef450SBaptiste Daroussin 	return -1;
316*a96ef450SBaptiste Daroussin     }
3174c8945a0SNathan Whitehorn 
318*a96ef450SBaptiste Daroussin     for (i = 0; i < tok_count; ++i)
319*a96ef450SBaptiste Daroussin 	trim_token(&tokens[i]);
3204c8945a0SNathan Whitehorn 
321*a96ef450SBaptiste Daroussin     /* validate */
322*a96ef450SBaptiste Daroussin     if (UNKNOWN_COLOR == (result->fg = from_color_name(tokens[0]))
323*a96ef450SBaptiste Daroussin 	|| UNKNOWN_COLOR == (result->bg = from_color_name(tokens[1]))
324*a96ef450SBaptiste Daroussin 	|| UNKNOWN_COLOR == (result->hilite = from_boolean(tokens[2]))
325*a96ef450SBaptiste Daroussin #ifdef HAVE_RC_FILE2
326*a96ef450SBaptiste Daroussin 	|| (tok_count >= 4 && (result->ul = from_boolean(tokens[3])) == -1)
327*a96ef450SBaptiste Daroussin 	|| (tok_count >= 5 && (result->rv = from_boolean(tokens[4])) == -1)
328*a96ef450SBaptiste Daroussin #endif /* HAVE_RC_FILE2 */
329*a96ef450SBaptiste Daroussin 	) {
330*a96ef450SBaptiste Daroussin 	/* invalid representation */
331*a96ef450SBaptiste Daroussin 	return -1;
332*a96ef450SBaptiste Daroussin     }
3334c8945a0SNathan Whitehorn 
3344c8945a0SNathan Whitehorn     return 0;
3354c8945a0SNathan Whitehorn }
3364c8945a0SNathan Whitehorn #endif /* HAVE_COLOR */
3374c8945a0SNathan Whitehorn 
3384c8945a0SNathan Whitehorn /*
3394c8945a0SNathan Whitehorn  * Check if the line begins with a special keyword; if so, return true while
3404c8945a0SNathan Whitehorn  * pointing params to its parameters.
3414c8945a0SNathan Whitehorn  */
3424c8945a0SNathan Whitehorn static int
begins_with(char * line,const char * keyword,char ** params)3434c8945a0SNathan Whitehorn begins_with(char *line, const char *keyword, char **params)
3444c8945a0SNathan Whitehorn {
3454c8945a0SNathan Whitehorn     int i = skip_whitespace(line, 0);
3464c8945a0SNathan Whitehorn     int j = skip_keyword(line, i);
3474c8945a0SNathan Whitehorn 
3484c8945a0SNathan Whitehorn     if ((j - i) == (int) strlen(keyword)) {
3494c8945a0SNathan Whitehorn 	char save = line[j];
3504c8945a0SNathan Whitehorn 	line[j] = 0;
3514c8945a0SNathan Whitehorn 	if (!dlg_strcmp(keyword, line + i)) {
3524c8945a0SNathan Whitehorn 	    *params = line + skip_whitespace(line, j + 1);
3534c8945a0SNathan Whitehorn 	    return 1;
3544c8945a0SNathan Whitehorn 	}
3554c8945a0SNathan Whitehorn 	line[j] = save;
3564c8945a0SNathan Whitehorn     }
3574c8945a0SNathan Whitehorn 
3584c8945a0SNathan Whitehorn     return 0;
3594c8945a0SNathan Whitehorn }
3604c8945a0SNathan Whitehorn 
3614c8945a0SNathan Whitehorn /*
3624c8945a0SNathan Whitehorn  * Parse a line in the configuration file
3634c8945a0SNathan Whitehorn  *
3644c8945a0SNathan Whitehorn  * Each line is of the form:  "variable = value". On exit, 'var' will contain
3654c8945a0SNathan Whitehorn  * the variable name, and 'value' will contain the value string.
3664c8945a0SNathan Whitehorn  *
3674c8945a0SNathan Whitehorn  * Return values:
3684c8945a0SNathan Whitehorn  *
3694c8945a0SNathan Whitehorn  * LINE_EMPTY   - line is blank or comment
3704c8945a0SNathan Whitehorn  * LINE_EQUALS  - line contains "variable = value"
3714c8945a0SNathan Whitehorn  * LINE_ERROR   - syntax error in line
3724c8945a0SNathan Whitehorn  */
3734c8945a0SNathan Whitehorn static PARSE_LINE
parse_line(char * line,char ** var,char ** value)3744c8945a0SNathan Whitehorn parse_line(char *line, char **var, char **value)
3754c8945a0SNathan Whitehorn {
3764c8945a0SNathan Whitehorn     int i = 0;
3774c8945a0SNathan Whitehorn 
3784c8945a0SNathan Whitehorn     /* ignore white space at beginning of line */
3794c8945a0SNathan Whitehorn     i = skip_whitespace(line, i);
3804c8945a0SNathan Whitehorn 
3814c8945a0SNathan Whitehorn     if (line[i] == '\0')	/* line is blank */
3824c8945a0SNathan Whitehorn 	return LINE_EMPTY;
3834c8945a0SNathan Whitehorn     else if (line[i] == '#')	/* line is comment */
3844c8945a0SNathan Whitehorn 	return LINE_EMPTY;
3854c8945a0SNathan Whitehorn     else if (line[i] == '=')	/* variable names cannot start with a '=' */
3864c8945a0SNathan Whitehorn 	return LINE_ERROR;
3874c8945a0SNathan Whitehorn 
3884c8945a0SNathan Whitehorn     /* set 'var' to variable name */
3894c8945a0SNathan Whitehorn     *var = line + i++;		/* skip to next character */
3904c8945a0SNathan Whitehorn 
3914c8945a0SNathan Whitehorn     /* find end of variable name */
392f4f33ea0SBaptiste Daroussin     while (!isblank(UCH(line[i])) && line[i] != '=' && line[i] != '\0')
3934c8945a0SNathan Whitehorn 	i++;
3944c8945a0SNathan Whitehorn 
3954c8945a0SNathan Whitehorn     if (line[i] == '\0')	/* syntax error */
3964c8945a0SNathan Whitehorn 	return LINE_ERROR;
3974c8945a0SNathan Whitehorn     else if (line[i] == '=')
3984c8945a0SNathan Whitehorn 	line[i++] = '\0';
3994c8945a0SNathan Whitehorn     else {
4004c8945a0SNathan Whitehorn 	line[i++] = '\0';
4014c8945a0SNathan Whitehorn 
4024c8945a0SNathan Whitehorn 	/* skip white space before '=' */
4034c8945a0SNathan Whitehorn 	i = skip_whitespace(line, i);
4044c8945a0SNathan Whitehorn 
4054c8945a0SNathan Whitehorn 	if (line[i] != '=')	/* syntax error */
4064c8945a0SNathan Whitehorn 	    return LINE_ERROR;
4074c8945a0SNathan Whitehorn 	else
4084c8945a0SNathan Whitehorn 	    i++;		/* skip the '=' */
4094c8945a0SNathan Whitehorn     }
4104c8945a0SNathan Whitehorn 
4114c8945a0SNathan Whitehorn     /* skip white space after '=' */
4124c8945a0SNathan Whitehorn     i = skip_whitespace(line, i);
4134c8945a0SNathan Whitehorn 
4144c8945a0SNathan Whitehorn     if (line[i] == '\0')
4154c8945a0SNathan Whitehorn 	return LINE_ERROR;
4164c8945a0SNathan Whitehorn     else
4174c8945a0SNathan Whitehorn 	*value = line + i;	/* set 'value' to value string */
4184c8945a0SNathan Whitehorn 
4194c8945a0SNathan Whitehorn     /* trim trailing white space from 'value' */
4204c8945a0SNathan Whitehorn     i = (int) strlen(*value) - 1;
421f4f33ea0SBaptiste Daroussin     while (isblank(UCH((*value)[i])) && i > 0)
4224c8945a0SNathan Whitehorn 	i--;
4234c8945a0SNathan Whitehorn     (*value)[i + 1] = '\0';
4244c8945a0SNathan Whitehorn 
4254c8945a0SNathan Whitehorn     return LINE_EQUALS;		/* no syntax error in line */
4264c8945a0SNathan Whitehorn }
4274c8945a0SNathan Whitehorn 
4284c8945a0SNathan Whitehorn /*
4294c8945a0SNathan Whitehorn  * Create the configuration file
4304c8945a0SNathan Whitehorn  */
4314c8945a0SNathan Whitehorn void
dlg_create_rc(const char * filename)4324c8945a0SNathan Whitehorn dlg_create_rc(const char *filename)
4334c8945a0SNathan Whitehorn {
4344c8945a0SNathan Whitehorn     unsigned i;
4354c8945a0SNathan Whitehorn     FILE *rc_file;
4364c8945a0SNathan Whitehorn 
4374c8945a0SNathan Whitehorn     if ((rc_file = fopen(filename, "wt")) == NULL)
4384c8945a0SNathan Whitehorn 	dlg_exiterr("Error opening file for writing in dlg_create_rc().");
4394c8945a0SNathan Whitehorn 
4404c8945a0SNathan Whitehorn     fprintf(rc_file, "#\n\
4414c8945a0SNathan Whitehorn # Run-time configuration file for dialog\n\
4424c8945a0SNathan Whitehorn #\n\
4434c8945a0SNathan Whitehorn # Automatically generated by \"dialog --create-rc <file>\"\n\
4444c8945a0SNathan Whitehorn #\n\
4454c8945a0SNathan Whitehorn #\n\
4464c8945a0SNathan Whitehorn # Types of values:\n\
4474c8945a0SNathan Whitehorn #\n\
4484c8945a0SNathan Whitehorn # Number     -  <number>\n\
4494c8945a0SNathan Whitehorn # String     -  \"string\"\n\
4504c8945a0SNathan Whitehorn # Boolean    -  <ON|OFF>\n"
4514c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
452*a96ef450SBaptiste Daroussin #ifdef HAVE_RC_FILE2
453*a96ef450SBaptiste Daroussin 	    "\
454*a96ef450SBaptiste Daroussin # Attribute  -  (foreground,background,highlight?,underline?,reverse?)\n"
455*a96ef450SBaptiste Daroussin #else /* HAVE_RC_FILE2 */
4564c8945a0SNathan Whitehorn 	    "\
4574c8945a0SNathan Whitehorn # Attribute  -  (foreground,background,highlight?)\n"
458*a96ef450SBaptiste Daroussin #endif /* HAVE_RC_FILE2 */
459*a96ef450SBaptiste Daroussin #endif /* HAVE_COLOR */
4604c8945a0SNathan Whitehorn 	);
4614c8945a0SNathan Whitehorn 
4624c8945a0SNathan Whitehorn     /* Print an entry for each configuration variable */
4634c8945a0SNathan Whitehorn     for (i = 0; i < VAR_COUNT; i++) {
4644c8945a0SNathan Whitehorn 	fprintf(rc_file, "\n# %s\n", vars[i].comment);
4654c8945a0SNathan Whitehorn 	switch (vars[i].type) {
4664c8945a0SNathan Whitehorn 	case VAL_INT:
4674c8945a0SNathan Whitehorn 	    fprintf(rc_file, "%s = %d\n", vars[i].name,
4684c8945a0SNathan Whitehorn 		    *((int *) vars[i].var));
4694c8945a0SNathan Whitehorn 	    break;
4704c8945a0SNathan Whitehorn 	case VAL_STR:
4714c8945a0SNathan Whitehorn 	    fprintf(rc_file, "%s = \"%s\"\n", vars[i].name,
4724c8945a0SNathan Whitehorn 		    (char *) vars[i].var);
4734c8945a0SNathan Whitehorn 	    break;
4744c8945a0SNathan Whitehorn 	case VAL_BOOL:
4754c8945a0SNathan Whitehorn 	    fprintf(rc_file, "%s = %s\n", vars[i].name,
4764c8945a0SNathan Whitehorn 		    *((bool *) vars[i].var) ? "ON" : "OFF");
4774c8945a0SNathan Whitehorn 	    break;
4784c8945a0SNathan Whitehorn 	}
4794c8945a0SNathan Whitehorn     }
4804c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
4814c8945a0SNathan Whitehorn     for (i = 0; i < (unsigned) dlg_color_count(); ++i) {
4822a3e3873SBaptiste Daroussin 	unsigned j;
4832a3e3873SBaptiste Daroussin 	bool repeat = FALSE;
4844c8945a0SNathan Whitehorn 
4854c8945a0SNathan Whitehorn 	fprintf(rc_file, "\n# %s\n", dlg_color_table[i].comment);
4862a3e3873SBaptiste Daroussin 	for (j = 0; j != i; ++j) {
4872a3e3873SBaptiste Daroussin 	    if (dlg_color_table[i].fg == dlg_color_table[j].fg
4882a3e3873SBaptiste Daroussin 		&& dlg_color_table[i].bg == dlg_color_table[j].bg
4892a3e3873SBaptiste Daroussin 		&& dlg_color_table[i].hilite == dlg_color_table[j].hilite) {
4902a3e3873SBaptiste Daroussin 		fprintf(rc_file, "%s = %s\n",
4912a3e3873SBaptiste Daroussin 			dlg_color_table[i].name,
4922a3e3873SBaptiste Daroussin 			dlg_color_table[j].name);
4932a3e3873SBaptiste Daroussin 		repeat = TRUE;
4942a3e3873SBaptiste Daroussin 		break;
4952a3e3873SBaptiste Daroussin 	    }
4962a3e3873SBaptiste Daroussin 	}
4972a3e3873SBaptiste Daroussin 
4982a3e3873SBaptiste Daroussin 	if (!repeat) {
499*a96ef450SBaptiste Daroussin 	    fprintf(rc_file, "%s = %c", dlg_color_table[i].name, L_PAREN);
500*a96ef450SBaptiste Daroussin 	    fprintf(rc_file, "%s", to_color_name(dlg_color_table[i].fg));
501*a96ef450SBaptiste Daroussin 	    fprintf(rc_file, ",%s", to_color_name(dlg_color_table[i].bg));
502*a96ef450SBaptiste Daroussin 	    fprintf(rc_file, ",%s", to_boolean(dlg_color_table[i].hilite));
503*a96ef450SBaptiste Daroussin #ifdef HAVE_RC_FILE2
504*a96ef450SBaptiste Daroussin 	    if (dlg_color_table[i].ul || dlg_color_table[i].rv)
505*a96ef450SBaptiste Daroussin 		fprintf(rc_file, ",%s", to_boolean(dlg_color_table[i].ul));
506*a96ef450SBaptiste Daroussin 	    if (dlg_color_table[i].rv)
507*a96ef450SBaptiste Daroussin 		fprintf(rc_file, ",%s", to_boolean(dlg_color_table[i].rv));
508*a96ef450SBaptiste Daroussin #endif /* HAVE_RC_FILE2 */
509*a96ef450SBaptiste Daroussin 	    fprintf(rc_file, "%c\n", R_PAREN);
5104c8945a0SNathan Whitehorn 	}
5112a3e3873SBaptiste Daroussin     }
5124c8945a0SNathan Whitehorn #endif /* HAVE_COLOR */
5134c8945a0SNathan Whitehorn     dlg_dump_keys(rc_file);
5144c8945a0SNathan Whitehorn 
5154c8945a0SNathan Whitehorn     (void) fclose(rc_file);
5164c8945a0SNathan Whitehorn }
5174c8945a0SNathan Whitehorn 
518*a96ef450SBaptiste Daroussin static void
report_error(const char * filename,int line_no,const char * msg)519*a96ef450SBaptiste Daroussin report_error(const char *filename, int line_no, const char *msg)
520*a96ef450SBaptiste Daroussin {
521*a96ef450SBaptiste Daroussin     fprintf(stderr, "%s:%d: %s\n", filename, line_no, msg);
522*a96ef450SBaptiste Daroussin     dlg_trace_msg("%s:%d: %s\n", filename, line_no, msg);
523*a96ef450SBaptiste Daroussin }
524*a96ef450SBaptiste Daroussin 
5254c8945a0SNathan Whitehorn /*
5264c8945a0SNathan Whitehorn  * Parse the configuration file and set up variables
5274c8945a0SNathan Whitehorn  */
5284c8945a0SNathan Whitehorn int
dlg_parse_rc(void)5294c8945a0SNathan Whitehorn dlg_parse_rc(void)
5304c8945a0SNathan Whitehorn {
5314c8945a0SNathan Whitehorn     int i;
5324c8945a0SNathan Whitehorn     int l = 1;
5334c8945a0SNathan Whitehorn     PARSE_LINE parse;
5344c8945a0SNathan Whitehorn     char str[MAX_LEN + 1];
5354c8945a0SNathan Whitehorn     char *var;
5364c8945a0SNathan Whitehorn     char *value;
537*a96ef450SBaptiste Daroussin     char *filename;
5384c8945a0SNathan Whitehorn     int result = 0;
5394c8945a0SNathan Whitehorn     FILE *rc_file = 0;
5404c8945a0SNathan Whitehorn     char *params;
5414c8945a0SNathan Whitehorn 
5424c8945a0SNathan Whitehorn     /*
5434c8945a0SNathan Whitehorn      *  At startup, dialog determines the settings to use as follows:
5444c8945a0SNathan Whitehorn      *
5454c8945a0SNathan Whitehorn      *  a) if the environment variable $DIALOGRC is set, its value determines
5464c8945a0SNathan Whitehorn      *     the name of the configuration file.
5474c8945a0SNathan Whitehorn      *
5484c8945a0SNathan Whitehorn      *  b) if the file in (a) can't be found, use the file $HOME/.dialogrc
5494c8945a0SNathan Whitehorn      *     as the configuration file.
5504c8945a0SNathan Whitehorn      *
5514c8945a0SNathan Whitehorn      *  c) if the file in (b) can't be found, try using the GLOBALRC file.
5524c8945a0SNathan Whitehorn      *     Usually this will be /etc/dialogrc.
5534c8945a0SNathan Whitehorn      *
5544c8945a0SNathan Whitehorn      *  d) if the file in (c) cannot be found, use the compiled-in defaults.
5554c8945a0SNathan Whitehorn      */
5564c8945a0SNathan Whitehorn 
5574c8945a0SNathan Whitehorn     /* try step (a) */
558*a96ef450SBaptiste Daroussin     if ((filename = dlg_getenv_str("DIALOGRC")) != NULL)
559*a96ef450SBaptiste Daroussin 	rc_file = fopen(filename, "rt");
5604c8945a0SNathan Whitehorn 
5614c8945a0SNathan Whitehorn     if (rc_file == NULL) {	/* step (a) failed? */
5624c8945a0SNathan Whitehorn 	/* try step (b) */
563*a96ef450SBaptiste Daroussin 	if ((filename = dlg_getenv_str("HOME")) != NULL
564*a96ef450SBaptiste Daroussin 	    && strlen(filename) < MAX_LEN - (sizeof(DIALOGRC) + 3)) {
565*a96ef450SBaptiste Daroussin 	    if (filename[0] == '\0' || lastch(filename) == '/')
566*a96ef450SBaptiste Daroussin 		sprintf(str, "%s%s", filename, DIALOGRC);
5674c8945a0SNathan Whitehorn 	    else
568*a96ef450SBaptiste Daroussin 		sprintf(str, "%s/%s", filename, DIALOGRC);
569*a96ef450SBaptiste Daroussin 	    rc_file = fopen(filename = str, "rt");
5704c8945a0SNathan Whitehorn 	}
5714c8945a0SNathan Whitehorn     }
5724c8945a0SNathan Whitehorn 
5734c8945a0SNathan Whitehorn     if (rc_file == NULL) {	/* step (b) failed? */
5744c8945a0SNathan Whitehorn 	/* try step (c) */
5754c8945a0SNathan Whitehorn 	strcpy(str, GLOBALRC);
576*a96ef450SBaptiste Daroussin 	if ((rc_file = fopen(filename = str, "rt")) == NULL)
5774c8945a0SNathan Whitehorn 	    return 0;		/* step (c) failed, use default values */
5784c8945a0SNathan Whitehorn     }
5794c8945a0SNathan Whitehorn 
580*a96ef450SBaptiste Daroussin     DLG_TRACE(("# opened rc file \"%s\"\n", filename));
5814c8945a0SNathan Whitehorn     /* Scan each line and set variables */
5824c8945a0SNathan Whitehorn     while ((result == 0) && (fgets(str, MAX_LEN, rc_file) != NULL)) {
583f4f33ea0SBaptiste Daroussin 	DLG_TRACE(("#\t%s", str));
5844c8945a0SNathan Whitehorn 	if (*str == '\0' || lastch(str) != '\n') {
5854c8945a0SNathan Whitehorn 	    /* ignore rest of file if line too long */
586*a96ef450SBaptiste Daroussin 	    report_error(filename, l, "line too long");
5874c8945a0SNathan Whitehorn 	    result = -1;	/* parse aborted */
5884c8945a0SNathan Whitehorn 	    break;
5894c8945a0SNathan Whitehorn 	}
5904c8945a0SNathan Whitehorn 
5914c8945a0SNathan Whitehorn 	lastch(str) = '\0';
5924c8945a0SNathan Whitehorn 	if (begins_with(str, "bindkey", &params)) {
5932a3e3873SBaptiste Daroussin 	    if (!dlg_parse_bindkey(params)) {
594*a96ef450SBaptiste Daroussin 		report_error(filename, l, "invalid bindkey");
5952a3e3873SBaptiste Daroussin 		result = -1;
5962a3e3873SBaptiste Daroussin 	    }
5974c8945a0SNathan Whitehorn 	    continue;
5984c8945a0SNathan Whitehorn 	}
5994c8945a0SNathan Whitehorn 	parse = parse_line(str, &var, &value);	/* parse current line */
6004c8945a0SNathan Whitehorn 
6014c8945a0SNathan Whitehorn 	switch (parse) {
6024c8945a0SNathan Whitehorn 	case LINE_EMPTY:	/* ignore blank lines and comments */
6034c8945a0SNathan Whitehorn 	    break;
6044c8945a0SNathan Whitehorn 	case LINE_EQUALS:
6054c8945a0SNathan Whitehorn 	    /* search table for matching config variable name */
6064c8945a0SNathan Whitehorn 	    if ((i = find_vars(var)) >= 0) {
6074c8945a0SNathan Whitehorn 		switch (vars[i].type) {
6084c8945a0SNathan Whitehorn 		case VAL_INT:
6094c8945a0SNathan Whitehorn 		    *((int *) vars[i].var) = atoi(value);
6104c8945a0SNathan Whitehorn 		    break;
6114c8945a0SNathan Whitehorn 		case VAL_STR:
6124c8945a0SNathan Whitehorn 		    if (!isquote(value[0]) || !isquote(lastch(value))
6134c8945a0SNathan Whitehorn 			|| strlen(value) < 2) {
614*a96ef450SBaptiste Daroussin 			report_error(filename, l, "expected string value");
6154c8945a0SNathan Whitehorn 			result = -1;	/* parse aborted */
6164c8945a0SNathan Whitehorn 		    } else {
6174c8945a0SNathan Whitehorn 			/* remove the (") quotes */
6184c8945a0SNathan Whitehorn 			value++;
6194c8945a0SNathan Whitehorn 			lastch(value) = '\0';
6204c8945a0SNathan Whitehorn 			strcpy((char *) vars[i].var, value);
6214c8945a0SNathan Whitehorn 		    }
6224c8945a0SNathan Whitehorn 		    break;
6234c8945a0SNathan Whitehorn 		case VAL_BOOL:
6244c8945a0SNathan Whitehorn 		    if (!dlg_strcmp(value, "ON"))
6254c8945a0SNathan Whitehorn 			*((bool *) vars[i].var) = TRUE;
6264c8945a0SNathan Whitehorn 		    else if (!dlg_strcmp(value, "OFF"))
6274c8945a0SNathan Whitehorn 			*((bool *) vars[i].var) = FALSE;
6284c8945a0SNathan Whitehorn 		    else {
629*a96ef450SBaptiste Daroussin 			report_error(filename, l, "expected boolean value");
6304c8945a0SNathan Whitehorn 			result = -1;	/* parse aborted */
6314c8945a0SNathan Whitehorn 		    }
6324c8945a0SNathan Whitehorn 		    break;
6334c8945a0SNathan Whitehorn 		}
6344c8945a0SNathan Whitehorn #ifdef HAVE_COLOR
6354c8945a0SNathan Whitehorn 	    } else if ((i = find_color(var)) >= 0) {
636*a96ef450SBaptiste Daroussin 		DIALOG_COLORS temp;
637*a96ef450SBaptiste Daroussin 		if (str_to_attr(value, &temp) == -1) {
638*a96ef450SBaptiste Daroussin 		    report_error(filename, l, "expected attribute value");
6394c8945a0SNathan Whitehorn 		    result = -1;	/* parse aborted */
6404c8945a0SNathan Whitehorn 		} else {
641*a96ef450SBaptiste Daroussin 		    dlg_color_table[i].fg = temp.fg;
642*a96ef450SBaptiste Daroussin 		    dlg_color_table[i].bg = temp.bg;
643*a96ef450SBaptiste Daroussin 		    dlg_color_table[i].hilite = temp.hilite;
644*a96ef450SBaptiste Daroussin #ifdef HAVE_RC_FILE2
645*a96ef450SBaptiste Daroussin 		    dlg_color_table[i].ul = temp.ul;
646*a96ef450SBaptiste Daroussin 		    dlg_color_table[i].rv = temp.rv;
647*a96ef450SBaptiste Daroussin #endif /* HAVE_RC_FILE2 */
6484c8945a0SNathan Whitehorn 		}
6494c8945a0SNathan Whitehorn 	    } else {
6504c8945a0SNathan Whitehorn #endif /* HAVE_COLOR */
651*a96ef450SBaptiste Daroussin 		report_error(filename, l, "unknown variable");
6524c8945a0SNathan Whitehorn 		result = -1;	/* parse aborted */
6534c8945a0SNathan Whitehorn 	    }
6544c8945a0SNathan Whitehorn 	    break;
6554c8945a0SNathan Whitehorn 	case LINE_ERROR:
656*a96ef450SBaptiste Daroussin 	    report_error(filename, l, "syntax error");
6574c8945a0SNathan Whitehorn 	    result = -1;	/* parse aborted */
6584c8945a0SNathan Whitehorn 	    break;
6594c8945a0SNathan Whitehorn 	}
6604c8945a0SNathan Whitehorn 	l++;			/* next line */
6614c8945a0SNathan Whitehorn     }
6624c8945a0SNathan Whitehorn 
6634c8945a0SNathan Whitehorn     (void) fclose(rc_file);
6644c8945a0SNathan Whitehorn     return result;
6654c8945a0SNathan Whitehorn }
666