Home | History | Annotate | Line # | Download | only in menuc
mdb.c revision 1.9
      1  1.9  phil /*	$NetBSD: mdb.c,v 1.9 1998/07/01 07:46:02 phil Exp $	*/
      2  1.1  phil 
      3  1.1  phil /*
      4  1.1  phil  * Copyright 1997 Piermont Information Systems Inc.
      5  1.1  phil  * All rights reserved.
      6  1.1  phil  *
      7  1.1  phil  * Written by Philip A. Nelson for Piermont Information Systems Inc.
      8  1.1  phil  *
      9  1.1  phil  * Redistribution and use in source and binary forms, with or without
     10  1.1  phil  * modification, are permitted provided that the following conditions
     11  1.1  phil  * are met:
     12  1.1  phil  * 1. Redistributions of source code must retain the above copyright
     13  1.1  phil  *    notice, this list of conditions and the following disclaimer.
     14  1.1  phil  * 2. Redistributions in binary form must reproduce the above copyright
     15  1.1  phil  *    notice, this list of conditions and the following disclaimer in the
     16  1.1  phil  *    documentation and/or other materials provided with the distribution.
     17  1.1  phil  * 3. All advertising materials mentioning features or use of this software
     18  1.1  phil  *    must display the following acknowledgement:
     19  1.1  phil  *      This product includes software develooped for the NetBSD Project by
     20  1.1  phil  *      Piermont Information Systems Inc.
     21  1.1  phil  * 4. The name of Piermont Information Systems Inc. may not be used to endorse
     22  1.1  phil  *    or promote products derived from this software without specific prior
     23  1.1  phil  *    written permission.
     24  1.1  phil  *
     25  1.1  phil  * THIS SOFTWARE IS PROVIDED BY PIERMONT INFORMATION SYSTEMS INC. ``AS IS''
     26  1.1  phil  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     27  1.1  phil  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     28  1.1  phil  * ARE DISCLAIMED. IN NO EVENT SHALL PIERMONT INFORMATION SYSTEMS INC. BE
     29  1.1  phil  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     30  1.1  phil  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     31  1.1  phil  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     32  1.1  phil  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     33  1.1  phil  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     34  1.1  phil  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
     35  1.1  phil  * THE POSSIBILITY OF SUCH DAMAGE.
     36  1.1  phil  *
     37  1.1  phil  */
     38  1.1  phil 
     39  1.1  phil /* mdb.c - menu database manipulation */
     40  1.1  phil 
     41  1.1  phil #include <stdio.h>
     42  1.1  phil #include <stdlib.h>
     43  1.1  phil #include <string.h>
     44  1.1  phil #include "mdb.h"
     45  1.1  phil #include "defs.h"
     46  1.1  phil 
     47  1.1  phil /* Data */
     48  1.1  phil #define MAX 500
     49  1.4   mrg static int menu_no = 0;
     50  1.1  phil static id_rec *menus[MAX];
     51  1.1  phil 
     52  1.1  phil 
     53  1.1  phil /* get_menu returns a pointer to a newly created id_rec or an old one. */
     54  1.1  phil 
     55  1.1  phil id_rec *
     56  1.1  phil get_menu (char *name)
     57  1.1  phil {
     58  1.1  phil 	id_rec *temp;
     59  1.1  phil 
     60  1.1  phil 	temp = find_id (root, name);
     61  1.1  phil 
     62  1.1  phil 	if (temp == NULL) {
     63  1.1  phil 		if (menu_no < MAX) {
     64  1.1  phil 			temp = (id_rec *) malloc (sizeof(id_rec));
     65  1.1  phil 			temp->id = strdup(name);
     66  1.1  phil 			temp->info = NULL;
     67  1.1  phil 			temp->menu_no = menu_no;
     68  1.1  phil 			menus[menu_no++] = temp;
     69  1.1  phil 			insert_id (&root, temp);
     70  1.1  phil 		} else {
     71  1.1  phil 			(void) fprintf (stderr, "Too many menus.  "
     72  1.1  phil 					"Increase MAX.\n");
     73  1.1  phil 			exit(1);
     74  1.1  phil 		}
     75  1.1  phil 	}
     76  1.1  phil 
     77  1.1  phil 	return temp;
     78  1.1  phil }
     79  1.1  phil 
     80  1.1  phil 
     81  1.1  phil /* Verify that all menus are defined. */
     82  1.1  phil 
     83  1.1  phil void
     84  1.1  phil check_defined (void)
     85  1.1  phil {
     86  1.1  phil 	int i;
     87  1.1  phil 
     88  1.1  phil 	for (i=0; i<menu_no; i++)
     89  1.1  phil 		if (!menus[i]->info)
     90  1.1  phil 			yyerror ("Menu '%s' undefined.", menus[i]->id);
     91  1.1  phil }
     92  1.1  phil 
     93  1.1  phil 
     94  1.1  phil /* Write out the menu file. */
     95  1.1  phil void
     96  1.1  phil write_menu_file (char *initcode)
     97  1.1  phil {
     98  1.1  phil 	FILE *out_file;
     99  1.1  phil 	FILE *sys_file;
    100  1.1  phil 	int i, j;
    101  1.1  phil 	char hname[1024];
    102  1.1  phil 	char cname[1024];
    103  1.1  phil 	char sname[1024];
    104  1.1  phil 	char *sys_prefix;
    105  1.5  phil 	char *tmpstr;
    106  1.1  phil 
    107  1.1  phil 	int nlen;
    108  1.1  phil 
    109  1.1  phil 	char opt_ch;
    110  1.3   cgd 	int ch;
    111  1.1  phil 
    112  1.1  phil 	optn_info *toptn;
    113  1.1  phil 
    114  1.1  phil 	/* Generate file names */
    115  1.1  phil 	snprintf (hname, 1024, "%s.h", out_name);
    116  1.1  phil 	nlen = strlen(hname);
    117  1.1  phil 	if (hname[nlen-2] != '.' || hname[nlen-1] != 'h') {
    118  1.1  phil 		(void) fprintf (stderr, "%s: name `%s` too long.\n",
    119  1.1  phil 				prog_name, out_name);
    120  1.1  phil 		exit(1);
    121  1.1  phil 	}
    122  1.1  phil 	snprintf (cname, 1024, "%s.c", out_name);
    123  1.1  phil 
    124  1.1  phil 	/* Open the menu_sys file first. */
    125  1.1  phil 	sys_prefix = getenv ("MENUDEF");
    126  1.1  phil 	if (sys_prefix == NULL)
    127  1.1  phil 		sys_prefix = "/usr/share/misc";
    128  1.1  phil 	snprintf (sname, 1024, "%s/%s", sys_prefix, sys_name);
    129  1.1  phil 	sys_file = fopen (sname, "r");
    130  1.1  phil 	if (sys_file == NULL) {
    131  1.1  phil 		(void) fprintf (stderr, "%s: could not open %s.\n",
    132  1.1  phil 				prog_name, sname);
    133  1.1  phil 		exit (1);
    134  1.1  phil 	}
    135  1.1  phil 
    136  1.1  phil 	/* Output the .h file first. */
    137  1.1  phil 	out_file = fopen (hname, "w");
    138  1.1  phil 	if (out_file == NULL) {
    139  1.1  phil 		(void) fprintf (stderr, "%s: could not open %s.\n",
    140  1.1  phil 				prog_name, hname);
    141  1.1  phil 		exit (1);
    142  1.1  phil 	}
    143  1.1  phil 
    144  1.1  phil 	/* Write it */
    145  1.1  phil 	(void) fprintf (out_file, "%s",
    146  1.1  phil 		"/* menu system definitions. */\n"
    147  1.1  phil 		"\n"
    148  1.1  phil 		"#ifndef MENU_DEFS_H\n"
    149  1.1  phil 		"#define MENU_DEFS_H\n"
    150  1.1  phil 		"#include <stdlib.h>\n"
    151  1.1  phil 		"#include <string.h>\n"
    152  1.1  phil 		"#include <ctype.h>\n"
    153  1.1  phil 		"#include <curses.h>\n"
    154  1.9  phil 		"\n");
    155  1.9  phil 
    156  1.9  phil 	if (do_dynamic)
    157  1.9  phil 		(void) fprintf (out_file,
    158  1.9  phil 			"\n#define DYNAMIC_MENUS\n\n"
    159  1.9  phil 			"typedef\n"
    160  1.9  phil 			"struct dyn_menu_ent {\n"
    161  1.9  phil 			"	char   *optname;\n"
    162  1.9  phil 			"	int	nextmenu;\n"
    163  1.9  phil 			"	int	submenu;\n"
    164  1.9  phil 			"	void	(*action)(void);\n"
    165  1.9  phil 			"} dyn_menu_ent ;\n");
    166  1.9  phil 
    167  1.9  phil 	(void) fprintf (out_file, "%s",
    168  1.1  phil 		"typedef\n"
    169  1.1  phil 		"struct menudesc {\n"
    170  1.1  phil 		"	char   *title;\n"
    171  1.1  phil 		"	int     y, x;\n"
    172  1.1  phil 		"	int	h, w;\n"
    173  1.1  phil 		"	int	mopt;\n"
    174  1.1  phil 		"	int     numopts;\n"
    175  1.1  phil 		"	int	cursel;\n"
    176  1.6  phil 		"	int	topline;\n"
    177  1.1  phil 		"	char   **opts;\n"
    178  1.1  phil 		"	WINDOW *mw;\n"
    179  1.9  phil 		"	char   *helpstr;\n");
    180  1.9  phil 
    181  1.9  phil 	if (do_dynamic)
    182  1.9  phil 		(void) fprintf (out_file,
    183  1.9  phil 			"	dyn_menu_ent   *dyn_opts;\n"
    184  1.9  phil 			"	void   (*post_act)(void);\n"
    185  1.9  phil 			"	void   (*exit_act)(void);\n");
    186  1.9  phil 
    187  1.9  phil 	(void) fprintf (out_file, "%s",
    188  1.1  phil 		"} menudesc ;\n"
    189  1.1  phil 		"\n"
    190  1.1  phil 		"/* defines for mopt field. */\n"
    191  1.1  phil 		"#define NOEXITOPT 1\n"
    192  1.1  phil 		"#define NOBOX 2\n"
    193  1.9  phil 		"#define SCROLL 4\n");
    194  1.9  phil 
    195  1.9  phil 	if (do_dynamic)
    196  1.9  phil 		(void) fprintf (out_file,
    197  1.9  phil 			"#define DYNAMIC 8\n");
    198  1.9  phil 
    199  1.9  phil 	(void) fprintf (out_file, "%s",
    200  1.1  phil 		"\n"
    201  1.1  phil 		"/* initilization flag */\n"
    202  1.1  phil 		"extern int __m_endwin;\n"
    203  1.1  phil 		"\n"
    204  1.1  phil 		"/* Prototypes */\n"
    205  1.1  phil 		"void process_menu (int num);\n"
    206  1.2  phil 		"void __menu_initerror (void);\n"
    207  1.1  phil 		"\n"
    208  1.1  phil 		"/* Menu names */\n"
    209  1.1  phil 	      );
    210  1.1  phil 	for (i=0; i<menu_no; i++) {
    211  1.1  phil 		(void) fprintf (out_file, "#define MENU_%s\t%d\n",
    212  1.1  phil 				menus[i]->id, i);
    213  1.1  phil 	}
    214  1.1  phil 	(void) fprintf (out_file, "\n#define MAX_STRLEN %d\n", max_strlen);
    215  1.9  phil 
    216  1.1  phil 	(void) fprintf (out_file, "#endif\n");
    217  1.1  phil 	fclose (out_file);
    218  1.1  phil 
    219  1.1  phil 	/* Now the C file */
    220  1.1  phil 	out_file = fopen (cname, "w");
    221  1.1  phil 	if (out_file == NULL) {
    222  1.1  phil 		(void) fprintf (stderr, "%s: could not open %s.\n",
    223  1.1  phil 				prog_name, cname);
    224  1.1  phil 		exit (1);
    225  1.1  phil 	}
    226  1.1  phil 
    227  1.1  phil 	/* initial code */
    228  1.1  phil 	fprintf (out_file, "#include \"%s\"\n\n", hname);
    229  1.1  phil 	fprintf (out_file, "%s\n\n", initcode);
    230  1.1  phil 
    231  1.1  phil 	/* data definitions */
    232  1.1  phil 
    233  1.1  phil 	/* optstrX */
    234  1.1  phil 	for (i=0; i<menu_no; i++) {
    235  1.1  phil 		(void) fprintf (out_file, "static char *optstr%d[] = {\n", i);
    236  1.1  phil 		toptn = menus[i]->info->optns;
    237  1.1  phil 		opt_ch = 'a';
    238  1.1  phil 		while (toptn != NULL) {
    239  1.8  phil 			if (opt_ch > 'Z' && opt_ch < 'a') {
    240  1.8  phil 				(void) fprintf (stderr, "Menu %s has "
    241  1.8  phil 					"too many options.\n",
    242  1.8  phil 					menus[i]->info->title);
    243  1.8  phil 				exit (1);
    244  1.8  phil 			}
    245  1.7  phil 			(void) fprintf (out_file, "\t\"%c: %s,\n", opt_ch,
    246  1.1  phil 					toptn->name+1);
    247  1.1  phil 			toptn = toptn->next;
    248  1.7  phil 			if (opt_ch == 'z')
    249  1.7  phil 				opt_ch = 'A';
    250  1.8  phil 			else if (opt_ch == 'w')
    251  1.7  phil 				opt_ch = 'y';
    252  1.8  phil 			else
    253  1.7  phil 				opt_ch++;
    254  1.1  phil 		}
    255  1.1  phil 		(void) fprintf (out_file, "\t(char *)NULL\n};\n\n");
    256  1.1  phil 	}
    257  1.1  phil 
    258  1.1  phil 	/* menus */
    259  1.1  phil 	(void) fprintf (out_file, "static struct menudesc menus[] = {\n");
    260  1.5  phil 	for (i=0; i<menu_no; i++) {
    261  1.1  phil 		(void) fprintf (out_file,
    262  1.6  phil 			"\t{%s,%d,%d,%d,%d,%d,%d,0,0,optstr%d,NULL,",
    263  1.1  phil 			menus[i]->info->title, 	menus[i]->info->y,
    264  1.1  phil 			menus[i]->info->x, menus[i]->info->h,
    265  1.1  phil 			menus[i]->info->w, menus[i]->info->mopt,
    266  1.1  phil 			menus[i]->info->numopt, i);
    267  1.5  phil 		if (menus[i]->info->helpstr == NULL)
    268  1.9  phil 			(void) fprintf (out_file, "NULL");
    269  1.5  phil 		else {
    270  1.5  phil 			tmpstr = menus[i]->info->helpstr;
    271  1.5  phil 			/* Skip an initial newline. */
    272  1.5  phil 			if (*tmpstr == '\n')
    273  1.5  phil 				tmpstr++;
    274  1.5  phil 			(void) fprintf (out_file, "\n\"");
    275  1.5  phil 			while (*tmpstr)
    276  1.5  phil 				if (*tmpstr != '\n')
    277  1.5  phil 				  fputc (*tmpstr++, out_file);
    278  1.5  phil 				else {
    279  1.5  phil 					(void) fprintf (out_file, "\\n\\\n");
    280  1.5  phil 					tmpstr++;
    281  1.5  phil 				}
    282  1.9  phil 			(void) fprintf (out_file, "\"");
    283  1.5  phil 		}
    284  1.9  phil 		if (do_dynamic)
    285  1.9  phil 			(void) fprintf (out_file, ",NULL,NULL,NULL},\n");
    286  1.9  phil 		else
    287  1.9  phil 			(void) fprintf (out_file, "},\n");
    288  1.9  phil 
    289  1.5  phil 	}
    290  1.1  phil 	(void) fprintf (out_file, "{NULL}};\n\n");
    291  1.1  phil 
    292  1.1  phil 	/* num_menus */
    293  1.1  phil 	(void) fprintf (out_file, "int num_menus = %d;\n\n", menu_no);
    294  1.1  phil 
    295  1.1  phil 	/* action code */
    296  1.1  phil 	(void) fprintf (out_file, "%s",
    297  1.1  phil 		"static int process_item (int *menu_no, int sel)\n"
    298  1.1  phil 		"{\n"
    299  1.1  phil 		"\tint retval = FALSE;\n"
    300  1.1  phil 		"\n"
    301  1.1  phil 		"\tswitch (*menu_no) {\n"
    302  1.1  phil 	);
    303  1.1  phil 	for (i=0; i<menu_no; i++) {
    304  1.1  phil 		(void) fprintf (out_file, "\tcase MENU_%s:\n",
    305  1.1  phil 				menus[i]->id);
    306  1.1  phil 		(void) fprintf (out_file, "\t\tswitch (sel) {\n"
    307  1.1  phil 				"\t\tcase -2:\n");
    308  1.1  phil 		if (menus[i]->info->postact.endwin)
    309  1.1  phil 			(void) fprintf (out_file, "\t\t\tendwin();\n"
    310  1.1  phil 					"\t\t__m_endwin = 1;\n");
    311  1.1  phil 		if (strlen(menus[i]->info->postact.code))
    312  1.1  phil 			(void) fprintf (out_file, "\t\t\t{%s}\n",
    313  1.1  phil 					menus[i]->info->postact.code);
    314  1.1  phil 		(void) fprintf (out_file, "\t\t\tbreak;\n");
    315  1.1  phil 		(void) fprintf (out_file, "\t\tcase -1:\n");
    316  1.1  phil 		if (menus[i]->info->exitact.endwin)
    317  1.1  phil 			(void) fprintf (out_file, "\t\t\tendwin();\n"
    318  1.1  phil 					"\t\t__m_endwin = 1;\n");
    319  1.1  phil 		if (strlen(menus[i]->info->exitact.code))
    320  1.1  phil 			(void) fprintf (out_file, "\t\t\t{%s}\n",
    321  1.1  phil 					menus[i]->info->exitact.code);
    322  1.1  phil 		(void) fprintf (out_file, "\t\t\tbreak;\n");
    323  1.1  phil 		j = 0;
    324  1.1  phil 		toptn = menus[i]->info->optns;
    325  1.1  phil 		while (toptn != NULL) {
    326  1.1  phil 			(void) fprintf (out_file, "\t\tcase %d:\n", j++);
    327  1.1  phil 			if (toptn->optact.endwin)
    328  1.1  phil 				(void) fprintf (out_file, "\t\t\tendwin();\n"
    329  1.1  phil 						"\t\t__m_endwin = 1;\n");
    330  1.1  phil 			if (strlen(toptn->optact.code))
    331  1.1  phil 				(void) fprintf (out_file, "\t\t\t{%s}\n",
    332  1.1  phil 						toptn->optact.code);
    333  1.1  phil 			if (toptn->menu >= 0)
    334  1.1  phil 				if (toptn->issub)
    335  1.1  phil 					(void) fprintf (out_file,
    336  1.1  phil 						"\t\t\tprocess_menu(%d);\n",
    337  1.1  phil 						toptn->menu);
    338  1.1  phil 				else
    339  1.1  phil 					(void) fprintf (out_file,
    340  1.1  phil 						"\t\t\t*menu_no = %d;\n",
    341  1.1  phil 						toptn->menu);
    342  1.1  phil 			if (toptn->doexit)
    343  1.1  phil 				(void) fprintf (out_file,
    344  1.1  phil 						"\t\t\tretval = TRUE;\n");
    345  1.1  phil 			(void) fprintf (out_file, "\t\t\tbreak;\n");
    346  1.1  phil 			toptn = toptn->next;
    347  1.1  phil 		}
    348  1.1  phil 
    349  1.1  phil 		(void) fprintf (out_file, "\t\t}\n\t\tbreak;\n");
    350  1.1  phil 	}
    351  1.1  phil 	(void) fprintf (out_file, "\t}\n\t return retval;\n}\n\n");
    352  1.1  phil 
    353  1.1  phil 	while ((ch = fgetc(sys_file)) != EOF)
    354  1.1  phil 		fputc(ch, out_file);
    355  1.1  phil 
    356  1.2  phil 	/* __menu_initerror: initscr failed. */
    357  1.2  phil 	(void) fprintf (out_file,
    358  1.2  phil 		"/* __menu_initerror: initscr failed. */\n"
    359  1.2  phil 		"void __menu_initerror (void) {\n");
    360  1.2  phil 	if (error_act.code == NULL) {
    361  1.2  phil 		(void) fprintf (out_file,
    362  1.2  phil 			"\t(void) fprintf (stderr, "
    363  1.2  phil 				"\"Could not initialize curses\\n\");\n"
    364  1.2  phil 			"\texit(1);\n"
    365  1.2  phil 			"}\n");
    366  1.2  phil 	} else {
    367  1.2  phil 		if (error_act.endwin)
    368  1.2  phil 			(void) fprintf (out_file, "\tendwin();\n");
    369  1.2  phil 		(void) fprintf (out_file, "%s\n}\n", error_act.code);
    370  1.2  phil 	}
    371  1.2  phil 
    372  1.1  phil 	fclose (out_file);
    373  1.1  phil 	fclose (sys_file);
    374  1.1  phil }
    375