e5372aca25
than at compile time. Should have same functionality as old libforms but with new mechanism. Lots of new features that use the new mechanism are still to be added.
503 lines
11 KiB
C
503 lines
11 KiB
C
/*-
|
||
* Copyright (c) 1995
|
||
* Paul Richards. 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
|
||
* 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.
|
||
* 3. All advertising materials mentioning features or use of this software
|
||
* must display the following acknowledgement:
|
||
* This product includes software developed by Paul Richards.
|
||
* 4. The name Paul Richards may not be used to endorse or promote products
|
||
* derived from this software without specific prior written permission.
|
||
*
|
||
* THIS SOFTWARE IS PROVIDED BY PAUL RICHARDS ``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 PAUL RICHARDS 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, 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 <string.h>
|
||
#include <stdlib.h>
|
||
#include <ctype.h>
|
||
#include <forms.h>
|
||
#include <ncurses.h>
|
||
|
||
#include "internal.h"
|
||
|
||
unsigned int f_keymap[] = {
|
||
KEY_UP, /* F_UP */
|
||
KEY_DOWN, /* F_DOWN */
|
||
9, /* F_RIGHT */
|
||
8, /* F_LEFT */
|
||
10, /* F_NEXT */
|
||
KEY_LEFT, /* F_CLEFT */
|
||
KEY_RIGHT, /* F_CRIGHT */
|
||
KEY_HOME, /* F_CHOME */
|
||
KEY_END, /* F_CEND */
|
||
263, /* F_CBS */
|
||
330, /* F_CDEL */
|
||
10 /* F_ACCEPT */
|
||
};
|
||
|
||
int done=0;
|
||
|
||
int
|
||
initfrm(struct form *form)
|
||
{
|
||
|
||
struct field *field = &form->field[0];
|
||
int i;
|
||
|
||
if (has_colors()) {
|
||
start_color();
|
||
if (form->color_table)
|
||
for (i=0; form->color_table[i].f != -1; i++) {
|
||
init_pair(i+1, form->color_table[i].f, form->color_table[i].b);
|
||
}
|
||
}
|
||
cbreak();
|
||
noecho();
|
||
|
||
if (!form->height)
|
||
form->height = LINES;
|
||
if (!form->width)
|
||
form->width = COLS;
|
||
|
||
form->window = newwin(form->height, form->width, form->y, form->x);
|
||
if (!form->window) {
|
||
print_status("Couldn't open window, closing form");
|
||
return (ERR);
|
||
}
|
||
form->no_fields = 0;
|
||
|
||
keypad(form->window, TRUE);
|
||
|
||
|
||
while (field->type != F_END) {
|
||
if (field->type == F_INPUT) {
|
||
field->field.input->input = malloc(field->field.input->limit);
|
||
if (!field->field.input->input){
|
||
print_status("Couldn't allocate memory, closing form");
|
||
endfrm(form);
|
||
return (ERR);
|
||
}
|
||
/*
|
||
* If it's a label then clear the input string
|
||
* otherwise copy the default string to the input string.
|
||
*/
|
||
if (field->field.input->lbl_flag)
|
||
field->field.input->input[0] = '\0';
|
||
else if (field->field.input->label) {
|
||
strncpy(field->field.input->input,
|
||
field->field.input->label,
|
||
field->field.input->limit);
|
||
field->field.input->input[field->field.input->limit] = 0;
|
||
}
|
||
} else if ((field->type != F_TEXT) && (field->type != F_MENU) &&
|
||
(field->type != F_ACTION)) {
|
||
print_status("Unknown field type, closing form");
|
||
endfrm(form);
|
||
return (ERR);
|
||
}
|
||
form->no_fields++;
|
||
field = &form->field[form->no_fields];
|
||
}
|
||
form->current_field = form->start_field;
|
||
show_form(form);
|
||
return (OK);
|
||
}
|
||
|
||
void
|
||
endfrm(struct form *form)
|
||
{
|
||
|
||
struct field *field = &form->field[0];
|
||
int i;
|
||
|
||
delwin(form->window);
|
||
|
||
for (i=0; i < form->no_fields; i++) {
|
||
if (field->type == F_INPUT)
|
||
free(field->field.input->input);
|
||
field = &form->field[i];
|
||
}
|
||
}
|
||
|
||
int
|
||
update_form(struct form *form)
|
||
{
|
||
show_form(form);
|
||
|
||
if (form->current_field == -1)
|
||
return (F_CANCEL);
|
||
|
||
wattrset(form->window, form->field[form->current_field].selattr);
|
||
switch (form->field[form->current_field].type) {
|
||
case F_MENU:
|
||
field_menu(form);
|
||
break;
|
||
case F_INPUT:
|
||
field_input(form);
|
||
break;
|
||
case F_ACTION:
|
||
field_action(form);
|
||
break;
|
||
case F_TEXT:
|
||
default:
|
||
print_status("Error, current field is invalid");
|
||
return (F_CANCEL);
|
||
}
|
||
wattrset(form->window, 0);
|
||
|
||
return (done);
|
||
}
|
||
|
||
static void
|
||
show_form(struct form *form)
|
||
{
|
||
int i;
|
||
int y, x;
|
||
|
||
wattrset(form->window, form->attr);
|
||
for (y = 0; y < form->height; y++)
|
||
for (x = 0; x < form->width; x++)
|
||
mvwaddch(form->window, y, x, ' ');
|
||
|
||
for (i=0; i < form->no_fields; i++) {
|
||
wattrset(form->window, form->field[i].attr);
|
||
wmove(form->window, form->field[i].y, form->field[i].x);
|
||
switch (form->field[i].type) {
|
||
case F_TEXT:
|
||
disp_text(form, i);
|
||
break;
|
||
case F_MENU:
|
||
disp_menu(form, i);
|
||
break;
|
||
case F_INPUT:
|
||
disp_input(form,i);
|
||
break;
|
||
case F_ACTION:
|
||
disp_action(form,i);
|
||
break;
|
||
case F_END:
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
wattrset(form->window, 0);
|
||
wrefresh(form->window);
|
||
}
|
||
|
||
static void
|
||
disp_text(struct form *form, int index)
|
||
{
|
||
|
||
struct field *field = &form->field[index];
|
||
|
||
if (print_string(form->window, field->y, field->x, field->height,
|
||
field->width, field->field.text->text) == ERR)
|
||
print_status("Illegal scroll in print_string");
|
||
}
|
||
|
||
static void
|
||
disp_input(struct form *form, int index)
|
||
{
|
||
|
||
struct field *field = &form->field[index];
|
||
|
||
if (field->field.input->lbl_flag) {
|
||
if (print_string(form->window, field->y, field->x, field->height,
|
||
field->width, field->field.input->label) == ERR)
|
||
print_status("Illegal scroll in print_string");
|
||
} else
|
||
if (print_string(form->window, field->y, field->x, field->height,
|
||
field->width, field->field.input->input) == ERR)
|
||
print_status("Illegal scroll in print_string");
|
||
}
|
||
|
||
static void
|
||
disp_menu(struct form *form, int index)
|
||
{
|
||
struct field *field = &form->field[index];
|
||
|
||
if (print_string(form->window, field->y, field->x, field->height,
|
||
field->width,
|
||
field->field.menu->options[field->field.menu->selected]) == ERR)
|
||
print_status("Illegal scroll in print_string");
|
||
}
|
||
|
||
static void
|
||
disp_action(struct form *form, int index)
|
||
{
|
||
struct field *field = &form->field[index];
|
||
|
||
if (print_string(form->window, field->y, field->x, field->height,
|
||
field->width,
|
||
field->field.action->text) == ERR)
|
||
print_status("Illegal scroll in print_string");
|
||
}
|
||
|
||
static void
|
||
field_action(struct form *form)
|
||
{
|
||
|
||
struct field *field = &form->field[form->current_field];
|
||
int ch;
|
||
|
||
for (;;) {
|
||
disp_action(form, form->current_field);
|
||
wmove(form->window, field->y, field->x);
|
||
ch = wgetch(form->window);
|
||
if (ch == F_ACCEPT) {
|
||
(*field->field.action->fn)();
|
||
return;
|
||
} else if (!next_field(form, ch))
|
||
beep();
|
||
else
|
||
return;
|
||
}
|
||
}
|
||
|
||
static void
|
||
field_menu(struct form *form)
|
||
{
|
||
struct field *field = &form->field[form->current_field];
|
||
int ch;
|
||
|
||
for (;;) {
|
||
disp_menu(form, form->current_field);
|
||
wmove(form->window, field->y, field->x);
|
||
switch (ch = wgetch(form->window)) {
|
||
case ' ':
|
||
print_status("");
|
||
field->field.menu->selected++;
|
||
if (field->field.menu->selected >= field->field.menu->no_options)
|
||
field->field.menu->selected = 0;
|
||
break;
|
||
default:
|
||
if (!next_field(form, ch)) {
|
||
print_status("Hit the space bar to toggle through options");
|
||
beep();
|
||
} else
|
||
return;
|
||
}
|
||
}
|
||
}
|
||
|
||
static int
|
||
next_field(struct form *form, int ch)
|
||
{
|
||
|
||
struct field *field = &form->field[form->current_field];
|
||
|
||
if (ch == F_UP) {
|
||
if (field->up == -1) {
|
||
print_status("Can't go up from here");
|
||
return (0);
|
||
} else
|
||
form->current_field = field->up;
|
||
} else if (ch == F_DOWN) {
|
||
if (field->down == -1) {
|
||
print_status("Can't go down from here");
|
||
return (0);
|
||
} else
|
||
form->current_field = field->down;
|
||
} else if (ch == F_NEXT) {
|
||
if (field->next == -1) {
|
||
print_status("Can't go to next from here");
|
||
return (0);
|
||
} else
|
||
form->current_field = field->next;
|
||
} else if (ch == F_RIGHT) {
|
||
if (field->right == -1) {
|
||
print_status("Can't go right from here");
|
||
return (0);
|
||
} else
|
||
form->current_field = field->right;
|
||
} else if (ch == F_LEFT) {
|
||
if (field->left == -1) {
|
||
print_status("Can't go left from here");
|
||
return (0);
|
||
} else
|
||
form->current_field = field->left;
|
||
} else
|
||
return (0);
|
||
|
||
print_status("");
|
||
return (1);
|
||
}
|
||
|
||
static int
|
||
print_string(WINDOW *window, int y, int x,
|
||
int height, int fwidth, char *string)
|
||
{
|
||
int len;
|
||
int width;
|
||
|
||
if (!string)
|
||
len = -1;
|
||
|
||
len = strlen(string);
|
||
|
||
if (wmove(window, y, x) == ERR)
|
||
return (ERR);
|
||
while (height--) {
|
||
width = fwidth;
|
||
while (width--) {
|
||
if (len-- > 0) {
|
||
if (waddch(window, *string++) == ERR)
|
||
return (ERR);
|
||
} else
|
||
if (waddch(window, ' ') == ERR)
|
||
return (ERR);
|
||
}
|
||
if (wmove(window, ++y, x) == ERR)
|
||
return (ERR);
|
||
|
||
}
|
||
return (OK);
|
||
}
|
||
|
||
void
|
||
print_status(char *msg)
|
||
{
|
||
if (wmove(stdscr, LINES-1, 0) == ERR) {
|
||
endwin();
|
||
exit(1);
|
||
}
|
||
|
||
wclrtoeol(stdscr);
|
||
|
||
wstandout(stdscr);
|
||
if (wprintw(stdscr, "%s",
|
||
msg) == ERR) {
|
||
endwin();
|
||
exit(1);
|
||
}
|
||
wstandend(stdscr);
|
||
wrefresh(stdscr);
|
||
}
|
||
|
||
|
||
void
|
||
field_input(struct form *form)
|
||
{
|
||
struct field *field = &form->field[form->current_field];
|
||
int len;
|
||
int ch;
|
||
int disp_off=0, abspos=0, cursor = 0;
|
||
|
||
#define DISPOFF ((len < field->width) ? 0 : len - field->width)
|
||
#define CURSPOS ((len < field->width) ? len : field->width)
|
||
|
||
len = strlen(field->field.input->input);
|
||
disp_input(form, form->current_field);
|
||
|
||
cursor = CURSPOS;
|
||
abspos = cursor;
|
||
|
||
for(;;) {
|
||
|
||
wmove(form->window, field->y, field->x+cursor);
|
||
wrefresh(form->window);
|
||
|
||
ch = wgetch(form->window);
|
||
if (next_field(form, ch)) {
|
||
print_string(form->window, field->y, field->x,
|
||
field->height, field->width,
|
||
field->field.input->input+DISPOFF);
|
||
return;
|
||
}
|
||
if (field->field.input->lbl_flag) {
|
||
field->field.input->lbl_flag = 0;
|
||
}
|
||
if ((ch == F_CHOME) || (ch == '')) {
|
||
disp_off = 0;
|
||
cursor = 0;
|
||
abspos = 0;
|
||
} else if ((ch == F_CEND) || (ch == '')) {
|
||
disp_off = DISPOFF;
|
||
abspos = len;
|
||
cursor = CURSPOS;
|
||
} else if (ch == F_CDEL) {
|
||
if (!(len-abspos))
|
||
beep();
|
||
else {
|
||
bcopy(field->field.input->input+abspos+1,
|
||
field->field.input->input+abspos,
|
||
len - abspos);
|
||
--len;
|
||
}
|
||
} else if ((ch == F_CLEFT) || (ch == F_CBS) || (ch == '')) {
|
||
if (!abspos)
|
||
beep();
|
||
else {
|
||
if (ch == F_CBS) {
|
||
bcopy(field->field.input->input+abspos,
|
||
field->field.input->input+abspos-1,
|
||
len-abspos+1);
|
||
--len;
|
||
}
|
||
--abspos;
|
||
--cursor;
|
||
if ((disp_off) && (cursor < 0)) {
|
||
--disp_off;
|
||
++cursor;
|
||
}
|
||
}
|
||
} else if ((ch == F_CRIGHT) || (ch == '')) {
|
||
if (abspos == len)
|
||
beep();
|
||
else {
|
||
++abspos;
|
||
if (++cursor == field->width) {
|
||
++disp_off;
|
||
--cursor;
|
||
}
|
||
}
|
||
} else if ((isprint(ch)) && (len < field->field.input->limit)){
|
||
bcopy(field->field.input->input+abspos,
|
||
field->field.input->input+abspos+1, len-abspos+1);
|
||
field->field.input->input[abspos++] = ch;
|
||
len++;
|
||
if (++cursor > field->width) {
|
||
++disp_off;
|
||
--cursor;
|
||
}
|
||
} else {
|
||
beep();
|
||
}
|
||
print_string(form->window, field->y, field->x, field->height,
|
||
field->width, field->field.input->input+disp_off);
|
||
}
|
||
/* Not Reached */
|
||
}
|
||
|
||
void
|
||
exit_form(void)
|
||
{
|
||
done = F_DONE;
|
||
}
|
||
|
||
void
|
||
cancel_form(void)
|
||
{
|
||
done = F_CANCEL;
|
||
}
|