1 /****************************************************************************
2 * Copyright (c) 1998-2013,2014 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.114 2014/07/26 20:46:51 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 = Get_Start_Of_Data(field->buf, Buffer_Length(field));
983 len = (int)(After_End_Of_Data(field->buf, Buffer_Length(field)) - bp);
987 assert(win && (field->drows == 1));
989 if (field->cols - len >= 0)
995 col = (field->cols - len) / 2;
998 col = field->cols - len;
1005 myADDNSTR(win, bp, len);
1009 /*---------------------------------------------------------------------------
1010 | Facility : libnform
1011 | Function : static void Undo_Justification(
1015 | Description : Display field without any justification, i.e.
1019 +--------------------------------------------------------------------------*/
1021 Undo_Justification(FIELD *field, WINDOW *win)
1026 bp = Get_Start_Of_Data(field->buf, Buffer_Length(field));
1027 len = (int)(After_End_Of_Data(field->buf, Buffer_Length(field)) - bp);
1033 myADDNSTR(win, bp, len);
1037 /*---------------------------------------------------------------------------
1038 | Facility : libnform
1039 | Function : static bool Check_Char(FORM *form,
1043 | TypeArgument *argp)
1045 | Description : Perform a single character check for character ch
1046 | according to the fieldtype instance.
1048 | Return Values : TRUE - Character is valid
1049 | FALSE - Character is invalid
1050 +--------------------------------------------------------------------------*/
1052 Check_Char(FORM *form,
1060 if (typ->status & _LINKED_TYPE)
1064 Check_Char(form, field, typ->left, ch, argp->left) ||
1065 Check_Char(form, field, typ->right, ch, argp->right));
1069 #if NCURSES_INTEROP_FUNCS
1070 if (typ->charcheck.occheck)
1072 if (typ->status & _GENERIC)
1073 return typ->charcheck.gccheck(ch, form, field, (void *)argp);
1075 return typ->charcheck.occheck(ch, (void *)argp);
1079 return typ->ccheck(ch, (void *)argp);
1083 return (!iscntrl(UChar(ch)) ? TRUE : FALSE);
1086 /*---------------------------------------------------------------------------
1087 | Facility : libnform
1088 | Function : static int Display_Or_Erase_Field(
1092 | Description : Create a subwindow for the field and display the
1093 | buffer contents (apply justification if required)
1094 | or simply erase the field.
1096 | Return Values : E_OK - on success
1097 | E_SYSTEM_ERROR - some error (typical no memory)
1098 +--------------------------------------------------------------------------*/
1100 Display_Or_Erase_Field(FIELD *field, bool bEraseFlag)
1106 return E_SYSTEM_ERROR;
1108 fwin = Get_Form_Window(field->form);
1110 field->rows, field->cols, field->frow, field->fcol);
1113 return E_SYSTEM_ERROR;
1116 if (Field_Has_Option(field, O_VISIBLE))
1118 Set_Field_Window_Attributes(field, win);
1122 (void)wattrset(win, (int)WINDOW_ATTRS(fwin));
1129 if (Field_Has_Option(field, O_PUBLIC))
1131 if (Justification_Allowed(field))
1132 Perform_Justification(field, win);
1134 Buffer_To_Window(field, win);
1136 ClrStatus(field, _NEWTOP);
1143 /* Macros to preset the bEraseFlag */
1144 #define Display_Field(field) Display_Or_Erase_Field(field,FALSE)
1145 #define Erase_Field(field) Display_Or_Erase_Field(field,TRUE)
1147 /*---------------------------------------------------------------------------
1148 | Facility : libnform
1149 | Function : static int Synchronize_Field(FIELD * field)
1151 | Description : Synchronize the windows content with the value in
1154 | Return Values : E_OK - success
1155 | E_BAD_ARGUMENT - invalid field pointer
1156 | E_SYSTEM_ERROR - some severe basic error
1157 +--------------------------------------------------------------------------*/
1159 Synchronize_Field(FIELD *field)
1165 return (E_BAD_ARGUMENT);
1167 if (((form = field->form) != (FORM *)0)
1168 && Field_Really_Appears(field))
1170 if (field == form->current)
1172 form->currow = form->curcol = form->toprow = form->begincol = 0;
1175 if ((Field_Has_Option(field, O_PUBLIC)) && Justification_Allowed(field))
1176 Undo_Justification(field, form->w);
1178 Buffer_To_Window(field, form->w);
1180 SetStatus(field, _NEWTOP);
1181 res = _nc_Refresh_Current_Field(form);
1184 res = Display_Field(field);
1186 SetStatus(field, _CHANGED);
1190 /*---------------------------------------------------------------------------
1191 | Facility : libnform
1192 | Function : static int Synchronize_Linked_Fields(FIELD * field)
1194 | Description : Propagate the Synchronize_Field function to all linked
1195 | fields. The first error that occurs in the sequence
1196 | of updates is the return value.
1198 | Return Values : E_OK - success
1199 | E_BAD_ARGUMENT - invalid field pointer
1200 | E_SYSTEM_ERROR - some severe basic error
1201 +--------------------------------------------------------------------------*/
1203 Synchronize_Linked_Fields(FIELD *field)
1205 FIELD *linked_field;
1210 return (E_BAD_ARGUMENT);
1213 return (E_SYSTEM_ERROR);
1215 for (linked_field = field->link;
1216 (linked_field != field) && (linked_field != 0);
1217 linked_field = linked_field->link)
1219 if (((syncres = Synchronize_Field(linked_field)) != E_OK) &&
1226 /*---------------------------------------------------------------------------
1227 | Facility : libnform
1228 | Function : int _nc_Synchronize_Attributes(FIELD * field)
1230 | Description : If a fields visual attributes have changed, this
1231 | routine is called to propagate those changes to the
1234 | Return Values : E_OK - success
1235 | E_BAD_ARGUMENT - invalid field pointer
1236 | E_SYSTEM_ERROR - some severe basic error
1237 +--------------------------------------------------------------------------*/
1239 _nc_Synchronize_Attributes(FIELD *field)
1245 T((T_CALLED("_nc_Synchronize_Attributes(%p)"), (void *)field));
1248 returnCode(E_BAD_ARGUMENT);
1250 CHECKPOS(field->form);
1251 if (((form = field->form) != (FORM *)0)
1252 && Field_Really_Appears(field))
1254 if (form->current == field)
1256 Synchronize_Buffer(form);
1257 Set_Field_Window_Attributes(field, form->w);
1259 wmove(form->w, form->currow, form->curcol);
1261 if (Field_Has_Option(field, O_PUBLIC))
1263 if (Justification_Allowed(field))
1264 Undo_Justification(field, form->w);
1266 Buffer_To_Window(field, form->w);
1270 formwin = Get_Form_Window(form);
1271 copywin(form->w, formwin,
1273 field->frow, field->fcol,
1274 field->rows - 1, field->cols - 1, 0);
1276 Buffer_To_Window(field, form->w);
1277 SetStatus(field, _NEWTOP); /* fake refresh to paint all */
1278 _nc_Refresh_Current_Field(form);
1283 res = Display_Field(field);
1290 /*---------------------------------------------------------------------------
1291 | Facility : libnform
1292 | Function : int _nc_Synchronize_Options(FIELD * field,
1293 | Field_Options newopts)
1295 | Description : If a fields options have changed, this routine is
1296 | called to propagate these changes to the screen and
1297 | to really change the behavior of the field.
1299 | Return Values : E_OK - success
1300 | E_BAD_ARGUMENT - invalid field pointer
1301 | E_CURRENT - field is the current one
1302 | E_SYSTEM_ERROR - some severe basic error
1303 +--------------------------------------------------------------------------*/
1305 _nc_Synchronize_Options(FIELD *field, Field_Options newopts)
1307 Field_Options oldopts;
1308 Field_Options changed_opts;
1312 T((T_CALLED("_nc_Synchronize_Options(%p,%#x)"), (void *)field, newopts));
1315 returnCode(E_BAD_ARGUMENT);
1317 oldopts = field->opts;
1318 changed_opts = oldopts ^ newopts;
1319 field->opts = newopts;
1324 if (form->status & _POSTED)
1326 if (form->current == field)
1328 field->opts = oldopts;
1329 returnCode(E_CURRENT);
1331 if (form->curpage == field->page)
1333 if ((unsigned)changed_opts & O_VISIBLE)
1335 if ((unsigned)newopts & O_VISIBLE)
1336 res = Display_Field(field);
1338 res = Erase_Field(field);
1342 if (((unsigned)changed_opts & O_PUBLIC) &&
1343 ((unsigned)newopts & O_VISIBLE))
1344 res = Display_Field(field);
1350 if ((unsigned)changed_opts & O_STATIC)
1352 bool single_line_field = Single_Line_Field(field);
1355 if ((unsigned)newopts & O_STATIC)
1357 /* the field becomes now static */
1358 ClrStatus(field, _MAY_GROW);
1359 /* if actually we have no hidden columns, justification may
1361 if (single_line_field &&
1362 (field->cols == field->dcols) &&
1363 (field->just != NO_JUSTIFICATION) &&
1364 Field_Really_Appears(field))
1366 res2 = Display_Field(field);
1371 /* field is no longer static */
1372 if ((field->maxgrow == 0) ||
1373 (single_line_field && (field->dcols < field->maxgrow)) ||
1374 (!single_line_field && (field->drows < field->maxgrow)))
1376 SetStatus(field, _MAY_GROW);
1377 /* a field with justification now changes its behavior,
1378 so we must redisplay it */
1379 if (single_line_field &&
1380 (field->just != NO_JUSTIFICATION) &&
1381 Field_Really_Appears(field))
1383 res2 = Display_Field(field);
1394 /*---------------------------------------------------------------------------
1395 | Facility : libnform
1396 | Function : int _nc_Set_Current_Field(FORM * form,
1399 | Description : Make the newfield the new current field.
1401 | Return Values : E_OK - success
1402 | E_BAD_ARGUMENT - invalid form or field pointer
1403 | E_SYSTEM_ERROR - some severe basic error
1404 | E_NOT_CONNECTED - no fields are connected to the form
1405 +--------------------------------------------------------------------------*/
1407 _nc_Set_Current_Field(FORM *form, FIELD *newfield)
1412 T((T_CALLED("_nc_Set_Current_Field(%p,%p)"), (void *)form, (void *)newfield));
1414 if (!form || !newfield || !form->current || (newfield->form != form))
1415 returnCode(E_BAD_ARGUMENT);
1417 if ((form->status & _IN_DRIVER))
1418 returnCode(E_BAD_STATE);
1421 returnCode(E_NOT_CONNECTED);
1423 field = form->current;
1425 if ((field != newfield) ||
1426 !(form->status & _POSTED))
1429 (Field_Has_Option(field, O_VISIBLE)) &&
1430 (field->form->curpage == field->page))
1432 _nc_Refresh_Current_Field(form);
1433 if (Field_Has_Option(field, O_PUBLIC))
1435 if (field->drows > field->rows)
1437 if (form->toprow == 0)
1438 ClrStatus(field, _NEWTOP);
1440 SetStatus(field, _NEWTOP);
1444 if (Justification_Allowed(field))
1446 Window_To_Buffer(form, field);
1448 Perform_Justification(field, form->w);
1449 if (Field_Has_Option(field, O_DYNAMIC_JUSTIFY) &&
1450 (form->w->_parent == 0))
1453 Get_Form_Window(form),
1459 field->cols + field->fcol - 1,
1461 wsyncup(Get_Form_Window(form));
1471 form->w = (WINDOW *)0;
1476 if (Has_Invisible_Parts(field))
1477 new_window = newpad(field->drows, field->dcols);
1479 new_window = derwin(Get_Form_Window(form),
1480 field->rows, field->cols, field->frow, field->fcol);
1483 returnCode(E_SYSTEM_ERROR);
1485 form->current = field;
1489 form->w = new_window;
1491 ClrStatus(form, _WINDOW_MODIFIED);
1492 Set_Field_Window_Attributes(field, form->w);
1494 if (Has_Invisible_Parts(field))
1497 Buffer_To_Window(field, form->w);
1501 if (Justification_Allowed(field))
1504 Undo_Justification(field, form->w);
1509 untouchwin(form->w);
1512 form->currow = form->curcol = form->toprow = form->begincol = 0;
1516 /*----------------------------------------------------------------------------
1517 Intra-Field Navigation routines
1518 --------------------------------------------------------------------------*/
1520 /*---------------------------------------------------------------------------
1521 | Facility : libnform
1522 | Function : static int IFN_Next_Character(FORM * form)
1524 | Description : Move to the next character in the field. In a multi-line
1525 | field this wraps at the end of the line.
1527 | Return Values : E_OK - success
1528 | E_REQUEST_DENIED - at the rightmost position
1529 +--------------------------------------------------------------------------*/
1531 IFN_Next_Character(FORM *form)
1533 FIELD *field = form->current;
1534 int step = myWCWIDTH(form->w, form->currow, form->curcol);
1536 T((T_CALLED("IFN_Next_Character(%p)"), (void *)form));
1537 if ((form->curcol += step) == field->dcols)
1539 if ((++(form->currow)) == field->drows)
1541 #if GROW_IF_NAVIGATE
1542 if (!Single_Line_Field(field) && Field_Grown(field, 1))
1549 #if GROW_IF_NAVIGATE
1550 if (Single_Line_Field(field) && Field_Grown(field, 1))
1553 form->curcol -= step;
1554 returnCode(E_REQUEST_DENIED);
1561 /*---------------------------------------------------------------------------
1562 | Facility : libnform
1563 | Function : static int IFN_Previous_Character(FORM * form)
1565 | Description : Move to the previous character in the field. In a
1566 | multi-line field this wraps and the beginning of the
1569 | Return Values : E_OK - success
1570 | E_REQUEST_DENIED - at the leftmost position
1571 +--------------------------------------------------------------------------*/
1573 IFN_Previous_Character(FORM *form)
1575 int amount = myWCWIDTH(form->w, form->currow, form->curcol - 1);
1576 int oldcol = form->curcol;
1578 T((T_CALLED("IFN_Previous_Character(%p)"), (void *)form));
1579 if ((form->curcol -= amount) < 0)
1581 if ((--(form->currow)) < 0)
1584 form->curcol = oldcol;
1585 returnCode(E_REQUEST_DENIED);
1587 form->curcol = form->current->dcols - 1;
1592 /*---------------------------------------------------------------------------
1593 | Facility : libnform
1594 | Function : static int IFN_Next_Line(FORM * form)
1596 | Description : Move to the beginning of the next line in the field
1598 | Return Values : E_OK - success
1599 | E_REQUEST_DENIED - at the last line
1600 +--------------------------------------------------------------------------*/
1602 IFN_Next_Line(FORM *form)
1604 FIELD *field = form->current;
1606 T((T_CALLED("IFN_Next_Line(%p)"), (void *)form));
1607 if ((++(form->currow)) == field->drows)
1609 #if GROW_IF_NAVIGATE
1610 if (!Single_Line_Field(field) && Field_Grown(field, 1))
1614 returnCode(E_REQUEST_DENIED);
1620 /*---------------------------------------------------------------------------
1621 | Facility : libnform
1622 | Function : static int IFN_Previous_Line(FORM * form)
1624 | Description : Move to the beginning of the previous line in the field
1626 | Return Values : E_OK - success
1627 | E_REQUEST_DENIED - at the first line
1628 +--------------------------------------------------------------------------*/
1630 IFN_Previous_Line(FORM *form)
1632 T((T_CALLED("IFN_Previous_Line(%p)"), (void *)form));
1633 if ((--(form->currow)) < 0)
1636 returnCode(E_REQUEST_DENIED);
1642 /*---------------------------------------------------------------------------
1643 | Facility : libnform
1644 | Function : static int IFN_Next_Word(FORM * form)
1646 | Description : Move to the beginning of the next word in the field.
1648 | Return Values : E_OK - success
1649 | E_REQUEST_DENIED - there is no next word
1650 +--------------------------------------------------------------------------*/
1652 IFN_Next_Word(FORM *form)
1654 FIELD *field = form->current;
1655 FIELD_CELL *bp = Address_Of_Current_Position_In_Buffer(form);
1659 T((T_CALLED("IFN_Next_Word(%p)"), (void *)form));
1661 /* We really need access to the data, so we have to synchronize */
1662 Synchronize_Buffer(form);
1664 /* Go to the first whitespace after the current position (including
1665 current position). This is then the starting point to look for the
1666 next non-blank data */
1667 s = Get_First_Whitespace_Character(bp, Buffer_Length(field) -
1668 (int)(bp - field->buf));
1670 /* Find the start of the next word */
1671 t = Get_Start_Of_Data(s, Buffer_Length(field) -
1672 (int)(s - field->buf));
1673 #if !FRIENDLY_PREV_NEXT_WORD
1675 returnCode(E_REQUEST_DENIED);
1679 Adjust_Cursor_Position(form, t);
1684 /*---------------------------------------------------------------------------
1685 | Facility : libnform
1686 | Function : static int IFN_Previous_Word(FORM * form)
1688 | Description : Move to the beginning of the previous word in the field.
1690 | Return Values : E_OK - success
1691 | E_REQUEST_DENIED - there is no previous word
1692 +--------------------------------------------------------------------------*/
1694 IFN_Previous_Word(FORM *form)
1696 FIELD *field = form->current;
1697 FIELD_CELL *bp = Address_Of_Current_Position_In_Buffer(form);
1702 T((T_CALLED("IFN_Previous_Word(%p)"), (void *)form));
1704 /* We really need access to the data, so we have to synchronize */
1705 Synchronize_Buffer(form);
1707 s = After_End_Of_Data(field->buf, (int)(bp - field->buf));
1708 /* s points now right after the last non-blank in the buffer before bp.
1709 If bp was in a word, s equals bp. In this case we must find the last
1710 whitespace in the buffer before bp and repeat the game to really find
1711 the previous word! */
1715 /* And next call now goes backward to look for the last whitespace
1716 before that, pointing right after this, so it points to the begin
1717 of the previous word.
1719 t = After_Last_Whitespace_Character(field->buf, (int)(s - field->buf));
1720 #if !FRIENDLY_PREV_NEXT_WORD
1722 returnCode(E_REQUEST_DENIED);
1726 /* and do it again, replacing bp by t */
1727 s = After_End_Of_Data(field->buf, (int)(t - field->buf));
1728 t = After_Last_Whitespace_Character(field->buf, (int)(s - field->buf));
1729 #if !FRIENDLY_PREV_NEXT_WORD
1731 returnCode(E_REQUEST_DENIED);
1734 Adjust_Cursor_Position(form, t);
1738 /*---------------------------------------------------------------------------
1739 | Facility : libnform
1740 | Function : static int IFN_Beginning_Of_Field(FORM * form)
1742 | Description : Place the cursor at the first non-pad character in
1745 | Return Values : E_OK - success
1746 +--------------------------------------------------------------------------*/
1748 IFN_Beginning_Of_Field(FORM *form)
1750 FIELD *field = form->current;
1752 T((T_CALLED("IFN_Beginning_Of_Field(%p)"), (void *)form));
1753 Synchronize_Buffer(form);
1754 Adjust_Cursor_Position(form,
1755 Get_Start_Of_Data(field->buf, Buffer_Length(field)));
1759 /*---------------------------------------------------------------------------
1760 | Facility : libnform
1761 | Function : static int IFN_End_Of_Field(FORM * form)
1763 | Description : Place the cursor after the last non-pad character in
1764 | the field. If the field occupies the last position in
1765 | the buffer, the cursor is positioned on the last
1768 | Return Values : E_OK - success
1769 +--------------------------------------------------------------------------*/
1771 IFN_End_Of_Field(FORM *form)
1773 FIELD *field = form->current;
1776 T((T_CALLED("IFN_End_Of_Field(%p)"), (void *)form));
1777 Synchronize_Buffer(form);
1778 pos = After_End_Of_Data(field->buf, Buffer_Length(field));
1779 if (pos == (field->buf + Buffer_Length(field)))
1781 Adjust_Cursor_Position(form, pos);
1785 /*---------------------------------------------------------------------------
1786 | Facility : libnform
1787 | Function : static int IFN_Beginning_Of_Line(FORM * form)
1789 | Description : Place the cursor on the first non-pad character in
1790 | the current line of the field.
1792 | Return Values : E_OK - success
1793 +--------------------------------------------------------------------------*/
1795 IFN_Beginning_Of_Line(FORM *form)
1797 FIELD *field = form->current;
1799 T((T_CALLED("IFN_Beginning_Of_Line(%p)"), (void *)form));
1800 Synchronize_Buffer(form);
1801 Adjust_Cursor_Position(form,
1802 Get_Start_Of_Data(Address_Of_Current_Row_In_Buffer(form),
1807 /*---------------------------------------------------------------------------
1808 | Facility : libnform
1809 | Function : static int IFN_End_Of_Line(FORM * form)
1811 | Description : Place the cursor after the last non-pad character in the
1812 | current line of the field. If the field occupies the
1813 | last column in the line, the cursor is positioned on the
1814 | last character of the line.
1816 | Return Values : E_OK - success
1817 +--------------------------------------------------------------------------*/
1819 IFN_End_Of_Line(FORM *form)
1821 FIELD *field = form->current;
1825 T((T_CALLED("IFN_End_Of_Line(%p)"), (void *)form));
1826 Synchronize_Buffer(form);
1827 bp = Address_Of_Current_Row_In_Buffer(form);
1828 pos = After_End_Of_Data(bp, field->dcols);
1829 if (pos == (bp + field->dcols))
1831 Adjust_Cursor_Position(form, pos);
1835 /*---------------------------------------------------------------------------
1836 | Facility : libnform
1837 | Function : static int IFN_Left_Character(FORM * form)
1839 | Description : Move one character to the left in the current line.
1840 | This doesn't cycle.
1842 | Return Values : E_OK - success
1843 | E_REQUEST_DENIED - already in first column
1844 +--------------------------------------------------------------------------*/
1846 IFN_Left_Character(FORM *form)
1848 int amount = myWCWIDTH(form->w, form->currow, form->curcol - 1);
1849 int oldcol = form->curcol;
1851 T((T_CALLED("IFN_Left_Character(%p)"), (void *)form));
1852 if ((form->curcol -= amount) < 0)
1854 form->curcol = oldcol;
1855 returnCode(E_REQUEST_DENIED);
1860 /*---------------------------------------------------------------------------
1861 | Facility : libnform
1862 | Function : static int IFN_Right_Character(FORM * form)
1864 | Description : Move one character to the right in the current line.
1865 | This doesn't cycle.
1867 | Return Values : E_OK - success
1868 | E_REQUEST_DENIED - already in last column
1869 +--------------------------------------------------------------------------*/
1871 IFN_Right_Character(FORM *form)
1873 int amount = myWCWIDTH(form->w, form->currow, form->curcol);
1874 int oldcol = form->curcol;
1876 T((T_CALLED("IFN_Right_Character(%p)"), (void *)form));
1877 if ((form->curcol += amount) >= form->current->dcols)
1879 #if GROW_IF_NAVIGATE
1880 FIELD *field = form->current;
1882 if (Single_Line_Field(field) && Field_Grown(field, 1))
1885 form->curcol = oldcol;
1886 returnCode(E_REQUEST_DENIED);
1891 /*---------------------------------------------------------------------------
1892 | Facility : libnform
1893 | Function : static int IFN_Up_Character(FORM * form)
1895 | Description : Move one line up. This doesn't cycle through the lines
1898 | Return Values : E_OK - success
1899 | E_REQUEST_DENIED - already in last column
1900 +--------------------------------------------------------------------------*/
1902 IFN_Up_Character(FORM *form)
1904 T((T_CALLED("IFN_Up_Character(%p)"), (void *)form));
1905 if ((--(form->currow)) < 0)
1908 returnCode(E_REQUEST_DENIED);
1913 /*---------------------------------------------------------------------------
1914 | Facility : libnform
1915 | Function : static int IFN_Down_Character(FORM * form)
1917 | Description : Move one line down. This doesn't cycle through the
1918 | lines of the field.
1920 | Return Values : E_OK - success
1921 | E_REQUEST_DENIED - already in last column
1922 +--------------------------------------------------------------------------*/
1924 IFN_Down_Character(FORM *form)
1926 FIELD *field = form->current;
1928 T((T_CALLED("IFN_Down_Character(%p)"), (void *)form));
1929 if ((++(form->currow)) == field->drows)
1931 #if GROW_IF_NAVIGATE
1932 if (!Single_Line_Field(field) && Field_Grown(field, 1))
1936 returnCode(E_REQUEST_DENIED);
1940 /*----------------------------------------------------------------------------
1941 END of Intra-Field Navigation routines
1942 --------------------------------------------------------------------------*/
1944 /*----------------------------------------------------------------------------
1945 Vertical scrolling helper routines
1946 --------------------------------------------------------------------------*/
1948 /*---------------------------------------------------------------------------
1949 | Facility : libnform
1950 | Function : static int VSC_Generic(FORM *form, int nlines)
1952 | Description : Scroll multi-line field forward (nlines>0) or
1953 | backward (nlines<0) this many lines.
1955 | Return Values : E_OK - success
1956 | E_REQUEST_DENIED - can't scroll
1957 +--------------------------------------------------------------------------*/
1959 VSC_Generic(FORM *form, int nlines)
1961 FIELD *field = form->current;
1962 int res = E_REQUEST_DENIED;
1963 int rows_to_go = (nlines > 0 ? nlines : -nlines);
1967 if ((rows_to_go + form->toprow) > (field->drows - field->rows))
1968 rows_to_go = (field->drows - field->rows - form->toprow);
1972 form->currow += rows_to_go;
1973 form->toprow += rows_to_go;
1979 if (rows_to_go > form->toprow)
1980 rows_to_go = form->toprow;
1984 form->currow -= rows_to_go;
1985 form->toprow -= rows_to_go;
1991 /*----------------------------------------------------------------------------
1992 End of Vertical scrolling helper routines
1993 --------------------------------------------------------------------------*/
1995 /*----------------------------------------------------------------------------
1996 Vertical scrolling routines
1997 --------------------------------------------------------------------------*/
1999 /*---------------------------------------------------------------------------
2000 | Facility : libnform
2001 | Function : static int Vertical_Scrolling(
2002 | int (* const fct) (FORM *),
2005 | Description : Performs the generic vertical scrolling routines.
2006 | This has to check for a multi-line field and to set
2007 | the _NEWTOP flag if scrolling really occurred.
2009 | Return Values : Propagated error code from low-level driver calls
2010 +--------------------------------------------------------------------------*/
2012 Vertical_Scrolling(int (*const fct) (FORM *), FORM *form)
2014 int res = E_REQUEST_DENIED;
2016 if (!Single_Line_Field(form->current))
2020 SetStatus(form->current, _NEWTOP);
2025 /*---------------------------------------------------------------------------
2026 | Facility : libnform
2027 | Function : static int VSC_Scroll_Line_Forward(FORM * form)
2029 | Description : Scroll multi-line field forward a line
2031 | Return Values : E_OK - success
2032 | E_REQUEST_DENIED - no data ahead
2033 +--------------------------------------------------------------------------*/
2035 VSC_Scroll_Line_Forward(FORM *form)
2037 T((T_CALLED("VSC_Scroll_Line_Forward(%p)"), (void *)form));
2038 returnCode(VSC_Generic(form, 1));
2041 /*---------------------------------------------------------------------------
2042 | Facility : libnform
2043 | Function : static int VSC_Scroll_Line_Backward(FORM * form)
2045 | Description : Scroll multi-line field backward a line
2047 | Return Values : E_OK - success
2048 | E_REQUEST_DENIED - no data behind
2049 +--------------------------------------------------------------------------*/
2051 VSC_Scroll_Line_Backward(FORM *form)
2053 T((T_CALLED("VSC_Scroll_Line_Backward(%p)"), (void *)form));
2054 returnCode(VSC_Generic(form, -1));
2057 /*---------------------------------------------------------------------------
2058 | Facility : libnform
2059 | Function : static int VSC_Scroll_Page_Forward(FORM * form)
2061 | Description : Scroll a multi-line field forward a page
2063 | Return Values : E_OK - success
2064 | E_REQUEST_DENIED - no data ahead
2065 +--------------------------------------------------------------------------*/
2067 VSC_Scroll_Page_Forward(FORM *form)
2069 T((T_CALLED("VSC_Scroll_Page_Forward(%p)"), (void *)form));
2070 returnCode(VSC_Generic(form, form->current->rows));
2073 /*---------------------------------------------------------------------------
2074 | Facility : libnform
2075 | Function : static int VSC_Scroll_Half_Page_Forward(FORM * form)
2077 | Description : Scroll a multi-line field forward half a page
2079 | Return Values : E_OK - success
2080 | E_REQUEST_DENIED - no data ahead
2081 +--------------------------------------------------------------------------*/
2083 VSC_Scroll_Half_Page_Forward(FORM *form)
2085 T((T_CALLED("VSC_Scroll_Half_Page_Forward(%p)"), (void *)form));
2086 returnCode(VSC_Generic(form, (form->current->rows + 1) / 2));
2089 /*---------------------------------------------------------------------------
2090 | Facility : libnform
2091 | Function : static int VSC_Scroll_Page_Backward(FORM * form)
2093 | Description : Scroll a multi-line field backward a page
2095 | Return Values : E_OK - success
2096 | E_REQUEST_DENIED - no data behind
2097 +--------------------------------------------------------------------------*/
2099 VSC_Scroll_Page_Backward(FORM *form)
2101 T((T_CALLED("VSC_Scroll_Page_Backward(%p)"), (void *)form));
2102 returnCode(VSC_Generic(form, -(form->current->rows)));
2105 /*---------------------------------------------------------------------------
2106 | Facility : libnform
2107 | Function : static int VSC_Scroll_Half_Page_Backward(FORM * form)
2109 | Description : Scroll a multi-line field backward half a page
2111 | Return Values : E_OK - success
2112 | E_REQUEST_DENIED - no data behind
2113 +--------------------------------------------------------------------------*/
2115 VSC_Scroll_Half_Page_Backward(FORM *form)
2117 T((T_CALLED("VSC_Scroll_Half_Page_Backward(%p)"), (void *)form));
2118 returnCode(VSC_Generic(form, -((form->current->rows + 1) / 2)));
2120 /*----------------------------------------------------------------------------
2121 End of Vertical scrolling routines
2122 --------------------------------------------------------------------------*/
2124 /*----------------------------------------------------------------------------
2125 Horizontal scrolling helper routines
2126 --------------------------------------------------------------------------*/
2128 /*---------------------------------------------------------------------------
2129 | Facility : libnform
2130 | Function : static int HSC_Generic(FORM *form, int ncolumns)
2132 | Description : Scroll single-line field forward (ncolumns>0) or
2133 | backward (ncolumns<0) this many columns.
2135 | Return Values : E_OK - success
2136 | E_REQUEST_DENIED - can't scroll
2137 +--------------------------------------------------------------------------*/
2139 HSC_Generic(FORM *form, int ncolumns)
2141 FIELD *field = form->current;
2142 int res = E_REQUEST_DENIED;
2143 int cols_to_go = (ncolumns > 0 ? ncolumns : -ncolumns);
2147 if ((cols_to_go + form->begincol) > (field->dcols - field->cols))
2148 cols_to_go = field->dcols - field->cols - form->begincol;
2152 form->curcol += cols_to_go;
2153 form->begincol += cols_to_go;
2159 if (cols_to_go > form->begincol)
2160 cols_to_go = form->begincol;
2164 form->curcol -= cols_to_go;
2165 form->begincol -= cols_to_go;
2171 /*----------------------------------------------------------------------------
2172 End of Horizontal scrolling helper routines
2173 --------------------------------------------------------------------------*/
2175 /*----------------------------------------------------------------------------
2176 Horizontal scrolling routines
2177 --------------------------------------------------------------------------*/
2179 /*---------------------------------------------------------------------------
2180 | Facility : libnform
2181 | Function : static int Horizontal_Scrolling(
2182 | int (* const fct) (FORM *),
2185 | Description : Performs the generic horizontal scrolling routines.
2186 | This has to check for a single-line field.
2188 | Return Values : Propagated error code from low-level driver calls
2189 +--------------------------------------------------------------------------*/
2191 Horizontal_Scrolling(int (*const fct) (FORM *), FORM *form)
2193 if (Single_Line_Field(form->current))
2196 return (E_REQUEST_DENIED);
2199 /*---------------------------------------------------------------------------
2200 | Facility : libnform
2201 | Function : static int HSC_Scroll_Char_Forward(FORM * form)
2203 | Description : Scroll single-line field forward a character
2205 | Return Values : E_OK - success
2206 | E_REQUEST_DENIED - no data ahead
2207 +--------------------------------------------------------------------------*/
2209 HSC_Scroll_Char_Forward(FORM *form)
2211 T((T_CALLED("HSC_Scroll_Char_Forward(%p)"), (void *)form));
2212 returnCode(HSC_Generic(form, 1));
2215 /*---------------------------------------------------------------------------
2216 | Facility : libnform
2217 | Function : static int HSC_Scroll_Char_Backward(FORM * form)
2219 | Description : Scroll single-line field backward a character
2221 | Return Values : E_OK - success
2222 | E_REQUEST_DENIED - no data behind
2223 +--------------------------------------------------------------------------*/
2225 HSC_Scroll_Char_Backward(FORM *form)
2227 T((T_CALLED("HSC_Scroll_Char_Backward(%p)"), (void *)form));
2228 returnCode(HSC_Generic(form, -1));
2231 /*---------------------------------------------------------------------------
2232 | Facility : libnform
2233 | Function : static int HSC_Horizontal_Line_Forward(FORM* form)
2235 | Description : Scroll single-line field forward a line
2237 | Return Values : E_OK - success
2238 | E_REQUEST_DENIED - no data ahead
2239 +--------------------------------------------------------------------------*/
2241 HSC_Horizontal_Line_Forward(FORM *form)
2243 T((T_CALLED("HSC_Horizontal_Line_Forward(%p)"), (void *)form));
2244 returnCode(HSC_Generic(form, form->current->cols));
2247 /*---------------------------------------------------------------------------
2248 | Facility : libnform
2249 | Function : static int HSC_Horizontal_Half_Line_Forward(FORM* form)
2251 | Description : Scroll single-line field forward half a line
2253 | Return Values : E_OK - success
2254 | E_REQUEST_DENIED - no data ahead
2255 +--------------------------------------------------------------------------*/
2257 HSC_Horizontal_Half_Line_Forward(FORM *form)
2259 T((T_CALLED("HSC_Horizontal_Half_Line_Forward(%p)"), (void *)form));
2260 returnCode(HSC_Generic(form, (form->current->cols + 1) / 2));
2263 /*---------------------------------------------------------------------------
2264 | Facility : libnform
2265 | Function : static int HSC_Horizontal_Line_Backward(FORM* form)
2267 | Description : Scroll single-line field backward a line
2269 | Return Values : E_OK - success
2270 | E_REQUEST_DENIED - no data behind
2271 +--------------------------------------------------------------------------*/
2273 HSC_Horizontal_Line_Backward(FORM *form)
2275 T((T_CALLED("HSC_Horizontal_Line_Backward(%p)"), (void *)form));
2276 returnCode(HSC_Generic(form, -(form->current->cols)));
2279 /*---------------------------------------------------------------------------
2280 | Facility : libnform
2281 | Function : static int HSC_Horizontal_Half_Line_Backward(FORM* form)
2283 | Description : Scroll single-line field backward half a line
2285 | Return Values : E_OK - success
2286 | E_REQUEST_DENIED - no data behind
2287 +--------------------------------------------------------------------------*/
2289 HSC_Horizontal_Half_Line_Backward(FORM *form)
2291 T((T_CALLED("HSC_Horizontal_Half_Line_Backward(%p)"), (void *)form));
2292 returnCode(HSC_Generic(form, -((form->current->cols + 1) / 2)));
2295 /*----------------------------------------------------------------------------
2296 End of Horizontal scrolling routines
2297 --------------------------------------------------------------------------*/
2299 /*----------------------------------------------------------------------------
2300 Helper routines for Field Editing
2301 --------------------------------------------------------------------------*/
2303 /*---------------------------------------------------------------------------
2304 | Facility : libnform
2305 | Function : static bool Is_There_Room_For_A_Line(FORM * form)
2307 | Description : Check whether or not there is enough room in the
2308 | buffer to enter a whole line.
2310 | Return Values : TRUE - there is enough space
2311 | FALSE - there is not enough space
2312 +--------------------------------------------------------------------------*/
2313 NCURSES_INLINE static bool
2314 Is_There_Room_For_A_Line(FORM *form)
2316 FIELD *field = form->current;
2317 FIELD_CELL *begin_of_last_line, *s;
2319 Synchronize_Buffer(form);
2320 begin_of_last_line = Address_Of_Row_In_Buffer(field, (field->drows - 1));
2321 s = After_End_Of_Data(begin_of_last_line, field->dcols);
2322 return ((s == begin_of_last_line) ? TRUE : FALSE);
2325 /*---------------------------------------------------------------------------
2326 | Facility : libnform
2327 | Function : static bool Is_There_Room_For_A_Char_In_Line(FORM * form)
2329 | Description : Checks whether or not there is room for a new character
2330 | in the current line.
2332 | Return Values : TRUE - there is room
2333 | FALSE - there is not enough room (line full)
2334 +--------------------------------------------------------------------------*/
2335 NCURSES_INLINE static bool
2336 Is_There_Room_For_A_Char_In_Line(FORM *form)
2338 int last_char_in_line;
2340 wmove(form->w, form->currow, form->current->dcols - 1);
2341 last_char_in_line = (int)(winch(form->w) & A_CHARTEXT);
2342 wmove(form->w, form->currow, form->curcol);
2343 return (((last_char_in_line == form->current->pad) ||
2344 is_blank(last_char_in_line)) ? TRUE : FALSE);
2347 #define There_Is_No_Room_For_A_Char_In_Line(f) \
2348 !Is_There_Room_For_A_Char_In_Line(f)
2350 /*---------------------------------------------------------------------------
2351 | Facility : libnform
2352 | Function : static int Insert_String(
2358 | Description : Insert the 'len' characters beginning at pointer 'txt'
2359 | into the 'row' of the 'form'. The insertion occurs
2360 | on the beginning of the row, all other characters are
2361 | moved to the right. After the text a pad character will
2362 | be inserted to separate the text from the rest. If
2363 | necessary the insertion moves characters on the next
2364 | line to make place for the requested insertion string.
2366 | Return Values : E_OK - success
2367 | E_REQUEST_DENIED -
2368 | E_SYSTEM_ERROR - system error
2369 +--------------------------------------------------------------------------*/
2371 Insert_String(FORM *form, int row, FIELD_CELL *txt, int len)
2373 FIELD *field = form->current;
2374 FIELD_CELL *bp = Address_Of_Row_In_Buffer(field, row);
2375 int datalen = (int)(After_End_Of_Data(bp, field->dcols) - bp);
2376 int freelen = field->dcols - datalen;
2377 int requiredlen = len + 1;
2379 int result = E_REQUEST_DENIED;
2381 if (freelen >= requiredlen)
2383 wmove(form->w, row, 0);
2384 myINSNSTR(form->w, txt, len);
2385 wmove(form->w, row, len);
2386 myINSNSTR(form->w, &myBLANK, 1);
2391 /* we have to move characters on the next line. If we are on the
2392 last line this may work, if the field is growable */
2393 if ((row == (field->drows - 1)) && Growable(field))
2395 if (!Field_Grown(field, 1))
2396 return (E_SYSTEM_ERROR);
2397 /* !!!Side-Effect : might be changed due to growth!!! */
2398 bp = Address_Of_Row_In_Buffer(field, row);
2401 if (row < (field->drows - 1))
2404 After_Last_Whitespace_Character(bp,
2405 (int)(Get_Start_Of_Data(bp
2410 /* split points now to the first character of the portion of the
2411 line that must be moved to the next line */
2412 datalen = (int)(split - bp); /* + freelen has to stay on this line */
2413 freelen = field->dcols - (datalen + freelen); /* for the next line */
2415 if ((result = Insert_String(form, row + 1, split, freelen)) == E_OK)
2417 wmove(form->w, row, datalen);
2419 wmove(form->w, row, 0);
2420 myINSNSTR(form->w, txt, len);
2421 wmove(form->w, row, len);
2422 myINSNSTR(form->w, &myBLANK, 1);
2430 /*---------------------------------------------------------------------------
2431 | Facility : libnform
2432 | Function : static int Wrapping_Not_Necessary_Or_Wrapping_Ok(
2435 | Description : If a character has been entered into a field, it may
2436 | be that wrapping has to occur. This routine checks
2437 | whether or not wrapping is required and if so, performs
2440 | Return Values : E_OK - no wrapping required or wrapping
2442 | E_REQUEST_DENIED -
2443 | E_SYSTEM_ERROR - some system error
2444 +--------------------------------------------------------------------------*/
2446 Wrapping_Not_Necessary_Or_Wrapping_Ok(FORM *form)
2448 FIELD *field = form->current;
2449 int result = E_REQUEST_DENIED;
2450 bool Last_Row = ((field->drows - 1) == form->currow);
2452 if ((Field_Has_Option(field, O_WRAP)) && /* wrapping wanted */
2453 (!Single_Line_Field(field)) && /* must be multi-line */
2454 (There_Is_No_Room_For_A_Char_In_Line(form)) && /* line is full */
2455 (!Last_Row || Growable(field))) /* there are more lines */
2459 int chars_to_be_wrapped;
2460 int chars_to_remain_on_line;
2464 /* the above logic already ensures, that in this case the field
2466 if (!Field_Grown(field, 1))
2467 return E_SYSTEM_ERROR;
2469 bp = Address_Of_Current_Row_In_Buffer(form);
2470 Window_To_Buffer(form, field);
2471 split = After_Last_Whitespace_Character(bp, field->dcols);
2472 /* split points to the first character of the sequence to be brought
2474 chars_to_remain_on_line = (int)(split - bp);
2475 chars_to_be_wrapped = field->dcols - chars_to_remain_on_line;
2476 if (chars_to_remain_on_line > 0)
2478 if ((result = Insert_String(form, form->currow + 1, split,
2479 chars_to_be_wrapped)) == E_OK)
2481 wmove(form->w, form->currow, chars_to_remain_on_line);
2483 if (form->curcol >= chars_to_remain_on_line)
2486 form->curcol -= chars_to_remain_on_line;
2496 Window_To_Buffer(form, field);
2497 result = E_REQUEST_DENIED;
2501 result = E_OK; /* wrapping was not necessary */
2505 /*----------------------------------------------------------------------------
2506 Field Editing routines
2507 --------------------------------------------------------------------------*/
2509 /*---------------------------------------------------------------------------
2510 | Facility : libnform
2511 | Function : static int Field_Editing(
2512 | int (* const fct) (FORM *),
2515 | Description : Generic routine for field editing requests. The driver
2516 | routines are only called for editable fields, the
2517 | _WINDOW_MODIFIED flag is set if editing occurred.
2518 | This is somewhat special due to the overload semantics
2519 | of the NEW_LINE and DEL_PREV requests.
2521 | Return Values : Error code from low level drivers.
2522 +--------------------------------------------------------------------------*/
2524 Field_Editing(int (*const fct) (FORM *), FORM *form)
2526 int res = E_REQUEST_DENIED;
2528 /* We have to deal here with the specific case of the overloaded
2529 behavior of New_Line and Delete_Previous requests.
2530 They may end up in navigational requests if we are on the first
2531 character in a field. But navigation is also allowed on non-
2534 if ((fct == FE_Delete_Previous) &&
2535 ((unsigned)form->opts & O_BS_OVERLOAD) &&
2536 First_Position_In_Current_Field(form))
2538 res = Inter_Field_Navigation(FN_Previous_Field, form);
2542 if (fct == FE_New_Line)
2544 if (((unsigned)form->opts & O_NL_OVERLOAD) &&
2545 First_Position_In_Current_Field(form))
2547 res = Inter_Field_Navigation(FN_Next_Field, form);
2550 /* FE_New_Line deals itself with the _WINDOW_MODIFIED flag */
2555 /* From now on, everything must be editable */
2556 if ((unsigned)form->current->opts & O_EDIT)
2560 SetStatus(form, _WINDOW_MODIFIED);
2567 /*---------------------------------------------------------------------------
2568 | Facility : libnform
2569 | Function : static int FE_New_Line(FORM * form)
2571 | Description : Perform a new line request. This is rather complex
2572 | compared to other routines in this code due to the
2573 | rather difficult to understand description in the
2576 | Return Values : E_OK - success
2577 | E_REQUEST_DENIED - new line not allowed
2578 | E_SYSTEM_ERROR - system error
2579 +--------------------------------------------------------------------------*/
2581 FE_New_Line(FORM *form)
2583 FIELD *field = form->current;
2585 bool Last_Row = ((field->drows - 1) == form->currow);
2587 T((T_CALLED("FE_New_Line(%p)"), (void *)form));
2588 if (form->status & _OVLMODE)
2591 (!(Growable(field) && !Single_Line_Field(field))))
2593 if (!((unsigned)form->opts & O_NL_OVERLOAD))
2594 returnCode(E_REQUEST_DENIED);
2595 wmove(form->w, form->currow, form->curcol);
2597 /* we have to set this here, although it is also
2598 handled in the generic routine. The reason is,
2599 that FN_Next_Field may fail, but the form is
2600 definitively changed */
2601 SetStatus(form, _WINDOW_MODIFIED);
2602 returnCode(Inter_Field_Navigation(FN_Next_Field, form));
2606 if (Last_Row && !Field_Grown(field, 1))
2608 /* N.B.: due to the logic in the 'if', LastRow==TRUE
2609 means here that the field is growable and not
2610 a single-line field */
2611 returnCode(E_SYSTEM_ERROR);
2613 wmove(form->w, form->currow, form->curcol);
2617 SetStatus(form, _WINDOW_MODIFIED);
2625 !(Growable(field) && !Single_Line_Field(field)))
2627 if (!((unsigned)form->opts & O_NL_OVERLOAD))
2628 returnCode(E_REQUEST_DENIED);
2629 returnCode(Inter_Field_Navigation(FN_Next_Field, form));
2633 bool May_Do_It = !Last_Row && Is_There_Room_For_A_Line(form);
2635 if (!(May_Do_It || Growable(field)))
2636 returnCode(E_REQUEST_DENIED);
2637 if (!May_Do_It && !Field_Grown(field, 1))
2638 returnCode(E_SYSTEM_ERROR);
2640 bp = Address_Of_Current_Position_In_Buffer(form);
2641 t = After_End_Of_Data(bp, field->dcols - form->curcol);
2642 wmove(form->w, form->currow, form->curcol);
2646 wmove(form->w, form->currow, form->curcol);
2648 myADDNSTR(form->w, bp, (int)(t - bp));
2649 SetStatus(form, _WINDOW_MODIFIED);
2655 /*---------------------------------------------------------------------------
2656 | Facility : libnform
2657 | Function : static int FE_Insert_Character(FORM * form)
2659 | Description : Insert blank character at the cursor position
2661 | Return Values : E_OK
2663 +--------------------------------------------------------------------------*/
2665 FE_Insert_Character(FORM *form)
2667 FIELD *field = form->current;
2668 int result = E_REQUEST_DENIED;
2670 T((T_CALLED("FE_Insert_Character(%p)"), (void *)form));
2671 if (Check_Char(form, field, field->type, (int)C_BLANK,
2672 (TypeArgument *)(field->arg)))
2674 bool There_Is_Room = Is_There_Room_For_A_Char_In_Line(form);
2676 if (There_Is_Room ||
2677 ((Single_Line_Field(field) && Growable(field))))
2679 if (!There_Is_Room && !Field_Grown(field, 1))
2680 result = E_SYSTEM_ERROR;
2683 winsch(form->w, (chtype)C_BLANK);
2684 result = Wrapping_Not_Necessary_Or_Wrapping_Ok(form);
2691 /*---------------------------------------------------------------------------
2692 | Facility : libnform
2693 | Function : static int FE_Insert_Line(FORM * form)
2695 | Description : Insert a blank line at the cursor position
2697 | Return Values : E_OK - success
2698 | E_REQUEST_DENIED - line can not be inserted
2699 +--------------------------------------------------------------------------*/
2701 FE_Insert_Line(FORM *form)
2703 FIELD *field = form->current;
2704 int result = E_REQUEST_DENIED;
2706 T((T_CALLED("FE_Insert_Line(%p)"), (void *)form));
2707 if (Check_Char(form, field,
2708 field->type, (int)C_BLANK, (TypeArgument *)(field->arg)))
2710 bool Maybe_Done = (form->currow != (field->drows - 1)) &&
2711 Is_There_Room_For_A_Line(form);
2713 if (!Single_Line_Field(field) &&
2714 (Maybe_Done || Growable(field)))
2716 if (!Maybe_Done && !Field_Grown(field, 1))
2717 result = E_SYSTEM_ERROR;
2729 /*---------------------------------------------------------------------------
2730 | Facility : libnform
2731 | Function : static int FE_Delete_Character(FORM * form)
2733 | Description : Delete character at the cursor position
2735 | Return Values : E_OK - success
2736 +--------------------------------------------------------------------------*/
2738 FE_Delete_Character(FORM *form)
2740 T((T_CALLED("FE_Delete_Character(%p)"), (void *)form));
2745 /*---------------------------------------------------------------------------
2746 | Facility : libnform
2747 | Function : static int FE_Delete_Previous(FORM * form)
2749 | Description : Delete character before cursor. Again this is a rather
2750 | difficult piece compared to others due to the overloading
2751 | semantics of backspace.
2752 | N.B.: The case of overloaded BS on first field position
2753 | is already handled in the generic routine.
2755 | Return Values : E_OK - success
2756 | E_REQUEST_DENIED - Character can't be deleted
2757 +--------------------------------------------------------------------------*/
2759 FE_Delete_Previous(FORM *form)
2761 FIELD *field = form->current;
2763 T((T_CALLED("FE_Delete_Previous(%p)"), (void *)form));
2764 if (First_Position_In_Current_Field(form))
2765 returnCode(E_REQUEST_DENIED);
2767 if ((--(form->curcol)) < 0)
2769 FIELD_CELL *this_line, *prev_line, *prev_end, *this_end;
2770 int this_row = form->currow;
2773 if (form->status & _OVLMODE)
2774 returnCode(E_REQUEST_DENIED);
2776 prev_line = Address_Of_Row_In_Buffer(field, (form->currow - 1));
2777 this_line = Address_Of_Row_In_Buffer(field, (form->currow));
2778 Synchronize_Buffer(form);
2779 prev_end = After_End_Of_Data(prev_line, field->dcols);
2780 this_end = After_End_Of_Data(this_line, field->dcols);
2781 if ((int)(this_end - this_line) >
2782 (field->cols - (int)(prev_end - prev_line)))
2783 returnCode(E_REQUEST_DENIED);
2784 wmove(form->w, form->currow, form->curcol);
2786 Adjust_Cursor_Position(form, prev_end);
2788 * If we did not really move to the previous line, help the user a
2789 * little. It is however a little inconsistent. Normally, when
2790 * backspacing around the point where text wraps to a new line in a
2791 * multi-line form, we absorb one keystroke for the wrapping point. That
2792 * is consistent with SVr4 forms. However, SVr4 does not allow typing
2793 * into the last column of the field, and requires the user to enter a
2794 * newline to move to the next line. Therefore it can consistently eat
2795 * that keystroke. Since ncurses allows the last column, it wraps
2796 * automatically (given the proper options). But we cannot eat the
2797 * keystroke to back over the wrapping point, since that would put the
2798 * cursor past the end of the form field. In this case, just delete the
2799 * character at the end of the field.
2801 if (form->currow == this_row && this_row > 0)
2804 form->curcol = field->dcols - 1;
2809 wmove(form->w, form->currow, form->curcol);
2810 myADDNSTR(form->w, this_line, (int)(this_end - this_line));
2820 /*---------------------------------------------------------------------------
2821 | Facility : libnform
2822 | Function : static int FE_Delete_Line(FORM * form)
2824 | Description : Delete line at cursor position.
2826 | Return Values : E_OK - success
2827 +--------------------------------------------------------------------------*/
2829 FE_Delete_Line(FORM *form)
2831 T((T_CALLED("FE_Delete_Line(%p)"), (void *)form));
2837 /*---------------------------------------------------------------------------
2838 | Facility : libnform
2839 | Function : static int FE_Delete_Word(FORM * form)
2841 | Description : Delete word at cursor position
2843 | Return Values : E_OK - success
2844 | E_REQUEST_DENIED - failure
2845 +--------------------------------------------------------------------------*/
2847 FE_Delete_Word(FORM *form)
2849 FIELD *field = form->current;
2850 FIELD_CELL *bp = Address_Of_Current_Row_In_Buffer(form);
2851 FIELD_CELL *ep = bp + field->dcols;
2852 FIELD_CELL *cp = bp + form->curcol;
2855 T((T_CALLED("FE_Delete_Word(%p)"), (void *)form));
2856 Synchronize_Buffer(form);
2858 returnCode(E_REQUEST_DENIED); /* not in word */
2860 /* move cursor to begin of word and erase to end of screen-line */
2861 Adjust_Cursor_Position(form,
2862 After_Last_Whitespace_Character(bp, form->curcol));
2863 wmove(form->w, form->currow, form->curcol);
2866 /* skip over word in buffer */
2867 s = Get_First_Whitespace_Character(cp, (int)(ep - cp));
2868 /* to begin of next word */
2869 s = Get_Start_Of_Data(s, (int)(ep - s));
2870 if ((s != cp) && !ISBLANK(*s))
2872 /* copy remaining line to window */
2873 myADDNSTR(form->w, s, (int)(s - After_End_Of_Data(s, (int)(ep - s))));
2878 /*---------------------------------------------------------------------------
2879 | Facility : libnform
2880 | Function : static int FE_Clear_To_End_Of_Line(FORM * form)
2882 | Description : Clear to end of current line.
2884 | Return Values : E_OK - success
2885 +--------------------------------------------------------------------------*/
2887 FE_Clear_To_End_Of_Line(FORM *form)
2889 T((T_CALLED("FE_Clear_To_End_Of_Line(%p)"), (void *)form));
2890 wmove(form->w, form->currow, form->curcol);
2895 /*---------------------------------------------------------------------------
2896 | Facility : libnform
2897 | Function : static int FE_Clear_To_End_Of_Field(FORM * form)
2899 | Description : Clear to end of field.
2901 | Return Values : E_OK - success
2902 +--------------------------------------------------------------------------*/
2904 FE_Clear_To_End_Of_Field(FORM *form)
2906 T((T_CALLED("FE_Clear_To_End_Of_Field(%p)"), (void *)form));
2907 wmove(form->w, form->currow, form->curcol);
2912 /*---------------------------------------------------------------------------
2913 | Facility : libnform
2914 | Function : static int FE_Clear_Field(FORM * form)
2916 | Description : Clear entire field.
2918 | Return Values : E_OK - success
2919 +--------------------------------------------------------------------------*/
2921 FE_Clear_Field(FORM *form)
2923 T((T_CALLED("FE_Clear_Field(%p)"), (void *)form));
2924 form->currow = form->curcol = 0;
2928 /*----------------------------------------------------------------------------
2929 END of Field Editing routines
2930 --------------------------------------------------------------------------*/
2932 /*----------------------------------------------------------------------------
2934 --------------------------------------------------------------------------*/
2936 /*---------------------------------------------------------------------------
2937 | Facility : libnform
2938 | Function : static int EM_Overlay_Mode(FORM * form)
2940 | Description : Switch to overlay mode.
2942 | Return Values : E_OK - success
2943 +--------------------------------------------------------------------------*/
2945 EM_Overlay_Mode(FORM *form)
2947 T((T_CALLED("EM_Overlay_Mode(%p)"), (void *)form));
2948 SetStatus(form, _OVLMODE);
2952 /*---------------------------------------------------------------------------
2953 | Facility : libnform
2954 | Function : static int EM_Insert_Mode(FORM * form)
2956 | Description : Switch to insert mode
2958 | Return Values : E_OK - success
2959 +--------------------------------------------------------------------------*/
2961 EM_Insert_Mode(FORM *form)
2963 T((T_CALLED("EM_Insert_Mode(%p)"), (void *)form));
2964 ClrStatus(form, _OVLMODE);
2968 /*----------------------------------------------------------------------------
2969 END of Edit Mode routines
2970 --------------------------------------------------------------------------*/
2972 /*----------------------------------------------------------------------------
2973 Helper routines for Choice Requests
2974 --------------------------------------------------------------------------*/
2976 /*---------------------------------------------------------------------------
2977 | Facility : libnform
2978 | Function : static bool Next_Choice(FORM * form,
2981 | TypeArgument *argp)
2983 | Description : Get the next field choice. For linked types this is
2986 | Return Values : TRUE - next choice successfully retrieved
2987 | FALSE - couldn't retrieve next choice
2988 +--------------------------------------------------------------------------*/
2990 Next_Choice(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
2992 if (!typ || !(typ->status & _HAS_CHOICE))
2995 if (typ->status & _LINKED_TYPE)
2999 Next_Choice(form, typ->left, field, argp->left) ||
3000 Next_Choice(form, typ->right, field, argp->right));
3004 #if NCURSES_INTEROP_FUNCS
3005 assert(typ->enum_next.onext);
3006 if (typ->status & _GENERIC)
3007 return typ->enum_next.gnext(form, field, (void *)argp);
3009 return typ->enum_next.onext(field, (void *)argp);
3012 return typ->next(field, (void *)argp);
3017 /*---------------------------------------------------------------------------
3018 | Facility : libnform
3019 | Function : static bool Previous_Choice(FORM * form,
3022 | TypeArgument *argp)
3024 | Description : Get the previous field choice. For linked types this
3025 | is done recursively.
3027 | Return Values : TRUE - previous choice successfully retrieved
3028 | FALSE - couldn't retrieve previous choice
3029 +--------------------------------------------------------------------------*/
3031 Previous_Choice(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
3033 if (!typ || !(typ->status & _HAS_CHOICE))
3036 if (typ->status & _LINKED_TYPE)
3040 Previous_Choice(form, typ->left, field, argp->left) ||
3041 Previous_Choice(form, typ->right, field, argp->right));
3045 #if NCURSES_INTEROP_FUNCS
3046 assert(typ->enum_prev.oprev);
3047 if (typ->status & _GENERIC)
3048 return typ->enum_prev.gprev(form, field, (void *)argp);
3050 return typ->enum_prev.oprev(field, (void *)argp);
3053 return typ->prev(field, (void *)argp);
3057 /*----------------------------------------------------------------------------
3058 End of Helper routines for Choice Requests
3059 --------------------------------------------------------------------------*/
3061 /*----------------------------------------------------------------------------
3062 Routines for Choice Requests
3063 --------------------------------------------------------------------------*/
3065 /*---------------------------------------------------------------------------
3066 | Facility : libnform
3067 | Function : static int CR_Next_Choice(FORM * form)
3069 | Description : Get the next field choice.
3071 | Return Values : E_OK - success
3072 | E_REQUEST_DENIED - next choice couldn't be retrieved
3073 +--------------------------------------------------------------------------*/
3075 CR_Next_Choice(FORM *form)
3077 FIELD *field = form->current;
3079 T((T_CALLED("CR_Next_Choice(%p)"), (void *)form));
3080 Synchronize_Buffer(form);
3081 returnCode((Next_Choice(form, field->type, field, (TypeArgument *)(field->arg)))
3083 : E_REQUEST_DENIED);
3086 /*---------------------------------------------------------------------------
3087 | Facility : libnform
3088 | Function : static int CR_Previous_Choice(FORM * form)
3090 | Description : Get the previous field choice.
3092 | Return Values : E_OK - success
3093 | E_REQUEST_DENIED - prev. choice couldn't be retrieved
3094 +--------------------------------------------------------------------------*/
3096 CR_Previous_Choice(FORM *form)
3098 FIELD *field = form->current;
3100 T((T_CALLED("CR_Previous_Choice(%p)"), (void *)form));
3101 Synchronize_Buffer(form);
3102 returnCode((Previous_Choice(form, field->type, field, (TypeArgument *)(field->arg)))
3104 : E_REQUEST_DENIED);
3106 /*----------------------------------------------------------------------------
3107 End of Routines for Choice Requests
3108 --------------------------------------------------------------------------*/
3110 /*----------------------------------------------------------------------------
3111 Helper routines for Field Validations.
3112 --------------------------------------------------------------------------*/
3114 /*---------------------------------------------------------------------------
3115 | Facility : libnform
3116 | Function : static bool Check_Field(FORM* form,
3119 | TypeArgument * argp)
3121 | Description : Check the field according to its fieldtype and its
3122 | actual arguments. For linked fieldtypes this is done
3125 | Return Values : TRUE - field is valid
3126 | FALSE - field is invalid.
3127 +--------------------------------------------------------------------------*/
3129 Check_Field(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
3133 if (Field_Has_Option(field, O_NULLOK))
3135 FIELD_CELL *bp = field->buf;
3138 while (ISBLANK(*bp))
3142 if (CharOf(*bp) == 0)
3146 if (typ->status & _LINKED_TYPE)
3150 Check_Field(form, typ->left, field, argp->left) ||
3151 Check_Field(form, typ->right, field, argp->right));
3155 #if NCURSES_INTEROP_FUNCS
3156 if (typ->fieldcheck.ofcheck)
3158 if (typ->status & _GENERIC)
3159 return typ->fieldcheck.gfcheck(form, field, (void *)argp);
3161 return typ->fieldcheck.ofcheck(field, (void *)argp);
3165 return typ->fcheck(field, (void *)argp);
3172 /*---------------------------------------------------------------------------
3173 | Facility : libnform
3174 | Function : bool _nc_Internal_Validation(FORM * form )
3176 | Description : Validate the current field of the form.
3178 | Return Values : TRUE - field is valid
3179 | FALSE - field is invalid
3180 +--------------------------------------------------------------------------*/
3181 NCURSES_EXPORT(bool)
3182 _nc_Internal_Validation(FORM *form)
3186 field = form->current;
3188 Synchronize_Buffer(form);
3189 if ((form->status & _FCHECK_REQUIRED) ||
3190 (!(Field_Has_Option(field, O_PASSOK))))
3192 if (!Check_Field(form, field->type, field, (TypeArgument *)(field->arg)))
3194 ClrStatus(form, _FCHECK_REQUIRED);
3195 SetStatus(field, _CHANGED);
3196 Synchronize_Linked_Fields(field);
3200 /*----------------------------------------------------------------------------
3201 End of Helper routines for Field Validations.
3202 --------------------------------------------------------------------------*/
3204 /*----------------------------------------------------------------------------
3205 Routines for Field Validation.
3206 --------------------------------------------------------------------------*/
3208 /*---------------------------------------------------------------------------
3209 | Facility : libnform
3210 | Function : static int FV_Validation(FORM * form)
3212 | Description : Validate the current field of the form.
3214 | Return Values : E_OK - field valid
3215 | E_INVALID_FIELD - field not valid
3216 +--------------------------------------------------------------------------*/
3218 FV_Validation(FORM *form)
3220 T((T_CALLED("FV_Validation(%p)"), (void *)form));
3221 if (_nc_Internal_Validation(form))
3224 returnCode(E_INVALID_FIELD);
3226 /*----------------------------------------------------------------------------
3227 End of routines for Field Validation.
3228 --------------------------------------------------------------------------*/
3230 /*----------------------------------------------------------------------------
3231 Helper routines for Inter-Field Navigation
3232 --------------------------------------------------------------------------*/
3234 /*---------------------------------------------------------------------------
3235 | Facility : libnform
3236 | Function : static FIELD *Next_Field_On_Page(FIELD * field)
3238 | Description : Get the next field after the given field on the current
3239 | page. The order of fields is the one defined by the
3240 | fields array. Only visible and active fields are
3243 | Return Values : Pointer to the next field.
3244 +--------------------------------------------------------------------------*/
3245 NCURSES_INLINE static FIELD *
3246 Next_Field_On_Page(FIELD *field)
3248 FORM *form = field->form;
3249 FIELD **field_on_page = &form->field[field->index];
3250 FIELD **first_on_page = &form->field[form->page[form->curpage].pmin];
3251 FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3256 (field_on_page == last_on_page) ? first_on_page : field_on_page + 1;
3257 if (Field_Is_Selectable(*field_on_page))
3260 while (field != (*field_on_page));
3261 return (*field_on_page);
3264 /*---------------------------------------------------------------------------
3265 | Facility : libnform
3266 | Function : FIELD* _nc_First_Active_Field(FORM * form)
3268 | Description : Get the first active field on the current page,
3269 | if there are such. If there are none, get the first
3270 | visible field on the page. If there are also none,
3271 | we return the first field on page and hope the best.
3273 | Return Values : Pointer to calculated field.
3274 +--------------------------------------------------------------------------*/
3275 NCURSES_EXPORT(FIELD *)
3276 _nc_First_Active_Field(FORM *form)
3278 FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3279 FIELD *proposed = Next_Field_On_Page(*last_on_page);
3281 if (proposed == *last_on_page)
3283 /* there might be the special situation, where there is no
3284 active and visible field on the current page. We then select
3285 the first visible field on this readonly page
3287 if (Field_Is_Not_Selectable(proposed))
3289 FIELD **field = &form->field[proposed->index];
3290 FIELD **first = &form->field[form->page[form->curpage].pmin];
3294 field = (field == last_on_page) ? first : field + 1;
3295 if (Field_Has_Option(*field, O_VISIBLE))
3298 while (proposed != (*field));
3302 if ((proposed == *last_on_page) &&
3303 !((unsigned)proposed->opts & O_VISIBLE))
3305 /* This means, there is also no visible field on the page.
3306 So we propose the first one and hope the very best...
3307 Some very clever user has designed a readonly and invisible
3317 /*---------------------------------------------------------------------------
3318 | Facility : libnform
3319 | Function : static FIELD *Previous_Field_On_Page(FIELD * field)
3321 | Description : Get the previous field before the given field on the
3322 | current page. The order of fields is the one defined by
3323 | the fields array. Only visible and active fields are
3326 | Return Values : Pointer to the previous field.
3327 +--------------------------------------------------------------------------*/
3328 NCURSES_INLINE static FIELD *
3329 Previous_Field_On_Page(FIELD *field)
3331 FORM *form = field->form;
3332 FIELD **field_on_page = &form->field[field->index];
3333 FIELD **first_on_page = &form->field[form->page[form->curpage].pmin];
3334 FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3339 (field_on_page == first_on_page) ? last_on_page : field_on_page - 1;
3340 if (Field_Is_Selectable(*field_on_page))
3343 while (field != (*field_on_page));
3345 return (*field_on_page);
3348 /*---------------------------------------------------------------------------
3349 | Facility : libnform
3350 | Function : static FIELD *Sorted_Next_Field(FIELD * field)
3352 | Description : Get the next field after the given field on the current
3353 | page. The order of fields is the one defined by the
3354 | (row,column) geometry, rows are major.
3356 | Return Values : Pointer to the next field.
3357 +--------------------------------------------------------------------------*/
3358 NCURSES_INLINE static FIELD *
3359 Sorted_Next_Field(FIELD *field)
3361 FIELD *field_on_page = field;
3365 field_on_page = field_on_page->snext;
3366 if (Field_Is_Selectable(field_on_page))
3369 while (field_on_page != field);
3371 return (field_on_page);
3374 /*---------------------------------------------------------------------------
3375 | Facility : libnform
3376 | Function : static FIELD *Sorted_Previous_Field(FIELD * field)
3378 | Description : Get the previous field before the given field on the
3379 | current page. The order of fields is the one defined
3380 | by the (row,column) geometry, rows are major.
3382 | Return Values : Pointer to the previous field.
3383 +--------------------------------------------------------------------------*/
3384 NCURSES_INLINE static FIELD *
3385 Sorted_Previous_Field(FIELD *field)
3387 FIELD *field_on_page = field;
3391 field_on_page = field_on_page->sprev;
3392 if (Field_Is_Selectable(field_on_page))
3395 while (field_on_page != field);
3397 return (field_on_page);
3400 /*---------------------------------------------------------------------------
3401 | Facility : libnform
3402 | Function : static FIELD *Left_Neighbor_Field(FIELD * field)
3404 | Description : Get the left neighbor of the field on the same line
3405 | and the same page. Cycles through the line.
3407 | Return Values : Pointer to left neighbor field.
3408 +--------------------------------------------------------------------------*/
3409 NCURSES_INLINE static FIELD *
3410 Left_Neighbor_Field(FIELD *field)
3412 FIELD *field_on_page = field;
3414 /* For a field that has really a left neighbor, the while clause
3415 immediately fails and the loop is left, positioned at the right
3416 neighbor. Otherwise we cycle backwards through the sorted field list
3417 until we enter the same line (from the right end).
3421 field_on_page = Sorted_Previous_Field(field_on_page);
3423 while (field_on_page->frow != field->frow);
3425 return (field_on_page);
3428 /*---------------------------------------------------------------------------
3429 | Facility : libnform
3430 | Function : static FIELD *Right_Neighbor_Field(FIELD * field)
3432 | Description : Get the right neighbor of the field on the same line
3433 | and the same page.
3435 | Return Values : Pointer to right neighbor field.
3436 +--------------------------------------------------------------------------*/
3437 NCURSES_INLINE static FIELD *
3438 Right_Neighbor_Field(FIELD *field)
3440 FIELD *field_on_page = field;
3442 /* See the comments on Left_Neighbor_Field to understand how it works */
3445 field_on_page = Sorted_Next_Field(field_on_page);
3447 while (field_on_page->frow != field->frow);
3449 return (field_on_page);
3452 /*---------------------------------------------------------------------------
3453 | Facility : libnform
3454 | Function : static FIELD *Upper_Neighbor_Field(FIELD * field)
3456 | Description : Because of the row-major nature of sorting the fields,
3457 | it is more difficult to define whats the upper neighbor
3458 | field really means. We define that it must be on a
3459 | 'previous' line (cyclic order!) and is the rightmost
3460 | field laying on the left side of the given field. If
3461 | this set is empty, we take the first field on the line.
3463 | Return Values : Pointer to the upper neighbor field.
3464 +--------------------------------------------------------------------------*/
3466 Upper_Neighbor_Field(FIELD *field)
3468 FIELD *field_on_page = field;
3469 int frow = field->frow;
3470 int fcol = field->fcol;
3472 /* Walk back to the 'previous' line. The second term in the while clause
3473 just guarantees that we stop if we cycled through the line because
3474 there might be no 'previous' line if the page has just one line.
3478 field_on_page = Sorted_Previous_Field(field_on_page);
3480 while (field_on_page->frow == frow && field_on_page->fcol != fcol);
3482 if (field_on_page->frow != frow)
3484 /* We really found a 'previous' line. We are positioned at the
3485 rightmost field on this line */
3486 frow = field_on_page->frow;
3488 /* We walk to the left as long as we are really right of the
3490 while (field_on_page->frow == frow && field_on_page->fcol > fcol)
3491 field_on_page = Sorted_Previous_Field(field_on_page);
3493 /* If we wrapped, just go to the right which is the first field on
3495 if (field_on_page->frow != frow)
3496 field_on_page = Sorted_Next_Field(field_on_page);
3499 return (field_on_page);
3502 /*---------------------------------------------------------------------------
3503 | Facility : libnform
3504 | Function : static FIELD *Down_Neighbor_Field(FIELD * field)
3506 | Description : Because of the row-major nature of sorting the fields,
3507 | its more difficult to define whats the down neighbor
3508 | field really means. We define that it must be on a
3509 | 'next' line (cyclic order!) and is the leftmost
3510 | field laying on the right side of the given field. If
3511 | this set is empty, we take the last field on the line.
3513 | Return Values : Pointer to the upper neighbor field.
3514 +--------------------------------------------------------------------------*/
3516 Down_Neighbor_Field(FIELD *field)
3518 FIELD *field_on_page = field;
3519 int frow = field->frow;
3520 int fcol = field->fcol;
3522 /* Walk forward to the 'next' line. The second term in the while clause
3523 just guarantees that we stop if we cycled through the line because
3524 there might be no 'next' line if the page has just one line.
3528 field_on_page = Sorted_Next_Field(field_on_page);
3530 while (field_on_page->frow == frow && field_on_page->fcol != fcol);
3532 if (field_on_page->frow != frow)
3534 /* We really found a 'next' line. We are positioned at the rightmost
3535 field on this line */
3536 frow = field_on_page->frow;
3538 /* We walk to the right as long as we are really left of the
3540 while (field_on_page->frow == frow && field_on_page->fcol < fcol)
3541 field_on_page = Sorted_Next_Field(field_on_page);
3543 /* If we wrapped, just go to the left which is the last field on
3545 if (field_on_page->frow != frow)
3546 field_on_page = Sorted_Previous_Field(field_on_page);
3549 return (field_on_page);
3552 /*----------------------------------------------------------------------------
3553 Inter-Field Navigation routines
3554 --------------------------------------------------------------------------*/
3556 /*---------------------------------------------------------------------------
3557 | Facility : libnform
3558 | Function : static int Inter_Field_Navigation(
3559 | int (* const fct) (FORM *),
3562 | Description : Generic behavior for changing the current field, the
3563 | field is left and a new field is entered. So the field
3564 | must be validated and the field init/term hooks must
3567 | Return Values : E_OK - success
3568 | E_INVALID_FIELD - field is invalid
3569 | some other - error from subordinate call
3570 +--------------------------------------------------------------------------*/
3572 Inter_Field_Navigation(int (*const fct) (FORM *), FORM *form)
3576 if (!_nc_Internal_Validation(form))
3577 res = E_INVALID_FIELD;
3580 Call_Hook(form, fieldterm);
3582 Call_Hook(form, fieldinit);
3587 /*---------------------------------------------------------------------------
3588 | Facility : libnform
3589 | Function : static int FN_Next_Field(FORM * form)
3591 | Description : Move to the next field on the current page of the form
3593 | Return Values : E_OK - success
3594 | != E_OK - error from subordinate call
3595 +--------------------------------------------------------------------------*/
3597 FN_Next_Field(FORM *form)
3599 T((T_CALLED("FN_Next_Field(%p)"), (void *)form));
3600 returnCode(_nc_Set_Current_Field(form,
3601 Next_Field_On_Page(form->current)));
3604 /*---------------------------------------------------------------------------
3605 | Facility : libnform
3606 | Function : static int FN_Previous_Field(FORM * form)
3608 | Description : Move to the previous field on the current page of the
3611 | Return Values : E_OK - success
3612 | != E_OK - error from subordinate call
3613 +--------------------------------------------------------------------------*/
3615 FN_Previous_Field(FORM *form)
3617 T((T_CALLED("FN_Previous_Field(%p)"), (void *)form));
3618 returnCode(_nc_Set_Current_Field(form,
3619 Previous_Field_On_Page(form->current)));
3622 /*---------------------------------------------------------------------------
3623 | Facility : libnform
3624 | Function : static int FN_First_Field(FORM * form)
3626 | Description : Move to the first field on the current page of the form
3628 | Return Values : E_OK - success
3629 | != E_OK - error from subordinate call
3630 +--------------------------------------------------------------------------*/
3632 FN_First_Field(FORM *form)
3634 T((T_CALLED("FN_First_Field(%p)"), (void *)form));
3635 returnCode(_nc_Set_Current_Field(form,
3636 Next_Field_On_Page(form->field[form->page[form->curpage].pmax])));
3639 /*---------------------------------------------------------------------------
3640 | Facility : libnform
3641 | Function : static int FN_Last_Field(FORM * form)
3643 | Description : Move to the last field on the current page of the form
3645 | Return Values : E_OK - success
3646 | != E_OK - error from subordinate call
3647 +--------------------------------------------------------------------------*/
3649 FN_Last_Field(FORM *form)
3651 T((T_CALLED("FN_Last_Field(%p)"), (void *)form));
3653 _nc_Set_Current_Field(form,
3654 Previous_Field_On_Page(form->field[form->page[form->curpage].pmin])));
3657 /*---------------------------------------------------------------------------
3658 | Facility : libnform
3659 | Function : static int FN_Sorted_Next_Field(FORM * form)
3661 | Description : Move to the sorted next field on the current page
3664 | Return Values : E_OK - success
3665 | != E_OK - error from subordinate call
3666 +--------------------------------------------------------------------------*/
3668 FN_Sorted_Next_Field(FORM *form)
3670 T((T_CALLED("FN_Sorted_Next_Field(%p)"), (void *)form));
3671 returnCode(_nc_Set_Current_Field(form,
3672 Sorted_Next_Field(form->current)));
3675 /*---------------------------------------------------------------------------
3676 | Facility : libnform
3677 | Function : static int FN_Sorted_Previous_Field(FORM * form)
3679 | Description : Move to the sorted previous field on the current page
3682 | Return Values : E_OK - success
3683 | != E_OK - error from subordinate call
3684 +--------------------------------------------------------------------------*/
3686 FN_Sorted_Previous_Field(FORM *form)
3688 T((T_CALLED("FN_Sorted_Previous_Field(%p)"), (void *)form));
3689 returnCode(_nc_Set_Current_Field(form,
3690 Sorted_Previous_Field(form->current)));
3693 /*---------------------------------------------------------------------------
3694 | Facility : libnform