freebsd-dev/usr.sbin/sade/dmenu.c

222 lines
5.5 KiB
C
Raw Normal View History

/*
* The new sysinstall program.
*
* This is probably the last attempt in the `sysinstall' line, the next
* generation being slated for what's essentially a complete rewrite.
*
* $Id: dmenu.c,v 1.15 1996/04/07 03:52:23 jkh Exp $
*
* Copyright (c) 1995
* Jordan Hubbard. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
1995-05-30 08:29:07 +00:00
* notice, this list of conditions and the following disclaimer,
* verbatim and that no modifications are made prior to this
* point in the file.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY JORDAN HUBBARD ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL JORDAN HUBBARD OR HIS PETS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, LIFE OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*
*/
#include "sysinstall.h"
1995-06-11 19:33:05 +00:00
#include <sys/types.h>
#define MAX_MENU 15
static Boolean cancelled;
int
dmenuDisplayFile(dialogMenuItem *tmp)
{
systemDisplayHelp((char *)tmp->data);
return DITEM_SUCCESS;
}
int
dmenuSubmenu(dialogMenuItem *tmp)
{
WINDOW *w;
w = savescr();
dialog_clear();
(void)dmenuOpenSimple((DMenu *)tmp->data);
restorescr(w);
return DITEM_SUCCESS;
}
int
dmenuSystemCommand(dialogMenuItem *tmp)
{
WINDOW *w;
w = savescr();
systemExecute((char *)tmp->data);
dialog_clear();
restorescr(w);
return DITEM_SUCCESS;
}
int
dmenuSystemCommandBox(dialogMenuItem *tmp)
{
WINDOW *w;
w = savescr();
use_helpfile(NULL);
use_helpline("Select OK to dismiss this dialog");
dialog_prgbox(tmp->title, (char *)tmp->data, 22, 76, 1, 1);
dialog_clear();
restorescr(w);
return DITEM_SUCCESS;
}
int
dmenuCancel(dialogMenuItem *tmp)
{
cancelled = TRUE;
return DITEM_LEAVE_MENU;
}
int
dmenuSetVariable(dialogMenuItem *tmp)
{
variable_set((char *)tmp->data);
msgInfo("Set %s", tmp->data);
return DITEM_SUCCESS;
}
int
dmenuSetFlag(dialogMenuItem *tmp)
{
if (*((unsigned int *)tmp->data) & tmp->aux)
*((unsigned int *)tmp->data) &= ~tmp->aux;
else
*((unsigned int *)tmp->data) |= tmp->aux;
return DITEM_SUCCESS;
}
int
dmenuSetValue(dialogMenuItem *tmp)
{
*((unsigned int *)tmp->data) = tmp->aux;
return DITEM_SUCCESS;
}
/* Traverse menu but give user no control over positioning */
1995-06-11 19:33:05 +00:00
Boolean
dmenuOpenSimple(DMenu *menu)
{
int choice, scroll, curr, max;
choice = scroll = curr = max = 0;
dialog_clear();
1995-06-11 19:33:05 +00:00
return dmenuOpen(menu, &choice, &scroll, &curr, &max);
}
/* Work functions for the state hook */
int
dmenuFlagCheck(dialogMenuItem *item)
1995-06-11 19:33:05 +00:00
{
return (*((unsigned int *)item->data) & item->aux);
1995-06-11 19:33:05 +00:00
}
int
dmenuVarCheck(dialogMenuItem *item)
1995-06-11 19:33:05 +00:00
{
char *w, *cp, *cp2, tmp[256];
1995-06-11 19:33:05 +00:00
w = (char *)item->aux;
if (!w)
w = (char *)item->data;
if (!w)
return FALSE;
strncpy(tmp, w, 256);
1995-06-11 19:33:05 +00:00
if ((cp = index(tmp, '=')) != NULL) {
*(cp++) = '\0';
cp2 = getenv(tmp);
if (cp2)
return !strcmp(cp, cp2);
else
return FALSE;
1995-06-11 19:33:05 +00:00
}
else
return (int)getenv(tmp);
1995-06-11 19:33:05 +00:00
}
int
dmenuRadioCheck(dialogMenuItem *item)
1995-06-11 19:33:05 +00:00
{
return (*((unsigned int *)item->data) == item->aux);
}
static int
menu_height(DMenu *menu, int n)
{
int max;
char *t;
for (t = menu->title, max = MAX_MENU; *t; t++) {
if (*t == '\n')
--max;
}
return n > max ? max : n;
}
/* Traverse over an internal menu */
1995-06-11 19:33:05 +00:00
Boolean
dmenuOpen(DMenu *menu, int *choice, int *scroll, int *curr, int *max)
{
int n, rval = 0;
/* Count up all the items */
for (n = 0; menu->items[n].title; n++);
while (1) {
char buf[FILENAME_MAX];
/* Any helpful hints, put 'em up! */
use_helpline(menu->helpline);
use_helpfile(systemHelpFile(menu->helpfile, buf));
/* Pop up that dialog! */
if (menu->type & DMENU_NORMAL_TYPE)
1995-06-11 19:33:05 +00:00
rval = dialog_menu((u_char *)menu->title, (u_char *)menu->prompt, -1, -1,
menu_height(menu, n), -n, menu->items, NULL, choice, scroll);
else if (menu->type & DMENU_RADIO_TYPE)
1995-06-11 19:33:05 +00:00
rval = dialog_radiolist((u_char *)menu->title, (u_char *)menu->prompt, -1, -1,
menu_height(menu, n), -n, menu->items, NULL);
else if (menu->type & DMENU_CHECKLIST_TYPE)
1995-06-11 19:33:05 +00:00
rval = dialog_checklist((u_char *)menu->title, (u_char *)menu->prompt, -1, -1,
menu_height(menu, n), -n, menu->items, NULL);
else
msgFatal("Menu: `%s' is of an unknown type\n", menu->title);
/* This seems to be the only technique that works for getting the display to look right */
dialog_clear();
if (rval || menu->type & (DMENU_SELECTION_RETURNS | DMENU_RADIO_TYPE | DMENU_CHECKLIST_TYPE))
1995-06-11 19:33:05 +00:00
return FALSE;
else if (cancelled) {
cancelled = FALSE;
return TRUE;
}
}
}