1 /****************************************************************************
2 * Copyright (c) 1998-2016,2017 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.121 2017/04/08 22:02:15 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->frow + field->rows - 1,
1279 field->fcol + field->cols - 1, 0);
1281 Buffer_To_Window(field, form->w);
1282 SetStatus(field, _NEWTOP); /* fake refresh to paint all */
1283 _nc_Refresh_Current_Field(form);
1288 res = Display_Field(field);
1295 /*---------------------------------------------------------------------------
1296 | Facility : libnform
1297 | Function : int _nc_Synchronize_Options(FIELD * field,
1298 | Field_Options newopts)
1300 | Description : If a fields options have changed, this routine is
1301 | called to propagate these changes to the screen and
1302 | to really change the behavior of the field.
1304 | Return Values : E_OK - success
1305 | E_BAD_ARGUMENT - invalid field pointer
1306 | E_CURRENT - field is the current one
1307 | E_SYSTEM_ERROR - some severe basic error
1308 +--------------------------------------------------------------------------*/
1310 _nc_Synchronize_Options(FIELD *field, Field_Options newopts)
1312 Field_Options oldopts;
1313 Field_Options changed_opts;
1317 T((T_CALLED("_nc_Synchronize_Options(%p,%#x)"), (void *)field, newopts));
1320 returnCode(E_BAD_ARGUMENT);
1322 oldopts = field->opts;
1323 changed_opts = oldopts ^ newopts;
1324 field->opts = newopts;
1329 if (form->status & _POSTED)
1331 if (form->current == field)
1333 field->opts = oldopts;
1334 returnCode(E_CURRENT);
1336 if (form->curpage == field->page)
1338 if ((unsigned)changed_opts & O_VISIBLE)
1340 if ((unsigned)newopts & O_VISIBLE)
1341 res = Display_Field(field);
1343 res = Erase_Field(field);
1347 if (((unsigned)changed_opts & O_PUBLIC) &&
1348 ((unsigned)newopts & O_VISIBLE))
1349 res = Display_Field(field);
1355 if ((unsigned)changed_opts & O_STATIC)
1357 bool single_line_field = Single_Line_Field(field);
1360 if ((unsigned)newopts & O_STATIC)
1362 /* the field becomes now static */
1363 ClrStatus(field, _MAY_GROW);
1364 /* if actually we have no hidden columns, justification may
1366 if (single_line_field &&
1367 (field->cols == field->dcols) &&
1368 (field->just != NO_JUSTIFICATION) &&
1369 Field_Really_Appears(field))
1371 res2 = Display_Field(field);
1376 /* field is no longer static */
1377 if ((field->maxgrow == 0) ||
1378 (single_line_field && (field->dcols < field->maxgrow)) ||
1379 (!single_line_field && (field->drows < field->maxgrow)))
1381 SetStatus(field, _MAY_GROW);
1382 /* a field with justification now changes its behavior,
1383 so we must redisplay it */
1384 if (single_line_field &&
1385 (field->just != NO_JUSTIFICATION) &&
1386 Field_Really_Appears(field))
1388 res2 = Display_Field(field);
1400 * Removes the focus from the current field of the form.
1403 _nc_Unset_Current_Field(FORM *form)
1405 FIELD *field = form->current;
1407 _nc_Refresh_Current_Field(form);
1408 if (Field_Has_Option(field, O_PUBLIC))
1410 if (field->drows > field->rows)
1412 if (form->toprow == 0)
1413 ClrStatus(field, _NEWTOP);
1415 SetStatus(field, _NEWTOP);
1419 if (Justification_Allowed(field))
1421 Window_To_Buffer(form, field);
1423 Perform_Justification(field, form->w);
1424 if (Field_Has_Option(field, O_DYNAMIC_JUSTIFY) &&
1425 (form->w->_parent == 0))
1428 Get_Form_Window(form),
1434 field->cols + field->fcol - 1,
1436 wsyncup(Get_Form_Window(form));
1446 form->w = (WINDOW *)0;
1450 /*---------------------------------------------------------------------------
1451 | Facility : libnform
1452 | Function : int _nc_Set_Current_Field(FORM * form,
1455 | Description : Make the newfield the new current field.
1457 | Return Values : E_OK - success
1458 | E_BAD_ARGUMENT - invalid form or field pointer
1459 | E_SYSTEM_ERROR - some severe basic error
1460 | E_NOT_CONNECTED - no fields are connected to the form
1461 +--------------------------------------------------------------------------*/
1463 _nc_Set_Current_Field(FORM *form, FIELD *newfield)
1468 T((T_CALLED("_nc_Set_Current_Field(%p,%p)"), (void *)form, (void *)newfield));
1470 if (!form || !newfield || (newfield->form != form))
1471 returnCode(E_BAD_ARGUMENT);
1473 if ((form->status & _IN_DRIVER))
1474 returnCode(E_BAD_STATE);
1477 returnCode(E_NOT_CONNECTED);
1479 field = form->current;
1481 if ((field != newfield) ||
1482 !(form->status & _POSTED))
1484 if (field && (form->w) &&
1485 (Field_Has_Option(field, O_VISIBLE)) &&
1486 (field->form->curpage == field->page))
1487 _nc_Unset_Current_Field(form);
1491 if (Has_Invisible_Parts(field))
1492 new_window = newpad(field->drows, field->dcols);
1494 new_window = derwin(Get_Form_Window(form),
1495 field->rows, field->cols, field->frow, field->fcol);
1498 returnCode(E_SYSTEM_ERROR);
1500 form->current = field;
1504 form->w = new_window;
1506 ClrStatus(form, _WINDOW_MODIFIED);
1507 Set_Field_Window_Attributes(field, form->w);
1509 if (Has_Invisible_Parts(field))
1512 Buffer_To_Window(field, form->w);
1516 if (Justification_Allowed(field))
1519 Undo_Justification(field, form->w);
1524 untouchwin(form->w);
1527 form->currow = form->curcol = form->toprow = form->begincol = 0;
1531 /*----------------------------------------------------------------------------
1532 Intra-Field Navigation routines
1533 --------------------------------------------------------------------------*/
1535 /*---------------------------------------------------------------------------
1536 | Facility : libnform
1537 | Function : static int IFN_Next_Character(FORM * form)
1539 | Description : Move to the next character in the field. In a multi-line
1540 | field this wraps at the end of the line.
1542 | Return Values : E_OK - success
1543 | E_REQUEST_DENIED - at the rightmost position
1544 +--------------------------------------------------------------------------*/
1546 IFN_Next_Character(FORM *form)
1548 FIELD *field = form->current;
1549 int step = myWCWIDTH(form->w, form->currow, form->curcol);
1551 T((T_CALLED("IFN_Next_Character(%p)"), (void *)form));
1552 if ((form->curcol += step) == field->dcols)
1554 if ((++(form->currow)) == field->drows)
1556 #if GROW_IF_NAVIGATE
1557 if (!Single_Line_Field(field) && Field_Grown(field, 1))
1564 #if GROW_IF_NAVIGATE
1565 if (Single_Line_Field(field) && Field_Grown(field, 1))
1568 form->curcol -= step;
1569 returnCode(E_REQUEST_DENIED);
1576 /*---------------------------------------------------------------------------
1577 | Facility : libnform
1578 | Function : static int IFN_Previous_Character(FORM * form)
1580 | Description : Move to the previous character in the field. In a
1581 | multi-line field this wraps and the beginning of the
1584 | Return Values : E_OK - success
1585 | E_REQUEST_DENIED - at the leftmost position
1586 +--------------------------------------------------------------------------*/
1588 IFN_Previous_Character(FORM *form)
1590 int amount = myWCWIDTH(form->w, form->currow, form->curcol - 1);
1591 int oldcol = form->curcol;
1593 T((T_CALLED("IFN_Previous_Character(%p)"), (void *)form));
1594 if ((form->curcol -= amount) < 0)
1596 if ((--(form->currow)) < 0)
1599 form->curcol = oldcol;
1600 returnCode(E_REQUEST_DENIED);
1602 form->curcol = form->current->dcols - 1;
1607 /*---------------------------------------------------------------------------
1608 | Facility : libnform
1609 | Function : static int IFN_Next_Line(FORM * form)
1611 | Description : Move to the beginning of the next line in the field
1613 | Return Values : E_OK - success
1614 | E_REQUEST_DENIED - at the last line
1615 +--------------------------------------------------------------------------*/
1617 IFN_Next_Line(FORM *form)
1619 FIELD *field = form->current;
1621 T((T_CALLED("IFN_Next_Line(%p)"), (void *)form));
1622 if ((++(form->currow)) == field->drows)
1624 #if GROW_IF_NAVIGATE
1625 if (!Single_Line_Field(field) && Field_Grown(field, 1))
1629 returnCode(E_REQUEST_DENIED);
1635 /*---------------------------------------------------------------------------
1636 | Facility : libnform
1637 | Function : static int IFN_Previous_Line(FORM * form)
1639 | Description : Move to the beginning of the previous line in the field
1641 | Return Values : E_OK - success
1642 | E_REQUEST_DENIED - at the first line
1643 +--------------------------------------------------------------------------*/
1645 IFN_Previous_Line(FORM *form)
1647 T((T_CALLED("IFN_Previous_Line(%p)"), (void *)form));
1648 if ((--(form->currow)) < 0)
1651 returnCode(E_REQUEST_DENIED);
1657 /*---------------------------------------------------------------------------
1658 | Facility : libnform
1659 | Function : static int IFN_Next_Word(FORM * form)
1661 | Description : Move to the beginning of the next word in the field.
1663 | Return Values : E_OK - success
1664 | E_REQUEST_DENIED - there is no next word
1665 +--------------------------------------------------------------------------*/
1667 IFN_Next_Word(FORM *form)
1669 FIELD *field = form->current;
1670 FIELD_CELL *bp = Address_Of_Current_Position_In_Buffer(form);
1674 T((T_CALLED("IFN_Next_Word(%p)"), (void *)form));
1676 /* We really need access to the data, so we have to synchronize */
1677 Synchronize_Buffer(form);
1679 /* Go to the first whitespace after the current position (including
1680 current position). This is then the starting point to look for the
1681 next non-blank data */
1682 s = Get_First_Whitespace_Character(bp, Buffer_Length(field) -
1683 (int)(bp - field->buf));
1685 /* Find the start of the next word */
1686 t = Get_Start_Of_Data(s, Buffer_Length(field) -
1687 (int)(s - field->buf));
1688 #if !FRIENDLY_PREV_NEXT_WORD
1690 returnCode(E_REQUEST_DENIED);
1694 Adjust_Cursor_Position(form, t);
1699 /*---------------------------------------------------------------------------
1700 | Facility : libnform
1701 | Function : static int IFN_Previous_Word(FORM * form)
1703 | Description : Move to the beginning of the previous word in the field.
1705 | Return Values : E_OK - success
1706 | E_REQUEST_DENIED - there is no previous word
1707 +--------------------------------------------------------------------------*/
1709 IFN_Previous_Word(FORM *form)
1711 FIELD *field = form->current;
1712 FIELD_CELL *bp = Address_Of_Current_Position_In_Buffer(form);
1717 T((T_CALLED("IFN_Previous_Word(%p)"), (void *)form));
1719 /* We really need access to the data, so we have to synchronize */
1720 Synchronize_Buffer(form);
1722 s = After_End_Of_Data(field->buf, (int)(bp - field->buf));
1723 /* s points now right after the last non-blank in the buffer before bp.
1724 If bp was in a word, s equals bp. In this case we must find the last
1725 whitespace in the buffer before bp and repeat the game to really find
1726 the previous word! */
1730 /* And next call now goes backward to look for the last whitespace
1731 before that, pointing right after this, so it points to the begin
1732 of the previous word.
1734 t = After_Last_Whitespace_Character(field->buf, (int)(s - field->buf));
1735 #if !FRIENDLY_PREV_NEXT_WORD
1737 returnCode(E_REQUEST_DENIED);
1741 /* and do it again, replacing bp by t */
1742 s = After_End_Of_Data(field->buf, (int)(t - field->buf));
1743 t = After_Last_Whitespace_Character(field->buf, (int)(s - field->buf));
1744 #if !FRIENDLY_PREV_NEXT_WORD
1746 returnCode(E_REQUEST_DENIED);
1749 Adjust_Cursor_Position(form, t);
1753 /*---------------------------------------------------------------------------
1754 | Facility : libnform
1755 | Function : static int IFN_Beginning_Of_Field(FORM * form)
1757 | Description : Place the cursor at the first non-pad character in
1760 | Return Values : E_OK - success
1761 +--------------------------------------------------------------------------*/
1763 IFN_Beginning_Of_Field(FORM *form)
1765 FIELD *field = form->current;
1767 T((T_CALLED("IFN_Beginning_Of_Field(%p)"), (void *)form));
1768 Synchronize_Buffer(form);
1769 Adjust_Cursor_Position(form,
1770 Get_Start_Of_Data(field->buf, Buffer_Length(field)));
1774 /*---------------------------------------------------------------------------
1775 | Facility : libnform
1776 | Function : static int IFN_End_Of_Field(FORM * form)
1778 | Description : Place the cursor after the last non-pad character in
1779 | the field. If the field occupies the last position in
1780 | the buffer, the cursor is positioned on the last
1783 | Return Values : E_OK - success
1784 +--------------------------------------------------------------------------*/
1786 IFN_End_Of_Field(FORM *form)
1788 FIELD *field = form->current;
1791 T((T_CALLED("IFN_End_Of_Field(%p)"), (void *)form));
1792 Synchronize_Buffer(form);
1793 pos = After_End_Of_Data(field->buf, Buffer_Length(field));
1794 if (pos == (field->buf + Buffer_Length(field)))
1796 Adjust_Cursor_Position(form, pos);
1800 /*---------------------------------------------------------------------------
1801 | Facility : libnform
1802 | Function : static int IFN_Beginning_Of_Line(FORM * form)
1804 | Description : Place the cursor on the first non-pad character in
1805 | the current line of the field.
1807 | Return Values : E_OK - success
1808 +--------------------------------------------------------------------------*/
1810 IFN_Beginning_Of_Line(FORM *form)
1812 FIELD *field = form->current;
1814 T((T_CALLED("IFN_Beginning_Of_Line(%p)"), (void *)form));
1815 Synchronize_Buffer(form);
1816 Adjust_Cursor_Position(form,
1817 Get_Start_Of_Data(Address_Of_Current_Row_In_Buffer(form),
1822 /*---------------------------------------------------------------------------
1823 | Facility : libnform
1824 | Function : static int IFN_End_Of_Line(FORM * form)
1826 | Description : Place the cursor after the last non-pad character in the
1827 | current line of the field. If the field occupies the
1828 | last column in the line, the cursor is positioned on the
1829 | last character of the line.
1831 | Return Values : E_OK - success
1832 +--------------------------------------------------------------------------*/
1834 IFN_End_Of_Line(FORM *form)
1836 FIELD *field = form->current;
1840 T((T_CALLED("IFN_End_Of_Line(%p)"), (void *)form));
1841 Synchronize_Buffer(form);
1842 bp = Address_Of_Current_Row_In_Buffer(form);
1843 pos = After_End_Of_Data(bp, field->dcols);
1844 if (pos == (bp + field->dcols))
1846 Adjust_Cursor_Position(form, pos);
1850 /*---------------------------------------------------------------------------
1851 | Facility : libnform
1852 | Function : static int IFN_Left_Character(FORM * form)
1854 | Description : Move one character to the left in the current line.
1855 | This doesn't cycle.
1857 | Return Values : E_OK - success
1858 | E_REQUEST_DENIED - already in first column
1859 +--------------------------------------------------------------------------*/
1861 IFN_Left_Character(FORM *form)
1863 int amount = myWCWIDTH(form->w, form->currow, form->curcol - 1);
1864 int oldcol = form->curcol;
1866 T((T_CALLED("IFN_Left_Character(%p)"), (void *)form));
1867 if ((form->curcol -= amount) < 0)
1869 form->curcol = oldcol;
1870 returnCode(E_REQUEST_DENIED);
1875 /*---------------------------------------------------------------------------
1876 | Facility : libnform
1877 | Function : static int IFN_Right_Character(FORM * form)
1879 | Description : Move one character to the right in the current line.
1880 | This doesn't cycle.
1882 | Return Values : E_OK - success
1883 | E_REQUEST_DENIED - already in last column
1884 +--------------------------------------------------------------------------*/
1886 IFN_Right_Character(FORM *form)
1888 int amount = myWCWIDTH(form->w, form->currow, form->curcol);
1889 int oldcol = form->curcol;
1891 T((T_CALLED("IFN_Right_Character(%p)"), (void *)form));
1892 if ((form->curcol += amount) >= form->current->dcols)
1894 #if GROW_IF_NAVIGATE
1895 FIELD *field = form->current;
1897 if (Single_Line_Field(field) && Field_Grown(field, 1))
1900 form->curcol = oldcol;
1901 returnCode(E_REQUEST_DENIED);
1906 /*---------------------------------------------------------------------------
1907 | Facility : libnform
1908 | Function : static int IFN_Up_Character(FORM * form)
1910 | Description : Move one line up. This doesn't cycle through the lines
1913 | Return Values : E_OK - success
1914 | E_REQUEST_DENIED - already in last column
1915 +--------------------------------------------------------------------------*/
1917 IFN_Up_Character(FORM *form)
1919 T((T_CALLED("IFN_Up_Character(%p)"), (void *)form));
1920 if ((--(form->currow)) < 0)
1923 returnCode(E_REQUEST_DENIED);
1928 /*---------------------------------------------------------------------------
1929 | Facility : libnform
1930 | Function : static int IFN_Down_Character(FORM * form)
1932 | Description : Move one line down. This doesn't cycle through the
1933 | lines of the field.
1935 | Return Values : E_OK - success
1936 | E_REQUEST_DENIED - already in last column
1937 +--------------------------------------------------------------------------*/
1939 IFN_Down_Character(FORM *form)
1941 FIELD *field = form->current;
1943 T((T_CALLED("IFN_Down_Character(%p)"), (void *)form));
1944 if ((++(form->currow)) == field->drows)
1946 #if GROW_IF_NAVIGATE
1947 if (!Single_Line_Field(field) && Field_Grown(field, 1))
1951 returnCode(E_REQUEST_DENIED);
1955 /*----------------------------------------------------------------------------
1956 END of Intra-Field Navigation routines
1957 --------------------------------------------------------------------------*/
1959 /*----------------------------------------------------------------------------
1960 Vertical scrolling helper routines
1961 --------------------------------------------------------------------------*/
1963 /*---------------------------------------------------------------------------
1964 | Facility : libnform
1965 | Function : static int VSC_Generic(FORM *form, int nlines)
1967 | Description : Scroll multi-line field forward (nlines>0) or
1968 | backward (nlines<0) this many lines.
1970 | Return Values : E_OK - success
1971 | E_REQUEST_DENIED - can't scroll
1972 +--------------------------------------------------------------------------*/
1974 VSC_Generic(FORM *form, int nlines)
1976 FIELD *field = form->current;
1977 int res = E_REQUEST_DENIED;
1978 int rows_to_go = (nlines > 0 ? nlines : -nlines);
1982 if ((rows_to_go + form->toprow) > (field->drows - field->rows))
1983 rows_to_go = (field->drows - field->rows - form->toprow);
1987 form->currow += rows_to_go;
1988 form->toprow += rows_to_go;
1994 if (rows_to_go > form->toprow)
1995 rows_to_go = form->toprow;
1999 form->currow -= rows_to_go;
2000 form->toprow -= rows_to_go;
2006 /*----------------------------------------------------------------------------
2007 End of Vertical scrolling helper routines
2008 --------------------------------------------------------------------------*/
2010 /*----------------------------------------------------------------------------
2011 Vertical scrolling routines
2012 --------------------------------------------------------------------------*/
2014 /*---------------------------------------------------------------------------
2015 | Facility : libnform
2016 | Function : static int Vertical_Scrolling(
2017 | int (* const fct) (FORM *),
2020 | Description : Performs the generic vertical scrolling routines.
2021 | This has to check for a multi-line field and to set
2022 | the _NEWTOP flag if scrolling really occurred.
2024 | Return Values : Propagated error code from low-level driver calls
2025 +--------------------------------------------------------------------------*/
2027 Vertical_Scrolling(int (*const fct) (FORM *), FORM *form)
2029 int res = E_REQUEST_DENIED;
2031 if (!Single_Line_Field(form->current))
2035 SetStatus(form->current, _NEWTOP);
2040 /*---------------------------------------------------------------------------
2041 | Facility : libnform
2042 | Function : static int VSC_Scroll_Line_Forward(FORM * form)
2044 | Description : Scroll multi-line field forward a line
2046 | Return Values : E_OK - success
2047 | E_REQUEST_DENIED - no data ahead
2048 +--------------------------------------------------------------------------*/
2050 VSC_Scroll_Line_Forward(FORM *form)
2052 T((T_CALLED("VSC_Scroll_Line_Forward(%p)"), (void *)form));
2053 returnCode(VSC_Generic(form, 1));
2056 /*---------------------------------------------------------------------------
2057 | Facility : libnform
2058 | Function : static int VSC_Scroll_Line_Backward(FORM * form)
2060 | Description : Scroll multi-line field backward a line
2062 | Return Values : E_OK - success
2063 | E_REQUEST_DENIED - no data behind
2064 +--------------------------------------------------------------------------*/
2066 VSC_Scroll_Line_Backward(FORM *form)
2068 T((T_CALLED("VSC_Scroll_Line_Backward(%p)"), (void *)form));
2069 returnCode(VSC_Generic(form, -1));
2072 /*---------------------------------------------------------------------------
2073 | Facility : libnform
2074 | Function : static int VSC_Scroll_Page_Forward(FORM * form)
2076 | Description : Scroll a multi-line field forward a page
2078 | Return Values : E_OK - success
2079 | E_REQUEST_DENIED - no data ahead
2080 +--------------------------------------------------------------------------*/
2082 VSC_Scroll_Page_Forward(FORM *form)
2084 T((T_CALLED("VSC_Scroll_Page_Forward(%p)"), (void *)form));
2085 returnCode(VSC_Generic(form, form->current->rows));
2088 /*---------------------------------------------------------------------------
2089 | Facility : libnform
2090 | Function : static int VSC_Scroll_Half_Page_Forward(FORM * form)
2092 | Description : Scroll a multi-line field forward half a page
2094 | Return Values : E_OK - success
2095 | E_REQUEST_DENIED - no data ahead
2096 +--------------------------------------------------------------------------*/
2098 VSC_Scroll_Half_Page_Forward(FORM *form)
2100 T((T_CALLED("VSC_Scroll_Half_Page_Forward(%p)"), (void *)form));
2101 returnCode(VSC_Generic(form, (form->current->rows + 1) / 2));
2104 /*---------------------------------------------------------------------------
2105 | Facility : libnform
2106 | Function : static int VSC_Scroll_Page_Backward(FORM * form)
2108 | Description : Scroll a multi-line field backward a page
2110 | Return Values : E_OK - success
2111 | E_REQUEST_DENIED - no data behind
2112 +--------------------------------------------------------------------------*/
2114 VSC_Scroll_Page_Backward(FORM *form)
2116 T((T_CALLED("VSC_Scroll_Page_Backward(%p)"), (void *)form));
2117 returnCode(VSC_Generic(form, -(form->current->rows)));
2120 /*---------------------------------------------------------------------------
2121 | Facility : libnform
2122 | Function : static int VSC_Scroll_Half_Page_Backward(FORM * form)
2124 | Description : Scroll a multi-line field backward half a page
2126 | Return Values : E_OK - success
2127 | E_REQUEST_DENIED - no data behind
2128 +--------------------------------------------------------------------------*/
2130 VSC_Scroll_Half_Page_Backward(FORM *form)
2132 T((T_CALLED("VSC_Scroll_Half_Page_Backward(%p)"), (void *)form));
2133 returnCode(VSC_Generic(form, -((form->current->rows + 1) / 2)));
2135 /*----------------------------------------------------------------------------
2136 End of Vertical scrolling routines
2137 --------------------------------------------------------------------------*/
2139 /*----------------------------------------------------------------------------
2140 Horizontal scrolling helper routines
2141 --------------------------------------------------------------------------*/
2143 /*---------------------------------------------------------------------------
2144 | Facility : libnform
2145 | Function : static int HSC_Generic(FORM *form, int ncolumns)
2147 | Description : Scroll single-line field forward (ncolumns>0) or
2148 | backward (ncolumns<0) this many columns.
2150 | Return Values : E_OK - success
2151 | E_REQUEST_DENIED - can't scroll
2152 +--------------------------------------------------------------------------*/
2154 HSC_Generic(FORM *form, int ncolumns)
2156 FIELD *field = form->current;
2157 int res = E_REQUEST_DENIED;
2158 int cols_to_go = (ncolumns > 0 ? ncolumns : -ncolumns);
2162 if ((cols_to_go + form->begincol) > (field->dcols - field->cols))
2163 cols_to_go = field->dcols - field->cols - form->begincol;
2167 form->curcol += cols_to_go;
2168 form->begincol += cols_to_go;
2174 if (cols_to_go > form->begincol)
2175 cols_to_go = form->begincol;
2179 form->curcol -= cols_to_go;
2180 form->begincol -= cols_to_go;
2186 /*----------------------------------------------------------------------------
2187 End of Horizontal scrolling helper routines
2188 --------------------------------------------------------------------------*/
2190 /*----------------------------------------------------------------------------
2191 Horizontal scrolling routines
2192 --------------------------------------------------------------------------*/
2194 /*---------------------------------------------------------------------------
2195 | Facility : libnform
2196 | Function : static int Horizontal_Scrolling(
2197 | int (* const fct) (FORM *),
2200 | Description : Performs the generic horizontal scrolling routines.
2201 | This has to check for a single-line field.
2203 | Return Values : Propagated error code from low-level driver calls
2204 +--------------------------------------------------------------------------*/
2206 Horizontal_Scrolling(int (*const fct) (FORM *), FORM *form)
2208 if (Single_Line_Field(form->current))
2211 return (E_REQUEST_DENIED);
2214 /*---------------------------------------------------------------------------
2215 | Facility : libnform
2216 | Function : static int HSC_Scroll_Char_Forward(FORM * form)
2218 | Description : Scroll single-line field forward a character
2220 | Return Values : E_OK - success
2221 | E_REQUEST_DENIED - no data ahead
2222 +--------------------------------------------------------------------------*/
2224 HSC_Scroll_Char_Forward(FORM *form)
2226 T((T_CALLED("HSC_Scroll_Char_Forward(%p)"), (void *)form));
2227 returnCode(HSC_Generic(form, 1));
2230 /*---------------------------------------------------------------------------
2231 | Facility : libnform
2232 | Function : static int HSC_Scroll_Char_Backward(FORM * form)
2234 | Description : Scroll single-line field backward a character
2236 | Return Values : E_OK - success
2237 | E_REQUEST_DENIED - no data behind
2238 +--------------------------------------------------------------------------*/
2240 HSC_Scroll_Char_Backward(FORM *form)
2242 T((T_CALLED("HSC_Scroll_Char_Backward(%p)"), (void *)form));
2243 returnCode(HSC_Generic(form, -1));
2246 /*---------------------------------------------------------------------------
2247 | Facility : libnform
2248 | Function : static int HSC_Horizontal_Line_Forward(FORM* form)
2250 | Description : Scroll single-line field forward a line
2252 | Return Values : E_OK - success
2253 | E_REQUEST_DENIED - no data ahead
2254 +--------------------------------------------------------------------------*/
2256 HSC_Horizontal_Line_Forward(FORM *form)
2258 T((T_CALLED("HSC_Horizontal_Line_Forward(%p)"), (void *)form));
2259 returnCode(HSC_Generic(form, form->current->cols));
2262 /*---------------------------------------------------------------------------
2263 | Facility : libnform
2264 | Function : static int HSC_Horizontal_Half_Line_Forward(FORM* form)
2266 | Description : Scroll single-line field forward half a line
2268 | Return Values : E_OK - success
2269 | E_REQUEST_DENIED - no data ahead
2270 +--------------------------------------------------------------------------*/
2272 HSC_Horizontal_Half_Line_Forward(FORM *form)
2274 T((T_CALLED("HSC_Horizontal_Half_Line_Forward(%p)"), (void *)form));
2275 returnCode(HSC_Generic(form, (form->current->cols + 1) / 2));
2278 /*---------------------------------------------------------------------------
2279 | Facility : libnform
2280 | Function : static int HSC_Horizontal_Line_Backward(FORM* form)
2282 | Description : Scroll single-line field backward a line
2284 | Return Values : E_OK - success
2285 | E_REQUEST_DENIED - no data behind
2286 +--------------------------------------------------------------------------*/
2288 HSC_Horizontal_Line_Backward(FORM *form)
2290 T((T_CALLED("HSC_Horizontal_Line_Backward(%p)"), (void *)form));
2291 returnCode(HSC_Generic(form, -(form->current->cols)));
2294 /*---------------------------------------------------------------------------
2295 | Facility : libnform
2296 | Function : static int HSC_Horizontal_Half_Line_Backward(FORM* form)
2298 | Description : Scroll single-line field backward half a line
2300 | Return Values : E_OK - success
2301 | E_REQUEST_DENIED - no data behind
2302 +--------------------------------------------------------------------------*/
2304 HSC_Horizontal_Half_Line_Backward(FORM *form)
2306 T((T_CALLED("HSC_Horizontal_Half_Line_Backward(%p)"), (void *)form));
2307 returnCode(HSC_Generic(form, -((form->current->cols + 1) / 2)));
2310 /*----------------------------------------------------------------------------
2311 End of Horizontal scrolling routines
2312 --------------------------------------------------------------------------*/
2314 /*----------------------------------------------------------------------------
2315 Helper routines for Field Editing
2316 --------------------------------------------------------------------------*/
2318 /*---------------------------------------------------------------------------
2319 | Facility : libnform
2320 | Function : static bool Is_There_Room_For_A_Line(FORM * form)
2322 | Description : Check whether or not there is enough room in the
2323 | buffer to enter a whole line.
2325 | Return Values : TRUE - there is enough space
2326 | FALSE - there is not enough space
2327 +--------------------------------------------------------------------------*/
2328 NCURSES_INLINE static bool
2329 Is_There_Room_For_A_Line(FORM *form)
2331 FIELD *field = form->current;
2332 FIELD_CELL *begin_of_last_line, *s;
2334 Synchronize_Buffer(form);
2335 begin_of_last_line = Address_Of_Row_In_Buffer(field, (field->drows - 1));
2336 s = After_End_Of_Data(begin_of_last_line, field->dcols);
2337 return ((s == begin_of_last_line) ? TRUE : FALSE);
2340 /*---------------------------------------------------------------------------
2341 | Facility : libnform
2342 | Function : static bool Is_There_Room_For_A_Char_In_Line(FORM * form)
2344 | Description : Checks whether or not there is room for a new character
2345 | in the current line.
2347 | Return Values : TRUE - there is room
2348 | FALSE - there is not enough room (line full)
2349 +--------------------------------------------------------------------------*/
2350 NCURSES_INLINE static bool
2351 Is_There_Room_For_A_Char_In_Line(FORM *form)
2353 int last_char_in_line;
2355 wmove(form->w, form->currow, form->current->dcols - 1);
2356 last_char_in_line = (int)(winch(form->w) & A_CHARTEXT);
2357 wmove(form->w, form->currow, form->curcol);
2358 return (((last_char_in_line == form->current->pad) ||
2359 is_blank(last_char_in_line)) ? TRUE : FALSE);
2362 #define There_Is_No_Room_For_A_Char_In_Line(f) \
2363 !Is_There_Room_For_A_Char_In_Line(f)
2365 /*---------------------------------------------------------------------------
2366 | Facility : libnform
2367 | Function : static int Insert_String(
2373 | Description : Insert the 'len' characters beginning at pointer 'txt'
2374 | into the 'row' of the 'form'. The insertion occurs
2375 | on the beginning of the row, all other characters are
2376 | moved to the right. After the text a pad character will
2377 | be inserted to separate the text from the rest. If
2378 | necessary the insertion moves characters on the next
2379 | line to make place for the requested insertion string.
2381 | Return Values : E_OK - success
2382 | E_REQUEST_DENIED -
2383 | E_SYSTEM_ERROR - system error
2384 +--------------------------------------------------------------------------*/
2386 Insert_String(FORM *form, int row, FIELD_CELL *txt, int len)
2388 FIELD *field = form->current;
2389 FIELD_CELL *bp = Address_Of_Row_In_Buffer(field, row);
2390 int datalen = (int)(After_End_Of_Data(bp, field->dcols) - bp);
2391 int freelen = field->dcols - datalen;
2392 int requiredlen = len + 1;
2394 int result = E_REQUEST_DENIED;
2396 if (freelen >= requiredlen)
2398 wmove(form->w, row, 0);
2399 myINSNSTR(form->w, txt, len);
2400 wmove(form->w, row, len);
2401 myINSNSTR(form->w, &myBLANK, 1);
2406 /* we have to move characters on the next line. If we are on the
2407 last line this may work, if the field is growable */
2408 if ((row == (field->drows - 1)) && Growable(field))
2410 if (!Field_Grown(field, 1))
2411 return (E_SYSTEM_ERROR);
2412 /* !!!Side-Effect : might be changed due to growth!!! */
2413 bp = Address_Of_Row_In_Buffer(field, row);
2416 if (row < (field->drows - 1))
2419 After_Last_Whitespace_Character(bp,
2420 (int)(Get_Start_Of_Data(bp
2425 /* split points now to the first character of the portion of the
2426 line that must be moved to the next line */
2427 datalen = (int)(split - bp); /* + freelen has to stay on this line */
2428 freelen = field->dcols - (datalen + freelen); /* for the next line */
2430 if ((result = Insert_String(form, row + 1, split, freelen)) == E_OK)
2432 wmove(form->w, row, datalen);
2434 wmove(form->w, row, 0);
2435 myINSNSTR(form->w, txt, len);
2436 wmove(form->w, row, len);
2437 myINSNSTR(form->w, &myBLANK, 1);
2445 /*---------------------------------------------------------------------------
2446 | Facility : libnform
2447 | Function : static int Wrapping_Not_Necessary_Or_Wrapping_Ok(
2450 | Description : If a character has been entered into a field, it may
2451 | be that wrapping has to occur. This routine checks
2452 | whether or not wrapping is required and if so, performs
2455 | Return Values : E_OK - no wrapping required or wrapping
2457 | E_REQUEST_DENIED -
2458 | E_SYSTEM_ERROR - some system error
2459 +--------------------------------------------------------------------------*/
2461 Wrapping_Not_Necessary_Or_Wrapping_Ok(FORM *form)
2463 FIELD *field = form->current;
2464 int result = E_REQUEST_DENIED;
2465 bool Last_Row = ((field->drows - 1) == form->currow);
2467 if ((Field_Has_Option(field, O_WRAP)) && /* wrapping wanted */
2468 (!Single_Line_Field(field)) && /* must be multi-line */
2469 (There_Is_No_Room_For_A_Char_In_Line(form)) && /* line is full */
2470 (!Last_Row || Growable(field))) /* there are more lines */
2474 int chars_to_be_wrapped;
2475 int chars_to_remain_on_line;
2479 /* the above logic already ensures, that in this case the field
2481 if (!Field_Grown(field, 1))
2482 return E_SYSTEM_ERROR;
2484 bp = Address_Of_Current_Row_In_Buffer(form);
2485 Window_To_Buffer(form, field);
2486 split = After_Last_Whitespace_Character(bp, field->dcols);
2487 /* split points to the first character of the sequence to be brought
2489 chars_to_remain_on_line = (int)(split - bp);
2490 chars_to_be_wrapped = field->dcols - chars_to_remain_on_line;
2491 if (chars_to_remain_on_line > 0)
2493 if ((result = Insert_String(form, form->currow + 1, split,
2494 chars_to_be_wrapped)) == E_OK)
2496 wmove(form->w, form->currow, chars_to_remain_on_line);
2498 if (form->curcol >= chars_to_remain_on_line)
2501 form->curcol -= chars_to_remain_on_line;
2511 Window_To_Buffer(form, field);
2512 result = E_REQUEST_DENIED;
2516 result = E_OK; /* wrapping was not necessary */
2520 /*----------------------------------------------------------------------------
2521 Field Editing routines
2522 --------------------------------------------------------------------------*/
2524 /*---------------------------------------------------------------------------
2525 | Facility : libnform
2526 | Function : static int Field_Editing(
2527 | int (* const fct) (FORM *),
2530 | Description : Generic routine for field editing requests. The driver
2531 | routines are only called for editable fields, the
2532 | _WINDOW_MODIFIED flag is set if editing occurred.
2533 | This is somewhat special due to the overload semantics
2534 | of the NEW_LINE and DEL_PREV requests.
2536 | Return Values : Error code from low level drivers.
2537 +--------------------------------------------------------------------------*/
2539 Field_Editing(int (*const fct) (FORM *), FORM *form)
2541 int res = E_REQUEST_DENIED;
2543 /* We have to deal here with the specific case of the overloaded
2544 behavior of New_Line and Delete_Previous requests.
2545 They may end up in navigational requests if we are on the first
2546 character in a field. But navigation is also allowed on non-
2549 if ((fct == FE_Delete_Previous) &&
2550 ((unsigned)form->opts & O_BS_OVERLOAD) &&
2551 First_Position_In_Current_Field(form))
2553 res = Inter_Field_Navigation(FN_Previous_Field, form);
2557 if (fct == FE_New_Line)
2559 if (((unsigned)form->opts & O_NL_OVERLOAD) &&
2560 First_Position_In_Current_Field(form))
2562 res = Inter_Field_Navigation(FN_Next_Field, form);
2565 /* FE_New_Line deals itself with the _WINDOW_MODIFIED flag */
2570 /* From now on, everything must be editable */
2571 if ((unsigned)form->current->opts & O_EDIT)
2575 SetStatus(form, _WINDOW_MODIFIED);
2582 /*---------------------------------------------------------------------------
2583 | Facility : libnform
2584 | Function : static int FE_New_Line(FORM * form)
2586 | Description : Perform a new line request. This is rather complex
2587 | compared to other routines in this code due to the
2588 | rather difficult to understand description in the
2591 | Return Values : E_OK - success
2592 | E_REQUEST_DENIED - new line not allowed
2593 | E_SYSTEM_ERROR - system error
2594 +--------------------------------------------------------------------------*/
2596 FE_New_Line(FORM *form)
2598 FIELD *field = form->current;
2600 bool Last_Row = ((field->drows - 1) == form->currow);
2602 T((T_CALLED("FE_New_Line(%p)"), (void *)form));
2603 if (form->status & _OVLMODE)
2606 (!(Growable(field) && !Single_Line_Field(field))))
2608 if (!((unsigned)form->opts & O_NL_OVERLOAD))
2609 returnCode(E_REQUEST_DENIED);
2610 wmove(form->w, form->currow, form->curcol);
2612 /* we have to set this here, although it is also
2613 handled in the generic routine. The reason is,
2614 that FN_Next_Field may fail, but the form is
2615 definitively changed */
2616 SetStatus(form, _WINDOW_MODIFIED);
2617 returnCode(Inter_Field_Navigation(FN_Next_Field, form));
2621 if (Last_Row && !Field_Grown(field, 1))
2623 /* N.B.: due to the logic in the 'if', LastRow==TRUE
2624 means here that the field is growable and not
2625 a single-line field */
2626 returnCode(E_SYSTEM_ERROR);
2628 wmove(form->w, form->currow, form->curcol);
2632 SetStatus(form, _WINDOW_MODIFIED);
2640 !(Growable(field) && !Single_Line_Field(field)))
2642 if (!((unsigned)form->opts & O_NL_OVERLOAD))
2643 returnCode(E_REQUEST_DENIED);
2644 returnCode(Inter_Field_Navigation(FN_Next_Field, form));
2648 bool May_Do_It = !Last_Row && Is_There_Room_For_A_Line(form);
2650 if (!(May_Do_It || Growable(field)))
2651 returnCode(E_REQUEST_DENIED);
2652 if (!May_Do_It && !Field_Grown(field, 1))
2653 returnCode(E_SYSTEM_ERROR);
2655 bp = Address_Of_Current_Position_In_Buffer(form);
2656 t = After_End_Of_Data(bp, field->dcols - form->curcol);
2657 wmove(form->w, form->currow, form->curcol);
2661 wmove(form->w, form->currow, form->curcol);
2663 myADDNSTR(form->w, bp, (int)(t - bp));
2664 SetStatus(form, _WINDOW_MODIFIED);
2670 /*---------------------------------------------------------------------------
2671 | Facility : libnform
2672 | Function : static int FE_Insert_Character(FORM * form)
2674 | Description : Insert blank character at the cursor position
2676 | Return Values : E_OK
2678 +--------------------------------------------------------------------------*/
2680 FE_Insert_Character(FORM *form)
2682 FIELD *field = form->current;
2683 int result = E_REQUEST_DENIED;
2685 T((T_CALLED("FE_Insert_Character(%p)"), (void *)form));
2686 if (Check_Char(form, field, field->type, (int)C_BLANK,
2687 (TypeArgument *)(field->arg)))
2689 bool There_Is_Room = Is_There_Room_For_A_Char_In_Line(form);
2691 if (There_Is_Room ||
2692 ((Single_Line_Field(field) && Growable(field))))
2694 if (!There_Is_Room && !Field_Grown(field, 1))
2695 result = E_SYSTEM_ERROR;
2698 winsch(form->w, (chtype)C_BLANK);
2699 result = Wrapping_Not_Necessary_Or_Wrapping_Ok(form);
2706 /*---------------------------------------------------------------------------
2707 | Facility : libnform
2708 | Function : static int FE_Insert_Line(FORM * form)
2710 | Description : Insert a blank line at the cursor position
2712 | Return Values : E_OK - success
2713 | E_REQUEST_DENIED - line can not be inserted
2714 +--------------------------------------------------------------------------*/
2716 FE_Insert_Line(FORM *form)
2718 FIELD *field = form->current;
2719 int result = E_REQUEST_DENIED;
2721 T((T_CALLED("FE_Insert_Line(%p)"), (void *)form));
2722 if (Check_Char(form, field,
2723 field->type, (int)C_BLANK, (TypeArgument *)(field->arg)))
2725 bool Maybe_Done = (form->currow != (field->drows - 1)) &&
2726 Is_There_Room_For_A_Line(form);
2728 if (!Single_Line_Field(field) &&
2729 (Maybe_Done || Growable(field)))
2731 if (!Maybe_Done && !Field_Grown(field, 1))
2732 result = E_SYSTEM_ERROR;
2744 /*---------------------------------------------------------------------------
2745 | Facility : libnform
2746 | Function : static int FE_Delete_Character(FORM * form)
2748 | Description : Delete character at the cursor position
2750 | Return Values : E_OK - success
2751 +--------------------------------------------------------------------------*/
2753 FE_Delete_Character(FORM *form)
2755 T((T_CALLED("FE_Delete_Character(%p)"), (void *)form));
2760 /*---------------------------------------------------------------------------
2761 | Facility : libnform
2762 | Function : static int FE_Delete_Previous(FORM * form)
2764 | Description : Delete character before cursor. Again this is a rather
2765 | difficult piece compared to others due to the overloading
2766 | semantics of backspace.
2767 | N.B.: The case of overloaded BS on first field position
2768 | is already handled in the generic routine.
2770 | Return Values : E_OK - success
2771 | E_REQUEST_DENIED - Character can't be deleted
2772 +--------------------------------------------------------------------------*/
2774 FE_Delete_Previous(FORM *form)
2776 FIELD *field = form->current;
2778 T((T_CALLED("FE_Delete_Previous(%p)"), (void *)form));
2779 if (First_Position_In_Current_Field(form))
2780 returnCode(E_REQUEST_DENIED);
2782 if ((--(form->curcol)) < 0)
2784 FIELD_CELL *this_line, *prev_line, *prev_end, *this_end;
2785 int this_row = form->currow;
2788 if (form->status & _OVLMODE)
2789 returnCode(E_REQUEST_DENIED);
2791 prev_line = Address_Of_Row_In_Buffer(field, (form->currow - 1));
2792 this_line = Address_Of_Row_In_Buffer(field, (form->currow));
2793 Synchronize_Buffer(form);
2794 prev_end = After_End_Of_Data(prev_line, field->dcols);
2795 this_end = After_End_Of_Data(this_line, field->dcols);
2796 if ((int)(this_end - this_line) >
2797 (field->cols - (int)(prev_end - prev_line)))
2798 returnCode(E_REQUEST_DENIED);
2799 wmove(form->w, form->currow, form->curcol);
2801 Adjust_Cursor_Position(form, prev_end);
2803 * If we did not really move to the previous line, help the user a
2804 * little. It is however a little inconsistent. Normally, when
2805 * backspacing around the point where text wraps to a new line in a
2806 * multi-line form, we absorb one keystroke for the wrapping point. That
2807 * is consistent with SVr4 forms. However, SVr4 does not allow typing
2808 * into the last column of the field, and requires the user to enter a
2809 * newline to move to the next line. Therefore it can consistently eat
2810 * that keystroke. Since ncurses allows the last column, it wraps
2811 * automatically (given the proper options). But we cannot eat the
2812 * keystroke to back over the wrapping point, since that would put the
2813 * cursor past the end of the form field. In this case, just delete the
2814 * character at the end of the field.
2816 if (form->currow == this_row && this_row > 0)
2819 form->curcol = field->dcols - 1;
2824 wmove(form->w, form->currow, form->curcol);
2825 myADDNSTR(form->w, this_line, (int)(this_end - this_line));
2835 /*---------------------------------------------------------------------------
2836 | Facility : libnform
2837 | Function : static int FE_Delete_Line(FORM * form)
2839 | Description : Delete line at cursor position.
2841 | Return Values : E_OK - success
2842 +--------------------------------------------------------------------------*/
2844 FE_Delete_Line(FORM *form)
2846 T((T_CALLED("FE_Delete_Line(%p)"), (void *)form));
2852 /*---------------------------------------------------------------------------
2853 | Facility : libnform
2854 | Function : static int FE_Delete_Word(FORM * form)
2856 | Description : Delete word at cursor position
2858 | Return Values : E_OK - success
2859 | E_REQUEST_DENIED - failure
2860 +--------------------------------------------------------------------------*/
2862 FE_Delete_Word(FORM *form)
2864 FIELD *field = form->current;
2865 FIELD_CELL *bp = Address_Of_Current_Row_In_Buffer(form);
2866 FIELD_CELL *ep = bp + field->dcols;
2867 FIELD_CELL *cp = bp + form->curcol;
2870 T((T_CALLED("FE_Delete_Word(%p)"), (void *)form));
2871 Synchronize_Buffer(form);
2873 returnCode(E_REQUEST_DENIED); /* not in word */
2875 /* move cursor to begin of word and erase to end of screen-line */
2876 Adjust_Cursor_Position(form,
2877 After_Last_Whitespace_Character(bp, form->curcol));
2878 wmove(form->w, form->currow, form->curcol);
2881 /* skip over word in buffer */
2882 s = Get_First_Whitespace_Character(cp, (int)(ep - cp));
2883 /* to begin of next word */
2884 s = Get_Start_Of_Data(s, (int)(ep - s));
2885 if ((s != cp) && !ISBLANK(*s))
2887 /* copy remaining line to window */
2888 myADDNSTR(form->w, s, (int)(s - After_End_Of_Data(s, (int)(ep - s))));
2893 /*---------------------------------------------------------------------------
2894 | Facility : libnform
2895 | Function : static int FE_Clear_To_End_Of_Line(FORM * form)
2897 | Description : Clear to end of current line.
2899 | Return Values : E_OK - success
2900 +--------------------------------------------------------------------------*/
2902 FE_Clear_To_End_Of_Line(FORM *form)
2904 T((T_CALLED("FE_Clear_To_End_Of_Line(%p)"), (void *)form));
2905 wmove(form->w, form->currow, form->curcol);
2910 /*---------------------------------------------------------------------------
2911 | Facility : libnform
2912 | Function : static int FE_Clear_To_End_Of_Field(FORM * form)
2914 | Description : Clear to end of field.
2916 | Return Values : E_OK - success
2917 +--------------------------------------------------------------------------*/
2919 FE_Clear_To_End_Of_Field(FORM *form)
2921 T((T_CALLED("FE_Clear_To_End_Of_Field(%p)"), (void *)form));
2922 wmove(form->w, form->currow, form->curcol);
2927 /*---------------------------------------------------------------------------
2928 | Facility : libnform
2929 | Function : static int FE_Clear_Field(FORM * form)
2931 | Description : Clear entire field.
2933 | Return Values : E_OK - success
2934 +--------------------------------------------------------------------------*/
2936 FE_Clear_Field(FORM *form)
2938 T((T_CALLED("FE_Clear_Field(%p)"), (void *)form));
2939 form->currow = form->curcol = 0;
2943 /*----------------------------------------------------------------------------
2944 END of Field Editing routines
2945 --------------------------------------------------------------------------*/
2947 /*----------------------------------------------------------------------------
2949 --------------------------------------------------------------------------*/
2951 /*---------------------------------------------------------------------------
2952 | Facility : libnform
2953 | Function : static int EM_Overlay_Mode(FORM * form)
2955 | Description : Switch to overlay mode.
2957 | Return Values : E_OK - success
2958 +--------------------------------------------------------------------------*/
2960 EM_Overlay_Mode(FORM *form)
2962 T((T_CALLED("EM_Overlay_Mode(%p)"), (void *)form));
2963 SetStatus(form, _OVLMODE);
2967 /*---------------------------------------------------------------------------
2968 | Facility : libnform
2969 | Function : static int EM_Insert_Mode(FORM * form)
2971 | Description : Switch to insert mode
2973 | Return Values : E_OK - success
2974 +--------------------------------------------------------------------------*/
2976 EM_Insert_Mode(FORM *form)
2978 T((T_CALLED("EM_Insert_Mode(%p)"), (void *)form));
2979 ClrStatus(form, _OVLMODE);
2983 /*----------------------------------------------------------------------------
2984 END of Edit Mode routines
2985 --------------------------------------------------------------------------*/
2987 /*----------------------------------------------------------------------------
2988 Helper routines for Choice Requests
2989 --------------------------------------------------------------------------*/
2991 /*---------------------------------------------------------------------------
2992 | Facility : libnform
2993 | Function : static bool Next_Choice(FORM * form,
2996 | TypeArgument *argp)
2998 | Description : Get the next field choice. For linked types this is
3001 | Return Values : TRUE - next choice successfully retrieved
3002 | FALSE - couldn't retrieve next choice
3003 +--------------------------------------------------------------------------*/
3005 Next_Choice(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
3007 if (!typ || !(typ->status & _HAS_CHOICE))
3010 if (typ->status & _LINKED_TYPE)
3014 Next_Choice(form, typ->left, field, argp->left) ||
3015 Next_Choice(form, typ->right, field, argp->right));
3019 #if NCURSES_INTEROP_FUNCS
3020 assert(typ->enum_next.onext);
3021 if (typ->status & _GENERIC)
3022 return typ->enum_next.gnext(form, field, (void *)argp);
3024 return typ->enum_next.onext(field, (void *)argp);
3027 return typ->next(field, (void *)argp);
3032 /*---------------------------------------------------------------------------
3033 | Facility : libnform
3034 | Function : static bool Previous_Choice(FORM * form,
3037 | TypeArgument *argp)
3039 | Description : Get the previous field choice. For linked types this
3040 | is done recursively.
3042 | Return Values : TRUE - previous choice successfully retrieved
3043 | FALSE - couldn't retrieve previous choice
3044 +--------------------------------------------------------------------------*/
3046 Previous_Choice(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
3048 if (!typ || !(typ->status & _HAS_CHOICE))
3051 if (typ->status & _LINKED_TYPE)
3055 Previous_Choice(form, typ->left, field, argp->left) ||
3056 Previous_Choice(form, typ->right, field, argp->right));
3060 #if NCURSES_INTEROP_FUNCS
3061 assert(typ->enum_prev.oprev);
3062 if (typ->status & _GENERIC)
3063 return typ->enum_prev.gprev(form, field, (void *)argp);
3065 return typ->enum_prev.oprev(field, (void *)argp);
3068 return typ->prev(field, (void *)argp);
3072 /*----------------------------------------------------------------------------
3073 End of Helper routines for Choice Requests
3074 --------------------------------------------------------------------------*/
3076 /*----------------------------------------------------------------------------
3077 Routines for Choice Requests
3078 --------------------------------------------------------------------------*/
3080 /*---------------------------------------------------------------------------
3081 | Facility : libnform
3082 | Function : static int CR_Next_Choice(FORM * form)
3084 | Description : Get the next field choice.
3086 | Return Values : E_OK - success
3087 | E_REQUEST_DENIED - next choice couldn't be retrieved
3088 +--------------------------------------------------------------------------*/
3090 CR_Next_Choice(FORM *form)
3092 FIELD *field = form->current;
3094 T((T_CALLED("CR_Next_Choice(%p)"), (void *)form));
3095 Synchronize_Buffer(form);
3096 returnCode((Next_Choice(form, field->type, field, (TypeArgument *)(field->arg)))
3098 : E_REQUEST_DENIED);
3101 /*---------------------------------------------------------------------------
3102 | Facility : libnform
3103 | Function : static int CR_Previous_Choice(FORM * form)
3105 | Description : Get the previous field choice.
3107 | Return Values : E_OK - success
3108 | E_REQUEST_DENIED - prev. choice couldn't be retrieved
3109 +--------------------------------------------------------------------------*/
3111 CR_Previous_Choice(FORM *form)
3113 FIELD *field = form->current;
3115 T((T_CALLED("CR_Previous_Choice(%p)"), (void *)form));
3116 Synchronize_Buffer(form);
3117 returnCode((Previous_Choice(form, field->type, field, (TypeArgument *)(field->arg)))
3119 : E_REQUEST_DENIED);
3121 /*----------------------------------------------------------------------------
3122 End of Routines for Choice Requests
3123 --------------------------------------------------------------------------*/
3125 /*----------------------------------------------------------------------------
3126 Helper routines for Field Validations.
3127 --------------------------------------------------------------------------*/
3129 /*---------------------------------------------------------------------------
3130 | Facility : libnform
3131 | Function : static bool Check_Field(FORM* form,
3134 | TypeArgument * argp)
3136 | Description : Check the field according to its fieldtype and its
3137 | actual arguments. For linked fieldtypes this is done
3140 | Return Values : TRUE - field is valid
3141 | FALSE - field is invalid.
3142 +--------------------------------------------------------------------------*/
3144 Check_Field(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
3148 if (Field_Has_Option(field, O_NULLOK))
3150 FIELD_CELL *bp = field->buf;
3153 while (ISBLANK(*bp))
3157 if (CharOf(*bp) == 0)
3161 if (typ->status & _LINKED_TYPE)
3165 Check_Field(form, typ->left, field, argp->left) ||
3166 Check_Field(form, typ->right, field, argp->right));
3170 #if NCURSES_INTEROP_FUNCS
3171 if (typ->fieldcheck.ofcheck)
3173 if (typ->status & _GENERIC)
3174 return typ->fieldcheck.gfcheck(form, field, (void *)argp);
3176 return typ->fieldcheck.ofcheck(field, (void *)argp);
3180 return typ->fcheck(field, (void *)argp);
3187 /*---------------------------------------------------------------------------
3188 | Facility : libnform
3189 | Function : bool _nc_Internal_Validation(FORM * form )
3191 | Description : Validate the current field of the form.
3193 | Return Values : TRUE - field is valid
3194 | FALSE - field is invalid
3195 +--------------------------------------------------------------------------*/
3196 NCURSES_EXPORT(bool)
3197 _nc_Internal_Validation(FORM *form)
3201 field = form->current;
3203 Synchronize_Buffer(form);
3204 if ((form->status & _FCHECK_REQUIRED) ||
3205 (!(Field_Has_Option(field, O_PASSOK))))
3207 if (!Check_Field(form, field->type, field, (TypeArgument *)(field->arg)))
3209 ClrStatus(form, _FCHECK_REQUIRED);
3210 SetStatus(field, _CHANGED);
3211 Synchronize_Linked_Fields(field);
3215 /*----------------------------------------------------------------------------
3216 End of Helper routines for Field Validations.
3217 --------------------------------------------------------------------------*/
3219 /*----------------------------------------------------------------------------
3220 Routines for Field Validation.
3221 --------------------------------------------------------------------------*/
3223 /*---------------------------------------------------------------------------
3224 | Facility : libnform
3225 | Function : static int FV_Validation(FORM * form)
3227 | Description : Validate the current field of the form.
3229 | Return Values : E_OK - field valid
3230 | E_INVALID_FIELD - field not valid
3231 +--------------------------------------------------------------------------*/
3233 FV_Validation(FORM *form)
3235 T((T_CALLED("FV_Validation(%p)"), (void *)form));
3236 if (_nc_Internal_Validation(form))
3239 returnCode(E_INVALID_FIELD);
3241 /*----------------------------------------------------------------------------
3242 End of routines for Field Validation.
3243 --------------------------------------------------------------------------*/
3245 /*----------------------------------------------------------------------------
3246 Helper routines for Inter-Field Navigation
3247 --------------------------------------------------------------------------*/
3249 /*---------------------------------------------------------------------------
3250 | Facility : libnform
3251 | Function : static FIELD *Next_Field_On_Page(FIELD * field)
3253 | Description : Get the next field after the given field on the current
3254 | page. The order of fields is the one defined by the
3255 | fields array. Only visible and active fields are
3258 | Return Values : Pointer to the next field.
3259 +--------------------------------------------------------------------------*/
3260 NCURSES_INLINE static FIELD *
3261 Next_Field_On_Page(FIELD *field)
3263 FORM *form = field->form;
3264 FIELD **field_on_page = &form->field[field->index];
3265 FIELD **first_on_page = &form->field[form->page[form->curpage].pmin];
3266 FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3271 (field_on_page == last_on_page) ? first_on_page : field_on_page + 1;
3272 if (Field_Is_Selectable(*field_on_page))
3275 while (field != (*field_on_page));
3276 return (*field_on_page);
3279 /*---------------------------------------------------------------------------
3280 | Facility : libnform
3281 | Function : FIELD* _nc_First_Active_Field(FORM * form)
3283 | Description : Get the first active field on the current page,
3284 | if there are such. If there are none, get the first
3285 | visible field on the page. If there are also none,
3286 | we return the first field on page and hope the best.
3288 | Return Values : Pointer to calculated field.
3289 +--------------------------------------------------------------------------*/
3290 NCURSES_EXPORT(FIELD *)
3291 _nc_First_Active_Field(FORM *form)
3293 FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3294 FIELD *proposed = Next_Field_On_Page(*last_on_page);
3296 if (proposed == *last_on_page)
3298 /* there might be the special situation, where there is no
3299 active and visible field on the current page. We then select
3300 the first visible field on this readonly page
3302 if (Field_Is_Not_Selectable(proposed))
3304 FIELD **field = &form->field[proposed->index];
3305 FIELD **first = &form->field[form->page[form->curpage].pmin];
3309 field = (field == last_on_page) ? first : field + 1;
3310 if (Field_Has_Option(*field, O_VISIBLE))
3313 while (proposed != (*field));
3317 if ((proposed == *last_on_page) &&
3318 !((unsigned)proposed->opts & O_VISIBLE))
3320 /* This means, there is also no visible field on the page.
3321 So we propose the first one and hope the very best...
3322 Some very clever user has designed a readonly and invisible
3332 /*---------------------------------------------------------------------------
3333 | Facility : libnform
3334 | Function : static FIELD *Previous_Field_On_Page(FIELD * field)
3336 | Description : Get the previous field before the given field on the
3337 | current page. The order of fields is the one defined by
3338 | the fields array. Only visible and active fields are
3341 | Return Values : Pointer to the previous field.
3342 +--------------------------------------------------------------------------*/
3343 NCURSES_INLINE static FIELD *
3344 Previous_Field_On_Page(FIELD *field)
3346 FORM *form = field->form;
3347 FIELD **field_on_page = &form->field[field->index];
3348 FIELD **first_on_page = &form->field[form->page[form->curpage].pmin];
3349 FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3354 (field_on_page == first_on_page) ? last_on_page : field_on_page - 1;
3355 if (Field_Is_Selectable(*field_on_page))
3358 while (field != (*field_on_page));
3360 return (*field_on_page);
3363 /*---------------------------------------------------------------------------
3364 | Facility : libnform
3365 | Function : static FIELD *Sorted_Next_Field(FIELD * field)
3367 | Description : Get the next field after the given field on the current
3368 | page. The order of fields is the one defined by the
3369 | (row,column) geometry, rows are major.
3371 | Return Values : Pointer to the next field.
3372 +--------------------------------------------------------------------------*/
3373 NCURSES_INLINE static FIELD *
3374 Sorted_Next_Field(FIELD *field)
3376 FIELD *field_on_page = field;
3380 field_on_page = field_on_page->snext;
3381 if (Field_Is_Selectable(field_on_page))
3384 while (field_on_page != field);
3386 return (field_on_page);
3389 /*---------------------------------------------------------------------------
3390 | Facility : libnform
3391 | Function : static FIELD *Sorted_Previous_Field(FIELD * field)
3393 | Description : Get the previous field before the given field on the
3394 | current page. The order of fields is the one defined
3395 | by the (row,column) geometry, rows are major.
3397 | Return Values : Pointer to the previous field.
3398 +--------------------------------------------------------------------------*/
3399 NCURSES_INLINE static FIELD *
3400 Sorted_Previous_Field(FIELD *field)
3402 FIELD *field_on_page = field;
3406 field_on_page = field_on_page->sprev;
3407 if (Field_Is_Selectable(field_on_page))
3410 while (field_on_page != field);
3412 return (field_on_page);
3415 /*---------------------------------------------------------------------------
3416 | Facility : libnform
3417 | Function : static FIELD *Left_Neighbor_Field(FIELD * field)
3419 | Description : Get the left neighbor of the field on the same line
3420 | and the same page. Cycles through the line.
3422 | Return Values : Pointer to left neighbor field.
3423 +--------------------------------------------------------------------------*/
3424 NCURSES_INLINE static FIELD *
3425 Left_Neighbor_Field(FIELD *field)
3427 FIELD *field_on_page = field;
3429 /* For a field that has really a left neighbor, the while clause
3430 immediately fails and the loop is left, positioned at the right
3431 neighbor. Otherwise we cycle backwards through the sorted field list
3432 until we enter the same line (from the right end).
3436 field_on_page = Sorted_Previous_Field(field_on_page);
3438 while (field_on_page->frow != field->frow);
3440 return (field_on_page);
3443 /*---------------------------------------------------------------------------
3444 | Facility : libnform
3445 | Function : static FIELD *Right_Neighbor_Field(FIELD * field)
3447 | Description : Get the right neighbor of the field on the same line
3448 | and the same page.
3450 | Return Values : Pointer to right neighbor field.
3451 +--------------------------------------------------------------------------*/
3452 NCURSES_INLINE static FIELD *
3453 Right_Neighbor_Field(FIELD *field)
3455 FIELD *field_on_page = field;
3457 /* See the comments on Left_Neighbor_Field to understand how it works */
3460 field_on_page = Sorted_Next_Field(field_on_page);
3462 while (field_on_page->frow != field->frow);
3464 return (field_on_page);
3467 /*---------------------------------------------------------------------------
3468 | Facility : libnform
3469 | Function : static FIELD *Upper_Neighbor_Field(FIELD * field)
3471 | Description : Because of the row-major nature of sorting the fields,
3472 | it is more difficult to define whats the upper neighbor
3473 | field really means. We define that it must be on a
3474 | 'previous' line (cyclic order!) and is the rightmost
3475 | field laying on the left side of the given field. If
3476 | this set is empty, we take the first field on the line.
3478 | Return Values : Pointer to the upper neighbor field.
3479 +--------------------------------------------------------------------------*/
3481 Upper_Neighbor_Field(FIELD *field)
3483 FIELD *field_on_page = field;
3484 int frow = field->frow;
3485 int fcol = field->fcol;
3487 /* Walk back to the 'previous' line. The second term in the while clause
3488 just guarantees that we stop if we cycled through the line because
3489 there might be no 'previous' line if the page has just one line.
3493 field_on_page = Sorted_Previous_Field(field_on_page);
3495 while (field_on_page->frow == frow && field_on_page->fcol != fcol);
3497 if (field_on_page->frow != frow)
3499 /* We really found a 'previous' line. We are positioned at the
3500 rightmost field on this line */
3501 frow = field_on_page->frow;
3503 /* We walk to the left as long as we are really right of the
3505 while (field_on_page->frow == frow && field_on_page->fcol > fcol)
3506 field_on_page = Sorted_Previous_Field(field_on_page);
3508 /* If we wrapped, just go to the right which is the first field on
3510 if (field_on_page->frow != frow)
3511 field_on_page = Sorted_Next_Field(field_on_page);
3514 return (field_on_page);
3517 /*---------------------------------------------------------------------------
3518 | Facility : libnform
3519 | Function : static FIELD *Down_Neighbor_Field(FIELD * field)
3521 | Description : Because of the row-major nature of sorting the fields,
3522 | its more difficult to define whats the down neighbor
3523 | field really means. We define that it must be on a
3524 | 'next' line (cyclic order!) and is the leftmost
3525 | field laying on the right side of the given field. If
3526 | this set is empty, we take the last field on the line.
3528 | Return Values : Pointer to the upper neighbor field.
3529 +--------------------------------------------------------------------------*/
3531 Down_Neighbor_Field(FIELD *field)
3533 FIELD *field_on_page = field;
3534 int frow = field->frow;
3535 int fcol = field->fcol;
3537 /* Walk forward to the 'next' line. The second term in the while clause
3538 just guarantees that we stop if we cycled through the line because
3539 there might be no 'next' line if the page has just one line.
3543 field_on_page = Sorted_Next_Field(field_on_page);
3545 while (field_on_page->frow == frow && field_on_page->fcol != fcol);
3547 if (field_on_page->frow != frow)
3549 /* We really found a 'next' line. We are positioned at the rightmost
3550 field on this line */
3551 frow = field_on_page->frow;
3553 /* We walk to the right as long as we are really left of the
3555 while (field_on_page->frow == frow && field_on_page->fcol < fcol)
3556 field_on_page = Sorted_Next_Field(field_on_page);
3558 /* If we wrapped, just go to the left which is the last field on
3560 if (field_on_page->frow != frow)
3561 field_on_page = Sorted_Previous_Field(field_on_page);
3564 return (field_on_page);
3567 /*----------------------------------------------------------------------------
3568 Inter-Field Navigation routines
3569 --------------------------------------------------------------------------*/
3571 /*---------------------------------------------------------------------------
3572 | Facility : libnform
3573 | Function : static int Inter_Field_Navigation(
3574 | int (* const fct) (FORM *),
3577 | Description : Generic behavior for changing the current field, the
3578 | field is left and a new field is entered. So the field
3579 | must be validated and the field init/term hooks must
3582 | Return Values : E_OK - success
3583 | E_INVALID_FIELD - field is invalid
3584 | some other - error from subordinate call
3585 +--------------------------------------------------------------------------*/
3587 Inter_Field_Navigation(int (*const fct) (FORM *), FORM *form)
3591 if (!_nc_Internal_Validation(form))
3592 res = E_INVALID_FIELD;
3595 Call_Hook(form, fieldterm);
3597 Call_Hook(form, fieldinit);
3602 /*---------------------------------------------------------------------------
3603 | Facility : libnform
3604 | Function : static int FN_Next_Field(FORM * form)
3606 | Description : Move to the next field on the current page of the form
3608 | Return Values : E_OK - success
3609 | != E_OK - error from subordinate call
3610 +--------------------------------------------------------------------------*/
3612 FN_Next_Field(FORM *form)
3614 T((T_CALLED("FN_Next_Field(%p)"), (void *)form));
3615 returnCode(_nc_Set_Current_Field(form,
3616 Next_Field_On_Page(form->current)));
3619 /*---------------------------------------------------------------------------
3620 | Facility : libnform
3621 | Function : static int FN_Previous_Field(FORM * form)
3623 | Description : Move to the previous field on the current page of the
3626 | Return Values : E_OK - success
3627 | != E_OK - error from subordinate call
3628 +--------------------------------------------------------------------------*/
3630 FN_Previous_Field(FORM *form)
3632 T((T_CALLED("FN_Previous_Field(%p)"), (void *)form));
3633 returnCode(_nc_Set_Current_Field(form,
3634 Previous_Field_On_Page(form->current)));
3637 /*---------------------------------------------------------------------------
3638 | Facility : libnform
3639 | Function : static int FN_First_Field(FORM * form)
3641 | Description : Move to the first field on the current page of the form
3643 | Return Values : E_OK - success
3644 | != E_OK - error from subordinate call
3645 +--------------------------------------------------------------------------*/
3647 FN_First_Field(FORM *form)
3649 T((T_CALLED("FN_First_Field(%p)"), (void *)form));
3650 returnCode(_nc_Set_Current_Field(form,
3651 Next_Field_On_Page(form->field[form->page[form->curpage].pmax])));
3654 /*---------------------------------------------------------------------------
3655 | Facility : libnform
3656 | Function : static int FN_Last_Field(FORM * form)
3658 | Description : Move to the last field on the current page of the form
3660 | Return Values : E_OK - success
3661 | != E_OK - error from subordinate call
3662 +--------------------------------------------------------------------------*/
3664 FN_Last_Field(FORM *form)
3666 T((T_CALLED("FN_Last_Field(%p)"), (void *)form));
3668 _nc_Set_Current_Field(form,
3669 Previous_Field_On_Page(form->field[form->page[form->curpage].pmin])));
3672 /*---------------------------------------------------------------------------
3673 | Facility : libnform
3674 | Function : static int FN_Sorted_Next_Field(FORM * form)
3676 | Description : Move to the sorted next field on the current page
3679 | Return Values : E_OK - success
3680 | != E_OK - error from subordinate call
3681 +--------------------------------------------------------------------------*/
3683 FN_Sorted_Next_Field(FORM *form)
3685 T((T_CALLED("FN_Sorted_Next_Field(%p)"), (void *)form));
3686 returnCode(_nc_Set_Current_Field(form,
3687 Sorted_Next_Field(form->current)));
3690 /*---------------------------------------------------------------------------
3691 | Facility : libnform
3692 | Function : static int FN_Sorted_Previous_Field(FORM * form)
3694 | Description : Move to the sorted previous field on the current page
3697 | Return Values : E_OK - success
3698 | != E_OK - error from subordinate call
3699 +--------------------------------------------------------------------------*/
3701 FN_Sorted_Previous_Field(FORM *form)
3703 T((T_CALLED("FN_Sorted_Previous_Field(%p)"), (void *)form));
3704 returnCode(_nc_Set_Current_Field(form,
3705 Sorted_Previous_Field(form->current)));
3708 /*---------------------------------------------------------------------------
3709 | Facility : libnform
3710 | Function : static int FN_Sorted_First_Field(FORM * form)
3712 | Description : Move to the sorted first field on the current page
3715 | Return Values : E_OK - success
3716 | != E_OK - error from subordinate call
3717 +--------------------------------------------------------------------------*/
3719 FN_Sorted_First_Field(FORM *form)
3721 T((T_CALLED("FN_Sorted_First_Field(%p)"), (void *)form));
3722 returnCode(_nc_Set_Current_Field(form,
3723 Sorted_Next_Field(form->field[form->page[form->curpage].smax])));
3726 /*---------------------------------------------------------------------------
3727 | Facility : libnform
3728 | Function : static int FN_Sorted_Last_Field(FORM * form)
3730 | Description : Move to the sorted last field on the current page
3733 | Return Values : E_OK - success
3734 | != E_OK - error from subordinate call
3735 +--------------------------------------------------------------------------*/
3737 FN_Sorted_Last_Field(FORM *form)
3739 T((T_CALLED("FN_Sorted_Last_Field(%p)"), (void *)form));
3740 returnCode(_nc_Set_Current_Field(form,
3741 Sorted_Previous_Field(form->field[form->page[form->curpage].smin])));
3744 /*---------------------------------------------------------------------------
3745 | Facility : libnform
3746 | Function : static int FN_Left_Field(FORM * form)
3748 | Description : Get the field on the left of the current field on the
3749 | same line and the same page. Cycles through the line.
3751 | Return Values : E_OK - success
3752 | != E_OK - error from subordinate call
3753 +--------------------------------------------------------------------------*/
3755 FN_Left_Field(FORM *form)
3757 T((T_CALLED("FN_Left_Field(%p)"), (void *)form));
3758 returnCode(_nc_Set_Current_Field(form,
3759 Left_Neighbor_Field(form->current)));
3762 /*---------------------------------------------------------------------------
3763 | Facility : libnform
3764 | Function : static int FN_Right_Field(FORM * form)
3766 | Description : Get the field on the right of the current field on the
3767 | same line and the same page. Cycles through the line.
3769 | Return Values : E_OK - success
3770 | != E_OK - error from subordinate call
3771 +--------------------------------------------------------------------------*/
3773 FN_Right_Field(FORM *form)
3775 T((T_CALLED("FN_Right_Field(%p)"), (void *)form));
3776 returnCode(_nc_Set_Current_Field(form,
3777 Right_Neighbor_Field(form->current)));
3780 /*---------------------------------------------------------------------------
3781 | Facility : libnform
3782 | Function : static int FN_Up_Field(FORM * form)
3784 | Description : Get the upper neighbor of the current field. This
3785 | cycles through the page. See the comments of the
3786 | Upper_Neighbor_Field function to understand how
3787 | 'upper' is defined.
3789 | Return Values : E_OK - success
3790 | != E_OK - error from subordinate call
3791 +--------------------------------------------------------------------------*/
3793 FN_Up_Field(FORM *form)
3795 T((T_CALLED("FN_Up_Field(%p)"), (void *)form));
3796 returnCode(_nc_Set_Current_Field(form,
3797 Upper_Neighbor_Field(form->current)));
3800 /*---------------------------------------------------------------------------
3801 | Facility : libnform
3802 | Function : static int FN_Down_Field(FORM * form)
3804 | Description : Get the down neighbor of the current field. This
3805 | cycles through the page. See the comments of the
3806 | Down_Neighbor_Field function to understand how
3807 | 'down' is defined.
3809 | Return Values : E_OK - success
3810 | != E_OK - error from subordinate call
3811 +--------------------------------------------------------------------------*/
3813 FN_Down_Field(FORM *form)
3815 T((T_CALLED("FN_Down_Field(%p)"), (void *)form));
3816 returnCode(_nc_Set_Current_Field(form,
3817 Down_Neighbor_Field(form->current)));
3819 /*----------------------------------------------------------------------------
3820 END of Field Navigation routines
3821 --------------------------------------------------------------------------*/
3823 /*----------------------------------------------------------------------------
3824 Helper routines for Page Navigation
3825 --------------------------------------------------------------------------*/
3827 /*---------------------------------------------------------------------------
3828 | Facility : libnform
3829 | Function : int _nc_Set_Form_Page(FORM * form,
3833 | Description : Make the given page number the current page and make
3834 | the given field the current field on the page. If
3835 | for the field NULL is given, make the first field on
3836 | the page the current field. The routine acts only
3837 | if the requested page is not the current page.
3839 | Return Values : E_OK - success
3840 | != E_OK - error from subordinate call
3841 | E_BAD_ARGUMENT - invalid field pointer
3842 | E_SYSTEM_ERROR - some severe basic error
3843 +--------------------------------------------------------------------------*/
3845 _nc_Set_Form_Page(FORM *form, int page, FIELD *field)
3849 if ((form->curpage != page))
3851 FIELD *last_field, *field_on_page;
3853 werase(Get_Form_Window(form));
3854 form->curpage = (short)page;
3855 last_field = field_on_page = form->field[form->page[page].smin];
3858 if ((unsigned)field_on_page->opts & O_VISIBLE)
3859 if ((res = Display_Field(field_on_page)) != E_OK)
3861 field_on_page = field_on_page->snext;
3863 while (field_on_page != last_field);
3866 res = _nc_Set_Current_Field(form, field);
3868 /* N.B.: we don't encapsulate this by Inter_Field_Navigation(),
3869 because this is already executed in a page navigation
3870 context that contains field navigation
3872 res = FN_First_Field(form);
3877 /*---------------------------------------------------------------------------
3878 | Facility : libnform
3879 | Function : static int Next_Page_Number(const FORM * form)
3881 | Description : Calculate the page number following the current page
3882 | number. This cycles if the highest page number is
3885 | Return Values : The next page number
3886 +--------------------------------------------------------------------------*/
3887 NCURSES_INLINE static int
3888 Next_Page_Number(const FORM *form)
3890 return (form->curpage + 1) % form->maxpage;
3893 /*---------------------------------------------------------------------------
3894 | Facility : libnform
3895 | Function : static int Previous_Page_Number(const FORM * form)
3897 | Description : Calculate the page number before the current page
3898 | number. This cycles if the first page number is
3901 | Return Values : The previous page number
3902 +--------------------------------------------------------------------------*/
3903 NCURSES_INLINE static int
3904 Previous_Page_Number(const FORM *form)
3906 return (form->curpage != 0 ? form->curpage - 1 : form->maxpage - 1);
3909 /*----------------------------------------------------------------------------
3910 Page Navigation routines
3911 --------------------------------------------------------------------------*/
3913 /*---------------------------------------------------------------------------
3914 | Facility : libnform
3915 | Function : static int Page_Navigation(
3916 | int (* const fct) (FORM *),
3919 | Description : Generic behavior for changing a page. This means
3920 | that the field is left and a new field is entered.
3921 | So the field must be validated and the field init/term
3922 | hooks must be called. Because also the page is changed,
3923 | the forms init/term hooks must be called also.
3925 | Return Values : E_OK - success
3926 | E_INVALID_FIELD - field is invalid
3927 | some other - error from subordinate call
3928 +--------------------------------------------------------------------------*/
3930 Page_Navigation(int (*const fct) (FORM *), FORM *form)
3934 if (!_nc_Internal_Validation(form))
3935 res = E_INVALID_FIELD;
3938 Call_Hook(form, fieldterm);
3939 Call_Hook(form, formterm);
3941 Call_Hook(form, forminit);
3942 Call_Hook(form, fieldinit);
3947 /*---------------------------------------------------------------------------
3948 | Facility : libnform
3949 | Function : static int PN_Next_Page(FORM * form)
3951 | Description : Move to the next page of the form
3953 | Return Values : E_OK - success
3954 | != E_OK - error from subordinate call
3955 +--------------------------------------------------------------------------*/
3957 PN_Next_Page(FORM *form)
3959 T((T_CALLED("PN_Next_Page(%p)"), (void *)form));
3960 returnCode(_nc_Set_Form_Page(form, Next_Page_Number(form), (FIELD *)0));
3963 /*---------------------------------------------------------------------------
3964 | Facility : libnform
3965 | Function : static int PN_Previous_Page(FORM * form)
3967 | Description : Move to the previous page of the form
3969 | Return Values : E_OK - success
3970 | != E_OK - error from subordinate call
3971 +--------------------------------------------------------------------------*/
3973 PN_Previous_Page(FORM *form)
3975 T((T_CALLED("PN_Previous_Page(%p)"), (void *)form));
3976 returnCode(_nc_Set_Form_Page(form, Previous_Page_Number(form), (FIELD *)0));
3979 /*---------------------------------------------------------------------------
3980 | Facility : libnform
3981 | Function : static int PN_First_Page(FORM * form)
3983 | Description : Move to the first page of the form
3985 | Return Values : E_OK - success
3986 | != E_OK - error from subordinate call
3987 +--------------------------------------------------------------------------*/
3989 PN_First_Page(FORM *form)
3991 T((T_CALLED("PN_First_Page(%p)"), (void *)form));
3992 returnCode(_nc_Set_Form_Page(form, 0, (FIELD *)0));
3995 /*---------------------------------------------------------------------------
3996 | Facility : libnform
3997 | Function : static int PN_Last_Page(FORM * form)
3999 | Description : Move to the last page of the form
4001 | Return Values : E_OK - success
4002 | != E_OK - error from subordinate call
4003 +--------------------------------------------------------------------------*/
4005 PN_Last_Page(FORM *form)
4007 T((T_CALLED("PN_Last_Page(%p)"), (void *)form));
4008 returnCode(_nc_Set_Form_Page(form, form->maxpage - 1, (FIELD *)0));
4011 /*----------------------------------------------------------------------------
4012 END of Field Navigation routines
4013 --------------------------------------------------------------------------*/
4015 /*----------------------------------------------------------------------------
4016 Helper routines for the core form driver.
4017 --------------------------------------------------------------------------*/
4019 # if USE_WIDEC_SUPPORT
4020 /*---------------------------------------------------------------------------
4021 | Facility : libnform
4022 | Function : static int Data_Entry_w(FORM * form, wchar_t c)
4024 | Description : Enter the wide character c into at the current
4025 | position of the current field of the form.
4027 | Return Values : E_OK - success
4028 | E_REQUEST_DENIED - driver could not process the request
4030 +--------------------------------------------------------------------------*/
4032 Data_Entry_w(FORM *form, wchar_t c)
4034 FIELD *field = form->current;
4035 int result = E_REQUEST_DENIED;
4037 T((T_CALLED("Data_Entry(%p,%s)"), (void *)form, _tracechtype((chtype)c)));
4038 if ((Field_Has_Option(field, O_EDIT))
4039 #if FIX_FORM_INACTIVE_BUG
4040 && (Field_Has_Option(field, O_ACTIVE))
4049 setcchar(&temp_ch, given, 0, 0, (void *)0);
4050 if ((Field_Has_Option(field, O_BLANK)) &&
4051 First_Position_In_Current_Field(form) &&
4052 !(form->status & _FCHECK_REQUIRED) &&
4053 !(form->status & _WINDOW_MODIFIED))
4056 if (form->status & _OVLMODE)
4058 wadd_wch(form->w, &temp_ch);
4063 bool There_Is_Room = Is_There_Room_For_A_Char_In_Line(form);
4065 if (!(There_Is_Room ||
4066 ((Single_Line_Field(field) && Growable(field)))))
4067 RETURN(E_REQUEST_DENIED);
4069 if (!There_Is_Room && !Field_Grown(field, 1))
4070 RETURN(E_SYSTEM_ERROR);
4072 wins_wch(form->w, &temp_ch);
4075 if ((result = Wrapping_Not_Necessary_Or_Wrapping_Ok(form)) == E_OK)
4077 bool End_Of_Field = (((field->drows - 1) == form->currow) &&
4078 ((field->dcols - 1) == form->curcol));
4080 form->status |= _WINDOW_MODIFIED;
4081 if (End_Of_Field && !Growable(field) && (Field_Has_Option(field, O_AUTOSKIP)))
4082 result = Inter_Field_Navigation(FN_Next_Field, form);
4085 if (End_Of_Field && Growable(field) && !Field_Grown(field, 1))
4086 result = E_SYSTEM_ERROR;
4090 * We have just added a byte to the form field. It may have
4091 * been part of a multibyte character. If it was, the
4092 * addch_used field is nonzero and we should not try to move
4095 if (WINDOW_EXT(form->w, addch_used) == 0)
4096 IFN_Next_Character(form);
4107 /*---------------------------------------------------------------------------
4108 | Facility : libnform
4109 | Function : static int Data_Entry(FORM * form,int c)
4111 | Description : Enter character c into at the current position of the
4112 | current field of the form.
4114 | Return Values : E_OK - success
4115 | E_REQUEST_DENIED - driver could not process the request
4117 +--------------------------------------------------------------------------*/
4119 Data_Entry(FORM *form, int c)
4121 FIELD *field = form->current;
4122 int result = E_REQUEST_DENIED;
4124 T((T_CALLED("Data_Entry(%p,%s)"), (void *)form, _tracechtype((chtype)c)));
4125 if ((Field_Has_Option(field, O_EDIT))
4126 #if FIX_FORM_INACTIVE_BUG
4127 && (Field_Has_Option(field, O_ACTIVE))
4131 if ((Field_Has_Option(field, O_BLANK)) &&
4132 First_Position_In_Current_Field(form) &&
4133 !(form->status & _FCHECK_REQUIRED) &&
4134 !(form->status & _WINDOW_MODIFIED))
4137 if (form->status & _OVLMODE)
4139 waddch(form->w, (chtype)c);
4144 bool There_Is_Room = Is_There_Room_For_A_Char_In_Line(form);
4146 if (!(There_Is_Room ||
4147 ((Single_Line_Field(field) && Growable(field)))))
4148 RETURN(E_REQUEST_DENIED);
4150 if (!There_Is_Room && !Field_Grown(field, 1))
4151 RETURN(E_SYSTEM_ERROR);
4153 winsch(form->w, (chtype)c);
4156 if ((result = Wrapping_Not_Necessary_Or_Wrapping_Ok(form)) == E_OK)
4158 bool End_Of_Field = (((field->drows - 1) == form->currow) &&
4159 ((field->dcols - 1) == form->curcol));
4161 SetStatus(form, _WINDOW_MODIFIED);
4162 if (End_Of_Field && !Growable(field) && (Field_Has_Option(field, O_AUTOSKIP)))
4163 result = Inter_Field_Navigation(FN_Next_Field, form);
4166 if (End_Of_Field && Growable(field) && !Field_Grown(field, 1))
4167 result = E_SYSTEM_ERROR;
4170 #if USE_WIDEC_SUPPORT
4172 * We have just added a byte to the form field. It may have
4173 * been part of a multibyte character. If it was, the
4174 * addch_used field is nonzero and we should not try to move
4177 if (WINDOW_EXT(form->w, addch_used) == 0)
4178 IFN_Next_Character(form);
4180 IFN_Next_Character(form);
4190 /* Structure to describe the binding of a request code to a function.
4191 The member keycode codes the request value as well as the generic
4192 routine to use for the request. The code for the generic routine
4193 is coded in the upper 16 Bits while the request code is coded in
4196 In terms of C++ you might think of a request as a class with a
4197 virtual method "perform". The different types of request are
4198 derived from this base class and overload (or not) the base class
4199 implementation of perform.
4203 int keycode; /* must be at least 32 bit: hi:mode, lo: key */
4204 int (*cmd) (FORM *); /* low level driver routine for this key */
4208 /* You may see this is the class-id of the request type class */
4209 #define ID_PN (0x00000000) /* Page navigation */
4210 #define ID_FN (0x00010000) /* Inter-Field navigation */
4211 #define ID_IFN (0x00020000) /* Intra-Field navigation */
4212 #define ID_VSC (0x00030000) /* Vertical Scrolling */
4213 #define ID_HSC (0x00040000) /* Horizontal Scrolling */
4214 #define ID_FE (0x00050000) /* Field Editing */
4215 #define ID_EM (0x00060000) /* Edit Mode */
4216 #define ID_FV (0x00070000) /* Field Validation */
4217 #define ID_CH (0x00080000) /* Choice */
4218 #define ID_Mask (0xffff0000)
4219 #define Key_Mask (0x0000ffff)
4220 #define ID_Shft (16)
4222 /* This array holds all the Binding Infos */
4224 static const Binding_Info bindings[MAX_FORM_COMMAND - MIN_FORM_COMMAND + 1] =
4226 { REQ_NEXT_PAGE |ID_PN ,PN_Next_Page},
4227 { REQ_PREV_PAGE |ID_PN ,PN_Previous_Page},
4228 { REQ_FIRST_PAGE |ID_PN ,PN_First_Page},
4229 { REQ_LAST_PAGE |ID_PN ,PN_Last_Page},
4231 { REQ_NEXT_FIELD |ID_FN ,FN_Next_Field},
4232 { REQ_PREV_FIELD |ID_FN ,FN_Previous_Field},
4233 { REQ_FIRST_FIELD |ID_FN ,FN_First_Field},
4234 { REQ_LAST_FIELD |ID_FN ,FN_Last_Field},
4235 { REQ_SNEXT_FIELD |ID_FN ,FN_Sorted_Next_Field},
4236 { REQ_SPREV_FIELD |ID_FN ,FN_Sorted_Previous_Field},
4237 { REQ_SFIRST_FIELD |ID_FN ,FN_Sorted_First_Field},
4238 { REQ_SLAST_FIELD |ID_FN ,FN_Sorted_Last_Field},
4239 { REQ_LEFT_FIELD |ID_FN ,FN_Left_Field},
4240 { REQ_RIGHT_FIELD |ID_FN ,FN_Right_Field},
4241 { REQ_UP_FIELD |ID_FN ,FN_Up_Field},
4242 { REQ_DOWN_FIELD |ID_FN ,FN_Down_Field},
4244 { REQ_NEXT_CHAR |ID_IFN ,IFN_Next_Character},
4245 { REQ_PREV_CHAR |ID_IFN ,IFN_Previous_Character},
4246 { REQ_NEXT_LINE |ID_IFN ,IFN_Next_Line},
4247 { REQ_PREV_LINE |ID_IFN ,IFN_Previous_Line},
4248 { REQ_NEXT_WORD |ID_IFN ,IFN_Next_Word},
4249 { REQ_PREV_WORD |ID_IFN ,IFN_Previous_Word},
4250 { REQ_BEG_FIELD |ID_IFN ,IFN_Beginning_Of_Field},
4251 { REQ_END_FIELD |ID_IFN ,IFN_End_Of_Field},
4252 { REQ_BEG_LINE |ID_IFN ,IFN_Beginning_Of_Line},
4253 { REQ_END_LINE |ID_IFN ,IFN_End_Of_Line},
4254 { REQ_LEFT_CHAR |ID_IFN ,IFN_Left_Character},
4255 { REQ_RIGHT_CHAR |ID_IFN ,IFN_Right_Character},
4256 { REQ_UP_CHAR |ID_IFN ,IFN_Up_Character},
4257 { REQ_DOWN_CHAR |ID_IFN ,IFN_Down_Character},
4259 { REQ_NEW_LINE |ID_FE ,FE_New_Line},
4260 { REQ_INS_CHAR |ID_FE ,FE_Insert_Character},
4261 { REQ_INS_LINE |ID_FE ,FE_Insert_Line},
4262 { REQ_DEL_CHAR |ID_FE ,FE_Delete_Character},
4263 { REQ_DEL_PREV |ID_FE ,FE_Delete_Previous},
4264 { REQ_DEL_LINE |ID_FE ,FE_Delete_Line},
4265 { REQ_DEL_WORD |ID_FE ,FE_Delete_Word},
4266 { REQ_CLR_EOL |ID_FE ,FE_Clear_To_End_Of_Line},
4267 { REQ_CLR_EOF |ID_FE ,FE_Clear_To_End_Of_Field},
4268 { REQ_CLR_FIELD |ID_FE ,FE_Clear_Field},
4270 { REQ_OVL_MODE |ID_EM ,EM_Overlay_Mode},
4271 { REQ_INS_MODE |ID_EM ,EM_Insert_Mode},
4273 { REQ_SCR_FLINE |ID_VSC ,VSC_Scroll_Line_Forward},
4274 { REQ_SCR_BLINE |ID_VSC ,VSC_Scroll_Line_Backward},
4275 { REQ_SCR_FPAGE |ID_VSC ,VSC_Scroll_Page_Forward},
4276 { REQ_SCR_BPAGE |ID_VSC ,VSC_Scroll_Page_Backward},
4277 { REQ_SCR_FHPAGE |ID_VSC ,VSC_Scroll_Half_Page_Forward},
4278 { REQ_SCR_BHPAGE |ID_VSC ,VSC_Scroll_Half_Page_Backward},
4280 { REQ_SCR_FCHAR |ID_HSC ,HSC_Scroll_Char_Forward},
4281 { REQ_SCR_BCHAR |ID_HSC ,HSC_Scroll_Char_Backward},
4282 { REQ_SCR_HFLINE |ID_HSC ,HSC_Horizontal_Line_Forward},
4283 { REQ_SCR_HBLINE |ID_HSC ,HSC_Horizontal_Line_Backward},
4284 { REQ_SCR_HFHALF |ID_HSC ,HSC_Horizontal_Half_Line_Forward},
4285 { REQ_SCR_HBHALF |ID_HSC ,HSC_Horizontal_Half_Line_Backward},
4287 { REQ_VALIDATION |ID_FV ,FV_Validation},
4289 { REQ_NEXT_CHOICE |ID_CH ,CR_Next_Choice},
4290 { REQ_PREV_CHOICE |ID_CH ,CR_Previous_Choice}
4294 /*---------------------------------------------------------------------------
4295 | Facility : libnform
4296 | Function : int form_driver(FORM * form,int c)
4298 | Description : This is the workhorse of the forms system. It checks
4299 | to determine whether the character c is a request or
4300 | data. If it is a request, the form driver executes
4301 | the request and returns the result. If it is data
4302 | (printable character), it enters the data into the
4303 | current position in the current field. If it is not
4304 | recognized, the form driver assumes it is an application
4305 | defined command and returns E_UNKNOWN_COMMAND.
4306 | Application defined command should be defined relative
4307 | to MAX_FORM_COMMAND, the maximum value of a request.
4309 | Return Values : E_OK - success
4310 | E_SYSTEM_ERROR - system error
4311 | E_BAD_ARGUMENT - an argument is incorrect
4312 | E_NOT_POSTED - form is not posted
4313 | E_INVALID_FIELD - field contents are invalid
4314 | E_BAD_STATE - called from inside a hook routine
4315 | E_REQUEST_DENIED - request failed
4316 | E_NOT_CONNECTED - no fields are connected to the form
4317 | E_UNKNOWN_COMMAND - command not known
4318 +--------------------------------------------------------------------------*/
4320 form_driver(FORM *form, int c)
4322 const Binding_Info *BI = (Binding_Info *) 0;
4323 int res = E_UNKNOWN_COMMAND;
4325 T((T_CALLED("form_driver(%p,%d)"), (void *)form, c));
4328 RETURN(E_BAD_ARGUMENT);
4331 RETURN(E_NOT_CONNECTED);
4335 if (c == FIRST_ACTIVE_MAGIC)
4337 form->current = _nc_First_Active_Field(form);
4341 assert(form->current &&
4342 form->current->buf &&
4343 (form->current->form == form)