614 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			614 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*	$NetBSD: form.c,v 1.15 2004/11/24 11:57:09 blymn Exp $	*/
 | |
| 
 | |
| /*-
 | |
|  * Copyright (c) 1998-1999 Brett Lymn
 | |
|  *                         (blymn@baea.com.au, brett_lymn@yahoo.com.au)
 | |
|  * All rights reserved.
 | |
|  *
 | |
|  * This code has been donated to The NetBSD Foundation by the Author.
 | |
|  *
 | |
|  * 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.
 | |
|  * 2. The name of the author may not be used to endorse or promote products
 | |
|  *    derived from this software without specific prior written permission
 | |
|  *
 | |
|  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 THE AUTHOR 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 <sys/cdefs.h>
 | |
| __RCSID("$NetBSD: form.c,v 1.15 2004/11/24 11:57:09 blymn Exp $");
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <strings.h>
 | |
| #include <form.h>
 | |
| #include "internals.h"
 | |
| 
 | |
| extern FIELD _formi_default_field;
 | |
| 
 | |
| FORM _formi_default_form = {
 | |
| 	FALSE, /* true if performing a init or term function */
 | |
| 	FALSE, /* the form is posted */
 | |
| 	FALSE, /* make field list circular if true */
 | |
| 	NULL, /* window for the form */
 | |
| 	NULL, /* subwindow for the form */
 | |
| 	NULL, /* use this window for output */
 | |
| 	NULL, /* user defined pointer */
 | |
| 	0, /* options for the form */
 | |
| 	NULL, /* function called when form posted and
 | |
| 				after page change */
 | |
| 	NULL, /* function called when form is unposted and
 | |
| 				before page change */
 | |
| 	NULL, /* function called when form posted and after
 | |
| 				 current field changes */
 | |
| 	NULL, /* function called when form unposted and
 | |
| 				 before current field changes */
 | |
| 	0, /* number of fields attached */
 | |
| 	0, /* current field */
 | |
| 	0, /* current page of form */
 | |
| 	0, /* number of pages in the form */
 | |
| 	NULL, /* dynamic array of fields that start
 | |
| 					   the pages */
 | |
| 	{NULL, NULL}, /* sorted field list */
 | |
| 	NULL /* array of fields attached to this form. */
 | |
| };
 | |
| 
 | |
| /*
 | |
|  * Set the window associated with the form
 | |
|  */
 | |
| int
 | |
| set_form_win(FORM *form, WINDOW *win)
 | |
