Home | History | Annotate | Line # | Download | only in menuc
mdb.c revision 1.2
      1  1.2  phil /*	$NetBSD: mdb.c,v 1.2 1997/11/09 20:59:14 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.1  phil static 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.1  phil 
    106  1.1  phil 	int nlen;
    107  1.1  phil 
    108  1.1  phil 	char opt_ch;
    109  1.1  phil 	char ch;
    110  1.1  phil 
    111  1.1  phil 	optn_info *toptn;
    112  1.1  phil 
    113  1.1  phil 	/* Generate file names */
    114  1.1  phil 	snprintf (hname, 1024, "%s.h", out_name);
    115  1.1  phil 	nlen = strlen(hname);
    116  1.1  phil 	if (hname[nlen-2] != '.' || hname[nlen-1] != 'h') {
    117  1.1  phil 		(void) fprintf (stderr, "%s: name `%s` too long.\n",
    118  1.1  phil 				prog_name, out_name);
    119  1.1  phil 		exit(1);
    120  1.1  phil 	}
    121  1.1  phil 	snprintf (cname, 1024, "%s.c", out_name);
    122  1.1  phil 
    123  1.1  phil 	/* Open the menu_sys file first. */
    124  1.1  phil 	sys_prefix = getenv ("MENUDEF");
    125  1.1  phil 	if (sys_prefix == NULL)
    126  1.1  phil 		sys_prefix = "/usr/share/misc";
    127  1.1  phil 	snprintf (sname, 1024, "%s/%s", sys_prefix, sys_name);
    128  1.1  phil 	sys_file = fopen (sname, "r");
    129  1.1  phil 	if (sys_file == NULL) {
    130  1.1  phil 		(void) fprintf (stderr, "%s: could not open %s.\n",
    131  1.1  phil 				prog_name, sname);
    132  1.1  phil 		exit (1);
    133  1.1  phil 	}
    134  1.1  phil 
    135  1.1  phil 	/* Output the .h file first. */
    136  1.1  phil 	out_file = fopen (hname, "w");
    137  1.1  phil 	if (out_file == NULL) {
    138  1.1  phil 		(void) fprintf (stderr, "%s: could not open %s.\n",
    139  1.1  phil 				prog_name, hname);
    140  1.1  phil 		exit (1);
    141  1.1  phil 	}
    142  1.1  phil 
    143  1.1  phil 	/* Write it */
    144  1.1  phil 	(void) fprintf (out_file, "%s",
    145  1.1  phil 		"/* menu system definitions. */\n"
    146  1.1  phil 		"\n"
    147  1.1  phil 		"#ifndef MENU_DEFS_H\n"
    148  1.1  phil 		"#define MENU_DEFS_H\n"
    149  1.1  phil 		"#include <stdlib.h>\n"
    150  1.1  phil 		"#include <string.h>\n"
    151  1.1  phil 		"#include <ctype.h>\n"
    152  1.1  phil 		"#include <curses.h>\n"
    153  1.1  phil 		"\n"
    154  1.1  phil 		"typedef\n"
    155  1.1  phil 		"struct menudesc {\n"
    156  1.1  phil 		"	char   *title;\n"
    157  1.1  phil 		"	int     y, x;\n"
    158  1.1  phil 		"	int	h, w;\n"
    159  1.1  phil 		"	int	mopt;\n"
    160  1.1  phil 		"	int     numopts;\n"
    161  1.1  phil 		"	int	cursel;\n"
    162  1.1  phil 		"	char   **opts;\n"
    163  1.1  phil 		"	WINDOW *mw;\n"
    164  1.1  phil 		"} menudesc ;\n"
    165  1.1  phil 		"\n"
    166  1.1  phil 		"/* defines for mopt field. */\n"
    167  1.1  phil 		"#define NOEXITOPT 1\n"
    168  1.1  phil 		"#define NOBOX 2\n"
    169  1.1  phil 		"\n"
    170  1.1  phil 		"/* initilization flag */\n"
    171  1.1  phil 		"extern int __m_endwin;\n"
    172  1.1  phil 		"\n"
    173  1.1  phil 		"/* Prototypes */\n"
    174  1.1  phil 		"void process_menu (int num);\n"
    175  1.2  phil 		"void __menu_initerror (void);\n"
    176  1.1  phil 		"\n"
    177  1.1  phil 		"/* Menu names */\n"
    178  1.1  phil 	      );
    179  1.1  phil 	for (i=0; i<menu_no; i++) {
    180  1.1  phil 		(void) fprintf (out_file, "#define MENU_%s\t%d\n",
    181  1.1  phil 				menus[i]->id, i);
    182  1.1  phil 	}
    183  1.1  phil 	(void) fprintf (out_file, "\n#define MAX_STRLEN %d\n", max_strlen);
    184  1.1  phil 	(void) fprintf (out_file, "#endif\n");
    185  1.1  phil 
    186  1.1  phil 	fclose (out_file);
    187  1.1  phil 
    188  1.1  phil 	/* Now the C file */
    189  1.1  phil 	out_file = fopen (cname, "w");
    190  1.1  phil 	if (out_file == NULL) {
    191  1.1  phil 		(void) fprintf (stderr, "%s: could not open %s.\n",
    192  1.1  phil 				prog_name, cname);
    193  1.1  phil 		exit (1);
    194  1.1  phil 	}
    195  1.1  phil 
    196  1.1  phil 	/* initial code */
    197  1.1  phil 	fprintf (out_file, "#include \"%s\"\n\n", hname);
    198  1.1  phil 	fprintf (out_file, "%s\n\n", initcode);
    199  1.1  phil 
    200  1.1  phil 	/* data definitions */
    201  1.1  phil 
    202  1.1  phil 	/* optstrX */
    203  1.1  phil 	for (i=0; i<menu_no; i++) {
    204  1.1  phil 		(void) fprintf (out_file, "static char *optstr%d[] = {\n", i);
    205  1.1  phil 		toptn = menus[i]->info->optns;
    206  1.1  phil 		opt_ch = 'a';
    207  1.1  phil 		while (toptn != NULL) {
    208  1.1  phil 			(void) fprintf (out_file, "\t\"%c: %s,\n", opt_ch++,
    209  1.1  phil 					toptn->name+1);
    210  1.1  phil 			toptn = toptn->next;
    211  1.1  phil 		}
    212  1.1  phil 		(void) fprintf (out_file, "\t(char *)NULL\n};\n\n");
    213  1.1  phil 	}
    214  1.1  phil 
    215  1.1  phil 	/* menus */
    216  1.1  phil 	(void) fprintf (out_file, "static struct menudesc menus[] = {\n");
    217  1.1  phil 	for (i=0; i<menu_no; i++)
    218  1.1  phil 		(void) fprintf (out_file,
    219  1.1  phil 			"\t{%s,%d,%d,%d,%d,%d,%d,0,optstr%d,NULL},\n",
    220  1.1  phil 			menus[i]->info->title, 	menus[i]->info->y,
    221  1.1  phil 			menus[i]->info->x, menus[i]->info->h,
    222  1.1  phil 			menus[i]->info->w, menus[i]->info->mopt,
    223  1.1  phil 			menus[i]->info->numopt, i);
    224  1.1  phil 	(void) fprintf (out_file, "{NULL}};\n\n");
    225  1.1  phil 
    226  1.1  phil 	/* num_menus */
    227  1.1  phil 	(void) fprintf (out_file, "int num_menus = %d;\n\n", menu_no);
    228  1.1  phil 
    229  1.1  phil 	/* action code */
    230  1.1  phil 	(void) fprintf (out_file, "%s",
    231  1.1  phil 		"static int process_item (int *menu_no, int sel)\n"
    232  1.1  phil 		"{\n"
    233  1.1  phil 		"\tint retval = FALSE;\n"
    234  1.1  phil 		"\n"
    235  1.1  phil 		"\tswitch (*menu_no) {\n"
    236  1.1  phil 	);
    237  1.1  phil 	for (i=0; i<menu_no; i++) {
    238  1.1  phil 		(void) fprintf (out_file, "\tcase MENU_%s:\n",
    239  1.1  phil 				menus[i]->id);
    240  1.1  phil 		(void) fprintf (out_file, "\t\tswitch (sel) {\n"
    241  1.1  phil 				"\t\tcase -2:\n");
    242  1.1  phil 		if (menus[i]->info->postact.endwin)
    243  1.1  phil 			(void) fprintf (out_file, "\t\t\tendwin();\n"
    244  1.1  phil 					"\t\t__m_endwin = 1;\n");
    245  1.1  phil 		if (strlen(menus[i]->info->postact.code))
    246  1.1  phil 			(void) fprintf (out_file, "\t\t\t{%s}\n",
    247  1.1  phil 					menus[i]->info->postact.code);
    248  1.1  phil 		(void) fprintf (out_file, "\t\t\tbreak;\n");
    249  1.1  phil 		(void) fprintf (out_file, "\t\tcase -1:\n");
    250  1.1  phil 		if (menus[i]->info->exitact.endwin)
    251  1.1  phil 			(void) fprintf (out_file, "\t\t\tendwin();\n"
    252  1.1  phil 					"\t\t__m_endwin = 1;\n");
    253  1.1  phil 		if (strlen(menus[i]->info->exitact.code))
    254  1.1  phil 			(void) fprintf (out_file, "\t\t\t{%s}\n",
    255  1.1  phil 					menus[i]->info->exitact.code);
    256  1.1  phil 		(void) fprintf (out_file, "\t\t\tbreak;\n");
    257  1.1  phil 		j = 0;
    258  1.1  phil 		toptn = menus[i]->info->optns;
    259  1.1  phil 		while (toptn != NULL) {
    260  1.1  phil 			(void) fprintf (out_file, "\t\tcase %d:\n", j++);
    261  1.1  phil 			if (toptn->optact.endwin)
    262  1.1  phil 				(void) fprintf (out_file, "\t\t\tendwin();\n"
    263  1.1  phil 						"\t\t__m_endwin = 1;\n");
    264  1.1  phil 			if (strlen(toptn->optact.code))
    265  1.1  phil 				(void) fprintf (out_file, "\t\t\t{%s}\n",
    266  1.1  phil 						toptn->optact.code);
    267  1.1  phil 			if (toptn->menu >= 0)
    268  1.1  phil 				if (toptn->issub)
    269  1.1  phil 					(void) fprintf (out_file,
    270  1.1  phil 						"\t\t\tprocess_menu(%d);\n",
    271  1.1  phil 						toptn->menu);
    272  1.1  phil 				else
    273  1.1  phil 					(void) fprintf (out_file,
    274  1.1  phil 						"\t\t\t*menu_no = %d;\n",
    275  1.1  phil 						toptn->menu);
    276  1.1  phil 			if (toptn->doexit)
    277  1.1  phil 				(void) fprintf (out_file,
    278  1.1  phil 						"\t\t\tretval = TRUE;\n");
    279  1.1  phil 			(void) fprintf (out_file, "\t\t\tbreak;\n");
    280  1.1  phil 			toptn = toptn->next;
    281  1.1  phil 		}
    282  1.1  phil 
    283  1.1  phil 		(void) fprintf (out_file, "\t\t}\n\t\tbreak;\n");
    284  1.1  phil 	}
    285  1.1  phil 	(void) fprintf (out_file, "\t}\n\t return retval;\n}\n\n");
    286  1.1  phil 
    287  1.1  phil 	while ((ch = fgetc(sys_file)) != EOF)
    288  1.1  phil 		fputc(ch, out_file);
    289  1.1  phil 
    290  1.2  phil 	/* __menu_initerror: initscr failed. */
    291  1.2  phil 	(void) fprintf (out_file,
    292  1.2  phil 		"/* __menu_initerror: initscr failed. */\n"
    293  1.2  phil 		"void __menu_initerror (void) {\n");
    294  1.2  phil 	if (error_act.code == NULL) {
    295  1.2  phil 		(void) fprintf (out_file,
    296  1.2  phil 			"\t(void) fprintf (stderr, "
    297  1.2  phil 				"\"Could not initialize curses\\n\");\n"
    298  1.2  phil 			"\texit(1);\n"
    299  1.2  phil 			"}\n");
    300  1.2  phil 	} else {
    301  1.2  phil 		if (error_act.endwin)
    302  1.2  phil 			(void) fprintf (out_file, "\tendwin();\n");
    303  1.2  phil 		(void) fprintf (out_file, "%s\n}\n", error_act.code);
    304  1.2  phil 	}
    305  1.2  phil 
    306  1.1  phil 	fclose (out_file);
    307  1.1  phil 	fclose (sys_file);
    308  1.1  phil }
    309