1 /****************************************************************************
2 * Copyright (c) 1998-2014,2015 Free Software Foundation, Inc. *
4 * Permission is hereby granted, free of charge, to any person obtaining a *
5 * copy of this software and associated documentation files (the *
6 * "Software"), to deal in the Software without restriction, including *
7 * without limitation the rights to use, copy, modify, merge, publish, *
8 * distribute, distribute with modifications, sublicense, and/or sell *
9 * copies of the Software, and to permit persons to whom the Software is *
10 * furnished to do so, subject to the following conditions: *
12 * The above copyright notice and this permission notice shall be included *
13 * in all copies or substantial portions of the Software. *
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS *
16 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF *
17 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. *
18 * IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, *
19 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR *
20 * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR *
21 * THE USE OR OTHER DEALINGS IN THE SOFTWARE. *
23 * Except as contained in this notice, the name(s) of the above copyright *
24 * holders shall not be used in advertising or otherwise to promote the *
25 * sale, use or other dealings in this Software without prior written *
27 ****************************************************************************/
29 /****************************************************************************
30 * Author: Juergen Pfeifer, 1995,1997 *
31 ****************************************************************************/
33 #include "form.priv.h"
35 MODULE_ID("$Id: frm_driver.c,v 1.117 2015/11/28 20:39:09 tom Exp $")
37 /*----------------------------------------------------------------------------
38 This is the core module of the form library. It contains the majority
39 of the driver routines as well as the form_driver function.
41 Essentially this module is nearly the whole library. This is because
42 all the functions in this module depends on some others in the module,
43 so it makes no sense to split them into separate files because they
44 will always be linked together. The only acceptable concern is turnaround
45 time for this module, but now we have all Pentiums or RISCs, so what!
47 The driver routines are grouped into nine generic categories:
49 a) Page Navigation ( all functions prefixed by PN_ )
50 The current page of the form is left and some new page is
52 b) Inter-Field Navigation ( all functions prefixed by FN_ )
53 The current field of the form is left and some new field is
55 c) Intra-Field Navigation ( all functions prefixed by IFN_ )
56 The current position in the current field is changed.
57 d) Vertical Scrolling ( all functions prefixed by VSC_ )
58 Essentially this is a specialization of Intra-Field navigation.
59 It has to check for a multi-line field.
60 e) Horizontal Scrolling ( all functions prefixed by HSC_ )
61 Essentially this is a specialization of Intra-Field navigation.
62 It has to check for a single-line field.
63 f) Field Editing ( all functions prefixed by FE_ )
64 The content of the current field is changed
65 g) Edit Mode requests ( all functions prefixed by EM_ )
66 Switching between insert and overlay mode
67 h) Field-Validation requests ( all functions prefixed by FV_ )
68 Perform verifications of the field.
69 i) Choice requests ( all functions prefixed by CR_ )
70 Requests to enumerate possible field values
71 --------------------------------------------------------------------------*/
73 /*----------------------------------------------------------------------------
74 Some remarks on the placements of assert() macros :
75 I use them only on "strategic" places, i.e. top level entries where
76 I want to make sure that things are set correctly. Throughout subordinate
77 routines I omit them mostly.
78 --------------------------------------------------------------------------*/
81 Some options that may effect compatibility in behavior to SVr4 forms,
82 but they are here to allow a more intuitive and user friendly behavior of
83 our form implementation. This doesn't affect the API, so we feel it is
86 The initial implementation tries to stay very close with the behavior
87 of the original SVr4 implementation, although in some areas it is quite
88 clear that this isn't the most appropriate way. As far as possible this
89 sources will allow you to build a forms lib that behaves quite similar
90 to SVr4, but now and in the future we will give you better options.
91 Perhaps at some time we will make this configurable at runtime.
94 /* Implement a more user-friendly previous/next word behavior */
95 #define FRIENDLY_PREV_NEXT_WORD (1)
96 /* Fix the wrong behavior for forms with all fields inactive */
97 #define FIX_FORM_INACTIVE_BUG (1)
98 /* Allow dynamic field growth also when navigating past the end */
99 #define GROW_IF_NAVIGATE (1)
101 #if USE_WIDEC_SUPPORT
102 #define myADDNSTR(w, s, n) wadd_wchnstr(w, s, n)
103 #define myINSNSTR(w, s, n) wins_wchnstr(w, s, n)
104 #define myINNSTR(w, s, n) fix_wchnstr(w, s, n)
105 #define myWCWIDTH(w, y, x) cell_width(w, y, x)
107 #define myADDNSTR(w, s, n) waddnstr(w, s, n)
108 #define myINSNSTR(w, s, n) winsnstr(w, s, n)
109 #define myINNSTR(w, s, n) winnstr(w, s, n)
110 #define myWCWIDTH(w, y, x) 1
113 /*----------------------------------------------------------------------------
114 Forward references to some internally used static functions
115 --------------------------------------------------------------------------*/
116 static int Inter_Field_Navigation(int (*const fct) (FORM *), FORM *form);
117 static int FN_Next_Field(FORM *form);
118 static int FN_Previous_Field(FORM *form);
119 static int FE_New_Line(FORM *);
120 static int FE_Delete_Previous(FORM *);
122 /*----------------------------------------------------------------------------
125 Some Remarks on that: I use the convention to use UPPERCASE for constants
126 defined by Macros. If I provide a macro as a kind of inline routine to
127 provide some logic, I use my Upper_Lower case style.
128 --------------------------------------------------------------------------*/
130 /* Calculate the position of a single row in a field buffer */
131 #define Position_Of_Row_In_Buffer(field,row) ((row)*(field)->dcols)
133 /* Calculate start address for the fields buffer# N */
134 #define Address_Of_Nth_Buffer(field,N) \
135 ((field)->buf + (N)*(1+Buffer_Length(field)))
137 /* Calculate the start address of the row in the fields specified buffer# N */
138 #define Address_Of_Row_In_Nth_Buffer(field,N,row) \
139 (Address_Of_Nth_Buffer(field,N) + Position_Of_Row_In_Buffer(field,row))
141 /* Calculate the start address of the row in the fields primary buffer */
142 #define Address_Of_Row_In_Buffer(field,row) \
143 Address_Of_Row_In_Nth_Buffer(field,0,row)
145 /* Calculate the start address of the row in the forms current field
147 #define Address_Of_Current_Row_In_Nth_Buffer(form,N) \
148 Address_Of_Row_In_Nth_Buffer((form)->current,N,(form)->currow)
150 /* Calculate the start address of the row in the forms current field
152 #define Address_Of_Current_Row_In_Buffer(form) \
153 Address_Of_Current_Row_In_Nth_Buffer(form,0)
155 /* Calculate the address of the cursor in the forms current field
157 #define Address_Of_Current_Position_In_Nth_Buffer(form,N) \
158 (Address_Of_Current_Row_In_Nth_Buffer(form,N) + (form)->curcol)
160 /* Calculate the address of the cursor in the forms current field
162 #define Address_Of_Current_Position_In_Buffer(form) \
163 Address_Of_Current_Position_In_Nth_Buffer(form,0)
165 /* Logic to decide whether or not a field is actually a field with
166 vertical or horizontal scrolling */
167 #define Is_Scroll_Field(field) \
168 (((field)->drows > (field)->rows) || \
169 ((field)->dcols > (field)->cols))
171 /* Logic to decide whether or not a field needs to have an individual window
172 instead of a derived window because it contains invisible parts.
173 This is true for non-public fields and for scrollable fields. */
174 #define Has_Invisible_Parts(field) \
175 (!(Field_Has_Option(field, O_PUBLIC)) || \
176 Is_Scroll_Field(field))
178 /* Logic to decide whether or not a field needs justification */
179 #define Justification_Allowed(field) \
180 (((field)->just != NO_JUSTIFICATION) && \
181 (Single_Line_Field(field)) && \
182 ((Field_Has_Option(field, O_STATIC) && \
183 ((field)->dcols == (field)->cols)) || \
184 Field_Has_Option(field, O_DYNAMIC_JUSTIFY)))
186 /* Logic to determine whether or not a dynamic field may still grow */
187 #define Growable(field) ((field)->status & _MAY_GROW)
189 /* Macro to set the attributes for a fields window */
190 #define Set_Field_Window_Attributes(field,win) \
191 ( wbkgdset((win),(chtype)((chtype)((field)->pad) | (field)->back)), \
192 (void) wattrset((win), (int)(field)->fore) )
194 /* Logic to decide whether or not a field really appears on the form */
195 #define Field_Really_Appears(field) \
197 (field->form->status & _POSTED) &&\
198 (Field_Has_Option(field, O_VISIBLE)) &&\
199 (field->page == field->form->curpage))
201 /* Logic to determine whether or not we are on the first position in the
203 #define First_Position_In_Current_Field(form) \
204 (((form)->currow==0) && ((form)->curcol==0))
206 #define Minimum(a,b) (((a)<=(b)) ? (a) : (b))
207 #define Maximum(a,b) (((a)>=(b)) ? (a) : (b))
209 /*----------------------------------------------------------------------------
211 --------------------------------------------------------------------------*/
212 static FIELD_CELL myBLANK = BLANK;
213 static FIELD_CELL myZEROS;
217 check_pos(FORM *form, int lineno)
223 getyx(form->w, y, x);
224 if (y != form->currow || x != form->curcol)
226 T(("CHECKPOS %s@%d have position %d,%d vs want %d,%d",
229 form->currow, form->curcol));
233 #define CHECKPOS(form) check_pos(form, __LINE__)
235 #define CHECKPOS(form) /* nothing */
238 /*----------------------------------------------------------------------------
239 Wide-character special functions
240 --------------------------------------------------------------------------*/
241 #if USE_WIDEC_SUPPORT
244 wins_wchnstr(WINDOW *w, cchar_t *s, int n)
252 if ((code = wins_wch(w, s++)) != OK)
254 if ((code = wmove(w, y, x + 1)) != OK)
260 /* win_wchnstr is inconsistent with winnstr, since it returns OK rather than
261 * the number of items transferred.
264 fix_wchnstr(WINDOW *w, cchar_t *s, int n)
268 win_wchnstr(w, s, n);
270 * This function is used to extract the text only from the window.
271 * Strip attributes and color from the string so they will not be added
272 * back when copying the string to the window.
274 for (x = 0; x < n; ++x)
276 RemAttr(s[x], A_ATTRIBUTES);
283 * Returns the column of the base of the given cell.
286 cell_base(WINDOW *win, int y, int x)
290 while (LEGALYX(win, y, x))
292 cchar_t *data = &(win->_line[y].text[x]);
294 if (isWidecBase(CHDEREF(data)) || !isWidecExt(CHDEREF(data)))
305 * Returns the number of columns needed for the given cell in a window.
308 cell_width(WINDOW *win, int y, int x)
312 if (LEGALYX(win, y, x))
314 cchar_t *data = &(win->_line[y].text[x]);
316 if (isWidecExt(CHDEREF(data)))
318 /* recur, providing the number of columns to the next character */
319 result = cell_width(win, y, x - 1);
323 result = wcwidth(CharOf(CHDEREF(data)));
330 * There is no wide-character function such as wdel_wch(), so we must find
331 * all of the cells that comprise a multi-column character and delete them
335 delete_char(FORM *form)
337 int cells = cell_width(form->w, form->currow, form->curcol);
339 form->curcol = cell_base(form->w, form->currow, form->curcol);
340 wmove(form->w, form->currow, form->curcol);
346 #define DeleteChar(form) delete_char(form)
348 #define DeleteChar(form) \
349 wmove((form)->w, (form)->currow, (form)->curcol), \
353 /*---------------------------------------------------------------------------
354 | Facility : libnform
355 | Function : static char *Get_Start_Of_Data(char * buf, int blen)
357 | Description : Return pointer to first non-blank position in buffer.
358 | If buffer is empty return pointer to buffer itself.
360 | Return Values : Pointer to first non-blank position in buffer
361 +--------------------------------------------------------------------------*/
362 NCURSES_INLINE static FIELD_CELL *
363 Get_Start_Of_Data(FIELD_CELL *buf, int blen)
366 FIELD_CELL *end = &buf[blen];
368 assert(buf && blen >= 0);
369 while ((p < end) && ISBLANK(*p))
371 return ((p == end) ? buf : p);
374 /*---------------------------------------------------------------------------
375 | Facility : libnform
376 | Function : static char *After_End_Of_Data(char * buf, int blen)
378 | Description : Return pointer after last non-blank position in buffer.
379 | If buffer is empty, return pointer to buffer itself.
381 | Return Values : Pointer to position after last non-blank position in
383 +--------------------------------------------------------------------------*/
384 NCURSES_INLINE static FIELD_CELL *
385 After_End_Of_Data(FIELD_CELL *buf, int blen)
387 FIELD_CELL *p = &buf[blen];
389 assert(buf && blen >= 0);
390 while ((p > buf) && ISBLANK(p[-1]))
395 /*---------------------------------------------------------------------------
396 | Facility : libnform
397 | Function : static char *Get_First_Whitespace_Character(
398 | char * buf, int blen)
400 | Description : Position to the first whitespace character.
402 | Return Values : Pointer to first whitespace character in buffer.
403 +--------------------------------------------------------------------------*/
404 NCURSES_INLINE static FIELD_CELL *
405 Get_First_Whitespace_Character(FIELD_CELL *buf, int blen)
408 FIELD_CELL *end = &p[blen];
410 assert(buf && blen >= 0);
411 while ((p < end) && !ISBLANK(*p))
413 return ((p == end) ? buf : p);
416 /*---------------------------------------------------------------------------
417 | Facility : libnform
418 | Function : static char *After_Last_Whitespace_Character(
419 | char * buf, int blen)
421 | Description : Get the position after the last whitespace character.
423 | Return Values : Pointer to position after last whitespace character in
425 +--------------------------------------------------------------------------*/
426 NCURSES_INLINE static FIELD_CELL *
427 After_Last_Whitespace_Character(FIELD_CELL *buf, int blen)
429 FIELD_CELL *p = &buf[blen];
431 assert(buf && blen >= 0);
432 while ((p > buf) && !ISBLANK(p[-1]))
437 /* Set this to 1 to use the div_t version. This is a good idea if your
438 compiler has an intrinsic div() support. Unfortunately GNU-C has it
440 N.B.: This only works if form->curcol follows immediately form->currow
441 and both are of type int.
443 #define USE_DIV_T (0)
445 /*---------------------------------------------------------------------------
446 | Facility : libnform
447 | Function : static void Adjust_Cursor_Position(
448 | FORM * form, const char * pos)
450 | Description : Set current row and column of the form to values
451 | corresponding to the buffer position.
454 +--------------------------------------------------------------------------*/
455 NCURSES_INLINE static void
456 Adjust_Cursor_Position(FORM *form, const FIELD_CELL *pos)
461 field = form->current;
462 assert(pos >= field->buf && field->dcols > 0);
463 idx = (int)(pos - field->buf);
465 *((div_t *) & (form->currow)) = div(idx, field->dcols);
467 form->currow = idx / field->dcols;
468 form->curcol = idx - field->cols * form->currow;
470 if (field->drows < form->currow)
474 /*---------------------------------------------------------------------------
475 | Facility : libnform
476 | Function : static void Buffer_To_Window(
477 | const FIELD * field,
480 | Description : Copy the buffer to the window. If it is a multi-line
481 | field, the buffer is split to the lines of the
482 | window without any editing.
485 +--------------------------------------------------------------------------*/
487 Buffer_To_Window(const FIELD *field, WINDOW *win)
495 assert(win && field);
498 width = getmaxx(win);
499 height = getmaxy(win);
501 for (row = 0, pBuffer = field->buf;
503 row++, pBuffer += width)
505 if ((len = (int)(After_End_Of_Data(pBuffer, width) - pBuffer)) > 0)
508 myADDNSTR(win, pBuffer, len);
514 /*---------------------------------------------------------------------------
515 | Facility : libnform
516 | Function : void _nc_get_fieldbuffer(
521 | Description : Copy the content of the window into the buffer.
522 | The multiple lines of a window are simply
523 | concatenated into the buffer. Pad characters in
524 | the window will be replaced by blanks in the buffer.
527 +--------------------------------------------------------------------------*/
529 _nc_get_fieldbuffer(FORM *form, FIELD *field, FIELD_CELL *buf)
537 assert(form && field && buf);
544 height = getmaxy(win);
546 for (row = 0; (row < height) && (row < field->drows); row++)
549 len += myINNSTR(win, p + len, field->dcols);
553 /* replace visual padding character by blanks in buffer */
558 for (i = 0; i < len; i++, p++)
560 if ((unsigned long)CharOf(*p) == ChCharOf(pad)
561 #if USE_WIDEC_SUPPORT
570 /*---------------------------------------------------------------------------
571 | Facility : libnform
572 | Function : static void Window_To_Buffer(
576 | Description : Copy the content of the window into the buffer.
577 | The multiple lines of a window are simply
578 | concatenated into the buffer. Pad characters in
579 | the window will be replaced by blanks in the buffer.
582 +--------------------------------------------------------------------------*/
584 Window_To_Buffer(FORM *form, FIELD *field)
586 _nc_get_fieldbuffer(form, field, field->buf);
589 /*---------------------------------------------------------------------------
590 | Facility : libnform
591 | Function : static void Synchronize_Buffer(FORM * form)
593 | Description : If there was a change, copy the content of the
594 | window into the buffer, so the buffer is synchronized
595 | with the windows content. We have to indicate that the
596 | buffer needs validation due to the change.
599 +--------------------------------------------------------------------------*/
600 NCURSES_INLINE static void
601 Synchronize_Buffer(FORM *form)
603 if (form->status & _WINDOW_MODIFIED)
605 ClrStatus(form, _WINDOW_MODIFIED);
606 SetStatus(form, _FCHECK_REQUIRED);
607 Window_To_Buffer(form, form->current);
608 wmove(form->w, form->currow, form->curcol);
612 /*---------------------------------------------------------------------------
613 | Facility : libnform
614 | Function : static bool Field_Grown( FIELD *field, int amount)
616 | Description : This function is called for growable dynamic fields
617 | only. It has to increase the buffers and to allocate
618 | a new window for this field.
619 | This function has the side effect to set a new
620 | field-buffer pointer, the dcols and drows values
621 | as well as a new current Window for the field.
623 | Return Values : TRUE - field successfully increased
624 | FALSE - there was some error
625 +--------------------------------------------------------------------------*/
627 Field_Grown(FIELD *field, int amount)
631 if (field && Growable(field))
633 bool single_line_field = Single_Line_Field(field);
634 int old_buflen = Buffer_Length(field);
636 int old_dcols = field->dcols;
637 int old_drows = field->drows;
638 FIELD_CELL *oldbuf = field->buf;
642 FORM *form = field->form;
643 bool need_visual_update = ((form != (FORM *)0) &&
644 (form->status & _POSTED) &&
645 (form->current == field));
647 if (need_visual_update)
648 Synchronize_Buffer(form);
650 if (single_line_field)
652 growth = field->cols * amount;
654 growth = Minimum(field->maxgrow - field->dcols, growth);
655 field->dcols += growth;
656 if (field->dcols == field->maxgrow)
657 ClrStatus(field, _MAY_GROW);
661 growth = (field->rows + field->nrow) * amount;
663 growth = Minimum(field->maxgrow - field->drows, growth);
664 field->drows += growth;
665 if (field->drows == field->maxgrow)
666 ClrStatus(field, _MAY_GROW);
668 /* drows, dcols changed, so we get really the new buffer length */
669 new_buflen = Buffer_Length(field);
670 newbuf = (FIELD_CELL *)malloc(Total_Buffer_Size(field));
673 /* restore to previous state */
674 field->dcols = old_dcols;
675 field->drows = old_drows;
676 if ((single_line_field && (field->dcols != field->maxgrow)) ||
677 (!single_line_field && (field->drows != field->maxgrow)))
678 SetStatus(field, _MAY_GROW);
682 /* Copy all the buffers. This is the reason why we can't just use
689 result = TRUE; /* allow sharing of recovery on failure */
691 T((T_CREATE("fieldcell %p"), (void *)newbuf));
693 for (i = 0; i <= field->nbuf; i++)
695 new_bp = Address_Of_Nth_Buffer(field, i);
696 old_bp = oldbuf + i * (1 + old_buflen);
697 for (j = 0; j < old_buflen; ++j)
698 new_bp[j] = old_bp[j];
699 while (j < new_buflen)
700 new_bp[j++] = myBLANK;
701 new_bp[new_buflen] = myZEROS;
704 #if USE_WIDEC_SUPPORT && NCURSES_EXT_FUNCS
705 if (wresize(field->working, 1, Buffer_Length(field) + 1) == ERR)
709 if (need_visual_update && result)
711 WINDOW *new_window = newpad(field->drows, field->dcols);
715 assert(form != (FORM *)0);
718 form->w = new_window;
719 Set_Field_Window_Attributes(field, form->w);
721 Buffer_To_Window(field, form->w);
723 wmove(form->w, form->currow, form->curcol);
732 /* reflect changes in linked fields */
733 if (field != field->link)
737 for (linked_field = field->link;
738 linked_field != field;
739 linked_field = linked_field->link)
741 linked_field->buf = field->buf;
742 linked_field->drows = field->drows;
743 linked_field->dcols = field->dcols;
749 /* restore old state */
750 field->dcols = old_dcols;
751 field->drows = old_drows;
753 if ((single_line_field &&
754 (field->dcols != field->maxgrow)) ||
755 (!single_line_field &&
756 (field->drows != field->maxgrow)))
757 SetStatus(field, _MAY_GROW);
765 #ifdef NCURSES_MOUSE_VERSION
766 /*---------------------------------------------------------------------------
767 | Facility : libnform
768 | Function : int Field_encloses(FIELD *field, int ry, int rx)
770 | Description : Check if the given coordinates lie within the given field.
772 | Return Values : E_OK - success
773 | E_BAD_ARGUMENT - invalid form pointer
774 | E_SYSTEM_ERROR - form has no current field or
776 +--------------------------------------------------------------------------*/
778 Field_encloses(FIELD *field, int ry, int rx)
780 T((T_CALLED("Field_encloses(%p)"), (void *)field));
783 && (field->frow + field->rows) > ry
785 && (field->fcol + field->cols) > rx)
789 RETURN(E_INVALID_FIELD);
793 /*---------------------------------------------------------------------------
794 | Facility : libnform
795 | Function : int _nc_Position_Form_Cursor(FORM * form)
797 | Description : Position the cursor in the window for the current
798 | field to be in sync. with the currow and curcol
801 | Return Values : E_OK - success
802 | E_BAD_ARGUMENT - invalid form pointer
803 | E_SYSTEM_ERROR - form has no current field or
805 +--------------------------------------------------------------------------*/
807 _nc_Position_Form_Cursor(FORM *form)
813 return (E_BAD_ARGUMENT);
815 if (!form->w || !form->current)
816 return (E_SYSTEM_ERROR);
818 field = form->current;
819 formwin = Get_Form_Window(form);
821 wmove(form->w, form->currow, form->curcol);
822 if (Has_Invisible_Parts(field))
824 /* in this case fieldwin isn't derived from formwin, so we have
825 to move the cursor in formwin by hand... */
827 field->frow + form->currow - form->toprow,
828 field->fcol + form->curcol - form->begincol);
836 /*---------------------------------------------------------------------------
837 | Facility : libnform
838 | Function : int _nc_Refresh_Current_Field(FORM * form)
840 | Description : Propagate the changes in the fields window to the
841 | window of the form.
843 | Return Values : E_OK - on success
844 | E_BAD_ARGUMENT - invalid form pointer
845 | E_SYSTEM_ERROR - general error
846 +--------------------------------------------------------------------------*/
848 _nc_Refresh_Current_Field(FORM *form)
853 T((T_CALLED("_nc_Refresh_Current_Field(%p)"), (void *)form));
856 RETURN(E_BAD_ARGUMENT);
858 if (!form->w || !form->current)
859 RETURN(E_SYSTEM_ERROR);
861 field = form->current;
862 formwin = Get_Form_Window(form);
864 if (Field_Has_Option(field, O_PUBLIC))
866 if (Is_Scroll_Field(field))
868 /* Again, in this case the fieldwin isn't derived from formwin,
869 so we have to perform a copy operation. */
870 if (Single_Line_Field(field))
872 /* horizontal scrolling */
873 if (form->curcol < form->begincol)
874 form->begincol = form->curcol;
877 if (form->curcol >= (form->begincol + field->cols))
878 form->begincol = form->curcol - field->cols + 1;
887 field->cols + field->fcol - 1,
892 /* A multi-line, i.e. vertical scrolling field */
893 int row_after_bottom, first_modified_row, first_unmodified_row;
895 if (field->drows > field->rows)
897 row_after_bottom = form->toprow + field->rows;
898 if (form->currow < form->toprow)
900 form->toprow = form->currow;
901 SetStatus(field, _NEWTOP);
903 if (form->currow >= row_after_bottom)
905 form->toprow = form->currow - field->rows + 1;
906 SetStatus(field, _NEWTOP);
908 if (field->status & _NEWTOP)
910 /* means we have to copy whole range */
911 first_modified_row = form->toprow;
912 first_unmodified_row = first_modified_row + field->rows;
913 ClrStatus(field, _NEWTOP);
917 /* we try to optimize : finding the range of touched
919 first_modified_row = form->toprow;
920 while (first_modified_row < row_after_bottom)
922 if (is_linetouched(form->w, first_modified_row))
924 first_modified_row++;
926 first_unmodified_row = first_modified_row;
927 while (first_unmodified_row < row_after_bottom)
929 if (!is_linetouched(form->w, first_unmodified_row))
931 first_unmodified_row++;
937 first_modified_row = form->toprow;
938 first_unmodified_row = first_modified_row + field->rows;
940 if (first_unmodified_row != first_modified_row)
945 field->frow + first_modified_row - form->toprow,
947 field->frow + first_unmodified_row - form->toprow - 1,
948 field->cols + field->fcol - 1,
955 /* if the field-window is simply a derived window, i.e. contains no
956 * invisible parts, the whole thing is trivial
962 returnCode(_nc_Position_Form_Cursor(form));
965 /*---------------------------------------------------------------------------
966 | Facility : libnform
967 | Function : static void Perform_Justification(
971 | Description : Output field with requested justification
974 +--------------------------------------------------------------------------*/
976 Perform_Justification(FIELD *field, WINDOW *win)
982 bp = (Field_Has_Option(field, O_NO_LEFT_STRIP)
984 : Get_Start_Of_Data(field->buf, Buffer_Length(field)));
985 len = (int)(After_End_Of_Data(field->buf, Buffer_Length(field)) - bp);
989 assert(win && (field->drows == 1));
991 if (field->cols - len >= 0)
997 col = (field->cols - len) / 2;
1000 col = field->cols - len;
1007 myADDNSTR(win, bp, len);
1011 /*---------------------------------------------------------------------------
1012 | Facility : libnform
1013 | Function : static void Undo_Justification(
1017 | Description : Display field without any justification, i.e.
1021 +--------------------------------------------------------------------------*/
1023 Undo_Justification(FIELD *field, WINDOW *win)
1028 bp = (Field_Has_Option(field, O_NO_LEFT_STRIP)
1030 : Get_Start_Of_Data(field->buf, Buffer_Length(field)));
1031 len = (int)(After_End_Of_Data(field->buf, Buffer_Length(field)) - bp);
1037 myADDNSTR(win, bp, len);
1041 /*---------------------------------------------------------------------------
1042 | Facility : libnform
1043 | Function : static bool Check_Char(FORM *form,
1047 | TypeArgument *argp)
1049 | Description : Perform a single character check for character ch
1050 | according to the fieldtype instance.
1052 | Return Values : TRUE - Character is valid
1053 | FALSE - Character is invalid
1054 +--------------------------------------------------------------------------*/
1056 Check_Char(FORM *form,
1064 if (typ->status & _LINKED_TYPE)
1068 Check_Char(form, field, typ->left, ch, argp->left) ||
1069 Check_Char(form, field, typ->right, ch, argp->right));
1073 #if NCURSES_INTEROP_FUNCS
1074 if (typ->charcheck.occheck)
1076 if (typ->status & _GENERIC)
1077 return typ->charcheck.gccheck(ch, form, field, (void *)argp);
1079 return typ->charcheck.occheck(ch, (void *)argp);
1083 return typ->ccheck(ch, (void *)argp);
1087 return (!iscntrl(UChar(ch)) ? TRUE : FALSE);
1090 /*---------------------------------------------------------------------------
1091 | Facility : libnform
1092 | Function : static int Display_Or_Erase_Field(
1096 | Description : Create a subwindow for the field and display the
1097 | buffer contents (apply justification if required)
1098 | or simply erase the field.
1100 | Return Values : E_OK - on success
1101 | E_SYSTEM_ERROR - some error (typical no memory)
1102 +--------------------------------------------------------------------------*/
1104 Display_Or_Erase_Field(FIELD *field, bool bEraseFlag)
1110 return E_SYSTEM_ERROR;
1112 fwin = Get_Form_Window(field->form);
1114 field->rows, field->cols, field->frow, field->fcol);
1117 return E_SYSTEM_ERROR;
1120 if (Field_Has_Option(field, O_VISIBLE))
1122 Set_Field_Window_Attributes(field, win);
1126 (void)wattrset(win, (int)WINDOW_ATTRS(fwin));
1133 if (Field_Has_Option(field, O_PUBLIC))
1135 if (Justification_Allowed(field))
1136 Perform_Justification(field, win);
1138 Buffer_To_Window(field, win);
1140 ClrStatus(field, _NEWTOP);
1147 /* Macros to preset the bEraseFlag */
1148 #define Display_Field(field) Display_Or_Erase_Field(field,FALSE)
1149 #define Erase_Field(field) Display_Or_Erase_Field(field,TRUE)
1151 /*---------------------------------------------------------------------------
1152 | Facility : libnform
1153 | Function : static int Synchronize_Field(FIELD * field)
1155 | Description : Synchronize the windows content with the value in
1158 | Return Values : E_OK - success
1159 | E_BAD_ARGUMENT - invalid field pointer
1160 | E_SYSTEM_ERROR - some severe basic error
1161 +--------------------------------------------------------------------------*/
1163 Synchronize_Field(FIELD *field)
1169 return (E_BAD_ARGUMENT);
1171 if (((form = field->form) != (FORM *)0)
1172 && Field_Really_Appears(field))
1174 if (field == form->current)
1176 form->currow = form->curcol = form->toprow = form->begincol = 0;
1179 if ((Field_Has_Option(field, O_PUBLIC)) && Justification_Allowed(field))
1180 Undo_Justification(field, form->w);
1182 Buffer_To_Window(field, form->w);
1184 SetStatus(field, _NEWTOP);
1185 res = _nc_Refresh_Current_Field(form);
1188 res = Display_Field(field);
1190 SetStatus(field, _CHANGED);
1194 /*---------------------------------------------------------------------------
1195 | Facility : libnform
1196 | Function : static int Synchronize_Linked_Fields(FIELD * field)
1198 | Description : Propagate the Synchronize_Field function to all linked
1199 | fields. The first error that occurs in the sequence
1200 | of updates is the return value.
1202 | Return Values : E_OK - success
1203 | E_BAD_ARGUMENT - invalid field pointer
1204 | E_SYSTEM_ERROR - some severe basic error
1205 +--------------------------------------------------------------------------*/
1207 Synchronize_Linked_Fields(FIELD *field)
1209 FIELD *linked_field;
1214 return (E_BAD_ARGUMENT);
1217 return (E_SYSTEM_ERROR);
1219 for (linked_field = field->link;
1220 (linked_field != field) && (linked_field != 0);
1221 linked_field = linked_field->link)
1223 if (((syncres = Synchronize_Field(linked_field)) != E_OK) &&
1230 /*---------------------------------------------------------------------------
1231 | Facility : libnform
1232 | Function : int _nc_Synchronize_Attributes(FIELD * field)
1234 | Description : If a fields visual attributes have changed, this
1235 | routine is called to propagate those changes to the
1238 | Return Values : E_OK - success
1239 | E_BAD_ARGUMENT - invalid field pointer
1240 | E_SYSTEM_ERROR - some severe basic error
1241 +--------------------------------------------------------------------------*/
1243 _nc_Synchronize_Attributes(FIELD *field)
1249 T((T_CALLED("_nc_Synchronize_Attributes(%p)"), (void *)field));
1252 returnCode(E_BAD_ARGUMENT);
1254 CHECKPOS(field->form);
1255 if (((form = field->form) != (FORM *)0)
1256 && Field_Really_Appears(field))
1258 if (form->current == field)
1260 Synchronize_Buffer(form);
1261 Set_Field_Window_Attributes(field, form->w);
1263 wmove(form->w, form->currow, form->curcol);
1265 if (Field_Has_Option(field, O_PUBLIC))
1267 if (Justification_Allowed(field))
1268 Undo_Justification(field, form->w);
1270 Buffer_To_Window(field, form->w);
1274 formwin = Get_Form_Window(form);
1275 copywin(form->w, formwin,
1277 field->frow, field->fcol,
1278 field->rows - 1, field->cols - 1, 0);
1280 Buffer_To_Window(field, form->w);
1281 SetStatus(field, _NEWTOP); /* fake refresh to paint all */
1282 _nc_Refresh_Current_Field(form);
1287 res = Display_Field(field);
1294 /*---------------------------------------------------------------------------
1295 | Facility : libnform
1296 | Function : int _nc_Synchronize_Options(FIELD * field,
1297 | Field_Options newopts)
1299 | Description : If a fields options have changed, this routine is
1300 | called to propagate these changes to the screen and
1301 | to really change the behavior of the field.
1303 | Return Values : E_OK - success
1304 | E_BAD_ARGUMENT - invalid field pointer
1305 | E_CURRENT - field is the current one
1306 | E_SYSTEM_ERROR - some severe basic error
1307 +--------------------------------------------------------------------------*/
1309 _nc_Synchronize_Options(FIELD *field, Field_Options newopts)
1311 Field_Options oldopts;
1312 Field_Options changed_opts;
1316 T((T_CALLED("_nc_Synchronize_Options(%p,%#x)"), (void *)field, newopts));
1319 returnCode(E_BAD_ARGUMENT);
1321 oldopts = field->opts;
1322 changed_opts = oldopts ^ newopts;
1323 field->opts = newopts;
1328 if (form->status & _POSTED)
1330 if (form->current == field)
1332 field->opts = oldopts;
1333 returnCode(E_CURRENT);
1335 if (form->curpage == field->page)
1337 if ((unsigned)changed_opts & O_VISIBLE)
1339 if ((unsigned)newopts & O_VISIBLE)
1340 res = Display_Field(field);
1342 res = Erase_Field(field);
1346 if (((unsigned)changed_opts & O_PUBLIC) &&
1347 ((unsigned)newopts & O_VISIBLE))
1348 res = Display_Field(field);
1354 if ((unsigned)changed_opts & O_STATIC)
1356 bool single_line_field = Single_Line_Field(field);
1359 if ((unsigned)newopts & O_STATIC)
1361 /* the field becomes now static */
1362 ClrStatus(field, _MAY_GROW);
1363 /* if actually we have no hidden columns, justification may
1365 if (single_line_field &&
1366 (field->cols == field->dcols) &&
1367 (field->just != NO_JUSTIFICATION) &&
1368 Field_Really_Appears(field))
1370 res2 = Display_Field(field);
1375 /* field is no longer static */
1376 if ((field->maxgrow == 0) ||
1377 (single_line_field && (field->dcols < field->maxgrow)) ||
1378 (!single_line_field && (field->drows < field->maxgrow)))
1380 SetStatus(field, _MAY_GROW);
1381 /* a field with justification now changes its behavior,
1382 so we must redisplay it */
1383 if (single_line_field &&
1384 (field->just != NO_JUSTIFICATION) &&
1385 Field_Really_Appears(field))
1387 res2 = Display_Field(field);
1398 /*---------------------------------------------------------------------------
1399 | Facility : libnform
1400 | Function : int _nc_Set_Current_Field(FORM * form,
1403 | Description : Make the newfield the new current field.
1405 | Return Values : E_OK - success
1406 | E_BAD_ARGUMENT - invalid form or field pointer
1407 | E_SYSTEM_ERROR - some severe basic error
1408 | E_NOT_CONNECTED - no fields are connected to the form
1409 +--------------------------------------------------------------------------*/
1411 _nc_Set_Current_Field(FORM *form, FIELD *newfield)
1416 T((T_CALLED("_nc_Set_Current_Field(%p,%p)"), (void *)form, (void *)newfield));
1418 if (!form || !newfield || !form->current || (newfield->form != form))
1419 returnCode(E_BAD_ARGUMENT);
1421 if ((form->status & _IN_DRIVER))
1422 returnCode(E_BAD_STATE);
1425 returnCode(E_NOT_CONNECTED);
1427 field = form->current;
1429 if ((field != newfield) ||
1430 !(form->status & _POSTED))
1433 (Field_Has_Option(field, O_VISIBLE)) &&
1434 (field->form->curpage == field->page))
1436 _nc_Refresh_Current_Field(form);
1437 if (Field_Has_Option(field, O_PUBLIC))
1439 if (field->drows > field->rows)
1441 if (form->toprow == 0)
1442 ClrStatus(field, _NEWTOP);
1444 SetStatus(field, _NEWTOP);
1448 if (Justification_Allowed(field))
1450 Window_To_Buffer(form, field);
1452 Perform_Justification(field, form->w);
1453 if (Field_Has_Option(field, O_DYNAMIC_JUSTIFY) &&
1454 (form->w->_parent == 0))
1457 Get_Form_Window(form),
1463 field->cols + field->fcol - 1,
1465 wsyncup(Get_Form_Window(form));
1475 form->w = (WINDOW *)0;
1480 if (Has_Invisible_Parts(field))
1481 new_window = newpad(field->drows, field->dcols);
1483 new_window = derwin(Get_Form_Window(form),
1484 field->rows, field->cols, field->frow, field->fcol);
1487 returnCode(E_SYSTEM_ERROR);
1489 form->current = field;
1493 form->w = new_window;
1495 ClrStatus(form, _WINDOW_MODIFIED);
1496 Set_Field_Window_Attributes(field, form->w);
1498 if (Has_Invisible_Parts(field))
1501 Buffer_To_Window(field, form->w);
1505 if (Justification_Allowed(field))
1508 Undo_Justification(field, form->w);
1513 untouchwin(form->w);
1516 form->currow = form->curcol = form->toprow = form->begincol = 0;
1520 /*----------------------------------------------------------------------------
1521 Intra-Field Navigation routines
1522 --------------------------------------------------------------------------*/
1524 /*---------------------------------------------------------------------------
1525 | Facility : libnform
1526 | Function : static int IFN_Next_Character(FORM * form)
1528 | Description : Move to the next character in the field. In a multi-line
1529 | field this wraps at the end of the line.
1531 | Return Values : E_OK - success
1532 | E_REQUEST_DENIED - at the rightmost position
1533 +--------------------------------------------------------------------------*/
1535 IFN_Next_Character(FORM *form)
1537 FIELD *field = form->current;
1538 int step = myWCWIDTH(form->w, form->currow, form->curcol);
1540 T((T_CALLED("IFN_Next_Character(%p)"), (void *)form));
1541 if ((form->curcol += step) == field->dcols)
1543 if ((++(form->currow)) == field->drows)
1545 #if GROW_IF_NAVIGATE
1546 if (!Single_Line_Field(field) && Field_Grown(field, 1))
1553 #if GROW_IF_NAVIGATE
1554 if (Single_Line_Field(field) && Field_Grown(field, 1))
1557 form->curcol -= step;
1558 returnCode(E_REQUEST_DENIED);
1565 /*---------------------------------------------------------------------------
1566 | Facility : libnform
1567 | Function : static int IFN_Previous_Character(FORM * form)
1569 | Description : Move to the previous character in the field. In a
1570 | multi-line field this wraps and the beginning of the
1573 | Return Values : E_OK - success
1574 | E_REQUEST_DENIED - at the leftmost position
1575 +--------------------------------------------------------------------------*/
1577 IFN_Previous_Character(FORM *form)
1579 int amount = myWCWIDTH(form->w, form->currow, form->curcol - 1);
1580 int oldcol = form->curcol;
1582 T((T_CALLED("IFN_Previous_Character(%p)"), (void *)form));
1583 if ((form->curcol -= amount) < 0)
1585 if ((--(form->currow)) < 0)
1588 form->curcol = oldcol;
1589 returnCode(E_REQUEST_DENIED);
1591 form->curcol = form->current->dcols - 1;
1596 /*---------------------------------------------------------------------------
1597 | Facility : libnform
1598 | Function : static int IFN_Next_Line(FORM * form)
1600 | Description : Move to the beginning of the next line in the field
1602 | Return Values : E_OK - success
1603 | E_REQUEST_DENIED - at the last line
1604 +--------------------------------------------------------------------------*/
1606 IFN_Next_Line(FORM *form)
1608 FIELD *field = form->current;
1610 T((T_CALLED("IFN_Next_Line(%p)"), (void *)form));
1611 if ((++(form->currow)) == field->drows)
1613 #if GROW_IF_NAVIGATE
1614 if (!Single_Line_Field(field) && Field_Grown(field, 1))
1618 returnCode(E_REQUEST_DENIED);
1624 /*---------------------------------------------------------------------------
1625 | Facility : libnform
1626 | Function : static int IFN_Previous_Line(FORM * form)
1628 | Description : Move to the beginning of the previous line in the field
1630 | Return Values : E_OK - success
1631 | E_REQUEST_DENIED - at the first line
1632 +--------------------------------------------------------------------------*/
1634 IFN_Previous_Line(FORM *form)
1636 T((T_CALLED("IFN_Previous_Line(%p)"), (void *)form));
1637 if ((--(form->currow)) < 0)
1640 returnCode(E_REQUEST_DENIED);
1646 /*---------------------------------------------------------------------------
1647 | Facility : libnform
1648 | Function : static int IFN_Next_Word(FORM * form)
1650 | Description : Move to the beginning of the next word in the field.
1652 | Return Values : E_OK - success
1653 | E_REQUEST_DENIED - there is no next word
1654 +--------------------------------------------------------------------------*/
1656 IFN_Next_Word(FORM *form)
1658 FIELD *field = form->current;
1659 FIELD_CELL *bp = Address_Of_Current_Position_In_Buffer(form);
1663 T((T_CALLED("IFN_Next_Word(%p)"), (void *)form));
1665 /* We really need access to the data, so we have to synchronize */
1666 Synchronize_Buffer(form);
1668 /* Go to the first whitespace after the current position (including
1669 current position). This is then the starting point to look for the
1670 next non-blank data */
1671 s = Get_First_Whitespace_Character(bp, Buffer_Length(field) -
1672 (int)(bp - field->buf));
1674 /* Find the start of the next word */
1675 t = Get_Start_Of_Data(s, Buffer_Length(field) -
1676 (int)(s - field->buf));
1677 #if !FRIENDLY_PREV_NEXT_WORD
1679 returnCode(E_REQUEST_DENIED);
1683 Adjust_Cursor_Position(form, t);
1688 /*---------------------------------------------------------------------------
1689 | Facility : libnform
1690 | Function : static int IFN_Previous_Word(FORM * form)
1692 | Description : Move to the beginning of the previous word in the field.
1694 | Return Values : E_OK - success
1695 | E_REQUEST_DENIED - there is no previous word
1696 +--------------------------------------------------------------------------*/
1698 IFN_Previous_Word(FORM *form)
1700 FIELD *field = form->current;
1701 FIELD_CELL *bp = Address_Of_Current_Position_In_Buffer(form);
1706 T((T_CALLED("IFN_Previous_Word(%p)"), (void *)form));
1708 /* We really need access to the data, so we have to synchronize */
1709 Synchronize_Buffer(form);
1711 s = After_End_Of_Data(field->buf, (int)(bp - field->buf));
1712 /* s points now right after the last non-blank in the buffer before bp.
1713 If bp was in a word, s equals bp. In this case we must find the last
1714 whitespace in the buffer before bp and repeat the game to really find
1715 the previous word! */
1719 /* And next call now goes backward to look for the last whitespace
1720 before that, pointing right after this, so it points to the begin
1721 of the previous word.
1723 t = After_Last_Whitespace_Character(field->buf, (int)(s - field->buf));
1724 #if !FRIENDLY_PREV_NEXT_WORD
1726 returnCode(E_REQUEST_DENIED);
1730 /* and do it again, replacing bp by t */
1731 s = After_End_Of_Data(field->buf, (int)(t - field->buf));
1732 t = After_Last_Whitespace_Character(field->buf, (int)(s - field->buf));
1733 #if !FRIENDLY_PREV_NEXT_WORD
1735 returnCode(E_REQUEST_DENIED);
1738 Adjust_Cursor_Position(form, t);
1742 /*---------------------------------------------------------------------------
1743 | Facility : libnform
1744 | Function : static int IFN_Beginning_Of_Field(FORM * form)
1746 | Description : Place the cursor at the first non-pad character in
1749 | Return Values : E_OK - success
1750 +--------------------------------------------------------------------------*/
1752 IFN_Beginning_Of_Field(FORM *form)
1754 FIELD *field = form->current;
1756 T((T_CALLED("IFN_Beginning_Of_Field(%p)"), (void *)form));
1757 Synchronize_Buffer(form);
1758 Adjust_Cursor_Position(form,
1759 Get_Start_Of_Data(field->buf, Buffer_Length(field)));
1763 /*---------------------------------------------------------------------------
1764 | Facility : libnform
1765 | Function : static int IFN_End_Of_Field(FORM * form)
1767 | Description : Place the cursor after the last non-pad character in
1768 | the field. If the field occupies the last position in
1769 | the buffer, the cursor is positioned on the last
1772 | Return Values : E_OK - success
1773 +--------------------------------------------------------------------------*/
1775 IFN_End_Of_Field(FORM *form)
1777 FIELD *field = form->current;
1780 T((T_CALLED("IFN_End_Of_Field(%p)"), (void *)form));
1781 Synchronize_Buffer(form);
1782 pos = After_End_Of_Data(field->buf, Buffer_Length(field));
1783 if (pos == (field->buf + Buffer_Length(field)))
1785 Adjust_Cursor_Position(form, pos);
1789 /*---------------------------------------------------------------------------
1790 | Facility : libnform
1791 | Function : static int IFN_Beginning_Of_Line(FORM * form)
1793 | Description : Place the cursor on the first non-pad character in
1794 | the current line of the field.
1796 | Return Values : E_OK - success
1797 +--------------------------------------------------------------------------*/
1799 IFN_Beginning_Of_Line(FORM *form)
1801 FIELD *field = form->current;
1803 T((T_CALLED("IFN_Beginning_Of_Line(%p)"), (void *)form));
1804 Synchronize_Buffer(form);
1805 Adjust_Cursor_Position(form,
1806 Get_Start_Of_Data(Address_Of_Current_Row_In_Buffer(form),
1811 /*---------------------------------------------------------------------------
1812 | Facility : libnform
1813 | Function : static int IFN_End_Of_Line(FORM * form)
1815 | Description : Place the cursor after the last non-pad character in the
1816 | current line of the field. If the field occupies the
1817 | last column in the line, the cursor is positioned on the
1818 | last character of the line.
1820 | Return Values : E_OK - success
1821 +--------------------------------------------------------------------------*/
1823 IFN_End_Of_Line(FORM *form)
1825 FIELD *field = form->current;
1829 T((T_CALLED("IFN_End_Of_Line(%p)"), (void *)form));
1830 Synchronize_Buffer(form);
1831 bp = Address_Of_Current_Row_In_Buffer(form);
1832 pos = After_End_Of_Data(bp, field->dcols);
1833 if (pos == (bp + field->dcols))
1835 Adjust_Cursor_Position(form, pos);
1839 /*---------------------------------------------------------------------------
1840 | Facility : libnform
1841 | Function : static int IFN_Left_Character(FORM * form)
1843 | Description : Move one character to the left in the current line.
1844 | This doesn't cycle.
1846 | Return Values : E_OK - success
1847 | E_REQUEST_DENIED - already in first column
1848 +--------------------------------------------------------------------------*/
1850 IFN_Left_Character(FORM *form)
1852 int amount = myWCWIDTH(form->w, form->currow, form->curcol - 1);
1853 int oldcol = form->curcol;
1855 T((T_CALLED("IFN_Left_Character(%p)"), (void *)form));
1856 if ((form->curcol -= amount) < 0)
1858 form->curcol = oldcol;
1859 returnCode(E_REQUEST_DENIED);
1864 /*---------------------------------------------------------------------------
1865 | Facility : libnform
1866 | Function : static int IFN_Right_Character(FORM * form)
1868 | Description : Move one character to the right in the current line.
1869 | This doesn't cycle.
1871 | Return Values : E_OK - success
1872 | E_REQUEST_DENIED - already in last column
1873 +--------------------------------------------------------------------------*/
1875 IFN_Right_Character(FORM *form)
1877 int amount = myWCWIDTH(form->w, form->currow, form->curcol);
1878 int oldcol = form->curcol;
1880 T((T_CALLED("IFN_Right_Character(%p)"), (void *)form));
1881 if ((form->curcol += amount) >= form->current->dcols)
1883 #if GROW_IF_NAVIGATE
1884 FIELD *field = form->current;
1886 if (Single_Line_Field(field) && Field_Grown(field, 1))
1889 form->curcol = oldcol;
1890 returnCode(E_REQUEST_DENIED);
1895 /*---------------------------------------------------------------------------
1896 | Facility : libnform
1897 | Function : static int IFN_Up_Character(FORM * form)
1899 | Description : Move one line up. This doesn't cycle through the lines
1902 | Return Values : E_OK - success
1903 | E_REQUEST_DENIED - already in last column
1904 +--------------------------------------------------------------------------*/
1906 IFN_Up_Character(FORM *form)
1908 T((T_CALLED("IFN_Up_Character(%p)"), (void *)form));
1909 if ((--(form->currow)) < 0)
1912 returnCode(E_REQUEST_DENIED);
1917 /*---------------------------------------------------------------------------
1918 | Facility : libnform
1919 | Function : static int IFN_Down_Character(FORM * form)
1921 | Description : Move one line down. This doesn't cycle through the
1922 | lines of the field.
1924 | Return Values : E_OK - success
1925 | E_REQUEST_DENIED - already in last column
1926 +--------------------------------------------------------------------------*/
1928 IFN_Down_Character(FORM *form)
1930 FIELD *field = form->current;
1932 T((T_CALLED("IFN_Down_Character(%p)"), (void *)form));
1933 if ((++(form->currow)) == field->drows)
1935 #if GROW_IF_NAVIGATE
1936 if (!Single_Line_Field(field) && Field_Grown(field, 1))
1940 returnCode(E_REQUEST_DENIED);
1944 /*----------------------------------------------------------------------------
1945 END of Intra-Field Navigation routines
1946 --------------------------------------------------------------------------*/
1948 /*----------------------------------------------------------------------------
1949 Vertical scrolling helper routines
1950 --------------------------------------------------------------------------*/
1952 /*---------------------------------------------------------------------------
1953 | Facility : libnform
1954 | Function : static int VSC_Generic(FORM *form, int nlines)
1956 | Description : Scroll multi-line field forward (nlines>0) or
1957 | backward (nlines<0) this many lines.
1959 | Return Values : E_OK - success
1960 | E_REQUEST_DENIED - can't scroll
1961 +--------------------------------------------------------------------------*/
1963 VSC_Generic(FORM *form, int nlines)
1965 FIELD *field = form->current;
1966 int res = E_REQUEST_DENIED;
1967 int rows_to_go = (nlines > 0 ? nlines : -nlines);
1971 if ((rows_to_go + form->toprow) > (field->drows - field->rows))
1972 rows_to_go = (field->drows - field->rows - form->toprow);
1976 form->currow += rows_to_go;
1977 form->toprow += rows_to_go;
1983 if (rows_to_go > form->toprow)
1984 rows_to_go = form->toprow;
1988 form->currow -= rows_to_go;
1989 form->toprow -= rows_to_go;
1995 /*----------------------------------------------------------------------------
1996 End of Vertical scrolling helper routines
1997 --------------------------------------------------------------------------*/
1999 /*----------------------------------------------------------------------------
2000 Vertical scrolling routines
2001 --------------------------------------------------------------------------*/
2003 /*---------------------------------------------------------------------------
2004 | Facility : libnform
2005 | Function : static int Vertical_Scrolling(
2006 | int (* const fct) (FORM *),
2009 | Description : Performs the generic vertical scrolling routines.
2010 | This has to check for a multi-line field and to set
2011 | the _NEWTOP flag if scrolling really occurred.
2013 | Return Values : Propagated error code from low-level driver calls
2014 +--------------------------------------------------------------------------*/
2016 Vertical_Scrolling(int (*const fct) (FORM *), FORM *form)
2018 int res = E_REQUEST_DENIED;
2020 if (!Single_Line_Field(form->current))
2024 SetStatus(form->current, _NEWTOP);
2029 /*---------------------------------------------------------------------------
2030 | Facility : libnform
2031 | Function : static int VSC_Scroll_Line_Forward(FORM * form)
2033 | Description : Scroll multi-line field forward a line
2035 | Return Values : E_OK - success
2036 | E_REQUEST_DENIED - no data ahead
2037 +--------------------------------------------------------------------------*/
2039 VSC_Scroll_Line_Forward(FORM *form)
2041 T((T_CALLED("VSC_Scroll_Line_Forward(%p)"), (void *)form));
2042 returnCode(VSC_Generic(form, 1));
2045 /*---------------------------------------------------------------------------
2046 | Facility : libnform
2047 | Function : static int VSC_Scroll_Line_Backward(FORM * form)
2049 | Description : Scroll multi-line field backward a line
2051 | Return Values : E_OK - success
2052 | E_REQUEST_DENIED - no data behind
2053 +--------------------------------------------------------------------------*/
2055 VSC_Scroll_Line_Backward(FORM *form)
2057 T((T_CALLED("VSC_Scroll_Line_Backward(%p)"), (void *)form));
2058 returnCode(VSC_Generic(form, -1));
2061 /*---------------------------------------------------------------------------
2062 | Facility : libnform
2063 | Function : static int VSC_Scroll_Page_Forward(FORM * form)
2065 | Description : Scroll a multi-line field forward a page
2067 | Return Values : E_OK - success
2068 | E_REQUEST_DENIED - no data ahead
2069 +--------------------------------------------------------------------------*/
2071 VSC_Scroll_Page_Forward(FORM *form)
2073 T((T_CALLED("VSC_Scroll_Page_Forward(%p)"), (void *)form));
2074 returnCode(VSC_Generic(form, form->current->rows));
2077 /*---------------------------------------------------------------------------
2078 | Facility : libnform
2079 | Function : static int VSC_Scroll_Half_Page_Forward(FORM * form)
2081 | Description : Scroll a multi-line field forward half a page
2083 | Return Values : E_OK - success
2084 | E_REQUEST_DENIED - no data ahead
2085 +--------------------------------------------------------------------------*/
2087 VSC_Scroll_Half_Page_Forward(FORM *form)
2089 T((T_CALLED("VSC_Scroll_Half_Page_Forward(%p)"), (void *)form));
2090 returnCode(VSC_Generic(form, (form->current->rows + 1) / 2));
2093 /*---------------------------------------------------------------------------
2094 | Facility : libnform
2095 | Function : static int VSC_Scroll_Page_Backward(FORM * form)
2097 | Description : Scroll a multi-line field backward a page
2099 | Return Values : E_OK - success
2100 | E_REQUEST_DENIED - no data behind
2101 +--------------------------------------------------------------------------*/
2103 VSC_Scroll_Page_Backward(FORM *form)
2105 T((T_CALLED("VSC_Scroll_Page_Backward(%p)"), (void *)form));
2106 returnCode(VSC_Generic(form, -(form->current->rows)));
2109 /*---------------------------------------------------------------------------
2110 | Facility : libnform
2111 | Function : static int VSC_Scroll_Half_Page_Backward(FORM * form)
2113 | Description : Scroll a multi-line field backward half a page
2115 | Return Values : E_OK - success
2116 | E_REQUEST_DENIED - no data behind
2117 +--------------------------------------------------------------------------*/
2119 VSC_Scroll_Half_Page_Backward(FORM *form)
2121 T((T_CALLED("VSC_Scroll_Half_Page_Backward(%p)"), (void *)form));
2122 returnCode(VSC_Generic(form, -((form->current->rows + 1) / 2)));
2124 /*----------------------------------------------------------------------------
2125 End of Vertical scrolling routines
2126 --------------------------------------------------------------------------*/
2128 /*----------------------------------------------------------------------------
2129 Horizontal scrolling helper routines
2130 --------------------------------------------------------------------------*/
2132 /*---------------------------------------------------------------------------
2133 | Facility : libnform
2134 | Function : static int HSC_Generic(FORM *form, int ncolumns)
2136 | Description : Scroll single-line field forward (ncolumns>0) or
2137 | backward (ncolumns<0) this many columns.
2139 | Return Values : E_OK - success
2140 | E_REQUEST_DENIED - can't scroll
2141 +--------------------------------------------------------------------------*/
2143 HSC_Generic(FORM *form, int ncolumns)
2145 FIELD *field = form->current;
2146 int res = E_REQUEST_DENIED;
2147 int cols_to_go = (ncolumns > 0 ? ncolumns : -ncolumns);
2151 if ((cols_to_go + form->begincol) > (field->dcols - field->cols))
2152 cols_to_go = field->dcols - field->cols - form->begincol;
2156 form->curcol += cols_to_go;
2157 form->begincol += cols_to_go;
2163 if (cols_to_go > form->begincol)
2164 cols_to_go = form->begincol;
2168 form->curcol -= cols_to_go;
2169 form->begincol -= cols_to_go;
2175 /*----------------------------------------------------------------------------
2176 End of Horizontal scrolling helper routines
2177 --------------------------------------------------------------------------*/
2179 /*----------------------------------------------------------------------------
2180 Horizontal scrolling routines
2181 --------------------------------------------------------------------------*/
2183 /*---------------------------------------------------------------------------
2184 | Facility : libnform
2185 | Function : static int Horizontal_Scrolling(
2186 | int (* const fct) (FORM *),
2189 | Description : Performs the generic horizontal scrolling routines.
2190 | This has to check for a single-line field.
2192 | Return Values : Propagated error code from low-level driver calls
2193 +--------------------------------------------------------------------------*/
2195 Horizontal_Scrolling(int (*const fct) (FORM *), FORM *form)
2197 if (Single_Line_Field(form->current))
2200 return (E_REQUEST_DENIED);
2203 /*---------------------------------------------------------------------------
2204 | Facility : libnform
2205 | Function : static int HSC_Scroll_Char_Forward(FORM * form)
2207 | Description : Scroll single-line field forward a character
2209 | Return Values : E_OK - success
2210 | E_REQUEST_DENIED - no data ahead
2211 +--------------------------------------------------------------------------*/
2213 HSC_Scroll_Char_Forward(FORM *form)
2215 T((T_CALLED("HSC_Scroll_Char_Forward(%p)"), (void *)form));
2216 returnCode(HSC_Generic(form, 1));
2219 /*---------------------------------------------------------------------------
2220 | Facility : libnform
2221 | Function : static int HSC_Scroll_Char_Backward(FORM * form)
2223 | Description : Scroll single-line field backward a character
2225 | Return Values : E_OK - success
2226 | E_REQUEST_DENIED - no data behind
2227 +--------------------------------------------------------------------------*/
2229 HSC_Scroll_Char_Backward(FORM *form)
2231 T((T_CALLED("HSC_Scroll_Char_Backward(%p)"), (void *)form));
2232 returnCode(HSC_Generic(form, -1));
2235 /*---------------------------------------------------------------------------
2236 | Facility : libnform
2237 | Function : static int HSC_Horizontal_Line_Forward(FORM* form)
2239 | Description : Scroll single-line field forward a line
2241 | Return Values : E_OK - success
2242 | E_REQUEST_DENIED - no data ahead
2243 +--------------------------------------------------------------------------*/
2245 HSC_Horizontal_Line_Forward(FORM *form)
2247 T((T_CALLED("HSC_Horizontal_Line_Forward(%p)"), (void *)form));
2248 returnCode(HSC_Generic(form, form->current->cols));
2251 /*---------------------------------------------------------------------------
2252 | Facility : libnform
2253 | Function : static int HSC_Horizontal_Half_Line_Forward(FORM* form)
2255 | Description : Scroll single-line field forward half a line
2257 | Return Values : E_OK - success
2258 | E_REQUEST_DENIED - no data ahead
2259 +--------------------------------------------------------------------------*/
2261 HSC_Horizontal_Half_Line_Forward(FORM *form)
2263 T((T_CALLED("HSC_Horizontal_Half_Line_Forward(%p)"), (void *)form));
2264 returnCode(HSC_Generic(form, (form->current->cols + 1) / 2));
2267 /*---------------------------------------------------------------------------
2268 | Facility : libnform
2269 | Function : static int HSC_Horizontal_Line_Backward(FORM* form)
2271 | Description : Scroll single-line field backward a line
2273 | Return Values : E_OK - success
2274 | E_REQUEST_DENIED - no data behind
2275 +--------------------------------------------------------------------------*/
2277 HSC_Horizontal_Line_Backward(FORM *form)
2279 T((T_CALLED("HSC_Horizontal_Line_Backward(%p)"), (void *)form));
2280 returnCode(HSC_Generic(form, -(form->current->cols)));
2283 /*---------------------------------------------------------------------------
2284 | Facility : libnform
2285 | Function : static int HSC_Horizontal_Half_Line_Backward(FORM* form)
2287 | Description : Scroll single-line field backward half a line
2289 | Return Values : E_OK - success
2290 | E_REQUEST_DENIED - no data behind
2291 +--------------------------------------------------------------------------*/
2293 HSC_Horizontal_Half_Line_Backward(FORM *form)
2295 T((T_CALLED("HSC_Horizontal_Half_Line_Backward(%p)"), (void *)form));
2296 returnCode(HSC_Generic(form, -((form->current->cols + 1) / 2)));
2299 /*----------------------------------------------------------------------------
2300 End of Horizontal scrolling routines
2301 --------------------------------------------------------------------------*/
2303 /*----------------------------------------------------------------------------
2304 Helper routines for Field Editing
2305 --------------------------------------------------------------------------*/
2307 /*---------------------------------------------------------------------------
2308 | Facility : libnform
2309 | Function : static bool Is_There_Room_For_A_Line(FORM * form)
2311 | Description : Check whether or not there is enough room in the
2312 | buffer to enter a whole line.
2314 | Return Values : TRUE - there is enough space
2315 | FALSE - there is not enough space
2316 +--------------------------------------------------------------------------*/
2317 NCURSES_INLINE static bool
2318 Is_There_Room_For_A_Line(FORM *form)
2320 FIELD *field = form->current;
2321 FIELD_CELL *begin_of_last_line, *s;
2323 Synchronize_Buffer(form);
2324 begin_of_last_line = Address_Of_Row_In_Buffer(field, (field->drows - 1));
2325 s = After_End_Of_Data(begin_of_last_line, field->dcols);
2326 return ((s == begin_of_last_line) ? TRUE : FALSE);
2329 /*---------------------------------------------------------------------------
2330 | Facility : libnform
2331 | Function : static bool Is_There_Room_For_A_Char_In_Line(FORM * form)
2333 | Description : Checks whether or not there is room for a new character
2334 | in the current line.
2336 | Return Values : TRUE - there is room
2337 | FALSE - there is not enough room (line full)
2338 +--------------------------------------------------------------------------*/
2339 NCURSES_INLINE static bool
2340 Is_There_Room_For_A_Char_In_Line(FORM *form)
2342 int last_char_in_line;
2344 wmove(form->w, form->currow, form->current->dcols - 1);
2345 last_char_in_line = (int)(winch(form->w) & A_CHARTEXT);
2346 wmove(form->w, form->currow, form->curcol);
2347 return (((last_char_in_line == form->current->pad) ||
2348 is_blank(last_char_in_line)) ? TRUE : FALSE);
2351 #define There_Is_No_Room_For_A_Char_In_Line(f) \
2352 !Is_There_Room_For_A_Char_In_Line(f)
2354 /*---------------------------------------------------------------------------
2355 | Facility : libnform
2356 | Function : static int Insert_String(
2362 | Description : Insert the 'len' characters beginning at pointer 'txt'
2363 | into the 'row' of the 'form'. The insertion occurs
2364 | on the beginning of the row, all other characters are
2365 | moved to the right. After the text a pad character will
2366 | be inserted to separate the text from the rest. If
2367 | necessary the insertion moves characters on the next
2368 | line to make place for the requested insertion string.
2370 | Return Values : E_OK - success
2371 | E_REQUEST_DENIED -
2372 | E_SYSTEM_ERROR - system error
2373 +--------------------------------------------------------------------------*/
2375 Insert_String(FORM *form, int row, FIELD_CELL *txt, int len)
2377 FIELD *field = form->current;
2378 FIELD_CELL *bp = Address_Of_Row_In_Buffer(field, row);
2379 int datalen = (int)(After_End_Of_Data(bp, field->dcols) - bp);
2380 int freelen = field->dcols - datalen;
2381 int requiredlen = len + 1;
2383 int result = E_REQUEST_DENIED;
2385 if (freelen >= requiredlen)
2387 wmove(form->w, row, 0);
2388 myINSNSTR(form->w, txt, len);
2389 wmove(form->w, row, len);
2390 myINSNSTR(form->w, &myBLANK, 1);
2395 /* we have to move characters on the next line. If we are on the
2396 last line this may work, if the field is growable */
2397 if ((row == (field->drows - 1)) && Growable(field))
2399 if (!Field_Grown(field, 1))
2400 return (E_SYSTEM_ERROR);
2401 /* !!!Side-Effect : might be changed due to growth!!! */
2402 bp = Address_Of_Row_In_Buffer(field, row);
2405 if (row < (field->drows - 1))
2408 After_Last_Whitespace_Character(bp,
2409 (int)(Get_Start_Of_Data(bp
2414 /* split points now to the first character of the portion of the
2415 line that must be moved to the next line */
2416 datalen = (int)(split - bp); /* + freelen has to stay on this line */
2417 freelen = field->dcols - (datalen + freelen); /* for the next line */
2419 if ((result = Insert_String(form, row + 1, split, freelen)) == E_OK)
2421 wmove(form->w, row, datalen);
2423 wmove(form->w, row, 0);
2424 myINSNSTR(form->w, txt, len);
2425 wmove(form->w, row, len);
2426 myINSNSTR(form->w, &myBLANK, 1);
2434 /*---------------------------------------------------------------------------
2435 | Facility : libnform
2436 | Function : static int Wrapping_Not_Necessary_Or_Wrapping_Ok(
2439 | Description : If a character has been entered into a field, it may
2440 | be that wrapping has to occur. This routine checks
2441 | whether or not wrapping is required and if so, performs
2444 | Return Values : E_OK - no wrapping required or wrapping
2446 | E_REQUEST_DENIED -
2447 | E_SYSTEM_ERROR - some system error
2448 +--------------------------------------------------------------------------*/
2450 Wrapping_Not_Necessary_Or_Wrapping_Ok(FORM *form)
2452 FIELD *field = form->current;
2453 int result = E_REQUEST_DENIED;
2454 bool Last_Row = ((field->drows - 1) == form->currow);
2456 if ((Field_Has_Option(field, O_WRAP)) && /* wrapping wanted */
2457 (!Single_Line_Field(field)) && /* must be multi-line */
2458 (There_Is_No_Room_For_A_Char_In_Line(form)) && /* line is full */
2459 (!Last_Row || Growable(field))) /* there are more lines */
2463 int chars_to_be_wrapped;
2464 int chars_to_remain_on_line;
2468 /* the above logic already ensures, that in this case the field
2470 if (!Field_Grown(field, 1))
2471 return E_SYSTEM_ERROR;
2473 bp = Address_Of_Current_Row_In_Buffer(form);
2474 Window_To_Buffer(form, field);
2475 split = After_Last_Whitespace_Character(bp, field->dcols);
2476 /* split points to the first character of the sequence to be brought
2478 chars_to_remain_on_line = (int)(split - bp);
2479 chars_to_be_wrapped = field->dcols - chars_to_remain_on_line;
2480 if (chars_to_remain_on_line > 0)
2482 if ((result = Insert_String(form, form->currow + 1, split,
2483 chars_to_be_wrapped)) == E_OK)
2485 wmove(form->w, form->currow, chars_to_remain_on_line);
2487 if (form->curcol >= chars_to_remain_on_line)
2490 form->curcol -= chars_to_remain_on_line;
2500 Window_To_Buffer(form, field);
2501 result = E_REQUEST_DENIED;
2505 result = E_OK; /* wrapping was not necessary */
2509 /*----------------------------------------------------------------------------
2510 Field Editing routines
2511 --------------------------------------------------------------------------*/
2513 /*---------------------------------------------------------------------------
2514 | Facility : libnform
2515 | Function : static int Field_Editing(
2516 | int (* const fct) (FORM *),
2519 | Description : Generic routine for field editing requests. The driver
2520 | routines are only called for editable fields, the
2521 | _WINDOW_MODIFIED flag is set if editing occurred.
2522 | This is somewhat special due to the overload semantics
2523 | of the NEW_LINE and DEL_PREV requests.
2525 | Return Values : Error code from low level drivers.
2526 +--------------------------------------------------------------------------*/
2528 Field_Editing(int (*const fct) (FORM *), FORM *form)
2530 int res = E_REQUEST_DENIED;
2532 /* We have to deal here with the specific case of the overloaded
2533 behavior of New_Line and Delete_Previous requests.
2534 They may end up in navigational requests if we are on the first
2535 character in a field. But navigation is also allowed on non-
2538 if ((fct == FE_Delete_Previous) &&
2539 ((unsigned)form->opts & O_BS_OVERLOAD) &&
2540 First_Position_In_Current_Field(form))
2542 res = Inter_Field_Navigation(FN_Previous_Field, form);
2546 if (fct == FE_New_Line)
2548 if (((unsigned)form->opts & O_NL_OVERLOAD) &&
2549 First_Position_In_Current_Field(form))
2551 res = Inter_Field_Navigation(FN_Next_Field, form);
2554 /* FE_New_Line deals itself with the _WINDOW_MODIFIED flag */
2559 /* From now on, everything must be editable */
2560 if ((unsigned)form->current->opts & O_EDIT)
2564 SetStatus(form, _WINDOW_MODIFIED);
2571 /*---------------------------------------------------------------------------
2572 | Facility : libnform
2573 | Function : static int FE_New_Line(FORM * form)
2575 | Description : Perform a new line request. This is rather complex
2576 | compared to other routines in this code due to the
2577 | rather difficult to understand description in the
2580 | Return Values : E_OK - success
2581 | E_REQUEST_DENIED - new line not allowed
2582 | E_SYSTEM_ERROR - system error
2583 +--------------------------------------------------------------------------*/
2585 FE_New_Line(FORM *form)
2587 FIELD *field = form->current;
2589 bool Last_Row = ((field->drows - 1) == form->currow);
2591 T((T_CALLED("FE_New_Line(%p)"), (void *)form));
2592 if (form->status & _OVLMODE)
2595 (!(Growable(field) && !Single_Line_Field(field))))
2597 if (!((unsigned)form->opts & O_NL_OVERLOAD))
2598 returnCode(E_REQUEST_DENIED);
2599 wmove(form->w, form->currow, form->curcol);
2601 /* we have to set this here, although it is also
2602 handled in the generic routine. The reason is,
2603 that FN_Next_Field may fail, but the form is
2604 definitively changed */
2605 SetStatus(form, _WINDOW_MODIFIED);
2606 returnCode(Inter_Field_Navigation(FN_Next_Field, form));
2610 if (Last_Row && !Field_Grown(field, 1))
2612 /* N.B.: due to the logic in the 'if', LastRow==TRUE
2613 means here that the field is growable and not
2614 a single-line field */
2615 returnCode(E_SYSTEM_ERROR);
2617 wmove(form->w, form->currow, form->curcol);
2621 SetStatus(form, _WINDOW_MODIFIED);
2629 !(Growable(field) && !Single_Line_Field(field)))
2631 if (!((unsigned)form->opts & O_NL_OVERLOAD))
2632 returnCode(E_REQUEST_DENIED);
2633 returnCode(Inter_Field_Navigation(FN_Next_Field, form));
2637 bool May_Do_It = !Last_Row && Is_There_Room_For_A_Line(form);
2639 if (!(May_Do_It || Growable(field)))
2640 returnCode(E_REQUEST_DENIED);
2641 if (!May_Do_It && !Field_Grown(field, 1))
2642 returnCode(E_SYSTEM_ERROR);
2644 bp = Address_Of_Current_Position_In_Buffer(form);
2645 t = After_End_Of_Data(bp, field->dcols - form->curcol);
2646 wmove(form->w, form->currow, form->curcol);
2650 wmove(form->w, form->currow, form->curcol);
2652 myADDNSTR(form->w, bp, (int)(t - bp));
2653 SetStatus(form, _WINDOW_MODIFIED);
2659 /*---------------------------------------------------------------------------
2660 | Facility : libnform
2661 | Function : static int FE_Insert_Character(FORM * form)
2663 | Description : Insert blank character at the cursor position
2665 | Return Values : E_OK
2667 +--------------------------------------------------------------------------*/
2669 FE_Insert_Character(FORM *form)
2671 FIELD *field = form->current;
2672 int result = E_REQUEST_DENIED;
2674 T((T_CALLED("FE_Insert_Character(%p)"), (void *)form));
2675 if (Check_Char(form, field, field->type, (int)C_BLANK,
2676 (TypeArgument *)(field->arg)))
2678 bool There_Is_Room = Is_There_Room_For_A_Char_In_Line(form);
2680 if (There_Is_Room ||
2681 ((Single_Line_Field(field) && Growable(field))))
2683 if (!There_Is_Room && !Field_Grown(field, 1))
2684 result = E_SYSTEM_ERROR;
2687 winsch(form->w, (chtype)C_BLANK);
2688 result = Wrapping_Not_Necessary_Or_Wrapping_Ok(form);
2695 /*---------------------------------------------------------------------------
2696 | Facility : libnform
2697 | Function : static int FE_Insert_Line(FORM * form)
2699 | Description : Insert a blank line at the cursor position
2701 | Return Values : E_OK - success
2702 | E_REQUEST_DENIED - line can not be inserted
2703 +--------------------------------------------------------------------------*/
2705 FE_Insert_Line(FORM *form)
2707 FIELD *field = form->current;
2708 int result = E_REQUEST_DENIED;
2710 T((T_CALLED("FE_Insert_Line(%p)"), (void *)form));
2711 if (Check_Char(form, field,
2712 field->type, (int)C_BLANK, (TypeArgument *)(field->arg)))
2714 bool Maybe_Done = (form->currow != (field->drows - 1)) &&
2715 Is_There_Room_For_A_Line(form);
2717 if (!Single_Line_Field(field) &&
2718 (Maybe_Done || Growable(field)))
2720 if (!Maybe_Done && !Field_Grown(field, 1))
2721 result = E_SYSTEM_ERROR;
2733 /*---------------------------------------------------------------------------
2734 | Facility : libnform
2735 | Function : static int FE_Delete_Character(FORM * form)
2737 | Description : Delete character at the cursor position
2739 | Return Values : E_OK - success
2740 +--------------------------------------------------------------------------*/
2742 FE_Delete_Character(FORM *form)
2744 T((T_CALLED("FE_Delete_Character(%p)"), (void *)form));
2749 /*---------------------------------------------------------------------------
2750 | Facility : libnform
2751 | Function : static int FE_Delete_Previous(FORM * form)
2753 | Description : Delete character before cursor. Again this is a rather
2754 | difficult piece compared to others due to the overloading
2755 | semantics of backspace.
2756 | N.B.: The case of overloaded BS on first field position
2757 | is already handled in the generic routine.
2759 | Return Values : E_OK - success
2760 | E_REQUEST_DENIED - Character can't be deleted
2761 +--------------------------------------------------------------------------*/
2763 FE_Delete_Previous(FORM *form)
2765 FIELD *field = form->current;
2767 T((T_CALLED("FE_Delete_Previous(%p)"), (void *)form));
2768 if (First_Position_In_Current_Field(form))
2769 returnCode(E_REQUEST_DENIED);
2771 if ((--(form->curcol)) < 0)
2773 FIELD_CELL *this_line, *prev_line, *prev_end, *this_end;
2774 int this_row = form->currow;
2777 if (form->status & _OVLMODE)
2778 returnCode(E_REQUEST_DENIED);
2780 prev_line = Address_Of_Row_In_Buffer(field, (form->currow - 1));
2781 this_line = Address_Of_Row_In_Buffer(field, (form->currow));
2782 Synchronize_Buffer(form);
2783 prev_end = After_End_Of_Data(prev_line, field->dcols);
2784 this_end = After_End_Of_Data(this_line, field->dcols);
2785 if ((int)(this_end - this_line) >
2786 (field->cols - (int)(prev_end - prev_line)))
2787 returnCode(E_REQUEST_DENIED);
2788 wmove(form->w, form->currow, form->curcol);
2790 Adjust_Cursor_Position(form, prev_end);
2792 * If we did not really move to the previous line, help the user a
2793 * little. It is however a little inconsistent. Normally, when
2794 * backspacing around the point where text wraps to a new line in a
2795 * multi-line form, we absorb one keystroke for the wrapping point. That
2796 * is consistent with SVr4 forms. However, SVr4 does not allow typing
2797 * into the last column of the field, and requires the user to enter a
2798 * newline to move to the next line. Therefore it can consistently eat
2799 * that keystroke. Since ncurses allows the last column, it wraps
2800 * automatically (given the proper options). But we cannot eat the
2801 * keystroke to back over the wrapping point, since that would put the
2802 * cursor past the end of the form field. In this case, just delete the
2803 * character at the end of the field.
2805 if (form->currow == this_row && this_row > 0)
2808 form->curcol = field->dcols - 1;
2813 wmove(form->w, form->currow, form->curcol);
2814 myADDNSTR(form->w, this_line, (int)(this_end - this_line));
2824 /*---------------------------------------------------------------------------
2825 | Facility : libnform
2826 | Function : static int FE_Delete_Line(FORM * form)
2828 | Description : Delete line at cursor position.
2830 | Return Values : E_OK - success
2831 +--------------------------------------------------------------------------*/
2833 FE_Delete_Line(FORM *form)
2835 T((T_CALLED("FE_Delete_Line(%p)"), (void *)form));
2841 /*---------------------------------------------------------------------------
2842 | Facility : libnform
2843 | Function : static int FE_Delete_Word(FORM * form)
2845 | Description : Delete word at cursor position
2847 | Return Values : E_OK - success
2848 | E_REQUEST_DENIED - failure
2849 +--------------------------------------------------------------------------*/
2851 FE_Delete_Word(FORM *form)
2853 FIELD *field = form->current;
2854 FIELD_CELL *bp = Address_Of_Current_Row_In_Buffer(form);
2855 FIELD_CELL *ep = bp + field->dcols;
2856 FIELD_CELL *cp = bp + form->curcol;
2859 T((T_CALLED("FE_Delete_Word(%p)"), (void *)form));
2860 Synchronize_Buffer(form);
2862 returnCode(E_REQUEST_DENIED); /* not in word */
2864 /* move cursor to begin of word and erase to end of screen-line */
2865 Adjust_Cursor_Position(form,
2866 After_Last_Whitespace_Character(bp, form->curcol));
2867 wmove(form->w, form->currow, form->curcol);
2870 /* skip over word in buffer */
2871 s = Get_First_Whitespace_Character(cp, (int)(ep - cp));
2872 /* to begin of next word */
2873 s = Get_Start_Of_Data(s, (int)(ep - s));
2874 if ((s != cp) && !ISBLANK(*s))
2876 /* copy remaining line to window */
2877 myADDNSTR(form->w, s, (int)(s - After_End_Of_Data(s, (int)(ep - s))));
2882 /*---------------------------------------------------------------------------
2883 | Facility : libnform
2884 | Function : static int FE_Clear_To_End_Of_Line(FORM * form)
2886 | Description : Clear to end of current line.
2888 | Return Values : E_OK - success
2889 +--------------------------------------------------------------------------*/
2891 FE_Clear_To_End_Of_Line(FORM *form)
2893 T((T_CALLED("FE_Clear_To_End_Of_Line(%p)"), (void *)form));
2894 wmove(form->w, form->currow, form->curcol);
2899 /*---------------------------------------------------------------------------
2900 | Facility : libnform
2901 | Function : static int FE_Clear_To_End_Of_Field(FORM * form)
2903 | Description : Clear to end of field.
2905 | Return Values : E_OK - success
2906 +--------------------------------------------------------------------------*/
2908 FE_Clear_To_End_Of_Field(FORM *form)
2910 T((T_CALLED("FE_Clear_To_End_Of_Field(%p)"), (void *)form));
2911 wmove(form->w, form->currow, form->curcol);
2916 /*---------------------------------------------------------------------------
2917 | Facility : libnform
2918 | Function : static int FE_Clear_Field(FORM * form)
2920 | Description : Clear entire field.
2922 | Return Values : E_OK - success
2923 +--------------------------------------------------------------------------*/
2925 FE_Clear_Field(FORM *form)
2927 T((T_CALLED("FE_Clear_Field(%p)"), (void *)form));
2928 form->currow = form->curcol = 0;
2932 /*----------------------------------------------------------------------------
2933 END of Field Editing routines
2934 --------------------------------------------------------------------------*/
2936 /*----------------------------------------------------------------------------
2938 --------------------------------------------------------------------------*/
2940 /*---------------------------------------------------------------------------
2941 | Facility : libnform
2942 | Function : static int EM_Overlay_Mode(FORM * form)
2944 | Description : Switch to overlay mode.
2946 | Return Values : E_OK - success
2947 +--------------------------------------------------------------------------*/
2949 EM_Overlay_Mode(FORM *form)
2951 T((T_CALLED("EM_Overlay_Mode(%p)"), (void *)form));
2952 SetStatus(form, _OVLMODE);
2956 /*---------------------------------------------------------------------------
2957 | Facility : libnform
2958 | Function : static int EM_Insert_Mode(FORM * form)
2960 | Description : Switch to insert mode
2962 | Return Values : E_OK - success
2963 +--------------------------------------------------------------------------*/
2965 EM_Insert_Mode(FORM *form)
2967 T((T_CALLED("EM_Insert_Mode(%p)"), (void *)form));
2968 ClrStatus(form, _OVLMODE);
2972 /*----------------------------------------------------------------------------
2973 END of Edit Mode routines
2974 --------------------------------------------------------------------------*/
2976 /*----------------------------------------------------------------------------
2977 Helper routines for Choice Requests
2978 --------------------------------------------------------------------------*/
2980 /*---------------------------------------------------------------------------
2981 | Facility : libnform
2982 | Function : static bool Next_Choice(FORM * form,
2985 | TypeArgument *argp)
2987 | Description : Get the next field choice. For linked types this is
2990 | Return Values : TRUE - next choice successfully retrieved
2991 | FALSE - couldn't retrieve next choice
2992 +--------------------------------------------------------------------------*/
2994 Next_Choice(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
2996 if (!typ || !(typ->status & _HAS_CHOICE))
2999 if (typ->status & _LINKED_TYPE)
3003 Next_Choice(form, typ->left, field, argp->left) ||
3004 Next_Choice(form, typ->right, field, argp->right));
3008 #if NCURSES_INTEROP_FUNCS
3009 assert(typ->enum_next.onext);
3010 if (typ->status & _GENERIC)
3011 return typ->enum_next.gnext(form, field, (void *)argp);
3013 return typ->enum_next.onext(field, (void *)argp);
3016 return typ->next(field, (void *)argp);
3021 /*---------------------------------------------------------------------------
3022 | Facility : libnform
3023 | Function : static bool Previous_Choice(FORM * form,
3026 | TypeArgument *argp)
3028 | Description : Get the previous field choice. For linked types this
3029 | is done recursively.
3031 | Return Values : TRUE - previous choice successfully retrieved
3032 | FALSE - couldn't retrieve previous choice
3033 +--------------------------------------------------------------------------*/
3035 Previous_Choice(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
3037 if (!typ || !(typ->status & _HAS_CHOICE))
3040 if (typ->status & _LINKED_TYPE)
3044 Previous_Choice(form, typ->left, field, argp->left) ||
3045 Previous_Choice(form, typ->right, field, argp->right));
3049 #if NCURSES_INTEROP_FUNCS
3050 assert(typ->enum_prev.oprev);
3051 if (typ->status & _GENERIC)
3052 return typ->enum_prev.gprev(form, field, (void *)argp);
3054 return typ->enum_prev.oprev(field, (void *)argp);
3057 return typ->prev(field, (void *)argp);
3061 /*----------------------------------------------------------------------------
3062 End of Helper routines for Choice Requests
3063 --------------------------------------------------------------------------*/
3065 /*----------------------------------------------------------------------------
3066 Routines for Choice Requests
3067 --------------------------------------------------------------------------*/
3069 /*---------------------------------------------------------------------------
3070 | Facility : libnform
3071 | Function : static int CR_Next_Choice(FORM * form)
3073 | Description : Get the next field choice.
3075 | Return Values : E_OK - success
3076 | E_REQUEST_DENIED - next choice couldn't be retrieved
3077 +--------------------------------------------------------------------------*/
3079 CR_Next_Choice(FORM *form)
3081 FIELD *field = form->current;
3083 T((T_CALLED("CR_Next_Choice(%p)"), (void *)form));
3084 Synchronize_Buffer(form);
3085 returnCode((Next_Choice(form, field->type, field, (TypeArgument *)(field->arg)))
3087 : E_REQUEST_DENIED);
3090 /*---------------------------------------------------------------------------
3091 | Facility : libnform
3092 | Function : static int CR_Previous_Choice(FORM * form)
3094 | Description : Get the previous field choice.
3096 | Return Values : E_OK - success
3097 | E_REQUEST_DENIED - prev. choice couldn't be retrieved
3098 +--------------------------------------------------------------------------*/
3100 CR_Previous_Choice(FORM *form)
3102 FIELD *field = form->current;
3104 T((T_CALLED("CR_Previous_Choice(%p)"), (void *)form));
3105 Synchronize_Buffer(form);
3106 returnCode((Previous_Choice(form, field->type, field, (TypeArgument *)(field->arg)))
3108 : E_REQUEST_DENIED);
3110 /*----------------------------------------------------------------------------
3111 End of Routines for Choice Requests
3112 --------------------------------------------------------------------------*/
3114 /*----------------------------------------------------------------------------
3115 Helper routines for Field Validations.
3116 --------------------------------------------------------------------------*/
3118 /*---------------------------------------------------------------------------
3119 | Facility : libnform
3120 | Function : static bool Check_Field(FORM* form,
3123 | TypeArgument * argp)
3125 | Description : Check the field according to its fieldtype and its
3126 | actual arguments. For linked fieldtypes this is done
3129 | Return Values : TRUE - field is valid
3130 | FALSE - field is invalid.
3131 +--------------------------------------------------------------------------*/
3133 Check_Field(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
3137 if (Field_Has_Option(field, O_NULLOK))
3139 FIELD_CELL *bp = field->buf;
3142 while (ISBLANK(*bp))
3146 if (CharOf(*bp) == 0)
3150 if (typ->status & _LINKED_TYPE)
3154 Check_Field(form, typ->left, field, argp->left) ||
3155 Check_Field(form, typ->right, field, argp->right));
3159 #if NCURSES_INTEROP_FUNCS
3160 if (typ->fieldcheck.ofcheck)
3162 if (typ->status & _GENERIC)
3163 return typ->fieldcheck.gfcheck(form, field, (void *)argp);
3165 return typ->fieldcheck.ofcheck(field, (void *)argp);
3169 return typ->fcheck(field, (void *)argp);
3176 /*---------------------------------------------------------------------------
3177 | Facility : libnform
3178 | Function : bool _nc_Internal_Validation(FORM * form )
3180 | Description : Validate the current field of the form.
3182 | Return Values : TRUE - field is valid
3183 | FALSE - field is invalid
3184 +--------------------------------------------------------------------------*/
3185 NCURSES_EXPORT(bool)
3186 _nc_Internal_Validation(FORM *form)
3190 field = form->current;
3192 Synchronize_Buffer(form);
3193 if ((form->status & _FCHECK_REQUIRED) ||
3194 (!(Field_Has_Option(field, O_PASSOK))))
3196 if (!Check_Field(form, field->type, field, (TypeArgument *)(field->arg)))
3198 ClrStatus(form, _FCHECK_REQUIRED);
3199 SetStatus(field, _CHANGED);
3200 Synchronize_Linked_Fields(field);
3204 /*----------------------------------------------------------------------------
3205 End of Helper routines for Field Validations.
3206 --------------------------------------------------------------------------*/
3208 /*----------------------------------------------------------------------------
3209 Routines for Field Validation.
3210 --------------------------------------------------------------------------*/
3212 /*---------------------------------------------------------------------------
3213 | Facility : libnform
3214 | Function : static int FV_Validation(FORM * form)
3216 | Description : Validate the current field of the form.
3218 | Return Values : E_OK - field valid
3219 | E_INVALID_FIELD - field not valid
3220 +--------------------------------------------------------------------------*/
3222 FV_Validation(FORM *form)
3224 T((T_CALLED("FV_Validation(%p)"), (void *)form));
3225 if (_nc_Internal_Validation(form))
3228 returnCode(E_INVALID_FIELD);
3230 /*----------------------------------------------------------------------------
3231 End of routines for Field Validation.
3232 --------------------------------------------------------------------------*/
3234 /*----------------------------------------------------------------------------
3235 Helper routines for Inter-Field Navigation
3236 --------------------------------------------------------------------------*/
3238 /*---------------------------------------------------------------------------
3239 | Facility : libnform
3240 | Function : static FIELD *Next_Field_On_Page(FIELD * field)
3242 | Description : Get the next field after the given field on the current
3243 | page. The order of fields is the one defined by the
3244 | fields array. Only visible and active fields are
3247 | Return Values : Pointer to the next field.
3248 +--------------------------------------------------------------------------*/
3249 NCURSES_INLINE static FIELD *
3250 Next_Field_On_Page(FIELD *field)
3252 FORM *form = field->form;
3253 FIELD **field_on_page = &form->field[field->index];
3254 FIELD **first_on_page = &form->field[form->page[form->curpage].pmin];
3255 FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3260 (field_on_page == last_on_page) ? first_on_page : field_on_page + 1;
3261 if (Field_Is_Selectable(*field_on_page))
3264 while (field != (*field_on_page));
3265 return (*field_on_page);
3268 /*---------------------------------------------------------------------------
3269 | Facility : libnform
3270 | Function : FIELD* _nc_First_Active_Field(FORM * form)
3272 | Description : Get the first active field on the current page,
3273 | if there are such. If there are none, get the first
3274 | visible field on the page. If there are also none,
3275 | we return the first field on page and hope the best.
3277 | Return Values : Pointer to calculated field.
3278 +--------------------------------------------------------------------------*/
3279 NCURSES_EXPORT(FIELD *)
3280 _nc_First_Active_Field(FORM *form)
3282 FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3283 FIELD *proposed = Next_Field_On_Page(*last_on_page);
3285 if (proposed == *last_on_page)
3287 /* there might be the special situation, where there is no
3288 active and visible field on the current page. We then select
3289 the first visible field on this readonly page
3291 if (Field_Is_Not_Selectable(proposed))
3293 FIELD **field = &form->field[proposed->index];
3294 FIELD **first = &form->field[form->page[form->curpage].pmin];
3298 field = (field == last_on_page) ? first : field + 1;
3299 if (Field_Has_Option(*field, O_VISIBLE))
3302 while (proposed != (*field));
3306 if ((proposed == *last_on_page) &&
3307 !((unsigned)proposed->opts & O_VISIBLE))
3309 /* This means, there is also no visible field on the page.
3310 So we propose the first one and hope the very best...
3311 Some very clever user has designed a readonly and invisible
3321 /*---------------------------------------------------------------------------
3322 | Facility : libnform
3323 | Function : static FIELD *Previous_Field_On_Page(FIELD * field)
3325 | Description : Get the previous field before the given field on the
3326 | current page. The order of fields is the one defined by
3327 | the fields array. Only visible and active fields are
3330 | Return Values : Pointer to the previous field.
3331 +--------------------------------------------------------------------------*/
3332 NCURSES_INLINE static FIELD *
3333 Previous_Field_On_Page(FIELD *field)
3335 FORM *form = field->form;
3336 FIELD **field_on_page = &form->field[field->index];
3337 FIELD **first_on_page = &form->field[form->page[form->curpage].pmin];
3338 FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3343 (field_on_page == first_on_page) ? last_on_page : field_on_page - 1;
3344 if (Field_Is_Selectable(*field_on_page))
3347 while (field != (*field_on_page));
3349 return (*field_on_page);
3352 /*---------------------------------------------------------------------------
3353 | Facility : libnform
3354 | Function : static FIELD *Sorted_Next_Field(FIELD * field)
3356 | Description : Get the next field after the given field on the current
3357 | page. The order of fields is the one defined by the
3358 | (row,column) geometry, rows are major.
3360 | Return Values : Pointer to the next field.
3361 +--------------------------------------------------------------------------*/
3362 NCURSES_INLINE static FIELD *
3363 Sorted_Next_Field(FIELD *field)
3365 FIELD *field_on_page = field;
3369 field_on_page = field_on_page->snext;
3370 if (Field_Is_Selectable(field_on_page))
3373 while (field_on_page != field);
3375 return (field_on_page);
3378 /*---------------------------------------------------------------------------
3379 | Facility : libnform
3380 | Function : static FIELD *Sorted_Previous_Field(FIELD * field)
3382 | Description : Get the previous field before the given field on the
3383 | current page. The order of fields is the one defined
3384 | by the (row,column) geometry, rows are major.
3386 | Return Values : Pointer to the previous field.
3387 +--------------------------------------------------------------------------*/
3388 NCURSES_INLINE static FIELD *
3389 Sorted_Previous_Field(FIELD *field)
3391 FIELD *field_on_page = field;
3395 field_on_page = field_on_page->sprev;
3396 if (Field_Is_Selectable(field_on_page))
3399 while (field_on_page != field);
3401 return (field_on_page);
3404 /*---------------------------------------------------------------------------
3405 | Facility : libnform
3406 | Function : static FIELD *Left_Neighbor_Field(FIELD * field)
3408 | Description : Get the left neighbor of the field on the same line
3409 | and the same page. Cycles through the line.
3411 | Return Values : Pointer to left neighbor field.
3412 +--------------------------------------------------------------------------*/
3413 NCURSES_INLINE static FIELD *
3414 Left_Neighbor_Field(FIELD *field)
3416 FIELD *field_on_page = field;
3418 /* For a field that has really a left neighbor, the while clause
3419 immediately fails and the loop is left, positioned at the right
3420 neighbor. Otherwise we cycle backwards through the sorted field list
3421 until we enter the same line (from the right end).
3425 field_on_page = Sorted_Previous_Field(field_on_page);
3427 while (field_on_page->frow != field->frow);
3429 return (field_on_page);
3432 /*---------------------------------------------------------------------------
3433 | Facility : libnform
3434 | Function : static FIELD *Right_Neighbor_Field(FIELD * field)
3436 | Description : Get the right neighbor of the field on the same line
3437 | and the same page.
3439 | Return Values : Pointer to right neighbor field.
3440 +--------------------------------------------------------------------------*/
3441 NCURSES_INLINE static FIELD *
3442 Right_Neighbor_Field(FIELD *field)
3444 FIELD *field_on_page = field;
3446 /* See the comments on Left_Neighbor_Field to understand how it works */
3449 field_on_page = Sorted_Next_Field(field_on_page);
3451 while (field_on_page->frow != field->frow);
3453 return (field_on_page);
3456 /*---------------------------------------------------------------------------
3457 | Facility : libnform
3458 | Function : static FIELD *Upper_Neighbor_Field(FIELD * field)
3460 | Description : Because of the row-major nature of sorting the fields,
3461 | it is more difficult to define whats the upper neighbor
3462 | field really means. We define that it must be on a
3463 | 'previous' line (cyclic order!) and is the rightmost
3464 | field laying on the left side of the given field. If
3465 | this set is empty, we take the first field on the line.
3467 | Return Values : Pointer to the upper neighbor field.
3468 +--------------------------------------------------------------------------*/
3470 Upper_Neighbor_Field(FIELD *field)
3472 FIELD *field_on_page = field;
3473 int frow = field->frow;
3474 int fcol = field->fcol;
3476 /* Walk back to the 'previous' line. The second term in the while clause
3477 just guarantees that we stop if we cycled through the line because
3478 there might be no 'previous' line if the page has just one line.
3482 field_on_page = Sorted_Previous_Field(field_on_page);
3484 while (field_on_page->frow == frow && field_on_page->fcol != fcol);
3486 if (field_on_page->frow != frow)
3488 /* We really found a 'previous' line. We are positioned at the
3489 rightmost field on this line */
3490 frow = field_on_page->frow;
3492 /* We walk to the left as long as we are really right of the
3494 while (field_on_page->frow == frow && field_on_page->fcol > fcol)
3495 field_on_page = Sorted_Previous_Field(field_on_page);
3497 /* If we wrapped, just go to the right which is the first field on
3499 if (field_on_page->frow != frow)
3500 field_on_page = Sorted_Next_Field(field_on_page);
3503 return (field_on_page);
3506 /*---------------------------------------------------------------------------
3507 | Facility : libnform
3508 | Function : static FIELD *Down_Neighbor_Field(FIELD * field)
3510 | Description : Because of the row-major nature of sorting the fields,
3511 | its more difficult to define whats the down neighbor
3512 | field really means. We define that it must be on a
3513 | 'next' line (cyclic order!) and is the leftmost
3514 | field laying on the right side of the given field. If
3515 | this set is empty, we take the last field on the line.
3517 | Return Values : Pointer to the upper neighbor field.
3518 +--------------------------------------------------------------------------*/
3520 Down_Neighbor_Field(FIELD *field)
3522 FIELD *field_on_page = field;
3523 int frow = field->frow;
3524 int fcol = field->fcol;
3526 /* Walk forward to the 'next' line. The second term in the while clause
3527 just guarantees that we stop if we cycled through the line because
3528 there might be no 'next' line if the page has just one line.
3532 field_on_page = Sorted_Next_Field(field_on_page);
3534 while (field_on_page->frow == frow && field_on_page->fcol != fcol);
3536 if (field_on_page->frow != frow)
3538 /* We really found a 'next' line. We are positioned at the rightmost
3539 field on this line */
3540 frow = field_on_page->frow;
3542 /* We walk to the right as long as we are really left of the
3544 while (field_on_page->frow == frow && field_on_page->fcol < fcol)
3545 field_on_page = Sorted_Next_Field(field_on_page);
3547 /* If we wrapped, just go to the left which is the last field on
3549 if (field_on_page->frow != frow)
3550 field_on_page = Sorted_Previous_Field(field_on_page);
3553 return (field_on_page);
3556 /*----------------------------------------------------------------------------
3557 Inter-Field Navigation routines
3558 --------------------------------------------------------------------------*/
3560 /*---------------------------------------------------------------------------
3561 | Facility : libnform
3562 | Function : static int Inter_Field_Navigation(
3563 | int (* const fct) (FORM *),
3566 | Description : Generic behavior for changing the current field, the
3567 | field is left and a new field is entered. So the field
3568 | must be validated and the field init/term hooks must
3571 | Return Values : E_OK - success
3572 | E_INVALID_FIELD - field is invalid
3573 | some other - error from subordinate call
3574 +--------------------------------------------------------------------------*/
3576 Inter_Field_Navigation(int (*const fct) (FORM *), FORM *form)
3580 if (!_nc_Internal_Validation(form))
3581 res = E_INVALID_FIELD;
3584 Call_Hook(form, fieldterm);
3586 Call_Hook(form, fieldinit);
3591 /*---------------------------------------------------------------------------
3592 | Facility : libnform
3593 | Function : static int FN_Next_Field(FORM * form)
3595 | Description : Move to the next field on the current page of the form
3597 | Return Values : E_OK - success
3598 | != E_OK - error from subordinate call
3599 +--------------------------------------------------------------------------*/
3601 FN_Next_Field(FORM *form)
3603 T((T_CALLED("FN_Next_Field(%p)"), (void *)form));
3604 returnCode(_nc_Set_Current_Field(form,
3605 Next_Field_On_Page(form->current)));
3608 /*---------------------------------------------------------------------------
3609 | Facility : libnform
3610 | Function : static int FN_Previous_Field(FORM * form)
3612 | Description : Move to the previous field on the current page of the
3615 | Return Values : E_OK - success
3616 | != E_OK - error from subordinate call
3617 +--------------------------------------------------------------------------*/
3619 FN_Previous_Field(FORM *form)
3621 T((T_CALLED("FN_Previous_Field(%p)"), (void *)form));
3622 returnCode(_nc_Set_Current_Field(form,
3623 Previous_Field_On_Page(form->current)));
3626 /*---------------------------------------------------------------------------
3627 | Facility : libnform
3628 | Function : static int FN_First_Field(FORM * form)
3630 | Description : Move to the first field on the current page of the form
3632 | Return Values : E_OK - success
3633 | != E_OK - error from subordinate call
3634 +--------------------------------------------------------------------------*/
3636 FN_First_Field(FORM *form)
3638 T((T_CALLED("FN_First_Field(%p)"), (void *)form));
3639 returnCode(_nc_Set_Current_Field(form,
3640 Next_Field_On_Page(form->field[form->page[form->curpage].pmax])));
3643 /*---------------------------------------------------------------------------
3644 | Facility : libnform
3645 | Function : static int FN_Last_Field(FORM * form)
3647 | Description : Move to the last field on the current page of the form
3649 | Return Values : E_OK - success
3650 | != E_OK - error from subordinate call
3651 +--------------------------------------------------------------------------*/
3653 FN_Last_Field(FORM *form)
3655 T((T_CALLED("FN_Last_Field(%p)"), (void *)form));
3657 _nc_Set_Current_Field(form,
3658 Previous_Field_On_Page(form->field[form->page[form->curpage].pmin])));
3661 /*---------------------------------------------------------------------------
3662 | Facility : libnform
3663 | Function : static int FN_Sorted_Next_Field(FORM * form)
3665 | Description : Move to the sorted next field on the current page
3668 | Return Values : E_OK - success
3669 | != E_OK - error from subordinate call
3670 +--------------------------------------------------------------------------*/
3672 FN_Sorted_Next_Field(FORM *form)
3674 T((T_CALLED("FN_Sorted_Next_Field(%p)"), (void *)form));
3675 returnCode(_nc_Set_Current_Field(form,
3676 Sorted_Next_Field(form->current)));
3679 /*---------------------------------------------------------------------------
3680 | Facility : libnform
3681 | Function : static int FN_Sorted_Previous_Field(FORM * form)
3683 | Description : Move to the sorted previous field on the current page
3686 | Return Values : E_OK - success
3687 | != E_OK - error from subordinate call
3688 +--------------------------------------------------------------------------*/
3690 FN_Sorted_Previous_Field(FORM *form)
3692 T((T_CALLED("FN_Sorted_Previous_Field(%p)"), (void *)form));
3693 returnCode(_nc_Set_Current_Field(form,
3694 Sorted_Previous_Field(form->current)));
3697 /*---------------------------------------------------------------------------
3698 | Facility : libnform
3699 | Function : static int FN_Sorted_First_Field(FORM * form)
3701 | Description : Move to the sorted first field on the current page
3704 | Return Values : E_OK - success
3705 | != E_OK - error from subordinate call
3706 +--------------------------------------------------------------------------*/
3708 FN_Sorted_First_Field(FORM *form)
3710 T((T_CALLED("FN_Sorted_First_Field(%p)"), (void *)form));
3711 returnCode(_nc_Set_Current_Field(form,
3712 Sorted_Next_Field(form->field[form->page[form->curpage].smax])));
3715 /*---------------------------------------------------------------------------
3716 | Facility : libnform
3717 | Function : static int FN_Sorted_Last_Field(FORM * form)
3719 | Description : Move to the sorted last field on the current page
3722 | Return Values : E_OK - success
3723 | != E_OK - error from subordinate call
3724 +--------------------------------------------------------------------------*/
3726 FN_Sorted_Last_Field(FORM *form)
3728 T((T_CALLED("FN_Sorted_Last_Field(%p)"), (void *)form));
3729 returnCode(_nc_Set_Current_Field(form,
3730 Sorted_Previous_Field(form->field[form->page[form->curpage].smin])));
3733 /*---------------------------------------------------------------------------
3734 | Facility : libnform
3735 | Function : static int FN_Left_Field(FORM * form)
3737 | Description : Get the field on the left of the current field on the
3738 | same line and the same page. Cycles through the line.
3740 | Return Values : E_OK - success
3741 | != E_OK - error from subordinate call
3742 +--------------------------------------------------------------------------*/
3744 FN_Left_Field(FORM *form)
3746 T((T_CALLED("FN_Left_Field(%p)"), (void *)form));
3747 returnCode(_nc_Set_Current_Field(form,
3748 Left_Neighbor_Field(form->current)));
3751 /*---------------------------------------------------------------------------
3752 | Facility : libnform
3753 | Function : static int FN_Right_Field(FORM * form)
3755 | Description : Get the field on the right of the current field on the
3756 | same line and the same page. Cycles through the line.
3758 | Return Values : E_OK - success
3759 | != E_OK - error from subordinate call
3760 +--------------------------------------------------------------------------*/
3762 FN_Right_Field(FORM *form)
3764 T((T_CALLED("FN_Right_Field(%p)"), (void *)form));
3765 returnCode(_nc_Set_Current_Field(form,
3766 Right_Neighbor_Field(form->current)));
3769 /*---------------------------------------------------------------------------
3770 | Facility : libnform
3771 | Function : static int FN_Up_Field(FORM * form)
3773 | Description : Get the upper neighbor of the current field. This
3774 | cycles through the page. See the comments of the
3775 | Upper_Neighbor_Field function to understand how
3776 | 'upper' is defined.
3778 | Return Values : E_OK - success
3779 | != E_OK - error from subordinate call
3780 +--------------------------------------------------------------------------*/
3782 FN_Up_Field(FORM *form)
3784 T((T_CALLED("FN_Up_Field(%p)"), (void *)form));
3785 returnCode(_nc_Set_Current_Field(form,
3786 Upper_Neighbor_Field(form->current)));
3789 /*---------------------------------------------------------------------------
3790 | Facility : libnform
3791 | Function : static int FN_Down_Field(FORM * form)
3793 | Description : Get the down neighbor of the current field. This
3794 | cycles through the page. See the comments of the
3795 | Down_Neighbor_Field function to understand how
3796 | 'down' is defined.
3798 | Return Values : E_OK - success
3799 | != E_OK - error from subordinate call
3800 +--------------------------------------------------------------------------*/
3802 FN_Down_Field(FORM *form)
3804 T((T_CALLED("FN_Down_Field(%p)"), (void *)form));
3805 returnCode(_nc_Set_Current_Field(form,
3806 Down_Neighbor_Field(form->current)));
3808 /*----------------------------------------------------------------------------
3809 END of Field Navigation routines
3810 --------------------------------------------------------------------------*/
3812 /*----------------------------------------------------------------------------
3813 Helper routines for Page Navigation
3814 --------------------------------------------------------------------------*/
3816 /*---------------------------------------------------------------------------
3817 | Facility : libnform
3818 | Function : int _nc_Set_Form_Page(FORM * form,
3822 | Description : Make the given page number the current page and make
3823 | the given field the current field on the page. If
3824 | for the field NULL is given, make the first field on
3825 | the page the current field. The routine acts only
3826 | if the requested page is not the current page.
3828 | Return Values : E_OK - success
3829 | != E_OK - error from subordinate call
3830 | E_BAD_ARGUMENT - invalid field pointer
3831 | E_SYSTEM_ERROR - some severe basic error
3832 +--------------------------------------------------------------------------*/
3834 _nc_Set_Form_Page(FORM *form, int page, FIELD *field)
3838 if ((form->curpage != page))
3840 FIELD *last_field, *field_on_page;
3842 werase(Get_Form_Window(form));
3843 form->curpage = (short)page;
3844 last_field = field_on_page = form->field[form->page[page].smin];
3847 if ((unsigned)field_on_page->opts & O_VISIBLE)
3848 if ((res = Display_Field(field_on_page)) != E_OK)
3850 field_on_page = field_on_page->snext;
3852 while (field_on_page != last_field);
3855 res = _nc_Set_Current_Field(form, field);
3857 /* N.B.: we don't encapsulate this by Inter_Field_Navigation(),
3858 because this is already executed in a page navigation
3859 context that contains field navigation
3861 res = FN_First_Field(form);
3866 /*---------------------------------------------------------------------------
3867 | Facility : libnform
3868 | Function : static int Next_Page_Number(const FORM * form)
3870 | Description : Calculate the page number following the current page
3871 | number. This cycles if the highest page number is
3874 | Return Values : The next page number
3875 +--------------------------------------------------------------------------*/
3876 NCURSES_INLINE static int
3877 Next_Page_Number(const FORM *form)
3879 return (form->curpage + 1) % form->maxpage;
3882 /*---------------------------------------------------------------------------
3883 | Facility : libnform
3884 | Function : static int Previous_Page_Number(const FORM * form)
3886 | Description : Calculate the page number before the current page
3887 | number. This cycles if the first page number is
3890 | Return Values : The previous page number
3891 +--------------------------------------------------------------------------*/
3892 NCURSES_INLINE static int
3893 Previous_Page_Number(const FORM *form)
3895 return (form->curpage != 0 ? form->curpage - 1 : form->maxpage - 1);
3898 /*----------------------------------------------------------------------------
3899 Page Navigation routines
3900 --------------------------------------------------------------------------*/
3902 /*---------------------------------------------------------------------------
3903 | Facility : libnform
3904 | Function : static int Page_Navigation(
3905 | int (* const fct) (FORM *),
3908 | Description : Generic behavior for changing a page. This means
3909 | that the field is left and a new field is entered.
3910 | So the field must be validated and the field init/term
3911 | hooks must be called. Because also the page is changed,
3912 | the forms init/term hooks must be called also.
3914 | Return Values : E_OK - success
3915 | E_INVALID_FIELD - field is invalid
3916 | some other - error from subordinate call
3917 +--------------------------------------------------------------------------*/
3919 Page_Navigation(int (*const fct) (FORM *), FORM *form)
3923 if (!_nc_Internal_Validation(form))
3924 res = E_INVALID_FIELD;
3927 Call_Hook(form, fieldterm);
3928 Call_Hook(form, formterm);
3930 Call_Hook(form, forminit);
3931 Call_Hook(form, fieldinit);
3936 /*---------------------------------------------------------------------------
3937 | Facility : libnform
3938 | Function : static int PN_Next_Page(FORM * form)
3940 | Description : Move to the next page of the form
3942 | Return Values : E_OK - success
3943 | != E_OK - error from subordinate call
3944 +--------------------------------------------------------------------------*/
3946 PN_Next_Page(FORM *form)
3948 T((T_CALLED("PN_Next_Page(%p)"), (void *)form));
3949 returnCode(_nc_Set_Form_Page(form, Next_Page_Number(form), (FIELD *)0));
3952 /*---------------------------------------------------------------------------
3953 | Facility : libnform
3954 | Function : static int PN_Previous_Page(FORM * form)
3956 | Description : Move to the previous page of the form
3958 | Return Values : E_OK - success
3959 | != E_OK - error from subordinate call
3960 +--------------------------------------------------------------------------*/
3962 PN_Previous_Page(FORM *form)
3964 T((T_CALLED("PN_Previous_Page(%p)"), (void *)form));
3965 returnCode(_nc_Set_Form_Page(form, Previous_Page_Number(form), (FIELD *)0));
3968 /*---------------------------------------------------------------------------
3969 | Facility : libnform
3970 | Function : static int PN_First_Page(FORM * form)
3972 | Description : Move to the first page of the form
3974 | Return Values : E_OK - success
3975 | != E_OK - error from subordinate call
3976 +--------------------------------------------------------------------------*/
3978 PN_First_Page(FORM *form)
3980 T((T_CALLED("PN_First_Page(%p)"), (void *)form));
3981 returnCode(_nc_Set_Form_Page(form, 0, (FIELD *)0));
3984 /*---------------------------------------------------------------------------
3985 | Facility : libnform
3986 | Function : static int PN_Last_Page(FORM * form)
3988 | Description : Move to the last page of the form
3990 | Return Values : E_OK - success
3991 | != E_OK - error from subordinate call
3992 +--------------------------------------------------------------------------*/
3994 PN_Last_Page(FORM *form)
3996 T((T_CALLED("PN_Last_Page(%p)"), (void *)form));
3997 returnCode(_nc_Set_Form_Page(form, form->maxpage - 1, (FIELD *)0));
4000 /*----------------------------------------------------------------------------
4001 END of Field Navigation routines
4002 --------------------------------------------------------------------------*/
4004 /*----------------------------------------------------------------------------
4005 Helper routines for the core form driver.
4006 --------------------------------------------------------------------------*/
4008 # if USE_WIDEC_SUPPORT
4009 /*---------------------------------------------------------------------------
4010 | Facility : libnform
4011 | Function : static int Data_Entry_w(FORM * form, wchar_t c)
4013 | Description : Enter the wide character c into at the current
4014 | position of the current field of the form.
4016 | Return Values : E_OK - success
4017 | E_REQUEST_DENIED - driver could not process the request
4019 +--------------------------------------------------------------------------*/
4021 Data_Entry_w(FORM *form, wchar_t c)
4023 FIELD *field = form->current;
4024 int result = E_REQUEST_DENIED;
4026 T((T_CALLED("Data_Entry(%p,%s)"), (void *)form, _tracechtype((chtype)c)));
4027 if ((Field_Has_Option(field, O_EDIT))
4028 #if FIX_FORM_INACTIVE_BUG
4029 && (Field_Has_Option(field, O_ACTIVE))
4038 setcchar(&temp_ch, given, 0, 0, (void *)0);
4039 if ((Field_Has_Option(field, O_BLANK)) &&
4040 First_Position_In_Current_Field(form) &&
4041 !(form->status & _FCHECK_REQUIRED) &&
4042 !(form->status & _WINDOW_MODIFIED))
4045 if (form->status & _OVLMODE)
4047 wadd_wch(form->w, &temp_ch);
4052 bool There_Is_Room = Is_There_Room_For_A_Char_In_Line(form);
4054 if (!(There_Is_Room ||
4055 ((Single_Line_Field(field) && Growable(field)))))
4056 RETURN(E_REQUEST_DENIED);
4058 if (!There_Is_Room && !Field_Grown(field, 1))
4059 RETURN(E_SYSTEM_ERROR);
4061 wins_wch(form->w, &temp_ch);
4064 if ((result = Wrapping_Not_Necessary_Or_Wrapping_Ok(form)) == E_OK)
4066 bool End_Of_Field = (((field->drows - 1) == form->currow) &&
4067 ((field->dcols - 1) == form->curcol));
4069 form->status |= _WINDOW_MODIFIED;
4070 if (End_Of_Field && !Growable(field) && (Field_Has_Option(field, O_AUTOSKIP)))
4071 result = Inter_Field_Navigation(FN_Next_Field, form);
4074 if (End_Of_Field && Growable(field) && !Field_Grown(field, 1))
4075 result = E_SYSTEM_ERROR;
4079 * We have just added a byte to the form field. It may have
4080 * been part of a multibyte character. If it was, the
4081 * addch_used field is nonzero and we should not try to move
4084 if (WINDOW_EXT(form->w, addch_used) == 0)
4085 IFN_Next_Character(form);
4096 /*---------------------------------------------------------------------------
4097 | Facility : libnform
4098 | Function : static int Data_Entry(FORM * form,int c)
4100 | Description : Enter character c into at the current position of the
4101 | current field of the form.
4103 | Return Values : E_OK - success
4104 | E_REQUEST_DENIED - driver could not process the request
4106 +--------------------------------------------------------------------------*/
4108 Data_Entry(FORM *form, int c)
4110 FIELD *field = form->current;
4111 int result = E_REQUEST_DENIED;
4113 T((T_CALLED("Data_Entry(%p,%s)"), (void *)form, _tracechtype((chtype)c)));
4114 if ((Field_Has_Option(field, O_EDIT))
4115 #if FIX_FORM_INACTIVE_BUG
4116 && (Field_Has_Option(field, O_ACTIVE))
4120 if ((Field_Has_Option(field, O_BLANK)) &&
4121 First_Position_In_Current_Field(form) &&
4122 !(form->status & _FCHECK_REQUIRED) &&
4123 !(form->status & _WINDOW_MODIFIED))
4126 if (form->status & _OVLMODE)
4128 waddch(form->w, (chtype)c);
4133 bool There_Is_Room = Is_There_Room_For_A_Char_In_Line(form);
4135 if (!(There_Is_Room ||
4136 ((Single_Line_Field(field) && Growable(field)))))
4137 RETURN(E_REQUEST_DENIED);
4139 if (!There_Is_Room && !Field_Grown(field, 1))
4140 RETURN(E_SYSTEM_ERROR);
4142 winsch(form->w, (chtype)c);
4145 if ((result = Wrapping_Not_Necessary_Or_Wrapping_Ok(form)) == E_OK)
4147 bool End_Of_Field = (((field->drows - 1) == form->currow) &&
4148 ((field->dcols - 1) == form->curcol));
4150 SetStatus(form, _WINDOW_MODIFIED);
4151 if (End_Of_Field && !Growable(field) && (Field_Has_Option(field, O_AUTOSKIP)))
4152 result = Inter_Field_Navigation(FN_Next_Field, form);
4155 if (End_Of_Field && Growable(field) && !Field_Grown(field, 1))
4156 result = E_SYSTEM_ERROR;
4159 #if USE_WIDEC_SUPPORT
4161 * We have just added a byte to the form field. It may have
4162 * been part of a multibyte character. If it was, the
4163 * addch_used field is nonzero and we should not try to move
4166 if (WINDOW_EXT(form->w, addch_used) == 0)
4167 IFN_Next_Character(form);
4169 IFN_Next_Character(form);
4179 /* Structure to describe the binding of a request code to a function.
4180 The member keycode codes the request value as well as the generic
4181 routine to use for the request. The code for the generic routine
4182 is coded in the upper 16 Bits while the request code is coded in
4185 In terms of C++ you might think of a request as a class with a
4186 virtual method "perform". The different types of request are
4187 derived from this base class and overload (or not) the base class
4188 implementation of perform.
4192 int keycode; /* must be at least 32 bit: hi:mode, lo: key */
4193 int (*cmd) (FORM *); /* low level driver routine for this key */
4197 /* You may see this is the class-id of the request type class */
4198 #define ID_PN (0x00000000) /* Page navigation */
4199 #define ID_FN (0x00010000) /* Inter-Field navigation */
4200 #define ID_IFN (0x00020000) /* Intra-Field navigation */
4201 #define ID_VSC (0x00030000) /* Vertical Scrolling */
4202 #define ID_HSC (0x00040000) /* Horizontal Scrolling */
4203 #define ID_FE (0x00050000) /* Field Editing */
4204 #define ID_EM (0x00060000) /* Edit Mode */
4205 #define ID_FV (0x00070000) /* Field Validation */
4206 #define ID_CH (0x00080000) /* Choice */
4207 #define ID_Mask (0xffff0000)
4208 #define Key_Mask (0x0000ffff)
4209 #define ID_Shft (16)
4211 /* This array holds all the Binding Infos */
4213 static const Binding_Info bindings[MAX_FORM_COMMAND - MIN_FORM_COMMAND + 1] =
4215 { REQ_NEXT_PAGE |ID_PN ,PN_Next_Page},
4216 { REQ_PREV_PAGE |ID_PN ,PN_Previous_Page},
4217 { REQ_FIRST_PAGE |ID_PN ,PN_First_Page},
4218 { REQ_LAST_PAGE |ID_PN ,PN_Last_Page},
4220 { REQ_NEXT_FIELD |ID_FN ,FN_Next_Field},
4221 { REQ_PREV_FIELD |ID_FN ,FN_Previous_Field},
4222 { REQ_FIRST_FIELD |ID_FN ,FN_First_Field},
4223 { REQ_LAST_FIELD |ID_FN ,FN_Last_Field},
4224 { REQ_SNEXT_FIELD |ID_FN ,FN_Sorted_Next_Field},
4225 { REQ_SPREV_FIELD |ID_FN ,FN_Sorted_Previous_Field},
4226 { REQ_SFIRST_FIELD |ID_FN ,FN_Sorted_First_Field},
4227 { REQ_SLAST_FIELD |ID_FN ,FN_Sorted_Last_Field},
4228 { REQ_LEFT_FIELD |ID_FN ,FN_Left_Field},
4229 { REQ_RIGHT_FIELD |ID_FN ,FN_Right_Field},
4230 { REQ_UP_FIELD |ID_FN ,FN_Up_Field},
4231 { REQ_DOWN_FIELD |ID_FN ,FN_Down_Field},
4233 { REQ_NEXT_CHAR |ID_IFN ,IFN_Next_Character},
4234 { REQ_PREV_CHAR |ID_IFN ,IFN_Previous_Character},
4235 { REQ_NEXT_LINE |ID_IFN ,IFN_Next_Line},
4236 { REQ_PREV_LINE |ID_IFN ,IFN_Previous_Line},
4237 { REQ_NEXT_WORD |ID_IFN ,IFN_Next_Word},
4238 { REQ_PREV_WORD |ID_IFN ,IFN_Previous_Word},
4239 { REQ_BEG_FIELD |ID_IFN ,IFN_Beginning_Of_Field},
4240 { REQ_END_FIELD |ID_IFN ,IFN_End_Of_Field},
4241 { REQ_BEG_LINE |ID_IFN ,IFN_Beginning_Of_Line},
4242 { REQ_END_LINE |ID_IFN ,IFN_End_Of_Line},
4243 { REQ_LEFT_CHAR |ID_IFN ,IFN_Left_Character},
4244 { REQ_RIGHT_CHAR |ID_IFN ,IFN_Right_Character},
4245 { REQ_UP_CHAR |ID_IFN ,IFN_Up_Character},
4246 { REQ_DOWN_CHAR |ID_IFN ,IFN_Down_Character},
4248 { REQ_NEW_LINE |ID_FE ,FE_New_Line},
4249 { REQ_INS_CHAR |ID_FE ,FE_Insert_Character},
4250 { REQ_INS_LINE |ID_FE ,FE_Insert_Line},
4251 { REQ_DEL_CHAR |ID_FE ,FE_Delete_Character},
4252 { REQ_DEL_PREV |ID_FE ,FE_Delete_Previous},
4253 { REQ_DEL_LINE |ID_FE ,FE_Delete_Line},
4254 { REQ_DEL_WORD |ID_FE ,FE_Delete_Word},
4255 { REQ_CLR_EOL |ID_FE ,FE_Clear_To_End_Of_Line},
4256 { REQ_CLR_EOF |ID_FE ,FE_Clear_To_End_Of_Field},
4257 { REQ_CLR_FIELD |ID_FE ,FE_Clear_Field},
4259 { REQ_OVL_MODE |ID_EM ,EM_Overlay_Mode},
4260 { REQ_INS_MODE |ID_EM ,EM_Insert_Mode},
4262 { REQ_SCR_FLINE |ID_VSC ,VSC_Scroll_Line_Forward},
4263 { REQ_SCR_BLINE |ID_VSC ,VSC_Scroll_Line_Backward},
4264 { REQ_SCR_FPAGE |ID_VSC ,VSC_Scroll_Page_Forward},
4265 { REQ_SCR_BPAGE |ID_VSC ,VSC_Scroll_Page_Backward},
4266 { REQ_SCR_FHPAGE |ID_VSC ,VSC_Scroll_Half_Page_Forward},
4267 { REQ_SCR_BHPAGE |ID_VSC ,VSC_Scroll_Half_Page_Backward},
4269 { REQ_SCR_FCHAR |ID_HSC ,HSC_Scroll_Char_Forward},
4270 { REQ_SCR_BCHAR |ID_HSC ,HSC_Scroll_Char_Backward},
4271 { REQ_SCR_HFLINE |ID_HSC ,HSC_Horizontal_Line_Forward},
4272 { REQ_SCR_HBLINE |ID_HSC ,HSC_Horizontal_Line_Backward},
4273 { REQ_SCR_HFHALF |ID_HSC ,HSC_Horizontal_Half_Line_Forward},
4274 { REQ_SCR_HBHALF |ID_HSC ,HSC_Horizontal_Half_Line_Backward},
4276 { REQ_VALIDATION |ID_FV ,FV_Validation},
4278 { REQ_NEXT_CHOICE |ID_CH ,CR_Next_Choice},
4279 { REQ_PREV_CHOICE |ID_CH ,CR_Previous_Choice}
4283 /*---------------------------------------------------------------------------
4284 | Facility : libnform
4285 | Function : int form_driver(FORM * form,int c)
4287 | Description : This is the workhorse of the forms system. It checks
4288 | to determine whether the character c is a request or
4289 | data. If it is a request, the form driver executes
4290 | the request and returns the result. If it is data
4291 | (printable character), it enters the data into the
4292 | current position in the current field. If it is not
4293 | recognized, the form driver assumes it is an application
4294 | defined command and returns E_UNKNOWN_COMMAND.
4295 | Application defined command should be defined relative
4296 | to MAX_FORM_COMMAND, the maximum value of a request.
4298 | Return Values : E_OK - success
4299 | E_SYSTEM_ERROR - system error
4300 | E_BAD_ARGUMENT - an argument is incorrect
4301 | E_NOT_POSTED - form is not posted
4302 | E_INVALID_FIELD - field contents are invalid
4303 | E_BAD_STATE - called from inside a hook routine
4304 | E_REQUEST_DENIED - request failed
4305 | E_NOT_CONNECTED - no fields are connected to the form
4306 | E_UNKNOWN_COMMAND - command not known
4307 +--------------------------------------------------------------------------*/
4309 form_driver(FORM *form, int c)
4311 const Binding_Info *BI = (Binding_Info *) 0;
4312 int res = E_UNKNOWN_COMMAND;
4314 T((T_CALLED("form_driver(%p,%d)"), (void *)form, c));
4317 RETURN(E_BAD_ARGUMENT);
4320 RETURN(E_NOT_CONNECTED);
4324 if (c == FIRST_ACTIVE_MAGIC)
4326 form->current = _nc_First_Active_Field(form);
4330 assert(form->current &&
4331 form->current->buf &&
4332 (form->current->form == form)
4335 if (form->status & _IN_DRIVER)