| {
 | |
| 	if (form == NULL) {
 | |
| 		_formi_default_form.win = win;
 | |
| 		_formi_default_form.scrwin = win;
 | |
| 	} else {
 | |
| 		if (form->posted == TRUE)
 | |
| 			return E_POSTED;
 | |
| 		else {
 | |
| 			form->win = win;
 | |
| 			form->scrwin = win;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the window used by the given form
 | |
|  */
 | |
| WINDOW *
 | |
| form_win(FORM *form)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return _formi_default_form.win;
 | |
| 	else
 | |
| 		return form->win;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Set the subwindow for the form.
 | |
|  */
 | |
| int
 | |
| set_form_sub(FORM *form, WINDOW *window)
 | |
| {
 | |
| 	if (form == NULL) {
 | |
| 		_formi_default_form.subwin = window;
 | |
| 		_formi_default_form.scrwin = window;
 | |
| 	} else {
 | |
| 		if (form->posted == TRUE)
 | |
| 			return E_POSTED;
 | |
| 		else {
 | |
| 			form->subwin = window;
 | |
| 			form->scrwin = window;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the subwindow for the given form.
 | |
|  */
 | |
| WINDOW *
 | |
| form_sub(FORM *form)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return _formi_default_form.subwin;
 | |
| 	else
 | |
| 		return form->subwin;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the minimum size required to contain the form.
 | |
|  */
 | |
| int
 | |
| scale_form(FORM *form, int *rows, int *cols)
 | |
| {
 | |
| 	int i, max_row, max_col, temp;
 | |
| 
 | |
| 	if ((form->fields == NULL) || (form->fields[0] == NULL))
 | |
| 		return E_NOT_CONNECTED;
 | |
| 
 | |
| 	max_row = 0;
 | |
| 	max_col = 0;
 | |
| 	
 | |
| 	for (i = 0; i < form->field_count; i++) {
 | |
| 		temp = form->fields[i]->form_row + form->fields[i]->rows;
 | |
| 		max_row = (temp > max_row)? temp : max_row;
 | |
| 		temp = form->fields[i]->form_col + form->fields[i]->cols;
 | |
| 		max_col = (temp > max_col)? temp : max_col;
 | |
| 	}
 | |
| 
 | |
| 	(*rows) = max_row;
 | |
| 	(*cols) = max_col;
 | |
| 	
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Set the user defined pointer for the form given.
 | |
|  */
 | |
| int
 | |
| set_form_userptr(FORM *form, void *ptr)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		_formi_default_form.userptr = ptr;
 | |
| 	else
 | |
| 		form->userptr = ptr;
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the user defined pointer associated with the given form.
 | |
|  */
 | |
| void *
 | |
| form_userptr(FORM *form)
 | |
| {
 | |
| 
 | |
| 	if (form == NULL)
 | |
| 		return _formi_default_form.userptr;
 | |
| 	else
 | |
| 		return form->userptr;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Set the form options to the given ones.
 | |
|  */
 | |
| int
 | |
| set_form_opts(FORM *form, Form_Options options)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		_formi_default_form.opts = options;
 | |
| 	else
 | |
| 		form->opts = options;
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Turn the given options on for the form.
 | |
|  */
 | |
| int
 | |
| form_opts_on(FORM *form, Form_Options options)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		_formi_default_form.opts |= options;
 | |
| 	else
 | |
| 		form->opts |= options;
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Turn the given options off for the form.
 | |
|  */
 | |
| int
 | |
| form_opts_off(FORM *form, Form_Options options)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		_formi_default_form.opts &= ~options;
 | |
| 	else
 | |
| 		form->opts &= ~options;
 | |
| 
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the options set for the given form.
 | |
|  */
 | |
| Form_Options
 | |
| form_opts(FORM *form)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return _formi_default_form.opts;
 | |
| 	else
 | |
| 		return form->opts;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Set the form init function for the given form
 | |
|  */
 | |
| int
 | |
| set_form_init(FORM *form, Form_Hook func)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		_formi_default_form.form_init = func;
 | |
| 	else
 | |
| 		form->form_init = func;
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the init function associated with the given form.
 | |
|  */
 | |
| Form_Hook
 | |
| form_init(FORM *form)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return _formi_default_form.form_init;
 | |
| 	else
 | |
| 		return form->form_init;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Set the function to be called on form termination.
 | |
|  */
 | |
| int
 | |
| set_form_term(FORM *form, Form_Hook function)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		_formi_default_form.form_term = function;
 | |
| 	else
 | |
| 		form->form_term = function;
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the function defined for the termination function.
 | |
|  */
 | |
| Form_Hook
 | |
| form_term(FORM *form)
 | |
| {
 | |
| 
 | |
| 	if (form == NULL)
 | |
| 		return _formi_default_form.form_term;
 | |
| 	else
 | |
| 		return form->form_term;
 | |
| }
 | |
| 
 | |
| 	
 | |
| /*
 | |
|  * Attach the given fields to the form.
 | |
|  */
 | |
| int
 | |
| set_form_fields(FORM *form, FIELD **fields)
 | |
| {
 | |
| 	int num_fields = 0, i, maxpg = 1, status;
 | |
| 
 | |
| 	if (form == NULL)
 | |
| 		return E_BAD_ARGUMENT;
 | |
| 
 | |
| 	if (form->posted == TRUE)
 | |
| 		return E_POSTED;
 | |
| 
 | |
| 	if (fields == NULL)
 | |
| 		return E_BAD_ARGUMENT;
 | |
| 
 | |
| 	while (fields[num_fields] != NULL) {
 | |
| 		if ((fields[num_fields]->parent != NULL) &&
 | |
| 		    (fields[num_fields]->parent != form))
 | |
| 			return E_CONNECTED;
 | |
| 		num_fields++;
 | |
| 	}
 | |
| 
 | |
| 	  /* disconnect old fields, if any */
 | |
| 	if (form->fields != NULL) {
 | |
| 		for (i = 0; i < form->field_count; i++) {
 | |
| 			form->fields[i]->parent = NULL;
 | |
| 			form->fields[i]->index = -1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	  /* kill old page pointers if any */
 | |
| 	if (form->page_starts != NULL)
 | |
| 		free(form->page_starts);
 | |
| 
 | |
| 	form->field_count = num_fields;
 | |
| 
 | |
| 	  /* now connect the new fields to the form */
 | |
| 	for (i = 0; i < num_fields; i++) {
 | |
| 		fields[i]->parent = form;
 | |
| 		fields[i]->index = i;
 | |
| 		  /* set the page number of the field */
 | |
| 		if (fields[i]->page_break == 1) 
 | |
| 			maxpg++;
 | |
| 		fields[i]->page = maxpg;
 | |
| 	}
 | |
| 
 | |
| 	form->fields = fields;
 | |
| 	form->cur_field = 0;
 | |
| 	form->max_page = maxpg;
 | |
| 	if ((status = _formi_find_pages(form)) != E_OK)
 | |
| 		return status;
 | |
| 
 | |
| 	  /* sort the fields and set the navigation pointers */
 | |
| 	_formi_sort_fields(form);
 | |
| 	_formi_stitch_fields(form);
 | |
| 	
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the fields attached to the form given.
 | |
|  */
 | |
| FIELD **
 | |
| form_fields(FORM *form)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return NULL;
 | |
| 
 | |
| 	return form->fields;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the number of fields attached to the given form.
 | |
|  */
 | |
| int
 | |
| field_count(FORM *form)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return -1;
 | |
| 
 | |
| 	return form->field_count;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Move the given field to the row and column given.
 | |
|  */
 | |
| int
 | |
| move_field(FIELD *fptr, int frow, int fcol)
 | |
| {
 | |
| 	FIELD *field = (fptr == NULL) ? &_formi_default_field : fptr;
 | |
| 
 | |
| 	if (field->parent != NULL)
 | |
| 		return E_CONNECTED;
 | |
| 
 | |
| 	field->form_row = frow;
 | |
| 	field->form_col = fcol;
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Set the page of the form to the given page.
 | |
|  */
 | |
| int
 | |
| set_form_page(FORM *form, int page)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return E_BAD_ARGUMENT;
 | |
| 
 | |
| 	if (form->in_init == TRUE)
 | |
| 		return E_BAD_STATE;
 | |
| 
 | |
| 	if (page > form->max_page)
 | |
| 		return E_BAD_ARGUMENT;
 | |
| 
 | |
| 	form->page = page;
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the maximum page of the form.
 | |
|  */
 | |
| int
 | |
| form_max_page(FORM *form)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return _formi_default_form.max_page;
 | |
| 	else
 | |
| 		return form->max_page;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the current page of the form.
 | |
|  */
 | |
| int
 | |
| form_page(FORM *form)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return E_BAD_ARGUMENT;
 | |
| 
 | |
| 	return form->page;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Set the current field to the field given.
 | |
|  */
 | |
| int
 | |
| set_current_field(FORM *form, FIELD *field)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return E_BAD_ARGUMENT;
 | |
| 
 | |
| 	if (form->in_init == TRUE)
 | |
| 		return E_BAD_STATE;
 | |
| 
 | |
| 	if (field == NULL)
 | |
| 		return E_INVALID_FIELD;
 | |
| 
 | |
| 	if ((field->parent == NULL) || (field->parent != form))
 | |
| 		return E_INVALID_FIELD; /* field is not of this form */
 | |
| 
 | |
| 	form->cur_field = field->index;
 | |
| 
 | |
| 	  /* XXX update page if posted??? */
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the current field of the given form.
 | |
|  */
 | |
| FIELD *
 | |
| current_field(FORM *form)
 | |
| {
 | |
| 	if (form == NULL)
 | |
| 		return NULL;
 | |
| 
 | |
| 	if (form->fields == NULL)
 | |
| 		return NULL;
 | |
| 
 | |
| 	return form->fields[form->cur_field];
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Allocate a new form with the given fields.
 | |
|  */
 | |
| FORM *
 | |
| new_form(FIELD **fields)
 | |
| {
 | |
| 	FORM *new;
 | |
| 
 | |
| 	if ((new = (FORM *) malloc(sizeof(FORM))) == NULL)
 | |
| 		return NULL;
 | |
| 
 | |
| 	
 | |
| 	  /* copy in the defaults... */
 | |
| 	bcopy(&_formi_default_form, new, sizeof(FORM));
 | |
| 
 | |
| 	if (new->win == NULL)
 | |
| 		new->scrwin = stdscr; /* something for curses to write to */
 | |
| 
 | |
| 	if (fields != NULL) { /* attach the fields, if any */
 | |
| 		if (set_form_fields(new, fields) < 0) {
 | |
| 			free(new); /* field attach failed, back out */
 | |
| 			return NULL;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return new;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Free the given form.
 | |
|  */
 | |
| int
 | |
| free_form(FORM *form)
 | |
| {
 | |
| 	int i;
 | |
| 	
 | |
| 	if (form == NULL)
 | |
| 		return E_BAD_ARGUMENT;
 | |
| 
 | |
| 	if (form->posted == TRUE)
 | |
| 		return E_POSTED;
 | |
| 
 | |
| 	for (i = 0; i < form->field_count; i++) {
 | |
| 		  /* detach all the fields from the form */
 | |
| 		form->fields[i]->parent = NULL;
 | |
| 		form->fields[i]->index = -1;
 | |
| 	}
 | |
| 
 | |
| 	free(form);
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Tell if the current field of the form has offscreen data ahead
 | |
|  */
 | |
| int
 | |
| data_ahead(FORM *form)
 | |
| {
 | |
| 	FIELD *cur;
 | |
| 	
 | |
| 	if ((form == NULL) || (form->fields == NULL)
 | |
| 	    || (form->fields[0] == NULL))
 | |
| 		return FALSE;
 | |
| 
 | |
| 	cur = form->fields[form->cur_field];
 | |
| 
 | |
| 	  /*XXXX wrong */
 | |
| 	if (cur->cur_line->expanded > cur->cols)
 | |
| 		return TRUE;
 | |
| 
 | |
| 	return FALSE;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Tell if current field of the form has offscreen data behind
 | |
|  */
 | |
| int
 | |
| data_behind(FORM *form)
 | |
| {
 | |
| 	FIELD *cur;
 | |
| 	
 | |
| 	if ((form == NULL) || (form->fields == NULL)
 | |
| 	    || (form->fields[0] == NULL))
 | |
| 		return FALSE;
 | |
| 
 | |
| 	cur = form->fields[form->cur_field];
 | |
| 
 | |
| 	if (cur->start_char > 0)
 | |
| 		return TRUE;
 | |
| 
 | |
| 	return FALSE;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Position the form cursor.
 | |
|  */
 | |
| int
 | |
| pos_form_cursor(FORM *form)
 | |
| {
 | |
| 	FIELD *cur;
 | |
| 	int row, col;
 | |
| 	
 | |
| 	if ((form == NULL) || (form->fields == NULL) ||
 | |
| 	    (form->fields[0] == NULL))
 | |
| 		return E_BAD_ARGUMENT;
 | |
| 
 | |
| 	if (form->posted != 1)
 | |
| 		return E_NOT_POSTED;
 | |
| 
 | |
| 	cur = form->fields[form->cur_field];
 | |
| 	row = cur->form_row;
 | |
| 	col = cur->form_col;
 | |
| 
 | |
| 	  /* if the field is public then show the cursor pos */
 | |
| 	if ((cur->opts & O_PUBLIC) == O_PUBLIC) {
 | |
| 		row += cur->cursor_ypos;
 | |
| 		col += cur->cursor_xpos;
 | |
| 		if (cur->cursor_xpos >= cur->cols) {
 | |
| 			col = cur->form_col;
 | |
| 			row++;
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| #ifdef DEBUG
 | |
| 	fprintf(dbg, "pos_cursor: row=%d, col=%d\n", row, col);
 | |
| #endif
 | |
| 	
 | |
| 	wmove(form->scrwin, row, col);
 | |
| 
 | |
| 	return E_OK;
 | |
| }
 | 
