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", ¶ms)) {
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