xref: /freebsd/contrib/ncurses/form/frm_driver.c (revision 9207f9d206a4017001f01ca27d3d25a26c268a95)
1 /****************************************************************************
2  * Copyright 2018-2020,2021 Thomas E. Dickey                                *
3  * Copyright 1998-2016,2017 Free Software Foundation, Inc.                  *
4  *                                                                          *
5  * Permission is hereby granted, free of charge, to any person obtaining a  *
6  * copy of this software and associated documentation files (the            *
7  * "Software"), to deal in the Software without restriction, including      *
8  * without limitation the rights to use, copy, modify, merge, publish,      *
9  * distribute, distribute with modifications, sublicense, and/or sell       *
10  * copies of the Software, and to permit persons to whom the Software is    *
11  * furnished to do so, subject to the following conditions:                 *
12  *                                                                          *
13  * The above copyright notice and this permission notice shall be included  *
14  * in all copies or substantial portions of the Software.                   *
15  *                                                                          *
16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS  *
17  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF               *
18  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.   *
19  * IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,   *
20  * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR    *
21  * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR    *
22  * THE USE OR OTHER DEALINGS IN THE SOFTWARE.                               *
23  *                                                                          *
24  * Except as contained in this notice, the name(s) of the above copyright   *
25  * holders shall not be used in advertising or otherwise to promote the     *
26  * sale, use or other dealings in this Software without prior written       *
27  * authorization.                                                           *
28  ****************************************************************************/
29 
30 /****************************************************************************
31  *   Author:  Juergen Pfeifer, 1995,1997                                    *
32  ****************************************************************************/
33 
34 #include "form.priv.h"
35 
36 MODULE_ID("$Id: frm_driver.c,v 1.135 2021/09/01 23:34:01 tom Exp $")
37 
38 /*----------------------------------------------------------------------------
39   This is the core module of the form library. It contains the majority
40   of the driver routines as well as the form_driver function.
41 
42   Essentially this module is nearly the whole library. This is because
43   all the functions in this module depends on some others in the module,
44   so it makes no sense to split them into separate files because they
45   will always be linked together. The only acceptable concern is turnaround
46   time for this module, but now we have all Pentiums or RISCs, so what!
47 
48   The driver routines are grouped into nine generic categories:
49 
50    a)   Page Navigation            ( all functions prefixed by PN_ )
51         The current page of the form is left and some new page is
52         entered.
53    b)   Inter-Field Navigation     ( all functions prefixed by FN_ )
54         The current field of the form is left and some new field is
55         entered.
56    c)   Intra-Field Navigation     ( all functions prefixed by IFN_ )
57         The current position in the current field is changed.
58    d)   Vertical Scrolling         ( all functions prefixed by VSC_ )
59         Essentially this is a specialization of Intra-Field navigation.
60         It has to check for a multi-line field.
61    e)   Horizontal Scrolling       ( all functions prefixed by HSC_ )
62         Essentially this is a specialization of Intra-Field navigation.
63         It has to check for a single-line field.
64    f)   Field Editing              ( all functions prefixed by FE_ )
65         The content of the current field is changed
66    g)   Edit Mode requests         ( all functions prefixed by EM_ )
67         Switching between insert and overlay mode
68    h)   Field-Validation requests  ( all functions prefixed by FV_ )
69         Perform verifications of the field.
70    i)   Choice requests            ( all functions prefixed by CR_ )
71         Requests to enumerate possible field values
72   --------------------------------------------------------------------------*/
73 
74 /*----------------------------------------------------------------------------
75   Some remarks on the placements of assert() macros :
76   I use them only on "strategic" places, i.e. top level entries where
77   I want to make sure that things are set correctly. Throughout subordinate
78   routines I omit them mostly.
79   --------------------------------------------------------------------------*/
80 
81 /*
82 Some options that may effect compatibility in behavior to SVr4 forms,
83 but they are here to allow a more intuitive and user friendly behavior of
84 our form implementation. This doesn't affect the API, so we feel it is
85 uncritical.
86 
87 The initial implementation tries to stay very close with the behavior
88 of the original SVr4 implementation, although in some areas it is quite
89 clear that this isn't the most appropriate way. As far as possible this
90 sources will allow you to build a forms lib that behaves quite similar
91 to SVr4, but now and in the future we will give you better options.
92 Perhaps at some time we will make this configurable at runtime.
93 */
94 
95 /* Implement a more user-friendly previous/next word behavior */
96 #define FRIENDLY_PREV_NEXT_WORD (1)
97 /* Fix the wrong behavior for forms with all fields inactive */
98 #define FIX_FORM_INACTIVE_BUG (1)
99 /* Allow dynamic field growth also when navigating past the end */
100 #define GROW_IF_NAVIGATE (1)
101 
102 #if USE_WIDEC_SUPPORT
103 #define myADDNSTR(w, s, n) wide_waddnstr(w, s, n)
104 #define myINSNSTR(w, s, n) wide_winsnstr(w, s, n)
105 #define myINNSTR(w, s, n)  wide_winnstr(w, s, n)
106 #define myWCWIDTH(w, y, x) cell_width(w, y, x)
107 #else
108 #define myADDNSTR(w, s, n) waddnstr(w, s, n)
109 #define myINSNSTR(w, s, n) winsnstr(w, s, n)
110 #define myINNSTR(w, s, n)  winnstr(w, s, n)
111 #define myWCWIDTH(w, y, x) 1
112 #endif
113 
114 /*----------------------------------------------------------------------------
115   Forward references to some internally used static functions
116   --------------------------------------------------------------------------*/
117 static int Inter_Field_Navigation(int (*const fct) (FORM *), FORM *form);
118 static int FN_Next_Field(FORM *form);
119 static int FN_Previous_Field(FORM *form);
120 static int FE_New_Line(FORM *);
121 static int FE_Delete_Previous(FORM *);
122 
123 /*----------------------------------------------------------------------------
124   Macro Definitions.
125 
126   Some Remarks on that: I use the convention to use UPPERCASE for constants
127   defined by Macros. If I provide a macro as a kind of inline routine to
128   provide some logic, I use my Upper_Lower case style.
129   --------------------------------------------------------------------------*/
130 
131 /* Calculate the position of a single row in a field buffer */
132 #define Position_Of_Row_In_Buffer(field,row) ((row)*(field)->dcols)
133 
134 /* Calculate start address for the field's buffer# N */
135 #define Address_Of_Nth_Buffer(field,N) \
136   ((field)->buf + (N)*(1+Buffer_Length(field)))
137 
138 /* Calculate the start address of the row in the field's specified buffer# N */
139 #define Address_Of_Row_In_Nth_Buffer(field,N,row) \
140   (Address_Of_Nth_Buffer(field,N) + Position_Of_Row_In_Buffer(field,row))
141 
142 /* Calculate the start address of the row in the field's primary buffer */
143 #define Address_Of_Row_In_Buffer(field,row) \
144   Address_Of_Row_In_Nth_Buffer(field,0,row)
145 
146 /* Calculate the start address of the row in the form's current field
147    buffer# N */
148 #define Address_Of_Current_Row_In_Nth_Buffer(form,N) \
149    Address_Of_Row_In_Nth_Buffer((form)->current,N,(form)->currow)
150 
151 /* Calculate the start address of the row in the form's current field
152    primary buffer */
153 #define Address_Of_Current_Row_In_Buffer(form) \
154    Address_Of_Current_Row_In_Nth_Buffer(form,0)
155 
156 /* Calculate the address of the cursor in the form's current field
157    primary buffer */
158 #define Address_Of_Current_Position_In_Nth_Buffer(form,N) \
159    (Address_Of_Current_Row_In_Nth_Buffer(form,N) + (form)->curcol)
160 
161 /* Calculate the address of the cursor in the form's current field
162    buffer# N */
163 #define Address_Of_Current_Position_In_Buffer(form) \
164   Address_Of_Current_Position_In_Nth_Buffer(form,0)
165 
166 /* Logic to decide whether or not a field is actually a field with
167    vertical or horizontal scrolling */
168 #define Is_Scroll_Field(field)          \
169    (((field)->drows > (field)->rows) || \
170     ((field)->dcols > (field)->cols))
171 
172 /* Logic to decide whether or not a field needs to have an individual window
173    instead of a derived window because it contains invisible parts.
174    This is true for non-public fields and for scrollable fields. */
175 #define Has_Invisible_Parts(field)     \
176   (!(Field_Has_Option(field, O_PUBLIC)) || \
177    Is_Scroll_Field(field))
178 
179 /* Logic to decide whether or not a field needs justification */
180 #define Justification_Allowed(field)        \
181    (((field)->just != NO_JUSTIFICATION)  && \
182     (Single_Line_Field(field))           && \
183     ((Field_Has_Option(field, O_STATIC)  && \
184      ((field)->dcols == (field)->cols))  || \
185     Field_Has_Option(field, O_DYNAMIC_JUSTIFY)))
186 
187 /* Logic to determine whether or not a dynamic field may still grow */
188 #define Growable(field) ((field)->status & _MAY_GROW)
189 
190 /* Macro to set the attributes for a field's window */
191 #define Set_Field_Window_Attributes(field,win) \
192 (  wbkgdset((win),(chtype)((chtype)((field)->pad) | (field)->back)), \
193    (void) wattrset((win), (int)(field)->fore) )
194 
195 /* Logic to decide whether or not a field really appears on the form */
196 #define Field_Really_Appears(field)         \
197   ((field->form)                          &&\
198    (field->form->status & _POSTED)        &&\
199    (Field_Has_Option(field, O_VISIBLE))   &&\
200    (field->page == field->form->curpage))
201 
202 /* Logic to determine whether or not we are on the first position in the
203    current field */
204 #define First_Position_In_Current_Field(form) \
205   (((form)->currow==0) && ((form)->curcol==0))
206 
207 #define Minimum(a,b) (((a)<=(b)) ? (a) : (b))
208 #define Maximum(a,b) (((a)>=(b)) ? (a) : (b))
209 
210 /*----------------------------------------------------------------------------
211   Useful constants
212   --------------------------------------------------------------------------*/
213 static FIELD_CELL myBLANK = BLANK;
214 static FIELD_CELL myZEROS;
215 
216 #ifdef TRACE
217 static void
218 check_pos(FORM *form, int lineno)
219 {
220   if (form && form->w)
221     {
222       int y, x;
223 
224       getyx(form->w, y, x);
225       if (y != form->currow || x != form->curcol)
226 	{
227 	  T(("CHECKPOS %s@%d have position %d,%d vs want %d,%d",
228 	     __FILE__, lineno,
229 	     y, x,
230 	     form->currow, form->curcol));
231 	}
232     }
233 }
234 #define CHECKPOS(form) check_pos(form, __LINE__)
235 #else
236 #define CHECKPOS(form)		/* nothing */
237 #endif
238 
239 /*----------------------------------------------------------------------------
240   Wide-character special functions
241   --------------------------------------------------------------------------*/
242 #if USE_WIDEC_SUPPORT
243 /* add like waddnstr, but using cchar_t* rather than char*
244  */
245 static int
246 wide_waddnstr(WINDOW *w, const cchar_t *s, int n)
247 {
248   int rc = OK;
249 
250   while (n-- > 0)
251     {
252       if ((rc = wadd_wch(w, s)) != OK)
253 	break;
254       ++s;
255     }
256   return rc;
257 }
258 
259 /* insert like winsnstr, but using cchar_t* rather than char*
260  *
261  * X/Open Curses has no close equivalent; inserts are done only with wchar_t
262  * strings.
263  */
264 static int
265 wide_winsnstr(WINDOW *w, const cchar_t *s, int n)
266 {
267   int code = ERR;
268 
269   while (n-- > 0)
270     {
271       int y, x;
272 
273       getyx(w, y, x);
274       if ((code = wins_wch(w, s++)) != OK)
275 	break;
276       if ((code = wmove(w, y, x + 1)) != OK)
277 	break;
278     }
279   return code;
280 }
281 
282 /* retrieve like winnstr, but using cchar_t*, rather than char*.
283  *
284  * X/Open Curses' closest equivalent, win_wchnstr(), is inconsistent with
285  * winnstr(), since it returns OK rather than the number of items transferred.
286  */
287 static int
288 wide_winnstr(WINDOW *w, cchar_t *s, int n)
289 {
290   int x;
291 
292   win_wchnstr(w, s, n);
293   /*
294    * This function is used to extract the text only from the window.
295    * Strip attributes and color from the string so they will not be added
296    * back when copying the string to the window.
297    */
298   for (x = 0; x < n; ++x)
299     {
300       RemAttr(s[x], A_ATTRIBUTES);
301       SetPair(s[x], 0);
302     }
303   return n;
304 }
305 
306 /*
307  * Returns the column of the base of the given cell.
308  */
309 static int
310 cell_base(WINDOW *win, int y, int x)
311 {
312   int result = x;
313 
314   while (LEGALYX(win, y, x))
315     {
316       cchar_t *data = &(win->_line[y].text[x]);
317 
318       if (isWidecBase(CHDEREF(data)) || !isWidecExt(CHDEREF(data)))
319 	{
320 	  result = x;
321 	  break;
322 	}
323       --x;
324     }
325   return result;
326 }
327 
328 /*
329  * Returns the number of columns needed for the given cell in a window.
330  */
331 static int
332 cell_width(WINDOW *win, int y, int x)
333 {
334   int result = 1;
335 
336   if (LEGALYX(win, y, x))
337     {
338       cchar_t *data = &(win->_line[y].text[x]);
339 
340       if (isWidecExt(CHDEREF(data)))
341 	{
342 	  /* recur, providing the number of columns to the next character */
343 	  result = cell_width(win, y, x - 1);
344 	}
345       else
346 	{
347 	  result = wcwidth(CharOf(CHDEREF(data)));
348 	}
349     }
350   return result;
351 }
352 
353 /*
354  * There is no wide-character function such as wdel_wch(), so we must find
355  * all of the cells that comprise a multi-column character and delete them
356  * one-by-one.
357  */
358 static void
359 delete_char(FORM *form)
360 {
361   int cells = cell_width(form->w, form->currow, form->curcol);
362 
363   form->curcol = cell_base(form->w, form->currow, form->curcol);
364   wmove(form->w, form->currow, form->curcol);
365   while (cells-- > 0)
366     {
367       wdelch(form->w);
368     }
369 }
370 #define DeleteChar(form) delete_char(form)
371 #else
372 #define DeleteChar(form) \
373 	  wmove((form)->w, (form)->currow, (form)->curcol), \
374 	  wdelch((form)->w)
375 #endif
376 
377 /*---------------------------------------------------------------------------
378 |   Facility      :  libnform
379 |   Function      :  static char *Get_Start_Of_Data(char * buf, int blen)
380 |
381 |   Description   :  Return pointer to first non-blank position in buffer.
382 |                    If buffer is empty return pointer to buffer itself.
383 |
384 |   Return Values :  Pointer to first non-blank position in buffer
385 +--------------------------------------------------------------------------*/
386 NCURSES_INLINE static FIELD_CELL *
387 Get_Start_Of_Data(FIELD_CELL *buf, int blen)
388 {
389   FIELD_CELL *p = buf;
390   FIELD_CELL *end = &buf[blen];
391 
392   assert(buf && blen >= 0);
393   while ((p < end) && ISBLANK(*p))
394     p++;
395   return ((p == end) ? buf : p);
396 }
397 
398 /*---------------------------------------------------------------------------
399 |   Facility      :  libnform
400 |   Function      :  static char *After_End_Of_Data(char * buf, int blen)
401 |
402 |   Description   :  Return pointer after last non-blank position in buffer.
403 |                    If buffer is empty, return pointer to buffer itself.
404 |
405 |   Return Values :  Pointer to position after last non-blank position in
406 |                    buffer.
407 +--------------------------------------------------------------------------*/
408 NCURSES_INLINE static FIELD_CELL *
409 After_End_Of_Data(FIELD_CELL *buf, int blen)
410 {
411   FIELD_CELL *p = &buf[blen];
412 
413   assert(buf && blen >= 0);
414   while ((p > buf) && ISBLANK(p[-1]))
415     p--;
416   return (p);
417 }
418 
419 /*---------------------------------------------------------------------------
420 |   Facility      :  libnform
421 |   Function      :  static char *Get_First_Whitespace_Character(
422 |                                     char * buf, int   blen)
423 |
424 |   Description   :  Position to the first whitespace character.
425 |
426 |   Return Values :  Pointer to first whitespace character in buffer.
427 +--------------------------------------------------------------------------*/
428 NCURSES_INLINE static FIELD_CELL *
429 Get_First_Whitespace_Character(FIELD_CELL *buf, int blen)
430 {
431   FIELD_CELL *p = buf;
432   FIELD_CELL *end = &p[blen];
433 
434   assert(buf && blen >= 0);
435   while ((p < end) && !ISBLANK(*p))
436     p++;
437   return ((p == end) ? buf : p);
438 }
439 
440 /*---------------------------------------------------------------------------
441 |   Facility      :  libnform
442 |   Function      :  static char *After_Last_Whitespace_Character(
443 |                                     char * buf, int blen)
444 |
445 |   Description   :  Get the position after the last whitespace character.
446 |
447 |   Return Values :  Pointer to position after last whitespace character in
448 |                    buffer.
449 +--------------------------------------------------------------------------*/
450 NCURSES_INLINE static FIELD_CELL *
451 After_Last_Whitespace_Character(FIELD_CELL *buf, int blen)
452 {
453   FIELD_CELL *p = &buf[blen];
454 
455   assert(buf && blen >= 0);
456   while ((p > buf) && !ISBLANK(p[-1]))
457     p--;
458   return (p);
459 }
460 
461 /* Set this to 1 to use the div_t version. This is a good idea if your
462    compiler has an intrinsic div() support. Unfortunately GNU-C has it
463    not yet.
464    N.B.: This only works if form->curcol follows immediately form->currow
465          and both are of type int.
466 */
467 #define USE_DIV_T (0)
468 
469 /*---------------------------------------------------------------------------
470 |   Facility      :  libnform
471 |   Function      :  static void Adjust_Cursor_Position(
472 |                                       FORM * form, const char * pos)
473 |
474 |   Description   :  Set current row and column of the form to values
475 |                    corresponding to the buffer position.
476 |
477 |   Return Values :  -
478 +--------------------------------------------------------------------------*/
479 NCURSES_INLINE static void
480 Adjust_Cursor_Position(FORM *form, const FIELD_CELL *pos)
481 {
482   FIELD *field;
483   int idx;
484 
485   field = form->current;
486   assert(pos >= field->buf && field->dcols > 0);
487   idx = (int)(pos - field->buf);
488 #if USE_DIV_T
489   *((div_t *) & (form->currow)) = div(idx, field->dcols);
490 #else
491   form->currow = idx / field->dcols;
492   form->curcol = idx - field->cols * form->currow;
493 #endif
494   if (field->drows < form->currow)
495     form->currow = 0;
496 }
497 
498 /*---------------------------------------------------------------------------
499 |   Facility      :  libnform
500 |   Function      :  static void Buffer_To_Window(
501 |                                      const FIELD  * field,
502 |                                      WINDOW * win)
503 |
504 |   Description   :  Copy the buffer to the window. If it is a multi-line
505 |                    field, the buffer is split to the lines of the
506 |                    window without any editing.
507 |
508 |   Return Values :  -
509 +--------------------------------------------------------------------------*/
510 static void
511 Buffer_To_Window(const FIELD *field, WINDOW *win)
512 {
513   int width, height;
514   int y, x;
515   int row;
516   FIELD_CELL *pBuffer;
517 
518   assert(win && field);
519 
520   getyx(win, y, x);
521   width = getmaxx(win);
522   height = getmaxy(win);
523 
524   for (row = 0, pBuffer = field->buf;
525        row < height;
526        row++, pBuffer += width)
527     {
528       int len;
529 
530       if ((len = (int)(After_End_Of_Data(pBuffer, width) - pBuffer)) > 0)
531 	{
532 	  wmove(win, row, 0);
533 	  myADDNSTR(win, pBuffer, len);
534 	}
535     }
536   wmove(win, y, x);
537 }
538 
539 /*---------------------------------------------------------------------------
540 |   Facility      :  libnform
541 |   Function      :  void _nc_get_fieldbuffer(
542 |                                          WINDOW * win,
543 |                                          FIELD  * field,
544 |                                          FIELD_CELL * buf)
545 |
546 |   Description   :  Copy the content of the window into the buffer.
547 |                    The multiple lines of a window are simply
548 |                    concatenated into the buffer. Pad characters in
549 |                    the window will be replaced by blanks in the buffer.
550 |
551 |   Return Values :  -
552 +--------------------------------------------------------------------------*/
553 FORM_EXPORT(void)
554 _nc_get_fieldbuffer(FORM *form, FIELD *field, FIELD_CELL *buf)
555 {
556   int pad;
557   int len = 0;
558   FIELD_CELL *p;
559   int row, height;
560   WINDOW *win;
561 
562   assert(form && field && buf);
563 
564   win = form->w;
565   assert(win);
566 
567   pad = field->pad;
568   p = buf;
569   height = getmaxy(win);
570 
571   for (row = 0; (row < height) && (row < field->drows); row++)
572     {
573       wmove(win, row, 0);
574       len += myINNSTR(win, p + len, field->dcols);
575     }
576   p[len] = myZEROS;
577 
578   /* replace visual padding character by blanks in buffer */
579   if (pad != C_BLANK)
580     {
581       int i;
582 
583       for (i = 0; i < len; i++, p++)
584 	{
585 	  if ((unsigned long)CharOf(*p) == ChCharOf(pad)
586 #if USE_WIDEC_SUPPORT
587 	      && p->chars[1] == 0
588 #endif
589 	    )
590 	    *p = myBLANK;
591 	}
592     }
593 }
594 
595 /*---------------------------------------------------------------------------
596 |   Facility      :  libnform
597 |   Function      :  static void Window_To_Buffer(
598 |                                          FORM   * form,
599 |                                          FIELD  * field)
600 |
601 |   Description   :  Copy the content of the window into the buffer.
602 |                    The multiple lines of a window are simply
603 |                    concatenated into the buffer. Pad characters in
604 |                    the window will be replaced by blanks in the buffer.
605 |
606 |   Return Values :  -
607 +--------------------------------------------------------------------------*/
608 static void
609 Window_To_Buffer(FORM *form, FIELD *field)
610 {
611   _nc_get_fieldbuffer(form, field, field->buf);
612 }
613 
614 /*---------------------------------------------------------------------------
615 |   Facility      :  libnform
616 |   Function      :  static void Synchronize_Buffer(FORM * form)
617 |
618 |   Description   :  If there was a change, copy the content of the
619 |                    window into the buffer, so the buffer is synchronized
620 |                    with the windows content. We have to indicate that the
621 |                    buffer needs validation due to the change.
622 |
623 |   Return Values :  -
624 +--------------------------------------------------------------------------*/
625 NCURSES_INLINE static void
626 Synchronize_Buffer(FORM *form)
627 {
628   if (form->status & _WINDOW_MODIFIED)
629     {
630       ClrStatus(form, _WINDOW_MODIFIED);
631       SetStatus(form, _FCHECK_REQUIRED);
632       Window_To_Buffer(form, form->current);
633       wmove(form->w, form->currow, form->curcol);
634     }
635 }
636 
637 /*---------------------------------------------------------------------------
638 |   Facility      :  libnform
639 |   Function      :  static bool Field_Grown( FIELD *field, int amount)
640 |
641 |   Description   :  This function is called for growable dynamic fields
642 |                    only. It has to increase the buffers and to allocate
643 |                    a new window for this field.
644 |                    This function has the side effect to set a new
645 |                    field-buffer pointer, the dcols and drows values
646 |                    as well as a new current Window for the field.
647 |
648 |   Return Values :  TRUE     - field successfully increased
649 |                    FALSE    - there was some error
650 +--------------------------------------------------------------------------*/
651 static bool
652 Field_Grown(FIELD *field, int amount)
653 {
654   bool result = FALSE;
655 
656   if (field && Growable(field))
657     {
658       bool single_line_field = Single_Line_Field(field);
659       int old_buflen = Buffer_Length(field);
660       int new_buflen;
661       int old_dcols = field->dcols;
662       int old_drows = field->drows;
663       FIELD_CELL *oldbuf = field->buf;
664       FIELD_CELL *newbuf;
665 
666       int growth;
667       FORM *form = field->form;
668       bool need_visual_update = ((form != (FORM *)0) &&
669 				 (form->status & _POSTED) &&
670 				 (form->current == field));
671 
672       if (need_visual_update)
673 	Synchronize_Buffer(form);
674 
675       if (single_line_field)
676 	{
677 	  growth = field->cols * amount;
678 	  if (field->maxgrow)
679 	    growth = Minimum(field->maxgrow - field->dcols, growth);
680 	  field->dcols += growth;
681 	  if (field->dcols == field->maxgrow)
682 	    ClrStatus(field, _MAY_GROW);
683 	}
684       else
685 	{
686 	  growth = (field->rows + field->nrow) * amount;
687 	  if (field->maxgrow)
688 	    growth = Minimum(field->maxgrow - field->drows, growth);
689 	  field->drows += growth;
690 	  if (field->drows == field->maxgrow)
691 	    ClrStatus(field, _MAY_GROW);
692 	}
693       /* drows, dcols changed, so we get really the new buffer length */
694       new_buflen = Buffer_Length(field);
695       newbuf = (FIELD_CELL *)malloc(Total_Buffer_Size(field));
696       if (!newbuf)
697 	{
698 	  /* restore to previous state */
699 	  field->dcols = old_dcols;
700 	  field->drows = old_drows;
701 	  if ((single_line_field && (field->dcols != field->maxgrow)) ||
702 	      (!single_line_field && (field->drows != field->maxgrow)))
703 	    SetStatus(field, _MAY_GROW);
704 	}
705       else
706 	{
707 	  /* Copy all the buffers.  This is the reason why we can't just use
708 	   * realloc().
709 	   */
710 	  int i, j;
711 
712 	  result = TRUE;	/* allow sharing of recovery on failure */
713 
714 	  T((T_CREATE("fieldcell %p"), (void *)newbuf));
715 	  field->buf = newbuf;
716 	  for (i = 0; i <= field->nbuf; i++)
717 	    {
718 	      FIELD_CELL *new_bp = Address_Of_Nth_Buffer(field, i);
719 	      FIELD_CELL *old_bp = oldbuf + i * (1 + old_buflen);
720 
721 	      for (j = 0; j < old_buflen; ++j)
722 		new_bp[j] = old_bp[j];
723 	      while (j < new_buflen)
724 		new_bp[j++] = myBLANK;
725 	      new_bp[new_buflen] = myZEROS;
726 	    }
727 
728 #if USE_WIDEC_SUPPORT && NCURSES_EXT_FUNCS
729 	  if (wresize(field->working, 1, Buffer_Length(field) + 1) == ERR)
730 	    result = FALSE;
731 #endif
732 
733 	  if (need_visual_update && result)
734 	    {
735 	      WINDOW *new_window = newpad(field->drows, field->dcols);
736 
737 	      if (new_window != 0)
738 		{
739 		  assert(form != (FORM *)0);
740 		  if (form->w)
741 		    delwin(form->w);
742 		  form->w = new_window;
743 		  Set_Field_Window_Attributes(field, form->w);
744 		  werase(form->w);
745 		  Buffer_To_Window(field, form->w);
746 		  untouchwin(form->w);
747 		  wmove(form->w, form->currow, form->curcol);
748 		}
749 	      else
750 		result = FALSE;
751 	    }
752 
753 	  if (result)
754 	    {
755 	      free(oldbuf);
756 	      /* reflect changes in linked fields */
757 	      if (field != field->link)
758 		{
759 		  FIELD *linked_field;
760 
761 		  for (linked_field = field->link;
762 		       linked_field != field;
763 		       linked_field = linked_field->link)
764 		    {
765 		      linked_field->buf = field->buf;
766 		      linked_field->drows = field->drows;
767 		      linked_field->dcols = field->dcols;
768 		    }
769 		}
770 	    }
771 	  else
772 	    {
773 	      /* restore old state */
774 	      field->dcols = old_dcols;
775 	      field->drows = old_drows;
776 	      field->buf = oldbuf;
777 	      if ((single_line_field &&
778 		   (field->dcols != field->maxgrow)) ||
779 		  (!single_line_field &&
780 		   (field->drows != field->maxgrow)))
781 		SetStatus(field, _MAY_GROW);
782 	      free(newbuf);
783 	    }
784 	}
785     }
786   return (result);
787 }
788 
789 #ifdef NCURSES_MOUSE_VERSION
790 /*---------------------------------------------------------------------------
791 |   Facility      :  libnform
792 |   Function      :  int Field_encloses(FIELD *field, int ry, int rx)
793 |
794 |   Description   :  Check if the given coordinates lie within the given field.
795 |
796 |   Return Values :  E_OK              - success
797 |                    E_BAD_ARGUMENT    - invalid form pointer
798 |                    E_SYSTEM_ERROR    - form has no current field or
799 |                                        field-window
800 +--------------------------------------------------------------------------*/
801 static int
802 Field_encloses(FIELD *field, int ry, int rx)
803 {
804   T((T_CALLED("Field_encloses(%p)"), (void *)field));
805   if (field != 0
806       && field->frow <= ry
807       && (field->frow + field->rows) > ry
808       && field->fcol <= rx
809       && (field->fcol + field->cols) > rx)
810     {
811       RETURN(E_OK);
812     }
813   RETURN(E_INVALID_FIELD);
814 }
815 #endif
816 
817 /*---------------------------------------------------------------------------
818 |   Facility      :  libnform
819 |   Function      :  int _nc_Position_Form_Cursor(FORM * form)
820 |
821 |   Description   :  Position the cursor in the window for the current
822 |                    field to be in sync. with the currow and curcol
823 |                    values.
824 |
825 |   Return Values :  E_OK              - success
826 |                    E_BAD_ARGUMENT    - invalid form pointer
827 |                    E_SYSTEM_ERROR    - form has no current field or
828 |                                        field-window
829 +--------------------------------------------------------------------------*/
830 FORM_EXPORT(int)
831 _nc_Position_Form_Cursor(FORM *form)
832 {
833   FIELD *field;
834   WINDOW *formwin;
835 
836   if (!form)
837     return (E_BAD_ARGUMENT);
838 
839   if (!form->w || !form->current)
840     return (E_SYSTEM_ERROR);
841 
842   field = form->current;
843   formwin = Get_Form_Window(form);
844 
845   wmove(form->w, form->currow, form->curcol);
846   if (Has_Invisible_Parts(field))
847     {
848       /* in this case fieldwin isn't derived from formwin, so we have
849          to move the cursor in formwin by hand... */
850       wmove(formwin,
851 	    field->frow + form->currow - form->toprow,
852 	    field->fcol + form->curcol - form->begincol);
853       wcursyncup(formwin);
854     }
855   else
856     wcursyncup(form->w);
857   return (E_OK);
858 }
859 
860 /*---------------------------------------------------------------------------
861 |   Facility      :  libnform
862 |   Function      :  int _nc_Refresh_Current_Field(FORM * form)
863 |
864 |   Description   :  Propagate the changes in the field's window to the
865 |                    window of the form.
866 |
867 |   Return Values :  E_OK              - on success
868 |                    E_BAD_ARGUMENT    - invalid form pointer
869 |                    E_SYSTEM_ERROR    - general error
870 +--------------------------------------------------------------------------*/
871 static bool move_after_insert = TRUE;
872 FORM_EXPORT(int)
873 _nc_Refresh_Current_Field(FORM *form)
874 {
875   WINDOW *formwin;
876   FIELD *field;
877   bool is_public;
878 
879   T((T_CALLED("_nc_Refresh_Current_Field(%p)"), (void *)form));
880 
881   if (!form)
882     RETURN(E_BAD_ARGUMENT);
883 
884   if (!form->w || !form->current)
885     RETURN(E_SYSTEM_ERROR);
886 
887   field = form->current;
888   formwin = Get_Form_Window(form);
889 
890   is_public = Field_Has_Option(field, O_PUBLIC);
891 
892   if (Is_Scroll_Field(field))
893     {
894       /* Again, in this case the fieldwin isn't derived from formwin,
895          so we have to perform a copy operation. */
896       if (Single_Line_Field(field))
897 	{
898 	  /* horizontal scrolling */
899 	  if (form->curcol < form->begincol)
900 	    form->begincol = form->curcol;
901 	  else
902 	    {
903 	      if (form->curcol >= (form->begincol + field->cols))
904 		form->begincol = form->curcol - field->cols
905 		  + (move_after_insert ? 1 : 0);
906 	    }
907 	  if (is_public)
908 	    copywin(form->w,
909 		    formwin,
910 		    0,
911 		    form->begincol,
912 		    field->frow,
913 		    field->fcol,
914 		    field->frow,
915 		    field->cols + field->fcol - 1,
916 		    0);
917 	}
918       else
919 	{
920 	  /* A multi-line, i.e. vertical scrolling field */
921 	  int first_modified_row, first_unmodified_row;
922 
923 	  if (field->drows > field->rows)
924 	    {
925 	      int row_after_bottom = form->toprow + field->rows;
926 
927 	      if (form->currow < form->toprow)
928 		{
929 		  form->toprow = form->currow;
930 		  SetStatus(field, _NEWTOP);
931 		}
932 	      if (form->currow >= row_after_bottom)
933 		{
934 		  form->toprow = form->currow - field->rows + 1;
935 		  SetStatus(field, _NEWTOP);
936 		}
937 	      if (field->status & _NEWTOP)
938 		{
939 		  /* means we have to copy whole range */
940 		  first_modified_row = form->toprow;
941 		  first_unmodified_row = first_modified_row + field->rows;
942 		  ClrStatus(field, _NEWTOP);
943 		}
944 	      else
945 		{
946 		  /* we try to optimize : finding the range of touched
947 		     lines */
948 		  first_modified_row = form->toprow;
949 		  while (first_modified_row < row_after_bottom)
950 		    {
951 		      if (is_linetouched(form->w, first_modified_row))
952 			break;
953 		      first_modified_row++;
954 		    }
955 		  first_unmodified_row = first_modified_row;
956 		  while (first_unmodified_row < row_after_bottom)
957 		    {
958 		      if (!is_linetouched(form->w, first_unmodified_row))
959 			break;
960 		      first_unmodified_row++;
961 		    }
962 		}
963 	    }
964 	  else
965 	    {
966 	      first_modified_row = form->toprow;
967 	      first_unmodified_row = first_modified_row + field->rows;
968 	    }
969 	  if (first_unmodified_row != first_modified_row && is_public)
970 	    copywin(form->w,
971 		    formwin,
972 		    first_modified_row,
973 		    0,
974 		    field->frow + first_modified_row - form->toprow,
975 		    field->fcol,
976 		    field->frow + first_unmodified_row - form->toprow - 1,
977 		    field->cols + field->fcol - 1,
978 		    0);
979 	}
980       if (is_public)
981 	wsyncup(formwin);
982     }
983   else
984     {
985       /* if the field-window is simply a derived window, i.e. contains no
986        * invisible parts, the whole thing is trivial
987        */
988       if (is_public)
989 	wsyncup(form->w);
990     }
991   untouchwin(form->w);
992   returnCode(_nc_Position_Form_Cursor(form));
993 }
994 
995 /*---------------------------------------------------------------------------
996 |   Facility      :  libnform
997 |   Function      :  static void Perform_Justification(
998 |                                        FIELD  * field,
999 |                                        WINDOW * win)
1000 |
1001 |   Description   :  Output field with requested justification
1002 |
1003 |   Return Values :  -
1004 +--------------------------------------------------------------------------*/
1005 static void
1006 Perform_Justification(FIELD *field, WINDOW *win)
1007 {
1008   FIELD_CELL *bp;
1009   int len;
1010 
1011   bp = (Field_Has_Option(field, O_NO_LEFT_STRIP)
1012 	? field->buf
1013 	: Get_Start_Of_Data(field->buf, Buffer_Length(field)));
1014   len = (int)(After_End_Of_Data(field->buf, Buffer_Length(field)) - bp);
1015 
1016   if (len > 0)
1017     {
1018       int col = 0;
1019 
1020       assert(win && (field->drows == 1));
1021 
1022       if (field->cols - len >= 0)
1023 	switch (field->just)
1024 	  {
1025 	  case JUSTIFY_LEFT:
1026 	    break;
1027 	  case JUSTIFY_CENTER:
1028 	    col = (field->cols - len) / 2;
1029 	    break;
1030 	  case JUSTIFY_RIGHT:
1031 	    col = field->cols - len;
1032 	    break;
1033 	  default:
1034 	    break;
1035 	  }
1036 
1037       wmove(win, 0, col);
1038       myADDNSTR(win, bp, len);
1039     }
1040 }
1041 
1042 /*---------------------------------------------------------------------------
1043 |   Facility      :  libnform
1044 |   Function      :  static void Undo_Justification(
1045 |                                     FIELD  * field,
1046 |                                     WINDOW * win)
1047 |
1048 |   Description   :  Display field without any justification, i.e.
1049 |                    left justified
1050 |
1051 |   Return Values :  -
1052 +--------------------------------------------------------------------------*/
1053 static void
1054 Undo_Justification(FIELD *field, WINDOW *win)
1055 {
1056   FIELD_CELL *bp;
1057   int y, x;
1058   int len;
1059 
1060   getyx(win, y, x);
1061 
1062   bp = (Field_Has_Option(field, O_NO_LEFT_STRIP)
1063 	? field->buf
1064 	: Get_Start_Of_Data(field->buf, Buffer_Length(field)));
1065   len = (int)(After_End_Of_Data(field->buf, Buffer_Length(field)) - bp);
1066 
1067   if (len > 0)
1068     {
1069       assert(win);
1070       wmove(win, 0, 0);
1071       myADDNSTR(win, bp, len);
1072     }
1073   wmove(win, y, x);
1074 }
1075 
1076 /*---------------------------------------------------------------------------
1077 |   Facility      :  libnform
1078 |   Function      :  static bool Check_Char(FORM  *form,
1079 |                                           FIELD *field,
1080 |                                           FIELDTYPE * typ,
1081 |                                           int ch,
1082 |                                           TypeArgument *argp)
1083 |
1084 |   Description   :  Perform a single character check for character ch
1085 |                    according to the fieldtype instance.
1086 |
1087 |   Return Values :  TRUE             - Character is valid
1088 |                    FALSE            - Character is invalid
1089 +--------------------------------------------------------------------------*/
1090 static bool
1091 Check_Char(FORM *form,
1092 	   FIELD *field,
1093 	   FIELDTYPE *typ,
1094 	   int ch,
1095 	   TypeArgument *argp)
1096 {
1097   if (typ)
1098     {
1099       if (typ->status & _LINKED_TYPE)
1100 	{
1101 	  assert(argp);
1102 	  return (
1103 		   Check_Char(form, field, typ->left, ch, argp->left) ||
1104 		   Check_Char(form, field, typ->right, ch, argp->right));
1105 	}
1106       else
1107 	{
1108 #if NCURSES_INTEROP_FUNCS
1109 	  if (typ->charcheck.occheck)
1110 	    {
1111 	      if (typ->status & _GENERIC)
1112 		return typ->charcheck.gccheck(ch, form, field, (void *)argp);
1113 	      else
1114 		return typ->charcheck.occheck(ch, (void *)argp);
1115 	    }
1116 #else
1117 	  if (typ->ccheck)
1118 	    return typ->ccheck(ch, (void *)argp);
1119 #endif
1120 	}
1121     }
1122   return (!iscntrl(UChar(ch)) ? TRUE : FALSE);
1123 }
1124 
1125 /*---------------------------------------------------------------------------
1126 |   Facility      :  libnform
1127 |   Function      :  static int Display_Or_Erase_Field(
1128 |                                           FIELD * field,
1129 |                                           bool bEraseFlag)
1130 |
1131 |   Description   :  Create a subwindow for the field and display the
1132 |                    buffer contents (apply justification if required)
1133 |                    or simply erase the field.
1134 |
1135 |   Return Values :  E_OK           - on success
1136 |                    E_SYSTEM_ERROR - some error (typical no memory)
1137 +--------------------------------------------------------------------------*/
1138 static int
1139 Display_Or_Erase_Field(FIELD *field, bool bEraseFlag)
1140 {
1141   WINDOW *win;
1142   WINDOW *fwin;
1143 
1144   if (!field)
1145     return E_SYSTEM_ERROR;
1146 
1147   fwin = Get_Form_Window(field->form);
1148   win = derwin(fwin,
1149 	       field->rows, field->cols, field->frow, field->fcol);
1150 
1151   if (!win)
1152     return E_SYSTEM_ERROR;
1153   else
1154     {
1155       if (Field_Has_Option(field, O_VISIBLE))
1156 	{
1157 	  Set_Field_Window_Attributes(field, win);
1158 	}
1159       else
1160 	{
1161 	  (void)wattrset(win, (int)WINDOW_ATTRS(fwin));
1162 	}
1163       werase(win);
1164     }
1165 
1166   if (!bEraseFlag)
1167     {
1168       if (Field_Has_Option(field, O_PUBLIC))
1169 	{
1170 	  if (Justification_Allowed(field))
1171 	    Perform_Justification(field, win);
1172 	  else
1173 	    Buffer_To_Window(field, win);
1174 	}
1175       ClrStatus(field, _NEWTOP);
1176     }
1177   wsyncup(win);
1178   delwin(win);
1179   return E_OK;
1180 }
1181 
1182 /* Macros to preset the bEraseFlag */
1183 #define Display_Field(field) Display_Or_Erase_Field(field,FALSE)
1184 #define Erase_Field(field)   Display_Or_Erase_Field(field,TRUE)
1185 
1186 /*---------------------------------------------------------------------------
1187 |   Facility      :  libnform
1188 |   Function      :  static int Synchronize_Field(FIELD * field)
1189 |
1190 |   Description   :  Synchronize the windows content with the value in
1191 |                    the buffer.
1192 |
1193 |   Return Values :  E_OK                - success
1194 |                    E_BAD_ARGUMENT      - invalid field pointer
1195 |                    E_SYSTEM_ERROR      - some severe basic error
1196 +--------------------------------------------------------------------------*/
1197 static int
1198 Synchronize_Field(FIELD *field)
1199 {
1200   FORM *form;
1201   int res = E_OK;
1202 
1203   if (!field)
1204     return (E_BAD_ARGUMENT);
1205 
1206   if (((form = field->form) != (FORM *)0)
1207       && Field_Really_Appears(field))
1208     {
1209       if (field == form->current)
1210 	{
1211 	  form->currow = form->curcol = form->toprow = form->begincol = 0;
1212 	  werase(form->w);
1213 
1214 	  if ((Field_Has_Option(field, O_PUBLIC)) && Justification_Allowed(field))
1215 	    Undo_Justification(field, form->w);
1216 	  else
1217 	    Buffer_To_Window(field, form->w);
1218 
1219 	  SetStatus(field, _NEWTOP);
1220 	  res = _nc_Refresh_Current_Field(form);
1221 	}
1222       else
1223 	res = Display_Field(field);
1224     }
1225   SetStatus(field, _CHANGED);
1226   return (res);
1227 }
1228 
1229 /*---------------------------------------------------------------------------
1230 |   Facility      :  libnform
1231 |   Function      :  static int Synchronize_Linked_Fields(FIELD * field)
1232 |
1233 |   Description   :  Propagate the Synchronize_Field function to all linked
1234 |                    fields. The first error that occurs in the sequence
1235 |                    of updates is the return value.
1236 |
1237 |   Return Values :  E_OK                - success
1238 |                    E_BAD_ARGUMENT      - invalid field pointer
1239 |                    E_SYSTEM_ERROR      - some severe basic error
1240 +--------------------------------------------------------------------------*/
1241 static int
1242 Synchronize_Linked_Fields(FIELD *field)
1243 {
1244   FIELD *linked_field;
1245   int res = E_OK;
1246 
1247   if (!field)
1248     return (E_BAD_ARGUMENT);
1249 
1250   if (!field->link)
1251     return (E_SYSTEM_ERROR);
1252 
1253   for (linked_field = field->link;
1254        (linked_field != field) && (linked_field != 0);
1255        linked_field = linked_field->link)
1256     {
1257       int syncres;
1258 
1259       if (((syncres = Synchronize_Field(linked_field)) != E_OK) &&
1260 	  (res == E_OK))
1261 	res = syncres;
1262     }
1263   return (res);
1264 }
1265 
1266 /*---------------------------------------------------------------------------
1267 |   Facility      :  libnform
1268 |   Function      :  int _nc_Synchronize_Attributes(FIELD * field)
1269 |
1270 |   Description   :  If a field's visual attributes have changed, this
1271 |                    routine is called to propagate those changes to the
1272 |                    screen.
1273 |
1274 |   Return Values :  E_OK             - success
1275 |                    E_BAD_ARGUMENT   - invalid field pointer
1276 |                    E_SYSTEM_ERROR   - some severe basic error
1277 +--------------------------------------------------------------------------*/
1278 FORM_EXPORT(int)
1279 _nc_Synchronize_Attributes(FIELD *field)
1280 {
1281   FORM *form;
1282   int res = E_OK;
1283 
1284   T((T_CALLED("_nc_Synchronize_Attributes(%p)"), (void *)field));
1285 
1286   if (!field)
1287     returnCode(E_BAD_ARGUMENT);
1288 
1289   CHECKPOS(field->form);
1290   if (((form = field->form) != (FORM *)0)
1291       && Field_Really_Appears(field))
1292     {
1293       if (form->current == field)
1294 	{
1295 	  Synchronize_Buffer(form);
1296 	  Set_Field_Window_Attributes(field, form->w);
1297 	  werase(form->w);
1298 	  wmove(form->w, form->currow, form->curcol);
1299 
1300 	  if (Field_Has_Option(field, O_PUBLIC))
1301 	    {
1302 	      if (Justification_Allowed(field))
1303 		Undo_Justification(field, form->w);
1304 	      else
1305 		Buffer_To_Window(field, form->w);
1306 	    }
1307 	  else
1308 	    {
1309 	      WINDOW *formwin = Get_Form_Window(form);
1310 
1311 	      copywin(form->w, formwin,
1312 		      0, 0,
1313 		      field->frow, field->fcol,
1314 		      field->frow + field->rows - 1,
1315 		      field->fcol + field->cols - 1, 0);
1316 	      wsyncup(formwin);
1317 	      Buffer_To_Window(field, form->w);
1318 	      SetStatus(field, _NEWTOP);	/* fake refresh to paint all */
1319 	      _nc_Refresh_Current_Field(form);
1320 	    }
1321 	}
1322       else
1323 	{
1324 	  res = Display_Field(field);
1325 	}
1326     }
1327   CHECKPOS(form);
1328   returnCode(res);
1329 }
1330 
1331 /*---------------------------------------------------------------------------
1332 |   Facility      :  libnform
1333 |   Function      :  int _nc_Synchronize_Options(FIELD * field,
1334 |                                                Field_Options newopts)
1335 |
1336 |   Description   :  If a field's options have changed, this routine is
1337 |                    called to propagate these changes to the screen and
1338 |                    to really change the behavior of the field.
1339 |
1340 |   Return Values :  E_OK                - success
1341 |                    E_BAD_ARGUMENT      - invalid field pointer
1342 |                    E_CURRENT           - field is the current one
1343 |                    E_SYSTEM_ERROR      - some severe basic error
1344 +--------------------------------------------------------------------------*/
1345 FORM_EXPORT(int)
1346 _nc_Synchronize_Options(FIELD *field, Field_Options newopts)
1347 {
1348   Field_Options oldopts;
1349   Field_Options changed_opts;
1350   FORM *form;
1351   int res = E_OK;
1352 
1353   T((T_CALLED("_nc_Synchronize_Options(%p,%#x)"), (void *)field, newopts));
1354 
1355   if (!field)
1356     returnCode(E_BAD_ARGUMENT);
1357 
1358   oldopts = field->opts;
1359   changed_opts = oldopts ^ newopts;
1360   field->opts = newopts;
1361   form = field->form;
1362 
1363   if (form)
1364     {
1365       if (form->status & _POSTED)
1366 	{
1367 	  if (form->current == field)
1368 	    {
1369 	      field->opts = oldopts;
1370 	      returnCode(E_CURRENT);
1371 	    }
1372 	  if (form->curpage == field->page)
1373 	    {
1374 	      if ((unsigned)changed_opts & O_VISIBLE)
1375 		{
1376 		  if ((unsigned)newopts & O_VISIBLE)
1377 		    res = Display_Field(field);
1378 		  else
1379 		    res = Erase_Field(field);
1380 		}
1381 	      else
1382 		{
1383 		  if (((unsigned)changed_opts & O_PUBLIC) &&
1384 		      ((unsigned)newopts & O_VISIBLE))
1385 		    res = Display_Field(field);
1386 		}
1387 	    }
1388 	}
1389     }
1390 
1391   if ((unsigned)changed_opts & O_STATIC)
1392     {
1393       bool single_line_field = Single_Line_Field(field);
1394       int res2 = E_OK;
1395 
1396       if ((unsigned)newopts & O_STATIC)
1397 	{
1398 	  /* the field becomes now static */
1399 	  ClrStatus(field, _MAY_GROW);
1400 	  /* if actually we have no hidden columns, justification may
1401 	     occur again */
1402 	  if (single_line_field &&
1403 	      (field->cols == field->dcols) &&
1404 	      (field->just != NO_JUSTIFICATION) &&
1405 	      Field_Really_Appears(field))
1406 	    {
1407 	      res2 = Display_Field(field);
1408 	    }
1409 	}
1410       else
1411 	{
1412 	  /* field is no longer static */
1413 	  if ((field->maxgrow == 0) ||
1414 	      (single_line_field && (field->dcols < field->maxgrow)) ||
1415 	      (!single_line_field && (field->drows < field->maxgrow)))
1416 	    {
1417 	      SetStatus(field, _MAY_GROW);
1418 	      /* a field with justification now changes its behavior,
1419 	         so we must redisplay it */
1420 	      if (single_line_field &&
1421 		  (field->just != NO_JUSTIFICATION) &&
1422 		  Field_Really_Appears(field))
1423 		{
1424 		  res2 = Display_Field(field);
1425 		}
1426 	    }
1427 	}
1428       if (res2 != E_OK)
1429 	res = res2;
1430     }
1431 
1432   returnCode(res);
1433 }
1434 
1435 /*
1436  * Removes the focus from the current field of the form.
1437  */
1438 void
1439 _nc_Unset_Current_Field(FORM *form)
1440 {
1441   FIELD *field = form->current;
1442 
1443   _nc_Refresh_Current_Field(form);
1444   if (Field_Has_Option(field, O_PUBLIC))
1445     {
1446       if (field->drows > field->rows)
1447 	{
1448 	  if (form->toprow == 0)
1449 	    ClrStatus(field, _NEWTOP);
1450 	  else
1451 	    SetStatus(field, _NEWTOP);
1452 	}
1453       else
1454 	{
1455 	  if (Justification_Allowed(field))
1456 	    {
1457 	      Window_To_Buffer(form, field);
1458 	      werase(form->w);
1459 	      Perform_Justification(field, form->w);
1460 	      if (Field_Has_Option(field, O_DYNAMIC_JUSTIFY) &&
1461 		  (form->w->_parent == 0))
1462 		{
1463 		  copywin(form->w,
1464 			  Get_Form_Window(form),
1465 			  0,
1466 			  0,
1467 			  field->frow,
1468 			  field->fcol,
1469 			  field->frow,
1470 			  field->cols + field->fcol - 1,
1471 			  0);
1472 		  wsyncup(Get_Form_Window(form));
1473 		}
1474 	      else
1475 		{
1476 		  wsyncup(form->w);
1477 		}
1478 	    }
1479 	}
1480     }
1481   delwin(form->w);
1482   form->w = (WINDOW *)0;
1483   form->current = 0;
1484 }
1485 
1486 /*---------------------------------------------------------------------------
1487 |   Facility      :  libnform
1488 |   Function      :  int _nc_Set_Current_Field(FORM  * form,
1489 |                                              FIELD * newfield)
1490 |
1491 |   Description   :  Make the newfield the new current field.
1492 |
1493 |   Return Values :  E_OK              - success
1494 |                    E_BAD_ARGUMENT    - invalid form or field pointer
1495 |                    E_SYSTEM_ERROR    - some severe basic error
1496 |                    E_NOT_CONNECTED   - no fields are connected to the form
1497 +--------------------------------------------------------------------------*/
1498 FORM_EXPORT(int)
1499 _nc_Set_Current_Field(FORM *form, FIELD *newfield)
1500 {
1501   FIELD *field;
1502   WINDOW *new_window;
1503 
1504   T((T_CALLED("_nc_Set_Current_Field(%p,%p)"), (void *)form, (void *)newfield));
1505 
1506   if (!form || !newfield || (newfield->form != form))
1507     returnCode(E_BAD_ARGUMENT);
1508 
1509   if ((form->status & _IN_DRIVER))
1510     returnCode(E_BAD_STATE);
1511 
1512   if (!(form->field))
1513     returnCode(E_NOT_CONNECTED);
1514 
1515   field = form->current;
1516 
1517   if ((field != newfield) ||
1518       !(form->status & _POSTED))
1519     {
1520       if (field && (form->w) &&
1521 	  (Field_Has_Option(field, O_VISIBLE)) &&
1522 	  (field->form->curpage == field->page))
1523 	_nc_Unset_Current_Field(form);
1524 
1525       field = newfield;
1526 
1527       if (Has_Invisible_Parts(field))
1528 	new_window = newpad(field->drows, field->dcols);
1529       else
1530 	new_window = derwin(Get_Form_Window(form),
1531 			    field->rows, field->cols, field->frow, field->fcol);
1532 
1533       if (!new_window)
1534 	returnCode(E_SYSTEM_ERROR);
1535 
1536       form->current = field;
1537 
1538       if (form->w)
1539 	delwin(form->w);
1540       form->w = new_window;
1541 
1542       ClrStatus(form, _WINDOW_MODIFIED);
1543       Set_Field_Window_Attributes(field, form->w);
1544 
1545       if (Has_Invisible_Parts(field))
1546 	{
1547 	  werase(form->w);
1548 	  Buffer_To_Window(field, form->w);
1549 	}
1550       else
1551 	{
1552 	  if (Justification_Allowed(field))
1553 	    {
1554 	      werase(form->w);
1555 	      Undo_Justification(field, form->w);
1556 	      wsyncup(form->w);
1557 	    }
1558 	}
1559 
1560       untouchwin(form->w);
1561     }
1562 
1563   form->currow = form->curcol = form->toprow = form->begincol = 0;
1564   returnCode(E_OK);
1565 }
1566 
1567 /*----------------------------------------------------------------------------
1568   Intra-Field Navigation routines
1569   --------------------------------------------------------------------------*/
1570 
1571 /*---------------------------------------------------------------------------
1572 |   Facility      :  libnform
1573 |   Function      :  static int IFN_Next_Character(FORM * form)
1574 |
1575 |   Description   :  Move to the next character in the field. In a multi-line
1576 |                    field this wraps at the end of the line.
1577 |
1578 |   Return Values :  E_OK                - success
1579 |                    E_REQUEST_DENIED    - at the rightmost position
1580 +--------------------------------------------------------------------------*/
1581 static int
1582 IFN_Next_Character(FORM *form)
1583 {
1584   FIELD *field = form->current;
1585   int step = myWCWIDTH(form->w, form->currow, form->curcol);
1586 
1587   T((T_CALLED("IFN_Next_Character(%p)"), (void *)form));
1588   if ((form->curcol += step) == field->dcols)
1589     {
1590       if ((++(form->currow)) == field->drows)
1591 	{
1592 #if GROW_IF_NAVIGATE
1593 	  if (!Single_Line_Field(field) && Field_Grown(field, 1))
1594 	    {
1595 	      form->curcol = 0;
1596 	      returnCode(E_OK);
1597 	    }
1598 #endif
1599 	  form->currow--;
1600 #if GROW_IF_NAVIGATE
1601 	  if (Single_Line_Field(field) && Field_Grown(field, 1))
1602 	    returnCode(E_OK);
1603 #endif
1604 	  form->curcol -= step;
1605 	  returnCode(E_REQUEST_DENIED);
1606 	}
1607       form->curcol = 0;
1608     }
1609   returnCode(E_OK);
1610 }
1611 
1612 /*---------------------------------------------------------------------------
1613 |   Facility      :  libnform
1614 |   Function      :  static int IFN_Previous_Character(FORM * form)
1615 |
1616 |   Description   :  Move to the previous character in the field. In a
1617 |                    multi-line field this wraps and the beginning of the
1618 |                    line.
1619 |
1620 |   Return Values :  E_OK                - success
1621 |                    E_REQUEST_DENIED    - at the leftmost position
1622 +--------------------------------------------------------------------------*/
1623 static int
1624 IFN_Previous_Character(FORM *form)
1625 {
1626   int amount = myWCWIDTH(form->w, form->currow, form->curcol - 1);
1627   int oldcol = form->curcol;
1628 
1629   T((T_CALLED("IFN_Previous_Character(%p)"), (void *)form));
1630   if ((form->curcol -= amount) < 0)
1631     {
1632       if ((--(form->currow)) < 0)
1633 	{
1634 	  form->currow++;
1635 	  form->curcol = oldcol;
1636 	  returnCode(E_REQUEST_DENIED);
1637 	}
1638       form->curcol = form->current->dcols - 1;
1639     }
1640   returnCode(E_OK);
1641 }
1642 
1643 /*---------------------------------------------------------------------------
1644 |   Facility      :  libnform
1645 |   Function      :  static int IFN_Next_Line(FORM * form)
1646 |
1647 |   Description   :  Move to the beginning of the next line in the field
1648 |
1649 |   Return Values :  E_OK                - success
1650 |                    E_REQUEST_DENIED    - at the last line
1651 +--------------------------------------------------------------------------*/
1652 static int
1653 IFN_Next_Line(FORM *form)
1654 {
1655   FIELD *field = form->current;
1656 
1657   T((T_CALLED("IFN_Next_Line(%p)"), (void *)form));
1658   if ((++(form->currow)) == field->drows)
1659     {
1660 #if GROW_IF_NAVIGATE
1661       if (!Single_Line_Field(field) && Field_Grown(field, 1))
1662 	returnCode(E_OK);
1663 #endif
1664       form->currow--;
1665       returnCode(E_REQUEST_DENIED);
1666     }
1667   form->curcol = 0;
1668   returnCode(E_OK);
1669 }
1670 
1671 /*---------------------------------------------------------------------------
1672 |   Facility      :  libnform
1673 |   Function      :  static int IFN_Previous_Line(FORM * form)
1674 |
1675 |   Description   :  Move to the beginning of the previous line in the field
1676 |
1677 |   Return Values :  E_OK                - success
1678 |                    E_REQUEST_DENIED    - at the first line
1679 +--------------------------------------------------------------------------*/
1680 static int
1681 IFN_Previous_Line(FORM *form)
1682 {
1683   T((T_CALLED("IFN_Previous_Line(%p)"), (void *)form));
1684   if ((--(form->currow)) < 0)
1685     {
1686       form->currow++;
1687       returnCode(E_REQUEST_DENIED);
1688     }
1689   form->curcol = 0;
1690   returnCode(E_OK);
1691 }
1692 
1693 /*---------------------------------------------------------------------------
1694 |   Facility      :  libnform
1695 |   Function      :  static int IFN_Next_Word(FORM * form)
1696 |
1697 |   Description   :  Move to the beginning of the next word in the field.
1698 |
1699 |   Return Values :  E_OK             - success
1700 |                    E_REQUEST_DENIED - there is no next word
1701 +--------------------------------------------------------------------------*/
1702 static int
1703 IFN_Next_Word(FORM *form)
1704 {
1705   FIELD *field = form->current;
1706   FIELD_CELL *bp = Address_Of_Current_Position_In_Buffer(form);
1707   FIELD_CELL *s;
1708   FIELD_CELL *t;
1709 
1710   T((T_CALLED("IFN_Next_Word(%p)"), (void *)form));
1711 
1712   /* We really need access to the data, so we have to synchronize */
1713   Synchronize_Buffer(form);
1714 
1715   /* Go to the first whitespace after the current position (including
1716      current position). This is then the starting point to look for the
1717      next non-blank data */
1718   s = Get_First_Whitespace_Character(bp, Buffer_Length(field) -
1719 				     (int)(bp - field->buf));
1720 
1721   /* Find the start of the next word */
1722   t = Get_Start_Of_Data(s, Buffer_Length(field) -
1723 			(int)(s - field->buf));
1724 #if !FRIENDLY_PREV_NEXT_WORD
1725   if (s == t)
1726     returnCode(E_REQUEST_DENIED);
1727   else
1728 #endif
1729     {
1730       Adjust_Cursor_Position(form, t);
1731       returnCode(E_OK);
1732     }
1733 }
1734 
1735 /*---------------------------------------------------------------------------
1736 |   Facility      :  libnform
1737 |   Function      :  static int IFN_Previous_Word(FORM * form)
1738 |
1739 |   Description   :  Move to the beginning of the previous word in the field.
1740 |
1741 |   Return Values :  E_OK             - success
1742 |                    E_REQUEST_DENIED - there is no previous word
1743 +--------------------------------------------------------------------------*/
1744 static int
1745 IFN_Previous_Word(FORM *form)
1746 {
1747   FIELD *field = form->current;
1748   FIELD_CELL *bp = Address_Of_Current_Position_In_Buffer(form);
1749   FIELD_CELL *s;
1750   FIELD_CELL *t;
1751   bool again = FALSE;
1752 
1753   T((T_CALLED("IFN_Previous_Word(%p)"), (void *)form));
1754 
1755   /* We really need access to the data, so we have to synchronize */
1756   Synchronize_Buffer(form);
1757 
1758   s = After_End_Of_Data(field->buf, (int)(bp - field->buf));
1759   /* s points now right after the last non-blank in the buffer before bp.
1760      If bp was in a word, s equals bp. In this case we must find the last
1761      whitespace in the buffer before bp and repeat the game to really find
1762      the previous word! */
1763   if (s == bp)
1764     again = TRUE;
1765 
1766   /* And next call now goes backward to look for the last whitespace
1767      before that, pointing right after this, so it points to the begin
1768      of the previous word.
1769    */
1770   t = After_Last_Whitespace_Character(field->buf, (int)(s - field->buf));
1771 #if !FRIENDLY_PREV_NEXT_WORD
1772   if (s == t)
1773     returnCode(E_REQUEST_DENIED);
1774 #endif
1775   if (again)
1776     {
1777       /* and do it again, replacing bp by t */
1778       s = After_End_Of_Data(field->buf, (int)(t - field->buf));
1779       t = After_Last_Whitespace_Character(field->buf, (int)(s - field->buf));
1780 #if !FRIENDLY_PREV_NEXT_WORD
1781       if (s == t)
1782 	returnCode(E_REQUEST_DENIED);
1783 #endif
1784     }
1785   Adjust_Cursor_Position(form, t);
1786   returnCode(E_OK);
1787 }
1788 
1789 /*---------------------------------------------------------------------------
1790 |   Facility      :  libnform
1791 |   Function      :  static int IFN_Beginning_Of_Field(FORM * form)
1792 |
1793 |   Description   :  Place the cursor at the first non-pad character in
1794 |                    the field.
1795 |
1796 |   Return Values :  E_OK             - success
1797 +--------------------------------------------------------------------------*/
1798 static int
1799 IFN_Beginning_Of_Field(FORM *form)
1800 {
1801   FIELD *field = form->current;
1802 
1803   T((T_CALLED("IFN_Beginning_Of_Field(%p)"), (void *)form));
1804   Synchronize_Buffer(form);
1805   Adjust_Cursor_Position(form,
1806 			 Get_Start_Of_Data(field->buf, Buffer_Length(field)));
1807   returnCode(E_OK);
1808 }
1809 
1810 /*---------------------------------------------------------------------------
1811 |   Facility      :  libnform
1812 |   Function      :  static int IFN_End_Of_Field(FORM * form)
1813 |
1814 |   Description   :  Place the cursor after the last non-pad character in
1815 |                    the field. If the field occupies the last position in
1816 |                    the buffer, the cursor is positioned on the last
1817 |                    character.
1818 |
1819 |   Return Values :  E_OK              - success
1820 +--------------------------------------------------------------------------*/
1821 static int
1822 IFN_End_Of_Field(FORM *form)
1823 {
1824   FIELD *field = form->current;
1825   FIELD_CELL *pos;
1826 
1827   T((T_CALLED("IFN_End_Of_Field(%p)"), (void *)form));
1828   Synchronize_Buffer(form);
1829   pos = After_End_Of_Data(field->buf, Buffer_Length(field));
1830   if (pos == (field->buf + Buffer_Length(field)))
1831     pos--;
1832   Adjust_Cursor_Position(form, pos);
1833   returnCode(E_OK);
1834 }
1835 
1836 /*---------------------------------------------------------------------------
1837 |   Facility      :  libnform
1838 |   Function      :  static int IFN_Beginning_Of_Line(FORM * form)
1839 |
1840 |   Description   :  Place the cursor on the first non-pad character in
1841 |                    the current line of the field.
1842 |
1843 |   Return Values :  E_OK         - success
1844 +--------------------------------------------------------------------------*/
1845 static int
1846 IFN_Beginning_Of_Line(FORM *form)
1847 {
1848   FIELD *field = form->current;
1849 
1850   T((T_CALLED("IFN_Beginning_Of_Line(%p)"), (void *)form));
1851   Synchronize_Buffer(form);
1852   Adjust_Cursor_Position(form,
1853 			 Get_Start_Of_Data(Address_Of_Current_Row_In_Buffer(form),
1854 					   field->dcols));
1855   returnCode(E_OK);
1856 }
1857 
1858 /*---------------------------------------------------------------------------
1859 |   Facility      :  libnform
1860 |   Function      :  static int IFN_End_Of_Line(FORM * form)
1861 |
1862 |   Description   :  Place the cursor after the last non-pad character in the
1863 |                    current line of the field. If the field occupies the
1864 |                    last column in the line, the cursor is positioned on the
1865 |                    last character of the line.
1866 |
1867 |   Return Values :  E_OK        - success
1868 +--------------------------------------------------------------------------*/
1869 static int
1870 IFN_End_Of_Line(FORM *form)
1871 {
1872   FIELD *field = form->current;
1873   FIELD_CELL *pos;
1874   FIELD_CELL *bp;
1875 
1876   T((T_CALLED("IFN_End_Of_Line(%p)"), (void *)form));
1877   Synchronize_Buffer(form);
1878   bp = Address_Of_Current_Row_In_Buffer(form);
1879   pos = After_End_Of_Data(bp, field->dcols);
1880   if (pos == (bp + field->dcols))
1881     pos--;
1882   Adjust_Cursor_Position(form, pos);
1883   returnCode(E_OK);
1884 }
1885 
1886 /*---------------------------------------------------------------------------
1887 |   Facility      :  libnform
1888 |   Function      :  static int IFN_Left_Character(FORM * form)
1889 |
1890 |   Description   :  Move one character to the left in the current line.
1891 |                    This doesn't cycle.
1892 |
1893 |   Return Values :  E_OK             - success
1894 |                    E_REQUEST_DENIED - already in first column
1895 +--------------------------------------------------------------------------*/
1896 static int
1897 IFN_Left_Character(FORM *form)
1898 {
1899   int amount = myWCWIDTH(form->w, form->currow, form->curcol - 1);
1900   int oldcol = form->curcol;
1901 
1902   T((T_CALLED("IFN_Left_Character(%p)"), (void *)form));
1903   if ((form->curcol -= amount) < 0)
1904     {
1905       form->curcol = oldcol;
1906       returnCode(E_REQUEST_DENIED);
1907     }
1908   returnCode(E_OK);
1909 }
1910 
1911 /*---------------------------------------------------------------------------
1912 |   Facility      :  libnform
1913 |   Function      :  static int IFN_Right_Character(FORM * form)
1914 |
1915 |   Description   :  Move one character to the right in the current line.
1916 |                    This doesn't cycle.
1917 |
1918 |   Return Values :  E_OK              - success
1919 |                    E_REQUEST_DENIED  - already in last column
1920 +--------------------------------------------------------------------------*/
1921 static int
1922 IFN_Right_Character(FORM *form)
1923 {
1924   int amount = myWCWIDTH(form->w, form->currow, form->curcol);
1925   int oldcol = form->curcol;
1926 
1927   T((T_CALLED("IFN_Right_Character(%p)"), (void *)form));
1928   if ((form->curcol += amount) >= form->current->dcols)
1929     {
1930 #if GROW_IF_NAVIGATE
1931       FIELD *field = form->current;
1932 
1933       if (Single_Line_Field(field) && Field_Grown(field, 1))
1934 	returnCode(E_OK);
1935 #endif
1936       form->curcol = oldcol;
1937       returnCode(E_REQUEST_DENIED);
1938     }
1939   returnCode(E_OK);
1940 }
1941 
1942 /*---------------------------------------------------------------------------
1943 |   Facility      :  libnform
1944 |   Function      :  static int IFN_Up_Character(FORM * form)
1945 |
1946 |   Description   :  Move one line up. This doesn't cycle through the lines
1947 |                    of the field.
1948 |
1949 |   Return Values :  E_OK              - success
1950 |                    E_REQUEST_DENIED  - already in last column
1951 +--------------------------------------------------------------------------*/
1952 static int
1953 IFN_Up_Character(FORM *form)
1954 {
1955   T((T_CALLED("IFN_Up_Character(%p)"), (void *)form));
1956   if ((--(form->currow)) < 0)
1957     {
1958       form->currow++;
1959       returnCode(E_REQUEST_DENIED);
1960     }
1961   returnCode(E_OK);
1962 }
1963 
1964 /*---------------------------------------------------------------------------
1965 |   Facility      :  libnform
1966 |   Function      :  static int IFN_Down_Character(FORM * form)
1967 |
1968 |   Description   :  Move one line down. This doesn't cycle through the
1969 |                    lines of the field.
1970 |
1971 |   Return Values :  E_OK              - success
1972 |                    E_REQUEST_DENIED  - already in last column
1973 +--------------------------------------------------------------------------*/
1974 static int
1975 IFN_Down_Character(FORM *form)
1976 {
1977   FIELD *field = form->current;
1978 
1979   T((T_CALLED("IFN_Down_Character(%p)"), (void *)form));
1980   if ((++(form->currow)) == field->drows)
1981     {
1982 #if GROW_IF_NAVIGATE
1983       if (!Single_Line_Field(field) && Field_Grown(field, 1))
1984 	returnCode(E_OK);
1985 #endif
1986       --(form->currow);
1987       returnCode(E_REQUEST_DENIED);
1988     }
1989   returnCode(E_OK);
1990 }
1991 /*----------------------------------------------------------------------------
1992   END of Intra-Field Navigation routines
1993   --------------------------------------------------------------------------*/
1994 
1995 /*----------------------------------------------------------------------------
1996   Vertical scrolling helper routines
1997   --------------------------------------------------------------------------*/
1998 
1999 /*---------------------------------------------------------------------------
2000 |   Facility      :  libnform
2001 |   Function      :  static int VSC_Generic(FORM *form, int nlines)
2002 |
2003 |   Description   :  Scroll multi-line field forward (nlines>0) or
2004 |                    backward (nlines<0) this many lines.
2005 |
2006 |   Return Values :  E_OK              - success
2007 |                    E_REQUEST_DENIED  - can't scroll
2008 +--------------------------------------------------------------------------*/
2009 static int
2010 VSC_Generic(FORM *form, int nlines)
2011 {
2012   FIELD *field = form->current;
2013   int res = E_REQUEST_DENIED;
2014   int rows_to_go = (nlines > 0 ? nlines : -nlines);
2015 
2016   if (nlines > 0)
2017     {
2018       if ((rows_to_go + form->toprow) > (field->drows - field->rows))
2019 	rows_to_go = (field->drows - field->rows - form->toprow);
2020 
2021       if (rows_to_go > 0)
2022 	{
2023 	  form->currow += rows_to_go;
2024 	  form->toprow += rows_to_go;
2025 	  res = E_OK;
2026 	}
2027     }
2028   else
2029     {
2030       if (rows_to_go > form->toprow)
2031 	rows_to_go = form->toprow;
2032 
2033       if (rows_to_go > 0)
2034 	{
2035 	  form->currow -= rows_to_go;
2036 	  form->toprow -= rows_to_go;
2037 	  res = E_OK;
2038 	}
2039     }
2040   return (res);
2041 }
2042 /*----------------------------------------------------------------------------
2043   End of Vertical scrolling helper routines
2044   --------------------------------------------------------------------------*/
2045 
2046 /*----------------------------------------------------------------------------
2047   Vertical scrolling routines
2048   --------------------------------------------------------------------------*/
2049 
2050 /*---------------------------------------------------------------------------
2051 |   Facility      :  libnform
2052 |   Function      :  static int Vertical_Scrolling(
2053 |                                           int (* const fct) (FORM *),
2054 |                                           FORM * form)
2055 |
2056 |   Description   :  Performs the generic vertical scrolling routines.
2057 |                    This has to check for a multi-line field and to set
2058 |                    the _NEWTOP flag if scrolling really occurred.
2059 |
2060 |   Return Values :  Propagated error code from low-level driver calls
2061 +--------------------------------------------------------------------------*/
2062 static int
2063 Vertical_Scrolling(int (*const fct) (FORM *), FORM *form)
2064 {
2065   int res = E_REQUEST_DENIED;
2066 
2067   if (!Single_Line_Field(form->current))
2068     {
2069       res = fct(form);
2070       if (res == E_OK)
2071 	SetStatus(form->current, _NEWTOP);
2072     }
2073   return (res);
2074 }
2075 
2076 /*---------------------------------------------------------------------------
2077 |   Facility      :  libnform
2078 |   Function      :  static int VSC_Scroll_Line_Forward(FORM * form)
2079 |
2080 |   Description   :  Scroll multi-line field forward a line
2081 |
2082 |   Return Values :  E_OK                - success
2083 |                    E_REQUEST_DENIED    - no data ahead
2084 +--------------------------------------------------------------------------*/
2085 static int
2086 VSC_Scroll_Line_Forward(FORM *form)
2087 {
2088   T((T_CALLED("VSC_Scroll_Line_Forward(%p)"), (void *)form));
2089   returnCode(VSC_Generic(form, 1));
2090 }
2091 
2092 /*---------------------------------------------------------------------------
2093 |   Facility      :  libnform
2094 |   Function      :  static int VSC_Scroll_Line_Backward(FORM * form)
2095 |
2096 |   Description   :  Scroll multi-line field backward a line
2097 |
2098 |   Return Values :  E_OK                - success
2099 |                    E_REQUEST_DENIED    - no data behind
2100 +--------------------------------------------------------------------------*/
2101 static int
2102 VSC_Scroll_Line_Backward(FORM *form)
2103 {
2104   T((T_CALLED("VSC_Scroll_Line_Backward(%p)"), (void *)form));
2105   returnCode(VSC_Generic(form, -1));
2106 }
2107 
2108 /*---------------------------------------------------------------------------
2109 |   Facility      :  libnform
2110 |   Function      :  static int VSC_Scroll_Page_Forward(FORM * form)
2111 |
2112 |   Description   :  Scroll a multi-line field forward a page
2113 |
2114 |   Return Values :  E_OK              - success
2115 |                    E_REQUEST_DENIED  - no data ahead
2116 +--------------------------------------------------------------------------*/
2117 static int
2118 VSC_Scroll_Page_Forward(FORM *form)
2119 {
2120   T((T_CALLED("VSC_Scroll_Page_Forward(%p)"), (void *)form));
2121   returnCode(VSC_Generic(form, form->current->rows));
2122 }
2123 
2124 /*---------------------------------------------------------------------------
2125 |   Facility      :  libnform
2126 |   Function      :  static int VSC_Scroll_Half_Page_Forward(FORM * form)
2127 |
2128 |   Description   :  Scroll a multi-line field forward half a page
2129 |
2130 |   Return Values :  E_OK              - success
2131 |                    E_REQUEST_DENIED  - no data ahead
2132 +--------------------------------------------------------------------------*/
2133 static int
2134 VSC_Scroll_Half_Page_Forward(FORM *form)
2135 {
2136   T((T_CALLED("VSC_Scroll_Half_Page_Forward(%p)"), (void *)form));
2137   returnCode(VSC_Generic(form, (form->current->rows + 1) / 2));
2138 }
2139 
2140 /*---------------------------------------------------------------------------
2141 |   Facility      :  libnform
2142 |   Function      :  static int VSC_Scroll_Page_Backward(FORM * form)
2143 |
2144 |   Description   :  Scroll a multi-line field backward a page
2145 |
2146 |   Return Values :  E_OK              - success
2147 |                    E_REQUEST_DENIED  - no data behind
2148 +--------------------------------------------------------------------------*/
2149 static int
2150 VSC_Scroll_Page_Backward(FORM *form)
2151 {
2152   T((T_CALLED("VSC_Scroll_Page_Backward(%p)"), (void *)form));
2153   returnCode(VSC_Generic(form, -(form->current->rows)));
2154 }
2155 
2156 /*---------------------------------------------------------------------------
2157 |   Facility      :  libnform
2158 |   Function      :  static int VSC_Scroll_Half_Page_Backward(FORM * form)
2159 |
2160 |   Description   :  Scroll a multi-line field backward half a page
2161 |
2162 |   Return Values :  E_OK              - success
2163 |                    E_REQUEST_DENIED  - no data behind
2164 +--------------------------------------------------------------------------*/
2165 static int
2166 VSC_Scroll_Half_Page_Backward(FORM *form)
2167 {
2168   T((T_CALLED("VSC_Scroll_Half_Page_Backward(%p)"), (void *)form));
2169   returnCode(VSC_Generic(form, -((form->current->rows + 1) / 2)));
2170 }
2171 /*----------------------------------------------------------------------------
2172   End of Vertical scrolling routines
2173   --------------------------------------------------------------------------*/
2174 
2175 /*----------------------------------------------------------------------------
2176   Horizontal scrolling helper routines
2177   --------------------------------------------------------------------------*/
2178 
2179 /*---------------------------------------------------------------------------
2180 |   Facility      :  libnform
2181 |   Function      :  static int HSC_Generic(FORM *form, int ncolumns)
2182 |
2183 |   Description   :  Scroll single-line field forward (ncolumns>0) or
2184 |                    backward (ncolumns<0) this many columns.
2185 |
2186 |   Return Values :  E_OK              - success
2187 |                    E_REQUEST_DENIED  - can't scroll
2188 +--------------------------------------------------------------------------*/
2189 static int
2190 HSC_Generic(FORM *form, int ncolumns)
2191 {
2192   FIELD *field = form->current;
2193   int res = E_REQUEST_DENIED;
2194   int cols_to_go = (ncolumns > 0 ? ncolumns : -ncolumns);
2195 
2196   if (ncolumns > 0)
2197     {
2198       if ((cols_to_go + form->begincol) > (field->dcols - field->cols))
2199 	cols_to_go = field->dcols - field->cols - form->begincol;
2200 
2201       if (cols_to_go > 0)
2202 	{
2203 	  form->curcol += cols_to_go;
2204 	  form->begincol += cols_to_go;
2205 	  res = E_OK;
2206 	}
2207     }
2208   else
2209     {
2210       if (cols_to_go > form->begincol)
2211 	cols_to_go = form->begincol;
2212 
2213       if (cols_to_go > 0)
2214 	{
2215 	  form->curcol -= cols_to_go;
2216 	  form->begincol -= cols_to_go;
2217 	  res = E_OK;
2218 	}
2219     }
2220   return (res);
2221 }
2222 /*----------------------------------------------------------------------------
2223   End of Horizontal scrolling helper routines
2224   --------------------------------------------------------------------------*/
2225 
2226 /*----------------------------------------------------------------------------
2227   Horizontal scrolling routines
2228   --------------------------------------------------------------------------*/
2229 
2230 /*---------------------------------------------------------------------------
2231 |   Facility      :  libnform
2232 |   Function      :  static int Horizontal_Scrolling(
2233 |                                          int (* const fct) (FORM *),
2234 |                                          FORM * form)
2235 |
2236 |   Description   :  Performs the generic horizontal scrolling routines.
2237 |                    This has to check for a single-line field.
2238 |
2239 |   Return Values :  Propagated error code from low-level driver calls
2240 +--------------------------------------------------------------------------*/
2241 static int
2242 Horizontal_Scrolling(int (*const fct) (FORM *), FORM *form)
2243 {
2244   if (Single_Line_Field(form->current))
2245     return fct(form);
2246   else
2247     return (E_REQUEST_DENIED);
2248 }
2249 
2250 /*---------------------------------------------------------------------------
2251 |   Facility      :  libnform
2252 |   Function      :  static int HSC_Scroll_Char_Forward(FORM * form)
2253 |
2254 |   Description   :  Scroll single-line field forward a character
2255 |
2256 |   Return Values :  E_OK                - success
2257 |                    E_REQUEST_DENIED    - no data ahead
2258 +--------------------------------------------------------------------------*/
2259 static int
2260 HSC_Scroll_Char_Forward(FORM *form)
2261 {
2262   T((T_CALLED("HSC_Scroll_Char_Forward(%p)"), (void *)form));
2263   returnCode(HSC_Generic(form, 1));
2264 }
2265 
2266 /*---------------------------------------------------------------------------
2267 |   Facility      :  libnform
2268 |   Function      :  static int HSC_Scroll_Char_Backward(FORM * form)
2269 |
2270 |   Description   :  Scroll single-line field backward a character
2271 |
2272 |   Return Values :  E_OK                - success
2273 |                    E_REQUEST_DENIED    - no data behind
2274 +--------------------------------------------------------------------------*/
2275 static int
2276 HSC_Scroll_Char_Backward(FORM *form)
2277 {
2278   T((T_CALLED("HSC_Scroll_Char_Backward(%p)"), (void *)form));
2279   returnCode(HSC_Generic(form, -1));
2280 }
2281 
2282 /*---------------------------------------------------------------------------
2283 |   Facility      :  libnform
2284 |   Function      :  static int HSC_Horizontal_Line_Forward(FORM* form)
2285 |
2286 |   Description   :  Scroll single-line field forward a line
2287 |
2288 |   Return Values :  E_OK                - success
2289 |                    E_REQUEST_DENIED    - no data ahead
2290 +--------------------------------------------------------------------------*/
2291 static int
2292 HSC_Horizontal_Line_Forward(FORM *form)
2293 {
2294   T((T_CALLED("HSC_Horizontal_Line_Forward(%p)"), (void *)form));
2295   returnCode(HSC_Generic(form, form->current->cols));
2296 }
2297 
2298 /*---------------------------------------------------------------------------
2299 |   Facility      :  libnform
2300 |   Function      :  static int HSC_Horizontal_Half_Line_Forward(FORM* form)
2301 |
2302 |   Description   :  Scroll single-line field forward half a line
2303 |
2304 |   Return Values :  E_OK               - success
2305 |                    E_REQUEST_DENIED   - no data ahead
2306 +--------------------------------------------------------------------------*/
2307 static int
2308 HSC_Horizontal_Half_Line_Forward(FORM *form)
2309 {
2310   T((T_CALLED("HSC_Horizontal_Half_Line_Forward(%p)"), (void *)form));
2311   returnCode(HSC_Generic(form, (form->current->cols + 1) / 2));
2312 }
2313 
2314 /*---------------------------------------------------------------------------
2315 |   Facility      :  libnform
2316 |   Function      :  static int HSC_Horizontal_Line_Backward(FORM* form)
2317 |
2318 |   Description   :  Scroll single-line field backward a line
2319 |
2320 |   Return Values :  E_OK                - success
2321 |                    E_REQUEST_DENIED    - no data behind
2322 +--------------------------------------------------------------------------*/
2323 static int
2324 HSC_Horizontal_Line_Backward(FORM *form)
2325 {
2326   T((T_CALLED("HSC_Horizontal_Line_Backward(%p)"), (void *)form));
2327   returnCode(HSC_Generic(form, -(form->current->cols)));
2328 }
2329 
2330 /*---------------------------------------------------------------------------
2331 |   Facility      :  libnform
2332 |   Function      :  static int HSC_Horizontal_Half_Line_Backward(FORM* form)
2333 |
2334 |   Description   :  Scroll single-line field backward half a line
2335 |
2336 |   Return Values :  E_OK                - success
2337 |                    E_REQUEST_DENIED    - no data behind
2338 +--------------------------------------------------------------------------*/
2339 static int
2340 HSC_Horizontal_Half_Line_Backward(FORM *form)
2341 {
2342   T((T_CALLED("HSC_Horizontal_Half_Line_Backward(%p)"), (void *)form));
2343   returnCode(HSC_Generic(form, -((form->current->cols + 1) / 2)));
2344 }
2345 
2346 /*----------------------------------------------------------------------------
2347   End of Horizontal scrolling routines
2348   --------------------------------------------------------------------------*/
2349 
2350 /*----------------------------------------------------------------------------
2351   Helper routines for Field Editing
2352   --------------------------------------------------------------------------*/
2353 
2354 /*---------------------------------------------------------------------------
2355 |   Facility      :  libnform
2356 |   Function      :  static bool Is_There_Room_For_A_Line(FORM * form)
2357 |
2358 |   Description   :  Check whether or not there is enough room in the
2359 |                    buffer to enter a whole line.
2360 |
2361 |   Return Values :  TRUE   - there is enough space
2362 |                    FALSE  - there is not enough space
2363 +--------------------------------------------------------------------------*/
2364 NCURSES_INLINE static bool
2365 Is_There_Room_For_A_Line(FORM *form)
2366 {
2367   FIELD *field = form->current;
2368   FIELD_CELL *begin_of_last_line, *s;
2369 
2370   Synchronize_Buffer(form);
2371   begin_of_last_line = Address_Of_Row_In_Buffer(field, (field->drows - 1));
2372   s = After_End_Of_Data(begin_of_last_line, field->dcols);
2373   return ((s == begin_of_last_line) ? TRUE : FALSE);
2374 }
2375 
2376 /*---------------------------------------------------------------------------
2377 |   Facility      :  libnform
2378 |   Function      :  static bool Is_There_Room_For_A_Char_In_Line(FORM * form)
2379 |
2380 |   Description   :  Checks whether or not there is room for a new character
2381 |                    in the current line.
2382 |
2383 |   Return Values :  TRUE    - there is room
2384 |                    FALSE   - there is not enough room (line full)
2385 +--------------------------------------------------------------------------*/
2386 NCURSES_INLINE static bool
2387 Is_There_Room_For_A_Char_In_Line(FORM *form)
2388 {
2389   int last_char_in_line;
2390 
2391   wmove(form->w, form->currow, form->current->dcols - 1);
2392   last_char_in_line = (int)(winch(form->w) & A_CHARTEXT);
2393   wmove(form->w, form->currow, form->curcol);
2394   return (((last_char_in_line == form->current->pad) ||
2395 	   is_blank(last_char_in_line)) ? TRUE : FALSE);
2396 }
2397 
2398 #define There_Is_No_Room_For_A_Char_In_Line(f) \
2399   !Is_There_Room_For_A_Char_In_Line(f)
2400 
2401 /*---------------------------------------------------------------------------
2402 |   Facility      :  libnform
2403 |   Function      :  static int Insert_String(
2404 |                                             FORM * form,
2405 |                                             int row,
2406 |                                             char *txt,
2407 |                                             int  len )
2408 |
2409 |   Description   :  Insert the 'len' characters beginning at pointer 'txt'
2410 |                    into the 'row' of the 'form'. The insertion occurs
2411 |                    on the beginning of the row, all other characters are
2412 |                    moved to the right. After the text a pad character will
2413 |                    be inserted to separate the text from the rest. If
2414 |                    necessary the insertion moves characters on the next
2415 |                    line to make place for the requested insertion string.
2416 |
2417 |   Return Values :  E_OK              - success
2418 |                    E_REQUEST_DENIED  -
2419 |                    E_SYSTEM_ERROR    - system error
2420 +--------------------------------------------------------------------------*/
2421 static int
2422 Insert_String(FORM *form, int row, FIELD_CELL *txt, int len)
2423 {
2424   FIELD *field = form->current;
2425   FIELD_CELL *bp = Address_Of_Row_In_Buffer(field, row);
2426   int datalen = (int)(After_End_Of_Data(bp, field->dcols) - bp);
2427   int freelen = field->dcols - datalen;
2428   int requiredlen = len + 1;
2429   int result = E_REQUEST_DENIED;
2430 
2431   if (freelen >= requiredlen)
2432     {
2433       wmove(form->w, row, 0);
2434       myINSNSTR(form->w, txt, len);
2435       wmove(form->w, row, len);
2436       myINSNSTR(form->w, &myBLANK, 1);
2437       result = E_OK;
2438     }
2439   else
2440     {
2441       /* we have to move characters on the next line. If we are on the
2442          last line this may work, if the field is growable */
2443       if ((row == (field->drows - 1)) && Growable(field))
2444 	{
2445 	  if (!Field_Grown(field, 1))
2446 	    return (E_SYSTEM_ERROR);
2447 	  /* !!!Side-Effect : might be changed due to growth!!! */
2448 	  bp = Address_Of_Row_In_Buffer(field, row);
2449 	}
2450 
2451       if (row < (field->drows - 1))
2452 	{
2453 	  FIELD_CELL *split;
2454 
2455 	  split =
2456 	    After_Last_Whitespace_Character(bp,
2457 					    (int)(Get_Start_Of_Data(bp
2458 								    + field->dcols
2459 								    - requiredlen,
2460 								    requiredlen)
2461 						  - bp));
2462 	  /* split points now to the first character of the portion of the
2463 	     line that must be moved to the next line */
2464 	  datalen = (int)(split - bp);	/* + freelen has to stay on this line   */
2465 	  freelen = field->dcols - (datalen + freelen);		/* for the next line */
2466 
2467 	  if ((result = Insert_String(form, row + 1, split, freelen)) == E_OK)
2468 	    {
2469 	      wmove(form->w, row, datalen);
2470 	      wclrtoeol(form->w);
2471 	      wmove(form->w, row, 0);
2472 	      myINSNSTR(form->w, txt, len);
2473 	      wmove(form->w, row, len);
2474 	      myINSNSTR(form->w, &myBLANK, 1);
2475 	      return E_OK;
2476 	    }
2477 	}
2478     }
2479   return (result);
2480 }
2481 
2482 /*---------------------------------------------------------------------------
2483 |   Facility      :  libnform
2484 |   Function      :  static int Wrapping_Not_Necessary_Or_Wrapping_Ok(
2485 |                                             FORM * form)
2486 |
2487 |   Description   :  If a character has been entered into a field, it may
2488 |                    be that wrapping has to occur. This routine checks
2489 |                    whether or not wrapping is required and if so, performs
2490 |                    the wrapping.
2491 |
2492 |   Return Values :  E_OK              - no wrapping required or wrapping
2493 |                                        was successful
2494 |                    E_REQUEST_DENIED  -
2495 |                    E_SYSTEM_ERROR    - some system error
2496 +--------------------------------------------------------------------------*/
2497 static int
2498 Wrapping_Not_Necessary_Or_Wrapping_Ok(FORM *form)
2499 {
2500   FIELD *field = form->current;
2501   int result = E_REQUEST_DENIED;
2502   bool Last_Row = ((field->drows - 1) == form->currow);
2503 
2504   if ((Field_Has_Option(field, O_WRAP)) &&	/* wrapping wanted     */
2505       (!Single_Line_Field(field)) &&	/* must be multi-line  */
2506       (There_Is_No_Room_For_A_Char_In_Line(form)) &&	/* line is full        */
2507       (!Last_Row || Growable(field)))	/* there are more lines */
2508     {
2509       FIELD_CELL *bp;
2510       FIELD_CELL *split;
2511       int chars_to_be_wrapped;
2512       int chars_to_remain_on_line;
2513 
2514       if (Last_Row)
2515 	{
2516 	  /* the above logic already ensures, that in this case the field
2517 	     is growable */
2518 	  if (!Field_Grown(field, 1))
2519 	    return E_SYSTEM_ERROR;
2520 	}
2521       bp = Address_Of_Current_Row_In_Buffer(form);
2522       Window_To_Buffer(form, field);
2523       split = After_Last_Whitespace_Character(bp, field->dcols);
2524       /* split points to the first character of the sequence to be brought
2525          on the next line */
2526       chars_to_remain_on_line = (int)(split - bp);
2527       chars_to_be_wrapped = field->dcols - chars_to_remain_on_line;
2528       if (chars_to_remain_on_line > 0)
2529 	{
2530 	  if ((result = Insert_String(form, form->currow + 1, split,
2531 				      chars_to_be_wrapped)) == E_OK)
2532 	    {
2533 	      wmove(form->w, form->currow, chars_to_remain_on_line);
2534 	      wclrtoeol(form->w);
2535 	      if (form->curcol >= chars_to_remain_on_line)
2536 		{
2537 		  form->currow++;
2538 		  form->curcol -= chars_to_remain_on_line;
2539 		}
2540 	      return E_OK;
2541 	    }
2542 	}
2543       else
2544 	return E_OK;
2545       if (result != E_OK)
2546 	{
2547 	  DeleteChar(form);
2548 	  Window_To_Buffer(form, field);
2549 	  result = E_REQUEST_DENIED;
2550 	}
2551     }
2552   else
2553     result = E_OK;		/* wrapping was not necessary */
2554   return (result);
2555 }
2556 
2557 /*----------------------------------------------------------------------------
2558   Field Editing routines
2559   --------------------------------------------------------------------------*/
2560 
2561 /*---------------------------------------------------------------------------
2562 |   Facility      :  libnform
2563 |   Function      :  static int Field_Editing(
2564 |                                    int (* const fct) (FORM *),
2565 |                                    FORM * form)
2566 |
2567 |   Description   :  Generic routine for field editing requests. The driver
2568 |                    routines are only called for editable fields, the
2569 |                    _WINDOW_MODIFIED flag is set if editing occurred.
2570 |                    This is somewhat special due to the overload semantics
2571 |                    of the NEW_LINE and DEL_PREV requests.
2572 |
2573 |   Return Values :  Error code from low level drivers.
2574 +--------------------------------------------------------------------------*/
2575 static int
2576 Field_Editing(int (*const fct) (FORM *), FORM *form)
2577 {
2578   int res = E_REQUEST_DENIED;
2579 
2580   /* We have to deal here with the specific case of the overloaded
2581      behavior of New_Line and Delete_Previous requests.
2582      They may end up in navigational requests if we are on the first
2583      character in a field. But navigation is also allowed on non-
2584      editable fields.
2585    */
2586   if ((fct == FE_Delete_Previous) &&
2587       ((unsigned)form->opts & O_BS_OVERLOAD) &&
2588       First_Position_In_Current_Field(form))
2589     {
2590       res = Inter_Field_Navigation(FN_Previous_Field, form);
2591     }
2592   else
2593     {
2594       if (fct == FE_New_Line)
2595 	{
2596 	  if (((unsigned)form->opts & O_NL_OVERLOAD) &&
2597 	      First_Position_In_Current_Field(form))
2598 	    {
2599 	      res = Inter_Field_Navigation(FN_Next_Field, form);
2600 	    }
2601 	  else
2602 	    /* FE_New_Line deals itself with the _WINDOW_MODIFIED flag */
2603 	    res = fct(form);
2604 	}
2605       else
2606 	{
2607 	  /* From now on, everything must be editable */
2608 	  if ((unsigned)form->current->opts & O_EDIT)
2609 	    {
2610 	      res = fct(form);
2611 	      if (res == E_OK)
2612 		SetStatus(form, _WINDOW_MODIFIED);
2613 	    }
2614 	}
2615     }
2616   return res;
2617 }
2618 
2619 /*---------------------------------------------------------------------------
2620 |   Facility      :  libnform
2621 |   Function      :  static int FE_New_Line(FORM * form)
2622 |
2623 |   Description   :  Perform a new line request. This is rather complex
2624 |                    compared to other routines in this code due to the
2625 |                    rather difficult to understand description in the
2626 |                    manuals.
2627 |
2628 |   Return Values :  E_OK               - success
2629 |                    E_REQUEST_DENIED   - new line not allowed
2630 |                    E_SYSTEM_ERROR     - system error
2631 +--------------------------------------------------------------------------*/
2632 static int
2633 FE_New_Line(FORM *form)
2634 {
2635   FIELD *field = form->current;
2636   FIELD_CELL *bp, *t;
2637   bool Last_Row = ((field->drows - 1) == form->currow);
2638 
2639   T((T_CALLED("FE_New_Line(%p)"), (void *)form));
2640   if (form->status & _OVLMODE)
2641     {
2642       if (Last_Row &&
2643 	  (!(Growable(field) && !Single_Line_Field(field))))
2644 	{
2645 	  if (!((unsigned)form->opts & O_NL_OVERLOAD))
2646 	    returnCode(E_REQUEST_DENIED);
2647 	  wmove(form->w, form->currow, form->curcol);
2648 	  wclrtoeol(form->w);
2649 	  /* we have to set this here, although it is also
2650 	     handled in the generic routine. The reason is,
2651 	     that FN_Next_Field may fail, but the form is
2652 	     definitively changed */
2653 	  SetStatus(form, _WINDOW_MODIFIED);
2654 	  returnCode(Inter_Field_Navigation(FN_Next_Field, form));
2655 	}
2656       else
2657 	{
2658 	  if (Last_Row && !Field_Grown(field, 1))
2659 	    {
2660 	      /* N.B.: due to the logic in the 'if', LastRow==TRUE
2661 	         means here that the field is growable and not
2662 	         a single-line field */
2663 	      returnCode(E_SYSTEM_ERROR);
2664 	    }
2665 	  wmove(form->w, form->currow, form->curcol);
2666 	  wclrtoeol(form->w);
2667 	  form->currow++;
2668 	  form->curcol = 0;
2669 	  SetStatus(form, _WINDOW_MODIFIED);
2670 	  returnCode(E_OK);
2671 	}
2672     }
2673   else
2674     {
2675       /* Insert Mode */
2676       if (Last_Row &&
2677 	  !(Growable(field) && !Single_Line_Field(field)))
2678 	{
2679 	  if (!((unsigned)form->opts & O_NL_OVERLOAD))
2680 	    returnCode(E_REQUEST_DENIED);
2681 	  returnCode(Inter_Field_Navigation(FN_Next_Field, form));
2682 	}
2683       else
2684 	{
2685 	  bool May_Do_It = !Last_Row && Is_There_Room_For_A_Line(form);
2686 
2687 	  if (!(May_Do_It || Growable(field)))
2688 	    returnCode(E_REQUEST_DENIED);
2689 	  if (!May_Do_It && !Field_Grown(field, 1))
2690 	    returnCode(E_SYSTEM_ERROR);
2691 
2692 	  bp = Address_Of_Current_Position_In_Buffer(form);
2693 	  t = After_End_Of_Data(bp, field->dcols - form->curcol);
2694 	  wmove(form->w, form->currow, form->curcol);
2695 	  wclrtoeol(form->w);
2696 	  form->currow++;
2697 	  form->curcol = 0;
2698 	  wmove(form->w, form->currow, form->curcol);
2699 	  winsertln(form->w);
2700 	  myADDNSTR(form->w, bp, (int)(t - bp));
2701 	  SetStatus(form, _WINDOW_MODIFIED);
2702 	  returnCode(E_OK);
2703 	}
2704     }
2705 }
2706 
2707 /*---------------------------------------------------------------------------
2708 |   Facility      :  libnform
2709 |   Function      :  static int FE_Insert_Character(FORM * form)
2710 |
2711 |   Description   :  Insert blank character at the cursor position
2712 |
2713 |   Return Values :  E_OK
2714 |                    E_REQUEST_DENIED
2715 +--------------------------------------------------------------------------*/
2716 static int
2717 FE_Insert_Character(FORM *form)
2718 {
2719   FIELD *field = form->current;
2720   int result = E_REQUEST_DENIED;
2721 
2722   T((T_CALLED("FE_Insert_Character(%p)"), (void *)form));
2723   if (Check_Char(form, field, field->type, (int)C_BLANK,
2724 		 (TypeArgument *)(field->arg)))
2725     {
2726       bool There_Is_Room = Is_There_Room_For_A_Char_In_Line(form);
2727 
2728       if (There_Is_Room ||
2729 	  ((Single_Line_Field(field) && Growable(field))))
2730 	{
2731 	  if (!There_Is_Room && !Field_Grown(field, 1))
2732 	    result = E_SYSTEM_ERROR;
2733 	  else
2734 	    {
2735 	      winsch(form->w, (chtype)C_BLANK);
2736 	      result = Wrapping_Not_Necessary_Or_Wrapping_Ok(form);
2737 	    }
2738 	}
2739     }
2740   returnCode(result);
2741 }
2742 
2743 /*---------------------------------------------------------------------------
2744 |   Facility      :  libnform
2745 |   Function      :  static int FE_Insert_Line(FORM * form)
2746 |
2747 |   Description   :  Insert a blank line at the cursor position
2748 |
2749 |   Return Values :  E_OK               - success
2750 |                    E_REQUEST_DENIED   - line can not be inserted
2751 +--------------------------------------------------------------------------*/
2752 static int
2753 FE_Insert_Line(FORM *form)
2754 {
2755   FIELD *field = form->current;
2756   int result = E_REQUEST_DENIED;
2757 
2758   T((T_CALLED("FE_Insert_Line(%p)"), (void *)form));
2759   if (Check_Char(form, field,
2760 		 field->type, (int)C_BLANK, (TypeArgument *)(field->arg)))
2761     {
2762       bool Maybe_Done = (form->currow != (field->drows - 1)) &&
2763       Is_There_Room_For_A_Line(form);
2764 
2765       if (!Single_Line_Field(field) &&
2766 	  (Maybe_Done || Growable(field)))
2767 	{
2768 	  if (!Maybe_Done && !Field_Grown(field, 1))
2769 	    result = E_SYSTEM_ERROR;
2770 	  else
2771 	    {
2772 	      form->curcol = 0;
2773 	      winsertln(form->w);
2774 	      result = E_OK;
2775 	    }
2776 	}
2777     }
2778   returnCode(result);
2779 }
2780 
2781 /*---------------------------------------------------------------------------
2782 |   Facility      :  libnform
2783 |   Function      :  static int FE_Delete_Character(FORM * form)
2784 |
2785 |   Description   :  Delete character at the cursor position
2786 |
2787 |   Return Values :  E_OK    - success
2788 +--------------------------------------------------------------------------*/
2789 static int
2790 FE_Delete_Character(FORM *form)
2791 {
2792   T((T_CALLED("FE_Delete_Character(%p)"), (void *)form));
2793   DeleteChar(form);
2794   returnCode(E_OK);
2795 }
2796 
2797 /*---------------------------------------------------------------------------
2798 |   Facility      :  libnform
2799 |   Function      :  static int FE_Delete_Previous(FORM * form)
2800 |
2801 |   Description   :  Delete character before cursor. Again this is a rather
2802 |                    difficult piece compared to others due to the overloading
2803 |                    semantics of backspace.
2804 |                    N.B.: The case of overloaded BS on first field position
2805 |                          is already handled in the generic routine.
2806 |
2807 |   Return Values :  E_OK                - success
2808 |                    E_REQUEST_DENIED    - Character can't be deleted
2809 +--------------------------------------------------------------------------*/
2810 static int
2811 FE_Delete_Previous(FORM *form)
2812 {
2813   FIELD *field = form->current;
2814 
2815   T((T_CALLED("FE_Delete_Previous(%p)"), (void *)form));
2816   if (First_Position_In_Current_Field(form))
2817     returnCode(E_REQUEST_DENIED);
2818 
2819   if ((--(form->curcol)) < 0)
2820     {
2821       FIELD_CELL *this_line, *prev_line, *prev_end, *this_end;
2822       int this_row = form->currow;
2823 
2824       form->curcol++;
2825       if (form->status & _OVLMODE)
2826 	returnCode(E_REQUEST_DENIED);
2827 
2828       prev_line = Address_Of_Row_In_Buffer(field, (form->currow - 1));
2829       this_line = Address_Of_Row_In_Buffer(field, (form->currow));
2830       Synchronize_Buffer(form);
2831       prev_end = After_End_Of_Data(prev_line, field->dcols);
2832       this_end = After_End_Of_Data(this_line, field->dcols);
2833       if ((int)(this_end - this_line) >
2834 	  (field->cols - (int)(prev_end - prev_line)))
2835 	returnCode(E_REQUEST_DENIED);
2836       wmove(form->w, form->currow, form->curcol);
2837       wdeleteln(form->w);
2838       Adjust_Cursor_Position(form, prev_end);
2839       /*
2840        * If we did not really move to the previous line, help the user a
2841        * little.  It is however a little inconsistent.  Normally, when
2842        * backspacing around the point where text wraps to a new line in a
2843        * multi-line form, we absorb one keystroke for the wrapping point.  That
2844        * is consistent with SVr4 forms.  However, SVr4 does not allow typing
2845        * into the last column of the field, and requires the user to enter a
2846        * newline to move to the next line.  Therefore it can consistently eat
2847        * that keystroke.  Since ncurses allows the last column, it wraps
2848        * automatically (given the proper options).  But we cannot eat the
2849        * keystroke to back over the wrapping point, since that would put the
2850        * cursor past the end of the form field.  In this case, just delete the
2851        * character at the end of the field.
2852        */
2853       if (form->currow == this_row && this_row > 0)
2854 	{
2855 	  form->currow -= 1;
2856 	  form->curcol = field->dcols - 1;
2857 	  DeleteChar(form);
2858 	}
2859       else
2860 	{
2861 	  wmove(form->w, form->currow, form->curcol);
2862 	  myADDNSTR(form->w, this_line, (int)(this_end - this_line));
2863 	}
2864     }
2865   else
2866     {
2867       DeleteChar(form);
2868     }
2869   returnCode(E_OK);
2870 }
2871 
2872 /*---------------------------------------------------------------------------
2873 |   Facility      :  libnform
2874 |   Function      :  static int FE_Delete_Line(FORM * form)
2875 |
2876 |   Description   :  Delete line at cursor position.
2877 |
2878 |   Return Values :  E_OK  - success
2879 +--------------------------------------------------------------------------*/
2880 static int
2881 FE_Delete_Line(FORM *form)
2882 {
2883   T((T_CALLED("FE_Delete_Line(%p)"), (void *)form));
2884   form->curcol = 0;
2885   wdeleteln(form->w);
2886   returnCode(E_OK);
2887 }
2888 
2889 /*---------------------------------------------------------------------------
2890 |   Facility      :  libnform
2891 |   Function      :  static int FE_Delete_Word(FORM * form)
2892 |
2893 |   Description   :  Delete word at cursor position
2894 |
2895 |   Return Values :  E_OK               - success
2896 |                    E_REQUEST_DENIED   - failure
2897 +--------------------------------------------------------------------------*/
2898 static int
2899 FE_Delete_Word(FORM *form)
2900 {
2901   FIELD *field = form->current;
2902   FIELD_CELL *bp = Address_Of_Current_Row_In_Buffer(form);
2903   FIELD_CELL *ep = bp + field->dcols;
2904   FIELD_CELL *cp = bp + form->curcol;
2905   FIELD_CELL *s;
2906 
2907   T((T_CALLED("FE_Delete_Word(%p)"), (void *)form));
2908   Synchronize_Buffer(form);
2909   if (ISBLANK(*cp))
2910     returnCode(E_REQUEST_DENIED);	/* not in word */
2911 
2912   /* move cursor to begin of word and erase to end of screen-line */
2913   Adjust_Cursor_Position(form,
2914 			 After_Last_Whitespace_Character(bp, form->curcol));
2915   wmove(form->w, form->currow, form->curcol);
2916   wclrtoeol(form->w);
2917 
2918   /* skip over word in buffer */
2919   s = Get_First_Whitespace_Character(cp, (int)(ep - cp));
2920   /* to begin of next word    */
2921   s = Get_Start_Of_Data(s, (int)(ep - s));
2922   if ((s != cp) && !ISBLANK(*s))
2923     {
2924       /* copy remaining line to window */
2925       myADDNSTR(form->w, s, (int)(s - After_End_Of_Data(s, (int)(ep - s))));
2926     }
2927   returnCode(E_OK);
2928 }
2929 
2930 /*---------------------------------------------------------------------------
2931 |   Facility      :  libnform
2932 |   Function      :  static int FE_Clear_To_End_Of_Line(FORM * form)
2933 |
2934 |   Description   :  Clear to end of current line.
2935 |
2936 |   Return Values :  E_OK   - success
2937 +--------------------------------------------------------------------------*/
2938 static int
2939 FE_Clear_To_End_Of_Line(FORM *form)
2940 {
2941   T((T_CALLED("FE_Clear_To_End_Of_Line(%p)"), (void *)form));
2942   wmove(form->w, form->currow, form->curcol);
2943   wclrtoeol(form->w);
2944   returnCode(E_OK);
2945 }
2946 
2947 /*---------------------------------------------------------------------------
2948 |   Facility      :  libnform
2949 |   Function      :  static int FE_Clear_To_End_Of_Field(FORM * form)
2950 |
2951 |   Description   :  Clear to end of field.
2952 |
2953 |   Return Values :  E_OK   - success
2954 +--------------------------------------------------------------------------*/
2955 static int
2956 FE_Clear_To_End_Of_Field(FORM *form)
2957 {
2958   T((T_CALLED("FE_Clear_To_End_Of_Field(%p)"), (void *)form));
2959   wmove(form->w, form->currow, form->curcol);
2960   wclrtobot(form->w);
2961   returnCode(E_OK);
2962 }
2963 
2964 /*---------------------------------------------------------------------------
2965 |   Facility      :  libnform
2966 |   Function      :  static int FE_Clear_Field(FORM * form)
2967 |
2968 |   Description   :  Clear entire field.
2969 |
2970 |   Return Values :  E_OK   - success
2971 +--------------------------------------------------------------------------*/
2972 static int
2973 FE_Clear_Field(FORM *form)
2974 {
2975   T((T_CALLED("FE_Clear_Field(%p)"), (void *)form));
2976   form->currow = form->curcol = 0;
2977   werase(form->w);
2978   returnCode(E_OK);
2979 }
2980 /*----------------------------------------------------------------------------
2981   END of Field Editing routines
2982   --------------------------------------------------------------------------*/
2983 
2984 /*----------------------------------------------------------------------------
2985   Edit Mode routines
2986   --------------------------------------------------------------------------*/
2987 
2988 /*---------------------------------------------------------------------------
2989 |   Facility      :  libnform
2990 |   Function      :  static int EM_Overlay_Mode(FORM * form)
2991 |
2992 |   Description   :  Switch to overlay mode.
2993 |
2994 |   Return Values :  E_OK   - success
2995 +--------------------------------------------------------------------------*/
2996 static int
2997 EM_Overlay_Mode(FORM *form)
2998 {
2999   T((T_CALLED("EM_Overlay_Mode(%p)"), (void *)form));
3000   SetStatus(form, _OVLMODE);
3001   returnCode(E_OK);
3002 }
3003 
3004 /*---------------------------------------------------------------------------
3005 |   Facility      :  libnform
3006 |   Function      :  static int EM_Insert_Mode(FORM * form)
3007 |
3008 |   Description   :  Switch to insert mode
3009 |
3010 |   Return Values :  E_OK   - success
3011 +--------------------------------------------------------------------------*/
3012 static int
3013 EM_Insert_Mode(FORM *form)
3014 {
3015   T((T_CALLED("EM_Insert_Mode(%p)"), (void *)form));
3016   ClrStatus(form, _OVLMODE);
3017   returnCode(E_OK);
3018 }
3019 
3020 /*----------------------------------------------------------------------------
3021   END of Edit Mode routines
3022   --------------------------------------------------------------------------*/
3023 
3024 /*----------------------------------------------------------------------------
3025   Helper routines for Choice Requests
3026   --------------------------------------------------------------------------*/
3027 
3028 /*---------------------------------------------------------------------------
3029 |   Facility      :  libnform
3030 |   Function      :  static bool Next_Choice(FORM * form,
3031 |                                            FIELDTYPE * typ,
3032 |                                            FIELD * field,
3033 |                                            TypeArgument *argp)
3034 |
3035 |   Description   :  Get the next field choice. For linked types this is
3036 |                    done recursively.
3037 |
3038 |   Return Values :  TRUE    - next choice successfully retrieved
3039 |                    FALSE   - couldn't retrieve next choice
3040 +--------------------------------------------------------------------------*/
3041 static bool
3042 Next_Choice(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
3043 {
3044   if (!typ || !(typ->status & _HAS_CHOICE))
3045     return FALSE;
3046 
3047   if (typ->status & _LINKED_TYPE)
3048     {
3049       assert(argp);
3050       return (
3051 	       Next_Choice(form, typ->left, field, argp->left) ||
3052 	       Next_Choice(form, typ->right, field, argp->right));
3053     }
3054   else
3055     {
3056 #if NCURSES_INTEROP_FUNCS
3057       assert(typ->enum_next.onext);
3058       if (typ->status & _GENERIC)
3059 	return typ->enum_next.gnext(form, field, (void *)argp);
3060       else
3061 	return typ->enum_next.onext(field, (void *)argp);
3062 #else
3063       assert(typ->next);
3064       return typ->next(field, (void *)argp);
3065 #endif
3066     }
3067 }
3068 
3069 /*---------------------------------------------------------------------------
3070 |   Facility      :  libnform
3071 |   Function      :  static bool Previous_Choice(FORM * form,
3072 |                                                FIELDTYPE * typ,
3073 |                                                FIELD * field,
3074 |                                                TypeArgument *argp)
3075 |
3076 |   Description   :  Get the previous field choice. For linked types this
3077 |                    is done recursively.
3078 |
3079 |   Return Values :  TRUE    - previous choice successfully retrieved
3080 |                    FALSE   - couldn't retrieve previous choice
3081 +--------------------------------------------------------------------------*/
3082 static bool
3083 Previous_Choice(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
3084 {
3085   if (!typ || !(typ->status & _HAS_CHOICE))
3086     return FALSE;
3087 
3088   if (typ->status & _LINKED_TYPE)
3089     {
3090       assert(argp);
3091       return (
3092 	       Previous_Choice(form, typ->left, field, argp->left) ||
3093 	       Previous_Choice(form, typ->right, field, argp->right));
3094     }
3095   else
3096     {
3097 #if NCURSES_INTEROP_FUNCS
3098       assert(typ->enum_prev.oprev);
3099       if (typ->status & _GENERIC)
3100 	return typ->enum_prev.gprev(form, field, (void *)argp);
3101       else
3102 	return typ->enum_prev.oprev(field, (void *)argp);
3103 #else
3104       assert(typ->prev);
3105       return typ->prev(field, (void *)argp);
3106 #endif
3107     }
3108 }
3109 /*----------------------------------------------------------------------------
3110   End of Helper routines for Choice Requests
3111   --------------------------------------------------------------------------*/
3112 
3113 /*----------------------------------------------------------------------------
3114   Routines for Choice Requests
3115   --------------------------------------------------------------------------*/
3116 
3117 /*---------------------------------------------------------------------------
3118 |   Facility      :  libnform
3119 |   Function      :  static int CR_Next_Choice(FORM * form)
3120 |
3121 |   Description   :  Get the next field choice.
3122 |
3123 |   Return Values :  E_OK              - success
3124 |                    E_REQUEST_DENIED  - next choice couldn't be retrieved
3125 +--------------------------------------------------------------------------*/
3126 static int
3127 CR_Next_Choice(FORM *form)
3128 {
3129   FIELD *field = form->current;
3130 
3131   T((T_CALLED("CR_Next_Choice(%p)"), (void *)form));
3132   Synchronize_Buffer(form);
3133   returnCode((Next_Choice(form, field->type, field, (TypeArgument *)(field->arg)))
3134 	     ? E_OK
3135 	     : E_REQUEST_DENIED);
3136 }
3137 
3138 /*---------------------------------------------------------------------------
3139 |   Facility      :  libnform
3140 |   Function      :  static int CR_Previous_Choice(FORM * form)
3141 |
3142 |   Description   :  Get the previous field choice.
3143 |
3144 |   Return Values :  E_OK              - success
3145 |                    E_REQUEST_DENIED  - prev. choice couldn't be retrieved
3146 +--------------------------------------------------------------------------*/
3147 static int
3148 CR_Previous_Choice(FORM *form)
3149 {
3150   FIELD *field = form->current;
3151 
3152   T((T_CALLED("CR_Previous_Choice(%p)"), (void *)form));
3153   Synchronize_Buffer(form);
3154   returnCode((Previous_Choice(form, field->type, field, (TypeArgument *)(field->arg)))
3155 	     ? E_OK
3156 	     : E_REQUEST_DENIED);
3157 }
3158 /*----------------------------------------------------------------------------
3159   End of Routines for Choice Requests
3160   --------------------------------------------------------------------------*/
3161 
3162 /*----------------------------------------------------------------------------
3163   Helper routines for Field Validations.
3164   --------------------------------------------------------------------------*/
3165 
3166 /*---------------------------------------------------------------------------
3167 |   Facility      :  libnform
3168 |   Function      :  static bool Check_Field(FORM* form,
3169 |                                            FIELDTYPE * typ,
3170 |                                            FIELD * field,
3171 |                                            TypeArgument * argp)
3172 |
3173 |   Description   :  Check the field according to its fieldtype and its
3174 |                    actual arguments. For linked fieldtypes this is done
3175 |                    recursively.
3176 |
3177 |   Return Values :  TRUE       - field is valid
3178 |                    FALSE      - field is invalid.
3179 +--------------------------------------------------------------------------*/
3180 static bool
3181 Check_Field(FORM *form, FIELDTYPE *typ, FIELD *field, TypeArgument *argp)
3182 {
3183   if (typ)
3184     {
3185       if (Field_Has_Option(field, O_NULLOK))
3186 	{
3187 	  FIELD_CELL *bp = field->buf;
3188 
3189 	  assert(bp);
3190 	  while (ISBLANK(*bp))
3191 	    {
3192 	      bp++;
3193 	    }
3194 	  if (CharOf(*bp) == 0)
3195 	    return TRUE;
3196 	}
3197 
3198       if (typ->status & _LINKED_TYPE)
3199 	{
3200 	  assert(argp);
3201 	  return (
3202 		   Check_Field(form, typ->left, field, argp->left) ||
3203 		   Check_Field(form, typ->right, field, argp->right));
3204 	}
3205       else
3206 	{
3207 #if NCURSES_INTEROP_FUNCS
3208 	  if (typ->fieldcheck.ofcheck)
3209 	    {
3210 	      if (typ->status & _GENERIC)
3211 		return typ->fieldcheck.gfcheck(form, field, (void *)argp);
3212 	      else
3213 		return typ->fieldcheck.ofcheck(field, (void *)argp);
3214 	    }
3215 #else
3216 	  if (typ->fcheck)
3217 	    return typ->fcheck(field, (void *)argp);
3218 #endif
3219 	}
3220     }
3221   return TRUE;
3222 }
3223 
3224 /*---------------------------------------------------------------------------
3225 |   Facility      :  libnform
3226 |   Function      :  bool _nc_Internal_Validation(FORM * form )
3227 |
3228 |   Description   :  Validate the current field of the form.
3229 |
3230 |   Return Values :  TRUE  - field is valid
3231 |                    FALSE - field is invalid
3232 +--------------------------------------------------------------------------*/
3233 FORM_EXPORT(bool)
3234 _nc_Internal_Validation(FORM *form)
3235 {
3236   FIELD *field;
3237 
3238   field = form->current;
3239 
3240   Synchronize_Buffer(form);
3241   if ((form->status & _FCHECK_REQUIRED) ||
3242       (!(Field_Has_Option(field, O_PASSOK))))
3243     {
3244       if (!Check_Field(form, field->type, field, (TypeArgument *)(field->arg)))
3245 	return FALSE;
3246       ClrStatus(form, _FCHECK_REQUIRED);
3247       SetStatus(field, _CHANGED);
3248       Synchronize_Linked_Fields(field);
3249     }
3250   return TRUE;
3251 }
3252 /*----------------------------------------------------------------------------
3253   End of Helper routines for Field Validations.
3254   --------------------------------------------------------------------------*/
3255 
3256 /*----------------------------------------------------------------------------
3257   Routines for Field Validation.
3258   --------------------------------------------------------------------------*/
3259 
3260 /*---------------------------------------------------------------------------
3261 |   Facility      :  libnform
3262 |   Function      :  static int FV_Validation(FORM * form)
3263 |
3264 |   Description   :  Validate the current field of the form.
3265 |
3266 |   Return Values :  E_OK             - field valid
3267 |                    E_INVALID_FIELD  - field not valid
3268 +--------------------------------------------------------------------------*/
3269 static int
3270 FV_Validation(FORM *form)
3271 {
3272   T((T_CALLED("FV_Validation(%p)"), (void *)form));
3273   if (_nc_Internal_Validation(form))
3274     returnCode(E_OK);
3275   else
3276     returnCode(E_INVALID_FIELD);
3277 }
3278 /*----------------------------------------------------------------------------
3279   End of routines for Field Validation.
3280   --------------------------------------------------------------------------*/
3281 
3282 /*----------------------------------------------------------------------------
3283   Helper routines for Inter-Field Navigation
3284   --------------------------------------------------------------------------*/
3285 
3286 /*---------------------------------------------------------------------------
3287 |   Facility      :  libnform
3288 |   Function      :  static FIELD *Next_Field_On_Page(FIELD * field)
3289 |
3290 |   Description   :  Get the next field after the given field on the current
3291 |                    page. The order of fields is the one defined by the
3292 |                    field's array. Only visible and active fields are
3293 |                    counted.
3294 |
3295 |   Return Values :  Pointer to the next field.
3296 +--------------------------------------------------------------------------*/
3297 NCURSES_INLINE static FIELD *
3298 Next_Field_On_Page(FIELD *field)
3299 {
3300   FORM *form = field->form;
3301   FIELD **field_on_page = &form->field[field->index];
3302   FIELD **first_on_page = &form->field[form->page[form->curpage].pmin];
3303   FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3304 
3305   do
3306     {
3307       field_on_page =
3308 	(field_on_page == last_on_page) ? first_on_page : field_on_page + 1;
3309       if (Field_Is_Selectable(*field_on_page))
3310 	break;
3311     }
3312   while (field != (*field_on_page));
3313   return (*field_on_page);
3314 }
3315 
3316 /*---------------------------------------------------------------------------
3317 |   Facility      :  libnform
3318 |   Function      :  FIELD* _nc_First_Active_Field(FORM * form)
3319 |
3320 |   Description   :  Get the first active field on the current page,
3321 |                    if there are such. If there are none, get the first
3322 |                    visible field on the page. If there are also none,
3323 |                    we return the first field on page and hope the best.
3324 |
3325 |   Return Values :  Pointer to calculated field.
3326 +--------------------------------------------------------------------------*/
3327 FORM_EXPORT(FIELD *)
3328 _nc_First_Active_Field(FORM *form)
3329 {
3330   FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3331   FIELD *proposed = Next_Field_On_Page(*last_on_page);
3332 
3333   if (proposed == *last_on_page)
3334     {
3335       /* there might be the special situation, where there is no
3336          active and visible field on the current page. We then select
3337          the first visible field on this readonly page
3338        */
3339       if (Field_Is_Not_Selectable(proposed))
3340 	{
3341 	  FIELD **field = &form->field[proposed->index];
3342 	  FIELD **first = &form->field[form->page[form->curpage].pmin];
3343 
3344 	  do
3345 	    {
3346 	      field = (field == last_on_page) ? first : field + 1;
3347 	      if (Field_Has_Option(*field, O_VISIBLE))
3348 		break;
3349 	    }
3350 	  while (proposed != (*field));
3351 
3352 	  proposed = *field;
3353 
3354 	  if ((proposed == *last_on_page) &&
3355 	      !((unsigned)proposed->opts & O_VISIBLE))
3356 	    {
3357 	      /* This means, there is also no visible field on the page.
3358 	         So we propose the first one and hope the very best...
3359 	         Some very clever user has designed a readonly and invisible
3360 	         page on this form.
3361 	       */
3362 	      proposed = *first;
3363 	    }
3364 	}
3365     }
3366   return (proposed);
3367 }
3368 
3369 /*---------------------------------------------------------------------------
3370 |   Facility      :  libnform
3371 |   Function      :  static FIELD *Previous_Field_On_Page(FIELD * field)
3372 |
3373 |   Description   :  Get the previous field before the given field on the
3374 |                    current page. The order of fields is the one defined by
3375 |                    the field's array. Only visible and active fields are
3376 |                    counted.
3377 |
3378 |   Return Values :  Pointer to the previous field.
3379 +--------------------------------------------------------------------------*/
3380 NCURSES_INLINE static FIELD *
3381 Previous_Field_On_Page(FIELD *field)
3382 {
3383   FORM *form = field->form;
3384   FIELD **field_on_page = &form->field[field->index];
3385   FIELD **first_on_page = &form->field[form->page[form->curpage].pmin];
3386   FIELD **last_on_page = &form->field[form->page[form->curpage].pmax];
3387 
3388   do
3389     {
3390       field_on_page =
3391 	(field_on_page == first_on_page) ? last_on_page : field_on_page - 1;
3392       if (Field_Is_Selectable(*field_on_page))
3393 	break;
3394     }
3395   while (field != (*field_on_page));
3396 
3397   return (*field_on_page);
3398 }
3399 
3400 /*---------------------------------------------------------------------------
3401 |   Facility      :  libnform
3402 |   Function      :  static FIELD *Sorted_Next_Field(FIELD * field)
3403 |
3404 |   Description   :  Get the next field after the given field on the current
3405 |                    page. The order of fields is the one defined by the
3406 |                    (row,column) geometry, rows are major.
3407 |
3408 |   Return Values :  Pointer to the next field.
3409 +--------------------------------------------------------------------------*/
3410 NCURSES_INLINE static FIELD *
3411 Sorted_Next_Field(FIELD *field)
3412 {
3413   FIELD *field_on_page = field;
3414 
3415   do
3416     {
3417       field_on_page = field_on_page->snext;
3418       if (Field_Is_Selectable(field_on_page))
3419 	break;
3420     }
3421   while (field_on_page != field);
3422 
3423   return (field_on_page);
3424 }
3425 
3426 /*---------------------------------------------------------------------------
3427 |   Facility      :  libnform
3428 |   Function      :  static FIELD *Sorted_Previous_Field(FIELD * field)
3429 |
3430 |   Description   :  Get the previous field before the given field on the
3431 |                    current page. The order of fields is the one defined
3432 |                    by the (row,column) geometry, rows are major.
3433 |
3434 |   Return Values :  Pointer to the previous field.
3435 +--------------------------------------------------------------------------*/
3436 NCURSES_INLINE static FIELD *
3437 Sorted_Previous_Field(FIELD *field)
3438 {
3439   FIELD *field_on_page = field;
3440 
3441   do
3442     {
3443       field_on_page = field_on_page->sprev;
3444       if (Field_Is_Selectable(field_on_page))
3445 	break;
3446     }
3447   while (field_on_page != field);
3448 
3449   return (field_on_page);
3450 }
3451 
3452 /*---------------------------------------------------------------------------
3453 |   Facility      :  libnform
3454 |   Function      :  static FIELD *Left_Neighbor_Field(FIELD * field)
3455 |
3456 |   Description   :  Get the left neighbor of the field on the same line
3457 |                    and the same page. Cycles through the line.
3458 |
3459 |   Return Values :  Pointer to left neighbor field.
3460 +--------------------------------------------------------------------------*/
3461 NCURSES_INLINE static FIELD *
3462 Left_Neighbor_Field(FIELD *field)
3463 {
3464   FIELD *field_on_page = field;
3465 
3466   /* For a field that has really a left neighbor, the while clause
3467      immediately fails and the loop is left, positioned at the right
3468      neighbor. Otherwise we cycle backwards through the sorted field list
3469      until we enter the same line (from the right end).
3470    */
3471   do
3472     {
3473       field_on_page = Sorted_Previous_Field(field_on_page);
3474     }
3475   while (field_on_page->frow != field->frow);
3476 
3477   return (field_on_page);
3478 }
3479 
3480 /*---------------------------------------------------------------------------
3481 |   Facility      :  libnform
3482 |   Function      :  static FIELD *Right_Neighbor_Field(FIELD * field)
3483 |
3484 |   Description   :  Get the right neighbor of the field on the same line
3485 |                    and the same page.
3486 |
3487 |   Return Values :  Pointer to right neighbor field.
3488 +--------------------------------------------------------------------------*/
3489 NCURSES_INLINE static FIELD *
3490 Right_Neighbor_Field(FIELD *field)
3491 {
3492   FIELD *field_on_page = field;
3493 
3494   /* See the comments on Left_Neighbor_Field to understand how it works */
3495   do
3496     {
3497       field_on_page = Sorted_Next_Field(field_on_page);
3498     }
3499   while (field_on_page->frow != field->frow);
3500 
3501   return (field_on_page);
3502 }
3503 
3504 /*---------------------------------------------------------------------------
3505 |   Facility      :  libnform
3506 |   Function      :  static FIELD *Upper_Neighbor_Field(FIELD * field)
3507 |
3508 |   Description   :  Because of the row-major nature of sorting the fields,
3509 |                    it is more difficult to define what the upper neighbor
3510 |                    field really means. We define that it must be on a
3511 |                    'previous' line (cyclic order!) and is the rightmost
3512 |                    field lying on the left side of the given field. If
3513 |                    this set is empty, we take the first field on the line.
3514 |
3515 |   Return Values :  Pointer to the upper neighbor field.
3516 +--------------------------------------------------------------------------*/
3517 static FIELD *
3518 Upper_Neighbor_Field(FIELD *field)
3519 {
3520   FIELD *field_on_page = field;
3521   int frow = field->frow;
3522   int fcol = field->fcol;
3523 
3524   /* Walk back to the 'previous' line. The second term in the while clause
3525      just guarantees that we stop if we cycled through the line because
3526      there might be no 'previous' line if the page has just one line.
3527    */
3528   do
3529     {
3530       field_on_page = Sorted_Previous_Field(field_on_page);
3531     }
3532   while (field_on_page->frow == frow && field_on_page->fcol != fcol);
3533 
3534   if (field_on_page->frow != frow)
3535     {
3536       /* We really found a 'previous' line. We are positioned at the
3537          rightmost field on this line */
3538       frow = field_on_page->frow;
3539 
3540       /* We walk to the left as long as we are really right of the
3541          field. */
3542       while (field_on_page->frow == frow && field_on_page->fcol > fcol)
3543 	field_on_page = Sorted_Previous_Field(field_on_page);
3544 
3545       /* If we wrapped, just go to the right which is the first field on
3546          the row */
3547       if (field_on_page->frow != frow)
3548 	field_on_page = Sorted_Next_Field(field_on_page);
3549     }
3550 
3551   return (field_on_page);
3552 }
3553 
3554 /*---------------------------------------------------------------------------
3555 |   Facility      :  libnform
3556 |   Function      :  static FIELD *Down_Neighbor_Field(FIELD * field)
3557 |
3558 |   Description   :  Because of the row-major nature of sorting the fields,
3559 |                    it is more difficult to define what the down neighbor
3560 |                    field really means. We define that it must be on a
3561 |                    'next' line (cyclic order!) and is the leftmost
3562 |                    field laying on the right side of the given field. If
3563 |                    this set is empty, we take the last field on the line.
3564 |
3565 |   Return Values :  Pointer to the upper neighbor field.
3566 +--------------------------------------------------------------------------*/
3567 static FIELD *
3568 Down_Neighbor_Field(FIELD *field)
3569 {
3570   FIELD *field_on_page = field;
3571   int frow = field->frow;
3572   int fcol = field->fcol;
3573 
3574   /* Walk forward to the 'next' line. The second term in the while clause
3575      just guarantees that we stop if we cycled through the line because
3576      there might be no 'next' line if the page has just one line.
3577    */
3578   do
3579     {
3580       field_on_page = Sorted_Next_Field(field_on_page);
3581     }
3582   while (field_on_page->frow == frow && field_on_page->fcol != fcol);
3583 
3584   if (field_on_page->frow != frow)
3585     {
3586       /* We really found a 'next' line. We are positioned at the rightmost
3587          field on this line */
3588       frow = field_on_page->frow;
3589 
3590       /* We walk to the right as long as we are really left of the
3591          field. */
3592       while (field_on_page->frow == frow && field_on_page->fcol < fcol)
3593 	field_on_page = Sorted_Next_Field(field_on_page);
3594 
3595       /* If we wrapped, just go to the left which is the last field on
3596          the row */
3597       if (field_on_page->frow != frow)
3598 	field_on_page = Sorted_Previous_Field(field_on_page);
3599     }
3600 
3601   return (field_on_page);
3602 }
3603 
3604 /*----------------------------------------------------------------------------
3605   Inter-Field Navigation routines
3606   --------------------------------------------------------------------------*/
3607 
3608 /*---------------------------------------------------------------------------
3609 |   Facility      :  libnform
3610 |   Function      :  static int Inter_Field_Navigation(
3611 |                                           int (* const fct) (FORM *),
3612 |                                           FORM * form)
3613 |
3614 |   Description   :  Generic behavior for changing the current field, the
3615 |                    field is left and a new field is entered. So the field
3616 |                    must be validated and the field init/term hooks must
3617 |                    be called.
3618 |
3619 |   Return Values :  E_OK                - success
3620 |                    E_INVALID_FIELD     - field is invalid
3621 |                    some other          - error from subordinate call
3622 +--------------------------------------------------------------------------*/
3623 static int
3624 Inter_Field_Navigation(int (*const fct) (FORM *), FORM *form)
3625 {
3626   int res;
3627 
3628   if (!_nc_Internal_Validation(form))
3629     res = E_INVALID_FIELD;
3630   else
3631     {
3632       Call_Hook(form, fieldterm);
3633       res = fct(form);
3634       Call_Hook(form, fieldinit);
3635     }
3636   return res;
3637 }
3638 
3639 /*---------------------------------------------------------------------------
3640 |   Facility      :  libnform
3641 |   Function      :  static int FN_Next_Field(FORM * form)
3642 |
3643 |   Description   :  Move to the next field on the current page of the form
3644 |
3645 |   Return Values :  E_OK                 - success
3646 |                    != E_OK              - error from subordinate call
3647 +--------------------------------------------------------------------------*/
3648 static int
3649 FN_Next_Field(FORM *form)
3650 {
3651   T((T_CALLED("FN_Next_Field(%p)"), (void *)form));
3652   returnCode(_nc_Set_Current_Field(form,
3653 				   Next_Field_On_Page(form->current)));
3654 }
3655 
3656 /*---------------------------------------------------------------------------
3657 |   Facility      :  libnform
3658 |   Function      :  static int FN_Previous_Field(FORM * form)
3659 |
3660 |   Description   :  Move to the previous field on the current page of the
3661 |                    form
3662 |
3663 |   Return Values :  E_OK                 - success
3664 |                    != E_OK              - error from subordinate call
3665 +--------------------------------------------------------------------------*/
3666 static int
3667 FN_Previous_Field(FORM *form)
3668 {
3669   T((T_CALLED("FN_Previous_Field(%p)"), (void *)form));
3670   returnCode(_nc_Set_Current_Field(form,
3671 				   Previous_Field_On_Page(form->current)));
3672 }
3673 
3674 /*---------------------------------------------------------------------------
3675 |   Facility      :  libnform
3676 |   Function      :  static int FN_First_Field(FORM * form)
3677 |
3678 |   Description   :  Move to the first field on the current page of the form
3679 |
3680 |   Return Values :  E_OK                 - success
3681 |                    != E_OK              - error from subordinate call
3682 +--------------------------------------------------------------------------*/
3683 static int
3684 FN_First_Field(FORM *form)
3685 {
3686   T((T_CALLED("FN_First_Field(%p)"), (void *)form));
3687   returnCode(_nc_Set_Current_Field(form,
3688 				   Next_Field_On_Page(form->field[form->page[form->curpage].pmax])));
3689 }
3690 
3691 /*---------------------------------------------------------------------------
3692 |   Facility      :  libnform
3693 |   Function      :  static int FN_Last_Field(FORM * form)
3694 |
3695 |   Description   :  Move to the last field on the current page of the form
3696 |
3697 |   Return Values :  E_OK                 - success
3698 |                    != E_OK              - error from subordinate call
3699 +--------------------------------------------------------------------------*/
3700 static int
3701 FN_Last_Field(FORM *form)
3702 {
3703   T((T_CALLED("FN_Last_Field(%p)"), (void *)form));
3704   returnCode(
3705 	      _nc_Set_Current_Field(form,
3706 				    Previous_Field_On_Page(form->field[form->page[form->curpage].pmin])));
3707 }
3708 
3709 /*---------------------------------------------------------------------------
3710 |   Facility      :  libnform
3711 |   Function      :  static int FN_Sorted_Next_Field(FORM * form)
3712 |
3713 |   Description   :  Move to the sorted next field on the current page
3714 |                    of the form.
3715 |
3716 |   Return Values :  E_OK            - success
3717 |                    != E_OK         - error from subordinate call
3718 +--------------------------------------------------------------------------*/
3719 static int
3720 FN_Sorted_Next_Field(FORM *form)
3721 {
3722   T((T_CALLED("FN_Sorted_Next_Field(%p)"), (void *)form));
3723   returnCode(_nc_Set_Current_Field(form,
3724 				   Sorted_Next_Field(form->current)));
3725 }
3726 
3727 /*---------------------------------------------------------------------------
3728 |   Facility      :  libnform
3729 |   Function      :  static int FN_Sorted_Previous_Field(FORM * form)
3730 |
3731 |   Description   :  Move to the sorted previous field on the current page
3732 |                    of the form.
3733 |
3734 |   Return Values :  E_OK            - success
3735 |                    != E_OK         - error from subordinate call
3736 +--------------------------------------------------------------------------*/
3737 static int
3738 FN_Sorted_Previous_Field(FORM *form)
3739 {
3740   T((T_CALLED("FN_Sorted_Previous_Field(%p)"), (void *)form));
3741   returnCode(_nc_Set_Current_Field(form,
3742 				   Sorted_Previous_Field(form->current)));
3743 }
3744 
3745 /*---------------------------------------------------------------------------
3746 |   Facility      :  libnform
3747 |   Function      :  static int FN_Sorted_First_Field(FORM * form)
3748 |
3749 |   Description   :  Move to the sorted first field on the current page
3750 |                    of the form.
3751 |
3752 |   Return Values :  E_OK            - success
3753 |                    != E_OK         - error from subordinate call
3754 +--------------------------------------------------------------------------*/
3755 static int
3756 FN_Sorted_First_Field(FORM *form)
3757 {
3758   T((T_CALLED("FN_Sorted_First_Field(%p)"), (void *)form));
3759   returnCode(_nc_Set_Current_Field(form,
3760 				   Sorted_Next_Field(form->field[form->page[form->curpage].smax])));
3761 }
3762 
3763 /*---------------------------------------------------------------------------
3764 |   Facility      :  libnform
3765 |   Function      :  static int FN_Sorted_Last_Field(FORM * form)
3766 |
3767 |   Description   :  Move to the sorted last field on the current page
3768 |                    of the form.
3769 |
3770 |   Return Values :  E_OK            - success
3771 |                    != E_OK         - error from subordinate call
3772 +--------------------------------------------------------------------------*/
3773 static int
3774 FN_Sorted_Last_Field(FORM *form)
3775 {
3776   T((T_CALLED("FN_Sorted_Last_Field(%p)"), (void *)form));
3777   returnCode(_nc_Set_Current_Field(form,
3778 				   Sorted_Previous_Field(form->field[form->page[form->curpage].smin])));
3779 }
3780 
3781 /*---------------------------------------------------------------------------
3782 |   Facility      :  libnform
3783 |   Function      :  static int FN_Left_Field(FORM * form)
3784 |
3785 |   Description   :  Get the field on the left of the current field on the
3786 |                    same line and the same page. Cycles through the line.
3787 |
3788 |   Return Values :  E_OK            - success
3789 |                    != E_OK         - error from subordinate call
3790 +--------------------------------------------------------------------------*/
3791 static int
3792 FN_Left_Field(FORM *form)
3793 {
3794   T((T_CALLED("FN_Left_Field(%p)"), (void *)form));
3795   returnCode(_nc_Set_Current_Field(form,
3796 				   Left_Neighbor_Field(form->current)));
3797 }
3798 
3799 /*---------------------------------------------------------------------------
3800 |   Facility      :  libnform
3801 |   Function      :  static int FN_Right_Field(FORM * form)
3802 |
3803 |   Description   :  Get the field on the right of the current field on the
3804 |                    same line and the same page. Cycles through the line.
3805 |
3806 |   Return Values :  E_OK            - success
3807 |                    != E_OK         - error from subordinate call
3808 +--------------------------------------------------------------------------*/
3809 static int
3810 FN_Right_Field(FORM *form)
3811 {
3812   T((T_CALLED("FN_Right_Field(%p)"), (void *)form));
3813   returnCode(_nc_Set_Current_Field(form,
3814 				   Right_Neighbor_Field(form->current)));
3815 }
3816 
3817 /*---------------------------------------------------------------------------
3818 |   Facility      :  libnform
3819 |   Function      :  static int FN_Up_Field(FORM * form)
3820 |
3821 |   Description   :  Get the upper neighbor of the current field. This
3822 |                    cycles through the page. See the comments of the
3823 |                    Upper_Neighbor_Field function to understand how
3824 |                    'upper' is defined.
3825 |
3826 |   Return Values :  E_OK            - success
3827 |                    != E_OK         - error from subordinate call
3828 +--------------------------------------------------------------------------*/
3829 static int
3830 FN_Up_Field(FORM *form)
3831 {
3832   T((T_CALLED("FN_Up_Field(%p)"), (void *)form));
3833   returnCode(_nc_Set_Current_Field(form,
3834 				   Upper_Neighbor_Field(form->current)));
3835 }
3836 
3837 /*---------------------------------------------------------------------------
3838 |   Facility      :  libnform
3839 |   Function      :  static int FN_Down_Field(FORM * form)
3840 |
3841 |   Description   :  Get the down neighbor of the current field. This
3842 |                    cycles through the page. See the comments of the
3843 |                    Down_Neighbor_Field function to understand how
3844 |                    'down' is defined.
3845 |
3846 |   Return Values :  E_OK            - success
3847 |                    != E_OK         - error from subordinate call
3848 +--------------------------------------------------------------------------*/
3849 static int
3850 FN_Down_Field(FORM *form)
3851 {
3852   T((T_CALLED("FN_Down_Field(%p)"), (void *)form));
3853   returnCode(_nc_Set_Current_Field(form,
3854 				   Down_Neighbor_Field(form->current)));
3855 }
3856 /*----------------------------------------------------------------------------
3857   END of Field Navigation routines
3858   --------------------------------------------------------------------------*/
3859 
3860 /*----------------------------------------------------------------------------
3861   Helper routines for Page Navigation
3862   --------------------------------------------------------------------------*/
3863 
3864 /*---------------------------------------------------------------------------
3865 |   Facility      :  libnform
3866 |   Function      :  int _nc_Set_Form_Page(FORM * form,
3867 |                                          int page,
3868 |                                          FIELD * field)
3869 |
3870 |   Description   :  Make the given page number the current page and make
3871 |                    the given field the current field on the page. If
3872 |                    for the field NULL is given, make the first field on
3873 |                    the page the current field. The routine acts only
3874 |                    if the requested page is not the current page.
3875 |
3876 |   Return Values :  E_OK                - success
3877 |                    != E_OK             - error from subordinate call
3878 |                    E_BAD_ARGUMENT      - invalid field pointer
3879 |                    E_SYSTEM_ERROR      - some severe basic error
3880 +--------------------------------------------------------------------------*/
3881 FORM_EXPORT(int)
3882 _nc_Set_Form_Page(FORM *form, int page, FIELD *field)
3883 {
3884   int res = E_OK;
3885 
3886   if ((form->curpage != page))
3887     {
3888       FIELD *last_field, *field_on_page;
3889 
3890       werase(Get_Form_Window(form));
3891       form->curpage = (short)page;
3892       last_field = field_on_page = form->field[form->page[page].smin];
3893       do
3894 	{
3895 	  if ((unsigned)field_on_page->opts & O_VISIBLE)
3896 	    if ((res = Display_Field(field_on_page)) != E_OK)
3897 	      return (res);
3898 	  field_on_page = field_on_page->snext;
3899 	}
3900       while (field_on_page != last_field);
3901 
3902       if (field)
3903 	res = _nc_Set_Current_Field(form, field);
3904       else
3905 	/* N.B.: we don't encapsulate this by Inter_Field_Navigation(),
3906 	   because this is already executed in a page navigation
3907 	   context that contains field navigation
3908 	 */
3909 	res = FN_First_Field(form);
3910     }
3911   return (res);
3912 }
3913 
3914 /*---------------------------------------------------------------------------
3915 |   Facility      :  libnform
3916 |   Function      :  static int Next_Page_Number(const FORM * form)
3917 |
3918 |   Description   :  Calculate the page number following the current page
3919 |                    number. This cycles if the highest page number is
3920 |                    reached.
3921 |
3922 |   Return Values :  The next page number
3923 +--------------------------------------------------------------------------*/
3924 NCURSES_INLINE static int
3925 Next_Page_Number(const FORM *form)
3926 {
3927   return (form->curpage + 1) % form->maxpage;
3928 }
3929 
3930 /*---------------------------------------------------------------------------
3931 |   Facility      :  libnform
3932 |   Function      :  static int Previous_Page_Number(const FORM * form)
3933 |
3934 |   Description   :  Calculate the page number before the current page
3935 |                    number. This cycles if the first page number is
3936 |                    reached.
3937 |
3938 |   Return Values :  The previous page number
3939 +--------------------------------------------------------------------------*/
3940 NCURSES_INLINE static int
3941 Previous_Page_Number(const FORM *form)
3942 {
3943   return (form->curpage != 0 ? form->curpage - 1 : form->maxpage - 1);
3944 }
3945 
3946 /*----------------------------------------------------------------------------
3947   Page Navigation routines
3948   --------------------------------------------------------------------------*/
3949 
3950 /*---------------------------------------------------------------------------
3951 |   Facility      :  libnform
3952 |   Function      :  static int Page_Navigation(
3953 |                                               int (* const fct) (FORM *),
3954 |                                               FORM * form)
3955 |
3956 |   Description   :  Generic behavior for changing a page. This means
3957 |                    that the field is left and a new field is entered.
3958 |                    So the field must be validated and the field init/term
3959 |                    hooks must be called. Because also the page is changed,
3960 |                    the form's init/term hooks must be called also.
3961 |
3962 |   Return Values :  E_OK                - success
3963 |                    E_INVALID_FIELD     - field is invalid
3964 |                    some other          - error from subordinate call
3965 +--------------------------------------------------------------------------*/
3966 static int
3967 Page_Navigation(int (*const fct) (FORM *), FORM *form)
3968 {
3969   int res;
3970 
3971   if (!_nc_Internal_Validation(form))
3972     res = E_INVALID_FIELD;
3973   else
3974     {
3975       Call_Hook(form, fieldterm);
3976       Call_Hook(form, formterm);
3977       res = fct(form);
3978       Call_Hook(form, forminit);
3979       Call_Hook(form, fieldinit);
3980     }
3981   return res;
3982 }
3983 
3984 /*---------------------------------------------------------------------------
3985 |   Facility      :  libnform
3986 |   Function      :  static int PN_Next_Page(FORM * form)
3987 |
3988 |   Description   :  Move to the next page of the form
3989 |
3990 |   Return Values :  E_OK                - success
3991 |                    != E_OK             - error from subordinate call
3992 +--------------------------------------------------------------------------*/
3993 static int
3994 PN_Next_Page(FORM *form)
3995 {
3996   T((T_CALLED("PN_Next_Page(%p)"), (void *)form));
3997   returnCode(_nc_Set_Form_Page(form, Next_Page_Number(form), (FIELD *)0));
3998 }
3999 
4000 /*---------------------------------------------------------------------------
4001 |   Facility      :  libnform
4002 |   Function      :  static int PN_Previous_Page(FORM * form)
4003 |
4004 |   Description   :  Move to the previous page of the form
4005 |
4006 |   Return Values :  E_OK              - success
4007 |                    != E_OK           - error from subordinate call
4008 +--------------------------------------------------------------------------*/
4009 static int
4010 PN_Previous_Page(FORM *form)
4011 {
4012   T((T_CALLED("PN_Previous_Page(%p)"), (void *)form));
4013   returnCode(_nc_Set_Form_Page(form, Previous_Page_Number(form), (FIELD *)0));
4014 }
4015 
4016 /*---------------------------------------------------------------------------
4017 |   Facility      :  libnform
4018 |   Function      :  static int PN_First_Page(FORM * form)
4019 |
4020 |   Description   :  Move to the first page of the form
4021 |
4022 |   Return Values :  E_OK              - success
4023 |                    != E_OK           - error from subordinate call
4024 +--------------------------------------------------------------------------*/
4025 static int
4026 PN_First_Page(FORM *form)
4027 {
4028   T((T_CALLED("PN_First_Page(%p)"), (void *)form));
4029   returnCode(_nc_Set_Form_Page(form, 0, (FIELD *)0));
4030 }
4031 
4032 /*---------------------------------------------------------------------------
4033 |   Facility      :  libnform
4034 |   Function      :  static int PN_Last_Page(FORM * form)
4035 |
4036 |   Description   :  Move to the last page of the form
4037 |
4038 |   Return Values :  E_OK              - success
4039 |                    != E_OK           - error from subordinate call
4040 +--------------------------------------------------------------------------*/
4041 static int
4042 PN_Last_Page(FORM *form)
4043 {
4044   T((T_CALLED("PN_Last_Page(%p)"), (void *)form));
4045   returnCode(_nc_Set_Form_Page(form, form->maxpage - 1, (FIELD *)0));
4046 }
4047 
4048 /*----------------------------------------------------------------------------
4049   END of Field Navigation routines
4050   --------------------------------------------------------------------------*/
4051 
4052 /*----------------------------------------------------------------------------
4053   Helper routines for the core form driver.
4054   --------------------------------------------------------------------------*/
4055 
4056 # if USE_WIDEC_SUPPORT
4057 /*---------------------------------------------------------------------------
4058 |   Facility      :  libnform
4059 |   Function      :  static int Data_Entry_w(FORM * form, wchar_t c)
4060 |
4061 |   Description   :  Enter the wide character c into at the current
4062 |                    position of the current field of the form.
4063 |
4064 |   Return Values :  E_OK              - success
4065 |                    E_REQUEST_DENIED  - driver could not process the request
4066 |                    E_SYSTEM_ERROR    -
4067 +--------------------------------------------------------------------------*/
4068 static int
4069 Data_Entry_w(FORM *form, wchar_t c)
4070 {
4071   FIELD *field = form->current;
4072   int result = E_REQUEST_DENIED;
4073 
4074   T((T_CALLED("Data_Entry(%p,%s)"), (void *)form, _tracechtype((chtype)c)));
4075   if ((Field_Has_Option(field, O_EDIT))
4076 #if FIX_FORM_INACTIVE_BUG
4077       && (Field_Has_Option(field, O_ACTIVE))
4078 #endif
4079     )
4080     {
4081       wchar_t given[2];
4082       cchar_t temp_ch;
4083 
4084       given[0] = c;
4085       given[1] = 0;
4086       setcchar(&temp_ch, given, 0, 0, (void *)0);
4087       if ((Field_Has_Option(field, O_BLANK)) &&
4088 	  First_Position_In_Current_Field(form) &&
4089 	  !(form->status & _FCHECK_REQUIRED) &&
4090 	  !(form->status & _WINDOW_MODIFIED))
4091 	werase(form->w);
4092 
4093       if (form->status & _OVLMODE)
4094 	{
4095 	  wadd_wch(form->w, &temp_ch);
4096 	}
4097       else
4098 	/* no _OVLMODE */
4099 	{
4100 	  bool There_Is_Room = Is_There_Room_For_A_Char_In_Line(form);
4101 
4102 	  if (!(There_Is_Room ||
4103 		((Single_Line_Field(field) && Growable(field)))))
4104 	    RETURN(E_REQUEST_DENIED);
4105 
4106 	  if (!There_Is_Room && !Field_Grown(field, 1))
4107 	    RETURN(E_SYSTEM_ERROR);
4108 
4109 	  wins_wch(form->w, &temp_ch);
4110 	}
4111 
4112       if ((result = Wrapping_Not_Necessary_Or_Wrapping_Ok(form)) == E_OK)
4113 	{
4114 	  bool End_Of_Field = (((field->drows - 1) == form->currow) &&
4115 			       ((field->dcols - 1) == form->curcol));
4116 
4117 	  form->status |= _WINDOW_MODIFIED;
4118 	  if (End_Of_Field && !Growable(field) && (Field_Has_Option(field, O_AUTOSKIP)))
4119 	    result = Inter_Field_Navigation(FN_Next_Field, form);
4120 	  else
4121 	    {
4122 	      if (End_Of_Field && Growable(field) && !Field_Grown(field, 1))
4123 		result = E_SYSTEM_ERROR;
4124 	      else
4125 		{
4126 		  /*
4127 		   * We have just added a byte to the form field.  It may have
4128 		   * been part of a multibyte character.  If it was, the
4129 		   * addch_used field is nonzero and we should not try to move
4130 		   * to a new column.
4131 		   */
4132 		  if (WINDOW_EXT(form->w, addch_used) == 0)
4133 		    IFN_Next_Character(form);
4134 
4135 		  result = E_OK;
4136 		}
4137 	    }
4138 	}
4139     }
4140   RETURN(result);
4141 }
4142 # endif
4143 
4144 /*---------------------------------------------------------------------------
4145 |   Facility      :  libnform
4146 |   Function      :  static int Data_Entry(FORM * form,int c)
4147 |
4148 |   Description   :  Enter character c into at the current position of the
4149 |                    current field of the form.
4150 |
4151 |   Return Values :  E_OK              - success
4152 |                    E_REQUEST_DENIED  - driver could not process the request
4153 |                    E_SYSTEM_ERROR    -
4154 +--------------------------------------------------------------------------*/
4155 static int
4156 Data_Entry(FORM *form, int c)
4157 {
4158   FIELD *field = form->current;
4159   int result = E_REQUEST_DENIED;
4160 
4161   T((T_CALLED("Data_Entry(%p,%s)"), (void *)form, _tracechtype((chtype)c)));
4162   if ((Field_Has_Option(field, O_EDIT))
4163 #if FIX_FORM_INACTIVE_BUG
4164       && (Field_Has_Option(field, O_ACTIVE))
4165 #endif
4166     )
4167     {
4168       if ((Field_Has_Option(field, O_BLANK)) &&
4169 	  First_Position_In_Current_Field(form) &&
4170 	  !(form->status & _FCHECK_REQUIRED) &&
4171 	  !(form->status & _WINDOW_MODIFIED))
4172 	werase(form->w);
4173 
4174       if (form->status & _OVLMODE)
4175 	{
4176 	  waddch(form->w, (chtype)c);
4177 	}
4178       else
4179 	/* no _OVLMODE */
4180 	{
4181 	  bool There_Is_Room = Is_There_Room_For_A_Char_In_Line(form);
4182 
4183 	  if (!(There_Is_Room ||
4184 		((Single_Line_Field(field) && Growable(field)))))
4185 	    RETURN(E_REQUEST_DENIED);
4186 
4187 	  if (!There_Is_Room && !Field_Grown(field, 1))
4188 	    RETURN(E_SYSTEM_ERROR);
4189 
4190 	  winsch(form->w, (chtype)c);
4191 	}
4192 
4193       if ((result = Wrapping_Not_Necessary_Or_Wrapping_Ok(form)) == E_OK)
4194 	{
4195 	  bool End_Of_Field = (((field->drows - 1) == form->currow) &&
4196 			       ((field->dcols - 1) == form->curcol));
4197 
4198 	  if (Field_Has_Option(field, O_EDGE_INSERT_STAY))
4199 	    move_after_insert = !!(form->curcol
4200 				   - form->begincol
4201 				   - field->cols
4202 				   + 1);
4203 
4204 	  SetStatus(form, _WINDOW_MODIFIED);
4205 	  if (End_Of_Field && !Growable(field) && (Field_Has_Option(field, O_AUTOSKIP)))
4206 	    result = Inter_Field_Navigation(FN_Next_Field, form);
4207 	  else
4208 	    {
4209 	      if (End_Of_Field && Growable(field) && !Field_Grown(field, 1))
4210 		result = E_SYSTEM_ERROR;
4211 	      else
4212 		{
4213 #if USE_WIDEC_SUPPORT
4214 		  /*
4215 		   * We have just added a byte to the form field.  It may have
4216 		   * been part of a multibyte character.  If it was, the
4217 		   * addch_used field is nonzero and we should not try to move
4218 		   * to a new column.
4219 		   */
4220 		  if (WINDOW_EXT(form->w, addch_used) == 0)
4221 		    IFN_Next_Character(form);
4222 #else
4223 		  IFN_Next_Character(form);
4224 #endif
4225 		  result = E_OK;
4226 		}
4227 	    }
4228 	}
4229     }
4230   RETURN(result);
4231 }
4232 
4233 /* Structure to describe the binding of a request code to a function.
4234    The member keycode codes the request value as well as the generic
4235    routine to use for the request. The code for the generic routine
4236    is coded in the upper 16 Bits while the request code is coded in
4237    the lower 16 bits.
4238 
4239    In terms of C++ you might think of a request as a class with a
4240    virtual method "perform". The different types of request are
4241    derived from this base class and overload (or not) the base class
4242    implementation of perform.
4243 */
4244 typedef struct
4245 {
4246   int keycode;			/* must be at least 32 bit: hi:mode, lo: key */
4247   int (*cmd) (FORM *);		/* low level driver routine for this key     */
4248 }
4249 Binding_Info;
4250 
4251 /* You may see this is the class-id of the request type class */
4252 #define ID_PN    (0x00000000)	/* Page navigation           */
4253 #define ID_FN    (0x00010000)	/* Inter-Field navigation    */
4254 #define ID_IFN   (0x00020000)	/* Intra-Field navigation    */
4255 #define ID_VSC   (0x00030000)	/* Vertical Scrolling        */
4256 #define ID_HSC   (0x00040000)	/* Horizontal Scrolling      */
4257 #define ID_FE    (0x00050000)	/* Field Editing             */
4258 #define ID_EM    (0x00060000)	/* Edit Mode                 */
4259 #define ID_FV    (0x00070000)	/* Field Validation          */
4260 #define ID_CH    (0x00080000)	/* Choice                    */
4261 #define ID_Mask  (0xffff0000)
4262 #define Key_Mask (0x0000ffff)
4263 #define ID_Shft  (16)
4264 
4265 /* This array holds all the Binding Infos */
4266 /* *INDENT-OFF* */
4267 static const Binding_Info bindings[MAX_FORM_COMMAND - MIN_FORM_COMMAND + 1] =
4268 {
4269   { REQ_NEXT_PAGE    |ID_PN  ,PN_Next_Page},
4270   { REQ_PREV_PAGE    |ID_PN  ,PN_Previous_Page},
4271   { REQ_FIRST_PAGE   |ID_PN  ,PN_First_Page},
4272   { REQ_LAST_PAGE    |ID_PN  ,PN_Last_Page},
4273 
4274   { REQ_NEXT_FIELD   |ID_FN  ,FN_Next_Field},
4275   { REQ_PREV_FIELD   |ID_FN  ,FN_Previous_Field},
4276   { REQ_FIRST_FIELD  |ID_FN  ,FN_First_Field},
4277   { REQ_LAST_FIELD   |ID_FN  ,FN_Last_Field},
4278   { REQ_SNEXT_FIELD  |ID_FN  ,FN_Sorted_Next_Field},
4279   { REQ_SPREV_FIELD  |ID_FN  ,FN_Sorted_Previous_Field},
4280   { REQ_SFIRST_FIELD |ID_FN  ,FN_Sorted_First_Field},
4281   { REQ_SLAST_FIELD  |ID_FN  ,FN_Sorted_Last_Field},
4282   { REQ_LEFT_FIELD   |ID_FN  ,FN_Left_Field},
4283   { REQ_RIGHT_FIELD  |ID_FN  ,FN_Right_Field},
4284   { REQ_UP_FIELD     |ID_FN  ,FN_Up_Field},
4285   { REQ_DOWN_FIELD   |ID_FN  ,FN_Down_Field},
4286 
4287   { REQ_NEXT_CHAR    |ID_IFN ,IFN_Next_Character},
4288   { REQ_PREV_CHAR    |ID_IFN ,IFN_Previous_Character},
4289   { REQ_NEXT_LINE    |ID_IFN ,IFN_Next_Line},
4290   { REQ_PREV_LINE    |ID_IFN ,IFN_Previous_Line},
4291   { REQ_NEXT_WORD    |ID_IFN ,IFN_Next_Word},
4292   { REQ_PREV_WORD    |ID_IFN ,IFN_Previous_Word},
4293   { REQ_BEG_FIELD    |ID_IFN ,IFN_Beginning_Of_Field},
4294   { REQ_END_FIELD    |ID_IFN ,IFN_End_Of_Field},
4295   { REQ_BEG_LINE     |ID_IFN ,IFN_Beginning_Of_Line},
4296   { REQ_END_LINE     |ID_IFN ,IFN_End_Of_Line},
4297   { REQ_LEFT_CHAR    |ID_IFN ,IFN_Left_Character},
4298   { REQ_RIGHT_CHAR   |ID_IFN ,IFN_Right_Character},
4299   { REQ_UP_CHAR      |ID_IFN ,IFN_Up_Character},
4300   { REQ_DOWN_CHAR    |ID_IFN ,IFN_Down_Character},
4301 
4302   { REQ_NEW_LINE     |ID_FE  ,FE_New_Line},
4303   { REQ_INS_CHAR     |ID_FE  ,FE_Insert_Character},
4304   { REQ_INS_LINE     |ID_FE  ,FE_Insert_Line},
4305   { REQ_DEL_CHAR     |ID_FE  ,FE_Delete_Character},
4306   { REQ_DEL_PREV     |ID_FE  ,FE_Delete_Previous},
4307   { REQ_DEL_LINE     |ID_FE  ,FE_Delete_Line},
4308   { REQ_DEL_WORD     |ID_FE  ,FE_Delete_Word},
4309   { REQ_CLR_EOL      |ID_FE  ,FE_Clear_To_End_Of_Line},
4310   { REQ_CLR_EOF      |ID_FE  ,FE_Clear_To_End_Of_Field},
4311   { REQ_CLR_FIELD    |ID_FE  ,FE_Clear_Field},
4312 
4313   { REQ_OVL_MODE     |ID_EM  ,EM_Overlay_Mode},
4314   { REQ_INS_MODE     |ID_EM  ,EM_Insert_Mode},
4315 
4316   { REQ_SCR_FLINE    |ID_VSC ,VSC_Scroll_Line_Forward},
4317   { REQ_SCR_BLINE    |ID_VSC ,VSC_Scroll_Line_Backward},
4318   { REQ_SCR_FPAGE    |ID_VSC ,VSC_Scroll_Page_Forward},
4319   { REQ_SCR_BPAGE    |ID_VSC ,VSC_Scroll_Page_Backward},
4320   { REQ_SCR_FHPAGE   |ID_VSC ,VSC_Scroll_Half_Page_Forward},
4321   { REQ_SCR_BHPAGE   |ID_VSC ,VSC_Scroll_Half_Page_Backward},
4322 
4323   { REQ_SCR_FCHAR    |ID_HSC ,HSC_Scroll_Char_Forward},
4324   { REQ_SCR_BCHAR    |ID_HSC ,HSC_Scroll_Char_Backward},
4325   { REQ_SCR_HFLINE   |ID_HSC ,HSC_Horizontal_Line_Forward},
4326   { REQ_SCR_HBLINE   |ID_HSC ,HSC_Horizontal_Line_Backward},
4327   { REQ_SCR_HFHALF   |ID_HSC ,HSC_Horizontal_Half_Line_Forward},
4328   { REQ_SCR_HBHALF   |ID_HSC ,HSC_Horizontal_Half_Line_Backward},
4329 
4330   { REQ_VALIDATION   |ID_FV  ,FV_Validation},
4331 
4332   { REQ_NEXT_CHOICE  |ID_CH  ,CR_Next_Choice},
4333   { REQ_PREV_CHOICE  |ID_CH  ,CR_Previous_Choice}
4334 };
4335 /* *INDENT-ON* */
4336 
4337 /*---------------------------------------------------------------------------
4338 |   Facility      :  libnform
4339 |   Function      :  int form_driver(FORM * form,int  c)
4340 |
4341 |   Description   :  This is the workhorse of the forms system. It checks
4342 |                    to determine whether the character c is a request or
4343 |                    data. If it is a request, the form driver executes
4344 |                    the request and returns the result. If it is data
4345 |                    (printable character), it enters the data into the
4346 |                    current position in the current field. If it is not
4347 |                    recognized, the form driver assumes it is an application
4348 |                    defined command and returns E_UNKNOWN_COMMAND.
4349 |                    Application defined command should be defined relative
4350 |                    to MAX_FORM_COMMAND, the maximum value of a request.
4351 |
4352 |   Return Values :  E_OK              - success
4353 |                    E_SYSTEM_ERROR    - system error
4354 |                    E_BAD_ARGUMENT    - an argument is incorrect
4355 |                    E_NOT_POSTED      - form is not posted
4356 |                    E_INVALID_FIELD   - field contents are invalid
4357 |                    E_BAD_STATE       - called from inside a hook routine
4358 |                    E_REQUEST_DENIED  - request failed
4359 |                    E_NOT_CONNECTED   - no fields are connected to the form
4360 |                    E_UNKNOWN_COMMAND - command not known
4361 +--------------------------------------------------------------------------*/
4362 FORM_EXPORT(int)
4363 form_driver(FORM *form, int c)
4364 {
4365   const Binding_Info *BI = (Binding_Info *)0;
4366   int res = E_UNKNOWN_COMMAND;
4367 
4368   move_after_insert = TRUE;
4369 
4370   T((T_CALLED("form_driver(%p,%d)"), (void *)form, c));
4371 
4372   if (!form)
4373     RETURN(E_BAD_ARGUMENT);
4374 
4375   if (!(form->field) || !(form->current))
4376     RETURN(E_NOT_CONNECTED);
4377 
4378   assert(form->page);
4379 
4380   if (c == FIRST_ACTIVE_MAGIC)
4381     {
4382       form->current = _nc_First_Active_Field(form);
4383       RETURN(E_OK);
4384     }
4385 
4386   assert(form->current &&
4387 	 form->current->buf &&
4388 	 (form->current->form == form)
4389     );
4390 
4391   if (form->status & _IN_DRIVER)
4392     RETURN(E_BAD_STATE);
4393 
4394   if (!(form->status & _POSTED))
4395     RETURN(E_NOT_POSTED);
4396 
4397   if ((c >= MIN_FORM_COMMAND && c <= MAX_FORM_COMMAND) &&
4398       ((bindings[c - MIN_FORM_COMMAND].keycode & Key_Mask) == c))
4399     {
4400       TR(TRACE_CALLS, ("form_request %s", form_request_name(c)));
4401       BI = &(bindings[c - MIN_FORM_COMMAND]);
4402     }
4403 
4404   if (BI)
4405     {
4406       typedef int (*Generic_Method) (int (*const) (FORM *), FORM *);
4407       static const Generic_Method Generic_Methods[] =
4408       {
4409 	Page_Navigation,	/* overloaded to call field&form hooks */
4410 	Inter_Field_Navigation,	/* overloaded to call field hooks      */
4411 	NULL,			/* Intra-Field is generic              */
4412 	Vertical_Scrolling,	/* Overloaded to check multi-line      */
4413 	Horizontal_Scrolling,	/* Overloaded to check single-line     */
4414 	Field_Editing,		/* Overloaded to mark modification     */
4415 	NULL,			/* Edit Mode is generic                */
4416 	NULL,			/* Field Validation is generic         */
4417 	NULL			/* Choice Request is generic           */
4418       };
4419       size_t nMethods = (sizeof(Generic_Methods) / sizeof(Generic_Methods[0]));
4420       size_t method = (size_t)((BI->keycode >> ID_Shft) & 0xffff);	/* see ID_Mask */
4421 
4422       if ((method >= nMethods) || !(BI->cmd))
4423 	res = E_SYSTEM_ERROR;
4424       else
4425 	{
4426 	  Generic_Method fct = Generic_Methods[method];
4427 
4428 	  if (fct)
4429 	    {
4430 	      res = fct(BI->cmd, form);
4431 	    }
4432 	  else
4433 	    {
4434 	      res = (BI->cmd) (form);
4435 	    }
4436 	}
4437     }
4438 #ifdef NCURSES_MOUSE_VERSION
4439   else if (KEY_MOUSE == c)
4440     {
4441       MEVENT event;
4442       WINDOW *win = form->win ? form->win : StdScreen(Get_Form_Screen(form));
4443       WINDOW *sub = form->sub ? form->sub : win;
4444 
4445       getmouse(&event);
4446       if ((event.bstate & (BUTTON1_CLICKED |
4447 			   BUTTON1_DOUBLE_CLICKED |
4448 			   BUTTON1_TRIPLE_CLICKED))
4449 	  && wenclose(win, event.y, event.x))
4450 	{			/* we react only if the click was in the userwin, that means
4451 				 * inside the form display area or at the decoration window.
4452 				 */
4453 	  int ry = event.y, rx = event.x;	/* screen coordinates */
4454 
4455 	  res = E_REQUEST_DENIED;
4456 	  if (mouse_trafo(&ry, &rx, FALSE))
4457 	    {			/* rx, ry are now "curses" coordinates */
4458 	      if (ry < sub->_begy)
4459 		{		/* we clicked above the display region; this is
4460 				 * interpreted as "scroll up" request
4461 				 */
4462 		  if (event.bstate & BUTTON1_CLICKED)
4463 		    res = form_driver(form, REQ_PREV_FIELD);
4464 		  else if (event.bstate & BUTTON1_DOUBLE_CLICKED)
4465 		    res = form_driver(form, REQ_PREV_PAGE);
4466 		  else if (event.bstate & BUTTON1_TRIPLE_CLICKED)
4467 		    res = form_driver(form, REQ_FIRST_FIELD);
4468 		}
4469 	      else if (ry > sub->_begy + sub->_maxy)
4470 		{		/* we clicked below the display region; this is
4471 				 * interpreted as "scroll down" request
4472 				 */
4473 		  if (event.bstate & BUTTON1_CLICKED)
4474 		    res = form_driver(form, REQ_NEXT_FIELD);
4475 		  else if (event.bstate & BUTTON1_DOUBLE_CLICKED)
4476 		    res = form_driver(form, REQ_NEXT_PAGE);
4477 		  else if (event.bstate & BUTTON1_TRIPLE_CLICKED)
4478 		    res = form_driver(form, REQ_LAST_FIELD);
4479 		}
4480 	      else if (wenclose(sub, event.y, event.x))
4481 		{		/* Inside the area we try to find the hit item */
4482 		  ry = event.y;
4483 		  rx = event.x;
4484 		  if (wmouse_trafo(sub, &ry, &rx, FALSE))
4485 		    {
4486 		      int min_field = form->page[form->curpage].pmin;
4487 		      int max_field = form->page[form->curpage].pmax;
4488 		      int i;
4489 
4490 		      for (i = min_field; i <= max_field; ++i)
4491 			{
4492 			  FIELD *field = form->field[i];
4493 
4494 			  if (Field_Is_Selectable(field)
4495 			      && Field_encloses(field, ry, rx) == E_OK)
4496 			    {
4497 			      res = _nc_Set_Current_Field(form, field);
4498 			      if (res == E_OK)
4499 				res = _nc_Position_Form_Cursor(form);
4500 			      if (res == E_OK
4501 				  && (event.bstate & BUTTON1_DOUBLE_CLICKED))
4502 				res = E_UNKNOWN_COMMAND;
4503 			      break;
4504 			    }
4505 			}
4506 		    }
4507 		}
4508 	    }
4509 	}
4510       else
4511 	res = E_REQUEST_DENIED;
4512     }
4513 #endif /* NCURSES_MOUSE_VERSION */
4514   else if (!(c & (~(int)MAX_REGULAR_CHARACTER)))
4515     {
4516       /*
4517        * If we're using 8-bit characters, iscntrl+isprint cover the whole set.
4518        * But with multibyte characters, there is a third possibility, i.e.,
4519        * parts of characters that build up into printable characters which are
4520        * not considered printable.
4521        *
4522        * FIXME: the wide-character branch should also use Check_Char().
4523        */
4524 #if USE_WIDEC_SUPPORT
4525       if (!iscntrl(UChar(c)))
4526 #else
4527       if (isprint(UChar(c)) &&
4528 	  Check_Char(form, form->current, form->current->type, c,
4529 		     (TypeArgument *)(form->current->arg)))
4530 #endif
4531 	res = Data_Entry(form, c);
4532     }
4533   _nc_Refresh_Current_Field(form);
4534   RETURN(res);
4535 }
4536 
4537 # if USE_WIDEC_SUPPORT
4538 /*---------------------------------------------------------------------------
4539 |   Facility      :  libnform
4540 |   Function      :  int form_driver_w(FORM * form,int type,wchar_t  c)
4541 |
4542 |   Description   :  This is the workhorse of the forms system.
4543 |
4544 |                    Input is either a key code (request) or a wide char
4545 |                    returned by e.g. get_wch (). The type must be passed
4546 |                    as well,so that we are able to determine whether the char
4547 |                    is a multibyte char or a request.
4548 
4549 |                    If it is a request, the form driver executes
4550 |                    the request and returns the result. If it is data
4551 |                    (printable character), it enters the data into the
4552 |                    current position in the current field. If it is not
4553 |                    recognized, the form driver assumes it is an application
4554 |                    defined command and returns E_UNKNOWN_COMMAND.
4555 |                    Application defined command should be defined relative
4556 |                    to MAX_FORM_COMMAND, the maximum value of a request.
4557 |
4558 |   Return Values :  E_OK              - success
4559 |                    E_SYSTEM_ERROR    - system error
4560 |                    E_BAD_ARGUMENT    - an argument is incorrect
4561 |                    E_NOT_POSTED      - form is not posted
4562 |                    E_INVALID_FIELD   - field contents are invalid
4563 |                    E_BAD_STATE       - called from inside a hook routine
4564 |                    E_REQUEST_DENIED  - request failed
4565 |                    E_NOT_CONNECTED   - no fields are connected to the form
4566 |                    E_UNKNOWN_COMMAND - command not known
4567 +--------------------------------------------------------------------------*/
4568 FORM_EXPORT(int)
4569 form_driver_w(FORM *form, int type, wchar_t c)
4570 {
4571   const Binding_Info *BI = (Binding_Info *)0;
4572   int res = E_UNKNOWN_COMMAND;
4573 
4574   T((T_CALLED("form_driver(%p,%d)"), (void *)form, (int)c));
4575 
4576   if (!form)
4577     RETURN(E_BAD_ARGUMENT);
4578 
4579   if (!(form->field))
4580     RETURN(E_NOT_CONNECTED);
4581 
4582   assert(form->page);
4583 
4584   if (c == (wchar_t)FIRST_ACTIVE_MAGIC)
4585     {
4586       form->current = _nc_First_Active_Field(form);
4587       RETURN(E_OK);
4588     }
4589 
4590   assert(form->current &&
4591 	 form->current->buf &&
4592 	 (form->current->form == form)
4593     );
4594 
4595   if (form->status & _IN_DRIVER)
4596     RETURN(E_BAD_STATE);
4597 
4598   if (!(form->status & _POSTED))
4599     RETURN(E_NOT_POSTED);
4600 
4601   /* check if this is a keycode or a (wide) char */
4602   if (type == KEY_CODE_YES)
4603     {
4604       if ((c >= MIN_FORM_COMMAND && c <= MAX_FORM_COMMAND) &&
4605 	  ((bindings[c - MIN_FORM_COMMAND].keycode & Key_Mask) == c))
4606 	BI = &(bindings[c - MIN_FORM_COMMAND]);
4607     }
4608 
4609   if (BI)
4610     {
4611       typedef int (*Generic_Method) (int (*const) (FORM *), FORM *);
4612       static const Generic_Method Generic_Methods[] =
4613       {
4614 	Page_Navigation,	/* overloaded to call field&form hooks */
4615 	Inter_Field_Navigation,	/* overloaded to call field hooks      */
4616 	NULL,			/* Intra-Field is generic              */
4617 	Vertical_Scrolling,	/* Overloaded to check multi-line      */
4618 	Horizontal_Scrolling,	/* Overloaded to check single-line     */
4619 	Field_Editing,		/* Overloaded to mark modification     */
4620 	NULL,			/* Edit Mode is generic                */
4621 	NULL,			/* Field Validation is generic         */
4622 	NULL			/* Choice Request is generic           */
4623       };
4624       size_t nMethods = (sizeof(Generic_Methods) / sizeof(Generic_Methods[0]));
4625       size_t method = (size_t)(BI->keycode >> ID_Shft) & 0xffff;	/* see ID_Mask */
4626 
4627       if ((method >= nMethods) || !(BI->cmd))
4628 	res = E_SYSTEM_ERROR;
4629       else
4630 	{
4631 	  Generic_Method fct = Generic_Methods[method];
4632 
4633 	  if (fct)
4634 	    res = fct(BI->cmd, form);
4635 	  else
4636 	    res = (BI->cmd) (form);
4637 	}
4638     }
4639 #ifdef NCURSES_MOUSE_VERSION
4640   else if (KEY_MOUSE == c)
4641     {
4642       MEVENT event;
4643       WINDOW *win = form->win ? form->win : StdScreen(Get_Form_Screen(form));
4644       WINDOW *sub = form->sub ? form->sub : win;
4645 
4646       getmouse(&event);
4647       if ((event.bstate & (BUTTON1_CLICKED |
4648 			   BUTTON1_DOUBLE_CLICKED |
4649 			   BUTTON1_TRIPLE_CLICKED))
4650 	  && wenclose(win, event.y, event.x))
4651 	{			/* we react only if the click was in the userwin, that means
4652 				   * inside the form display area or at the decoration window.
4653 				 */
4654 	  int ry = event.y, rx = event.x;	/* screen coordinates */
4655 
4656 	  res = E_REQUEST_DENIED;
4657 	  if (mouse_trafo(&ry, &rx, FALSE))
4658 	    {			/* rx, ry are now "curses" coordinates */
4659 	      if (ry < sub->_begy)
4660 		{		/* we clicked above the display region; this is
4661 				   * interpreted as "scroll up" request
4662 				 */
4663 		  if (event.bstate & BUTTON1_CLICKED)
4664 		    res = form_driver(form, REQ_PREV_FIELD);
4665 		  else if (event.bstate & BUTTON1_DOUBLE_CLICKED)
4666 		    res = form_driver(form, REQ_PREV_PAGE);
4667 		  else if (event.bstate & BUTTON1_TRIPLE_CLICKED)
4668 		    res = form_driver(form, REQ_FIRST_FIELD);
4669 		}
4670 	      else if (ry > sub->_begy + sub->_maxy)
4671 		{		/* we clicked below the display region; this is
4672 				   * interpreted as "scroll down" request
4673 				 */
4674 		  if (event.bstate & BUTTON1_CLICKED)
4675 		    res = form_driver(form, REQ_NEXT_FIELD);
4676 		  else if (event.bstate & BUTTON1_DOUBLE_CLICKED)
4677 		    res = form_driver(form, REQ_NEXT_PAGE);
4678 		  else if (event.bstate & BUTTON1_TRIPLE_CLICKED)
4679 		    res = form_driver(form, REQ_LAST_FIELD);
4680 		}
4681 	      else if (wenclose(sub, event.y, event.x))
4682 		{		/* Inside the area we try to find the hit item */
4683 		  ry = event.y;
4684 		  rx = event.x;
4685 		  if (wmouse_trafo(sub, &ry, &rx, FALSE))
4686 		    {
4687 		      int min_field = form->page[form->curpage].pmin;
4688 		      int max_field = form->page[form->curpage].pmax;
4689 		      int i;
4690 
4691 		      for (i = min_field; i <= max_field; ++i)
4692 			{
4693 			  FIELD *field = form->field[i];
4694 
4695 			  if (Field_Is_Selectable(field)
4696 			      && Field_encloses(field, ry, rx) == E_OK)
4697 			    {
4698 			      res = _nc_Set_Current_Field(form, field);
4699 			      if (res == E_OK)
4700 				res = _nc_Position_Form_Cursor(form);
4701 			      if (res == E_OK
4702 				  && (event.bstate & BUTTON1_DOUBLE_CLICKED))
4703 				res = E_UNKNOWN_COMMAND;
4704 			      break;
4705 			    }
4706 			}
4707 		    }
4708 		}
4709 	    }
4710 	}
4711       else
4712 	res = E_REQUEST_DENIED;
4713     }
4714 #endif /* NCURSES_MOUSE_VERSION */
4715   else if (type == OK)
4716     {
4717       res = Data_Entry_w(form, c);
4718     }
4719 
4720   _nc_Refresh_Current_Field(form);
4721   RETURN(res);
4722 }
4723 # endif	/* USE_WIDEC_SUPPORT */
4724 
4725 /*----------------------------------------------------------------------------
4726   Field-Buffer manipulation routines.
4727   The effects of setting a buffer are tightly coupled to the core of the form
4728   driver logic. This is especially true in the case of growable fields.
4729   So I don't separate this into a separate module.
4730   --------------------------------------------------------------------------*/
4731 
4732 /*---------------------------------------------------------------------------
4733 |   Facility      :  libnform
4734 |   Function      :  int set_field_buffer(FIELD *field,
4735 |                                         int buffer, char *value)
4736 |
4737 |   Description   :  Set the given buffer of the field to the given value.
4738 |                    Buffer 0 stores the displayed content of the field.
4739 |                    For dynamic fields this may grow the fieldbuffers if
4740 |                    the length of the value exceeds the current buffer
4741 |                    length. For buffer 0 only printable values are allowed.
4742 |                    For static fields, the value must not be zero terminated.
4743 |                    It is copied up to the length of the buffer.
4744 |
4745 |   Return Values :  E_OK            - success
4746 |                    E_BAD_ARGUMENT  - invalid argument
4747 |                    E_SYSTEM_ERROR  - system error
4748 +--------------------------------------------------------------------------*/
4749 FORM_EXPORT(int)
4750 set_field_buffer(FIELD *field, int buffer, const char *value)
4751 {
4752   FIELD_CELL *p;
4753   int res = E_OK;
4754   int i;
4755   int len;
4756 
4757 #if USE_WIDEC_SUPPORT
4758   FIELD_CELL *widevalue = 0;
4759 #endif
4760 
4761   T((T_CALLED("set_field_buffer(%p,%d,%s)"), (void *)field, buffer, _nc_visbuf(value)));
4762 
4763   if (!field || !value || ((buffer < 0) || (buffer > field->nbuf)))
4764     RETURN(E_BAD_ARGUMENT);
4765 
4766   len = Buffer_Length(field);
4767 
4768   if (Growable(field))
4769     {
4770       /* for a growable field we must assume zero terminated strings, because
4771          somehow we have to detect the length of what should be copied.
4772        */
4773       int vlen = (int)strlen(value);
4774 
4775       if (vlen > len)
4776 	{
4777 	  if (!Field_Grown(field,
4778 			   (int)(1 + (vlen - len) / ((field->rows + field->nrow)
4779 						     * field->cols))))
4780 	    RETURN(E_SYSTEM_ERROR);
4781 
4782 #if !USE_WIDEC_SUPPORT
4783 	  len = vlen;
4784 #endif
4785 	}
4786     }
4787 
4788   p = Address_Of_Nth_Buffer(field, buffer);
4789 
4790 #if USE_WIDEC_SUPPORT
4791   /*
4792    * Use addstr's logic for converting a string to an array of cchar_t's.
4793    * There should be a better way, but this handles nonspacing characters
4794    * and other special cases that we really do not want to handle here.
4795    */
4796 #if NCURSES_EXT_FUNCS
4797   if (wresize(field->working, 1, Buffer_Length(field) + 1) == ERR)
4798 #endif
4799     {
4800       delwin(field->working);
4801       field->working = newpad(1, Buffer_Length(field) + 1);
4802     }
4803   len = Buffer_Length(field);
4804   wclear(field->working);
4805   (void)mvwaddstr(field->working, 0, 0, value);
4806 
4807   if ((widevalue = typeCalloc(FIELD_CELL, len + 1)) == 0)
4808     {
4809       RETURN(E_SYSTEM_ERROR);
4810     }
4811   else
4812     {
4813       for (i = 0; i < field->drows; ++i)
4814 	{
4815 	  (void)mvwin_wchnstr(field->working, 0, (int)i * field->dcols,
4816 			      widevalue + ((int)i * field->dcols),
4817 			      field->dcols);
4818 	}
4819       for (i = 0; i < len; ++i)
4820 	{
4821 	  if (CharEq(myZEROS, widevalue[i]))
4822 	    {
4823 	      while (i < len)
4824 		p[i++] = myBLANK;
4825 	      break;
4826 	    }
4827 	  p[i] = widevalue[i];
4828 	}
4829       free(widevalue);
4830     }
4831 #else
4832   for (i = 0; i < len; ++i)
4833     {
4834       if (value[i] == '\0')
4835 	{
4836 	  while (i < len)
4837 	    p[i++] = myBLANK;
4838 	  break;
4839 	}
4840       p[i] = value[i];
4841     }
4842 #endif
4843 
4844   if (buffer == 0)
4845     {
4846       int syncres;
4847 
4848       if (((syncres = Synchronize_Field(field)) != E_OK) &&
4849 	  (res == E_OK))
4850 	res = syncres;
4851       if (((syncres = Synchronize_Linked_Fields(field)) != E_OK) &&
4852 	  (res == E_OK))
4853 	res = syncres;
4854     }
4855   RETURN(res);
4856 }
4857 
4858 /*---------------------------------------------------------------------------
4859 |   Facility      :  libnform
4860 |   Function      :  char *field_buffer(const FIELD *field,int buffer)
4861 |
4862 |   Description   :  Return the address of the buffer for the field.
4863 |
4864 |   Return Values :  Pointer to buffer or NULL if arguments were invalid.
4865 +--------------------------------------------------------------------------*/
4866 FORM_EXPORT(char *)
4867 field_buffer(const FIELD *field, int buffer)
4868 {
4869   char *result = 0;
4870 
4871   T((T_CALLED("field_buffer(%p,%d)"), (const void *)field, buffer));
4872 
4873   if (field && (buffer >= 0) && (buffer <= field->nbuf))
4874     {
4875 #if USE_WIDEC_SUPPORT
4876       FIELD_CELL *data = Address_Of_Nth_Buffer(field, buffer);
4877       size_t need = 0;
4878       int size = Buffer_Length(field);
4879       int n;
4880 
4881       /* determine the number of bytes needed to store the expanded string */
4882       for (n = 0; n < size; ++n)
4883 	{
4884 	  if (!isWidecExt(data[n]) && data[n].chars[0] != L'\0')
4885 	    {
4886 	      mbstate_t state;
4887 	      size_t next;
4888 
4889 	      init_mb(state);
4890 	      next = _nc_wcrtomb(0, data[n].chars[0], &state);
4891 	      if (next > 0)
4892 		need += next;
4893 	    }
4894 	}
4895 
4896       /* allocate a place to store the expanded string */
4897       if (field->expanded[buffer] != 0)
4898 	free(field->expanded[buffer]);
4899       field->expanded[buffer] = typeMalloc(char, need + 1);
4900 
4901       /*
4902        * Expand the multibyte data.
4903        *
4904        * It may also be multi-column data.  In that case, the data for a row
4905        * may be null-padded to align to the dcols/drows layout (or it may
4906        * contain embedded wide-character extensions).  Change the null-padding
4907        * to blanks as needed.
4908        */
4909       if ((result = field->expanded[buffer]) != 0)
4910 	{
4911 	  wclear(field->working);
4912 	  wmove(field->working, 0, 0);
4913 	  for (n = 0; n < size; ++n)
4914 	    {
4915 	      if (!isWidecExt(data[n]) && data[n].chars[0] != L'\0')
4916 		wadd_wch(field->working, &data[n]);
4917 	    }
4918 	  wmove(field->working, 0, 0);
4919 	  winnstr(field->working, result, (int)need);
4920 	}
4921 #else
4922       result = Address_Of_Nth_Buffer(field, buffer);
4923 #endif
4924     }
4925   returnPtr(result);
4926 }
4927 
4928 #if USE_WIDEC_SUPPORT
4929 
4930 /*---------------------------------------------------------------------------
4931 | Convert a multibyte string to a wide-character string.  The result must be
4932 | freed by the caller.
4933 +--------------------------------------------------------------------------*/
4934 FORM_EXPORT(wchar_t *)
4935 _nc_Widen_String(char *source, int *lengthp)
4936 {
4937   wchar_t *result = 0;
4938   wchar_t wch;
4939   size_t given = strlen(source);
4940   size_t tries;
4941   int pass;
4942   int status;
4943 
4944 #ifndef state_unused
4945   mbstate_t state;
4946 #endif
4947 
4948   for (pass = 0; pass < 2; ++pass)
4949     {
4950       unsigned need = 0;
4951       size_t passed = 0;
4952 
4953       while (passed < given)
4954 	{
4955 	  bool found = FALSE;
4956 
4957 	  for (tries = 1, status = 0; tries <= (given - passed); ++tries)
4958 	    {
4959 	      int save = source[passed + tries];
4960 
4961 	      source[passed + tries] = 0;
4962 	      reset_mbytes(state);
4963 	      status = check_mbytes(wch, source + passed, tries, state);
4964 	      source[passed + tries] = (char)save;
4965 
4966 	      if (status > 0)
4967 		{
4968 		  found = TRUE;
4969 		  break;
4970 		}
4971 	    }
4972 	  if (found)
4973 	    {
4974 	      if (pass)
4975 		{
4976 		  result[need] = wch;
4977 		}
4978 	      passed += (size_t)status;
4979 	      ++need;
4980 	    }
4981 	  else
4982 	    {
4983 	      if (pass)
4984 		{
4985 		  result[need] = (wchar_t)source[passed];
4986 		}
4987 	      ++need;
4988 	      ++passed;
4989 	    }
4990 	}
4991 
4992       if (!pass)
4993 	{
4994 	  if (!need)
4995 	    break;
4996 	  result = typeCalloc(wchar_t, need);
4997 
4998 	  *lengthp = (int)need;
4999 	  if (result == 0)
5000 	    break;
5001 	}
5002     }
5003 
5004   return result;
5005 }
5006 #endif
5007 
5008 /* frm_driver.c ends here */
5009