libressl: Fix validation errors in certificate chains with expired certificates
[dragonfly.git] / contrib / dialog / formbox.c
1 /*
2  *  $Id: formbox.c,v 1.101 2020/03/27 20:42:19 tom Exp $
3  *
4  *  formbox.c -- implements the form (i.e., some pairs label/editbox)
5  *
6  *  Copyright 2003-2019,2020    Thomas E. Dickey
7  *
8  *  This program is free software; you can redistribute it and/or modify
9  *  it under the terms of the GNU Lesser General Public License, version 2.1
10  *  as published by the Free Software Foundation.
11  *
12  *  This program is distributed in the hope that it will be useful, but
13  *  WITHOUT ANY WARRANTY; without even the implied warranty of
14  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  *  Lesser General Public License for more details.
16  *
17  *  You should have received a copy of the GNU Lesser General Public
18  *  License along with this program; if not, write to
19  *      Free Software Foundation, Inc.
20  *      51 Franklin St., Fifth Floor
21  *      Boston, MA 02110, USA.
22  *
23  *  This is adapted from source contributed by
24  *      Valery Reznic (valery_reznic@users.sourceforge.net)
25  */
26
27 #include <dialog.h>
28 #include <dlg_keys.h>
29
30 #define LLEN(n) ((n) * FORMBOX_TAGS)
31
32 #define ItemName(i)     items[LLEN(i) + 0]
33 #define ItemNameY(i)    items[LLEN(i) + 1]
34 #define ItemNameX(i)    items[LLEN(i) + 2]
35 #define ItemText(i)     items[LLEN(i) + 3]
36 #define ItemTextY(i)    items[LLEN(i) + 4]
37 #define ItemTextX(i)    items[LLEN(i) + 5]
38 #define ItemTextFLen(i) items[LLEN(i) + 6]
39 #define ItemTextILen(i) items[LLEN(i) + 7]
40 #define ItemHelp(i)     (dialog_vars.item_help ? items[LLEN(i) + 8] : dlg_strempty())
41
42 static bool
43 is_readonly(DIALOG_FORMITEM * item)
44 {
45     return ((item->type & 2) != 0) || (item->text_flen <= 0);
46 }
47
48 static bool
49 is_hidden(DIALOG_FORMITEM * item)
50 {
51     return ((item->type & 1) != 0);
52 }
53
54 static bool
55 in_window(WINDOW *win, int scrollamt, int y)
56 {
57     return (y >= scrollamt && y - scrollamt < getmaxy(win));
58 }
59
60 static bool
61 ok_move(WINDOW *win, int scrollamt, int y, int x)
62 {
63     return in_window(win, scrollamt, y)
64         && (wmove(win, y - scrollamt, x) != ERR);
65 }
66
67 static void
68 move_past(WINDOW *win, int y, int x)
69 {
70     if (wmove(win, y, x) == ERR)
71         wmove(win, y, getmaxx(win) - 1);
72 }
73
74 /*
75  * Print form item
76  */
77 static int
78 print_item(WINDOW *win, DIALOG_FORMITEM * item, int scrollamt, bool choice)
79 {
80     int count = 0;
81     int len;
82
83     if (ok_move(win, scrollamt, item->name_y, item->name_x)) {
84         len = item->name_len;
85         len = MIN(len, getmaxx(win) - item->name_x);
86         if (len > 0) {
87             dlg_show_string(win,
88                             item->name,
89                             0,
90                             menubox_attr,
91                             item->name_y - scrollamt,
92                             item->name_x,
93                             len,
94                             FALSE,
95                             FALSE);
96             move_past(win, item->name_y - scrollamt, item->name_x + len);
97             count = 1;
98         }
99     }
100     if (item->text_len && ok_move(win, scrollamt, item->text_y, item->text_x)) {
101         chtype this_item_attribute;
102
103         len = item->text_len;
104         len = MIN(len, getmaxx(win) - item->text_x);
105
106         if (!is_readonly(item)) {
107             this_item_attribute = choice
108                 ? form_active_text_attr
109                 : form_text_attr;
110         } else {
111             this_item_attribute = form_item_readonly_attr;
112         }
113
114         if (len > 0) {
115             dlg_show_string(win,
116                             item->text,
117                             0,
118                             this_item_attribute,
119                             item->text_y - scrollamt,
120                             item->text_x,
121                             len,
122                             is_hidden(item),
123                             FALSE);
124             move_past(win, item->text_y - scrollamt, item->text_x + len);
125             count = 1;
126         }
127     }
128     return count;
129 }
130
131 /*
132  * Print the entire form.
133  */
134 static void
135 print_form(WINDOW *win, DIALOG_FORMITEM * item, int total, int scrollamt, int choice)
136 {
137     int n;
138     int count = 0;
139
140     for (n = 0; n < total; ++n) {
141         count += print_item(win, item + n, scrollamt, n == choice);
142     }
143     if (count) {
144         wbkgdset(win, menubox_attr | ' ');
145         wclrtobot(win);
146         (void) wnoutrefresh(win);
147     }
148 }
149
150 static int
151 set_choice(DIALOG_FORMITEM item[], int choice, int item_no, bool * noneditable)
152 {
153     int result = -1;
154
155     *noneditable = FALSE;
156     if (!is_readonly(&item[choice])) {
157         result = choice;
158     } else {
159         int i;
160
161         for (i = 0; i < item_no; i++) {
162             if (!is_readonly(&(item[i]))) {
163                 result = i;
164                 break;
165             }
166         }
167         if (result < 0) {
168             *noneditable = TRUE;
169             result = 0;
170         }
171     }
172     return result;
173 }
174
175 /*
176  * Find the last y-value in the form.
177  */
178 static int
179 form_limit(DIALOG_FORMITEM item[])
180 {
181     int n;
182     int limit = 0;
183     for (n = 0; item[n].name != 0; ++n) {
184         if (limit < item[n].name_y)
185             limit = item[n].name_y;
186         if (limit < item[n].text_y)
187             limit = item[n].text_y;
188     }
189     return limit;
190 }
191
192 static int
193 is_first_field(DIALOG_FORMITEM item[], int choice)
194 {
195     int count = 0;
196     while (choice >= 0) {
197         if (item[choice].text_flen > 0) {
198             ++count;
199         }
200         --choice;
201     }
202
203     return (count == 1);
204 }
205
206 static int
207 is_last_field(DIALOG_FORMITEM item[], int choice, int item_no)
208 {
209     int count = 0;
210     while (choice < item_no) {
211         if (item[choice].text_flen > 0) {
212             ++count;
213         }
214         ++choice;
215     }
216
217     return (count == 1);
218 }
219
220 /*
221  * Tab to the next field.
222  */
223 static bool
224 tab_next(WINDOW *win,
225          DIALOG_FORMITEM item[],
226          int item_no,
227          int stepsize,
228          int *choice,
229          int *scrollamt)
230 {
231     int old_choice = *choice;
232     int old_scroll = *scrollamt;
233     bool wrapped = FALSE;
234
235     do {
236         do {
237             *choice += stepsize;
238             if (*choice < 0) {
239                 *choice = item_no - 1;
240                 wrapped = TRUE;
241             } else if (*choice >= item_no) {
242                 *choice = 0;
243                 wrapped = TRUE;
244             }
245         } while ((*choice != old_choice) && is_readonly(&(item[*choice])));
246
247         if (item[*choice].text_flen > 0) {
248             int lo = MIN(item[*choice].name_y, item[*choice].text_y);
249             int hi = MAX(item[*choice].name_y, item[*choice].text_y);
250
251             if (old_choice == *choice)
252                 break;
253             print_item(win, item + old_choice, *scrollamt, FALSE);
254
255             if (*scrollamt < lo + 1 - getmaxy(win))
256                 *scrollamt = lo + 1 - getmaxy(win);
257             if (*scrollamt > hi)
258                 *scrollamt = hi;
259             /*
260              * If we have to scroll to show a wrap-around, it does get
261              * confusing.  Just give up rather than scroll.  Tab'ing to the
262              * next field in a multi-column form is a different matter.  Scroll
263              * for that.
264              */
265             if (*scrollamt != old_scroll) {
266                 if (wrapped) {
267                     beep();
268                     *scrollamt = old_scroll;
269                     *choice = old_choice;
270                 } else {
271                     scrollok(win, TRUE);
272                     wscrl(win, *scrollamt - old_scroll);
273                     scrollok(win, FALSE);
274                 }
275             }
276             break;
277         }
278     } while (*choice != old_choice);
279
280     return (old_choice != *choice) || (old_scroll != *scrollamt);
281 }
282
283 /*
284  * Scroll to the next page, putting the choice at the first editable field
285  * in that page.  Note that fields are not necessarily in top-to-bottom order,
286  * nor is there necessarily a field on each row of the window.
287  */
288 static bool
289 scroll_next(WINDOW *win, DIALOG_FORMITEM item[], int stepsize, int *choice, int *scrollamt)
290 {
291     bool result = TRUE;
292     int old_choice = *choice;
293     int old_scroll = *scrollamt;
294     int old_row = MIN(item[old_choice].text_y, item[old_choice].name_y);
295     int target = old_scroll + stepsize;
296
297     if (stepsize < 0) {
298         if (old_row != old_scroll)
299             target = old_scroll;
300         else
301             target = old_scroll + stepsize;
302         if (target < 0) {
303             result = FALSE;
304         }
305     } else {
306         if (target > form_limit(item)) {
307             result = FALSE;
308         }
309     }
310
311     if (result) {
312         int n;
313
314         for (n = 0; item[n].name != 0; ++n) {
315             if (item[n].text_flen > 0) {
316                 int new_row = MIN(item[n].text_y, item[n].name_y);
317                 if (abs(new_row - target) < abs(old_row - target)) {
318                     old_row = new_row;
319                     *choice = n;
320                 }
321             }
322         }
323
324         if (old_choice != *choice)
325             print_item(win, item + old_choice, *scrollamt, FALSE);
326
327         *scrollamt = *choice;
328         if (*scrollamt != old_scroll) {
329             scrollok(win, TRUE);
330             wscrl(win, *scrollamt - old_scroll);
331             scrollok(win, FALSE);
332         }
333         result = (old_choice != *choice) || (old_scroll != *scrollamt);
334     }
335     if (!result)
336         beep();
337     return result;
338 }
339
340 /*
341  * Do a sanity check on the field length, and return the "right" value.
342  */
343 static int
344 real_length(DIALOG_FORMITEM * item)
345 {
346     return (item->text_flen > 0
347             ? item->text_flen
348             : (item->text_flen < 0
349                ? -item->text_flen
350                : item->text_len));
351 }
352
353 /*
354  * Compute the form size, setup field buffers.
355  */
356 static void
357 make_FORM_ELTs(DIALOG_FORMITEM * item,
358                int item_no,
359                int *min_height,
360                int *min_width)
361 {
362     int i;
363     int min_w = 0;
364     int min_h = 0;
365
366     for (i = 0; i < item_no; ++i) {
367         int real_len = real_length(item + i);
368
369         /*
370          * Special value '0' for text_flen: no input allowed
371          * Special value '0' for text_ilen: 'be the same as text_flen'
372          */
373         if (item[i].text_ilen == 0)
374             item[i].text_ilen = real_len;
375
376         min_h = MAX(min_h, item[i].name_y + 1);
377         min_h = MAX(min_h, item[i].text_y + 1);
378         min_w = MAX(min_w, item[i].name_x + 1 + item[i].name_len);
379         min_w = MAX(min_w, item[i].text_x + 1 + real_len);
380
381         item[i].text_len = real_length(item + i);
382
383         /*
384          * We do not know the actual length of .text, so we allocate it here
385          * to ensure it is big enough.
386          */
387         if (item[i].text_flen > 0) {
388             int max_len = dlg_max_input(MAX(item[i].text_ilen + 1, MAX_LEN));
389             char *old_text = item[i].text;
390
391             item[i].text = dlg_malloc(char, (size_t) max_len + 1);
392             assert_ptr(item[i].text, "make_FORM_ELTs");
393
394             sprintf(item[i].text, "%.*s", item[i].text_ilen, old_text);
395
396             if (item[i].text_free) {
397                 item[i].text_free = FALSE;
398                 free(old_text);
399             }
400             item[i].text_free = TRUE;
401         }
402     }
403
404     *min_height = min_h;
405     *min_width = min_w;
406 }
407
408 int
409 dlg_default_formitem(DIALOG_FORMITEM * items)
410 {
411     int result = 0;
412
413     if (dialog_vars.default_item != 0) {
414         int count = 0;
415         while (items->name != 0) {
416             if (!strcmp(dialog_vars.default_item, items->name)) {
417                 result = count;
418                 break;
419             }
420             ++items;
421             count++;
422         }
423     }
424     return result;
425 }
426
427 #define sTEXT -1
428
429 static int
430 next_valid_buttonindex(int state, int extra, bool non_editable)
431 {
432     state = dlg_next_ok_buttonindex(state, extra);
433     while (non_editable && state == sTEXT)
434         state = dlg_next_ok_buttonindex(state, sTEXT);
435     return state;
436 }
437
438 static int
439 prev_valid_buttonindex(int state, int extra, bool non_editable)
440 {
441     state = dlg_prev_ok_buttonindex(state, extra);
442     while (non_editable && state == sTEXT)
443         state = dlg_prev_ok_buttonindex(state, sTEXT);
444     return state;
445 }
446
447 #define NAVIGATE_BINDINGS \
448         DLG_KEYS_DATA( DLGK_FIELD_NEXT, TAB ), \
449         DLG_KEYS_DATA( DLGK_FIELD_PREV, KEY_BTAB ), \
450         DLG_KEYS_DATA( DLGK_ITEM_NEXT,  CHR_NEXT ), \
451         DLG_KEYS_DATA( DLGK_ITEM_NEXT,  KEY_DOWN ), \
452         DLG_KEYS_DATA( DLGK_ITEM_NEXT,  KEY_RIGHT ), \
453         DLG_KEYS_DATA( DLGK_ITEM_NEXT,  KEY_NEXT ), \
454         DLG_KEYS_DATA( DLGK_ITEM_PREV,  CHR_PREVIOUS ), \
455         DLG_KEYS_DATA( DLGK_ITEM_PREV,  KEY_PREVIOUS ), \
456         DLG_KEYS_DATA( DLGK_ITEM_PREV,  KEY_LEFT ), \
457         DLG_KEYS_DATA( DLGK_ITEM_PREV,  KEY_UP ), \
458         DLG_KEYS_DATA( DLGK_PAGE_NEXT,  KEY_NPAGE ), \
459         DLG_KEYS_DATA( DLGK_PAGE_PREV,  KEY_PPAGE )
460 /*
461  * Display a form for entering a number of fields
462  */
463 int
464 dlg_form(const char *title,
465          const char *cprompt,
466          int height,
467          int width,
468          int form_height,
469          int item_no,
470          DIALOG_FORMITEM * items,
471          int *current_item)
472 {
473     /* *INDENT-OFF* */
474     static DLG_KEYS_BINDING binding[] = {
475         HELPKEY_BINDINGS,
476         ENTERKEY_BINDINGS,
477         NAVIGATE_BINDINGS,
478         TOGGLEKEY_BINDINGS,
479         END_KEYS_BINDING
480     };
481     static DLG_KEYS_BINDING binding2[] = {
482         INPUTSTR_BINDINGS,
483         HELPKEY_BINDINGS,
484         ENTERKEY_BINDINGS,
485         NAVIGATE_BINDINGS,
486         /* no TOGGLEKEY_BINDINGS, since that includes space... */
487         END_KEYS_BINDING
488     };
489     /* *INDENT-ON* */
490
491 #ifdef KEY_RESIZE
492     int old_height = height;
493     int old_width = width;
494 #endif
495
496     int form_width;
497     bool first = TRUE;
498     bool first_trace = TRUE;
499     int chr_offset = 0;
500     int state = (dialog_vars.default_button >= 0
501                  ? dlg_default_button()
502                  : sTEXT);
503     int x, y, cur_x, cur_y, box_x, box_y;
504     int code;
505     int fkey;
506     int choice = dlg_default_formitem(items);
507     int new_choice, new_scroll;
508     int scrollamt = 0;
509     int result = DLG_EXIT_UNKNOWN;
510     int min_width = 0, min_height = 0;
511     bool was_autosize = (height == 0 || width == 0);
512     bool show_buttons = FALSE;
513     bool scroll_changed = FALSE;
514     bool field_changed = FALSE;
515     bool non_editable = FALSE;
516     WINDOW *dialog, *form;
517     char *prompt;
518     const char **buttons = dlg_ok_labels();
519     DIALOG_FORMITEM *current;
520
521     DLG_TRACE(("# %sform args:\n", (dialog_vars.formitem_type
522                                     ? "password"
523                                     : "")));
524     DLG_TRACE2S("title", title);
525     DLG_TRACE2S("message", cprompt);
526     DLG_TRACE2N("height", height);
527     DLG_TRACE2N("width", width);
528     DLG_TRACE2N("lheight", form_height);
529     DLG_TRACE2N("llength", item_no);
530     /* FIXME dump the items[][] too */
531     DLG_TRACE2N("current", *current_item);
532
533     make_FORM_ELTs(items, item_no, &min_height, &min_width);
534     dlg_button_layout(buttons, &min_width);
535     dlg_does_output();
536
537 #ifdef KEY_RESIZE
538   retry:
539 #endif
540
541     prompt = dlg_strclone(cprompt);
542     dlg_tab_correct_str(prompt);
543     dlg_auto_size(title, prompt, &height, &width,
544                   1 + 3 * MARGIN,
545                   MAX(26, 2 + min_width));
546
547     if (form_height == 0)
548         form_height = min_height;
549
550     if (was_autosize) {
551         form_height = MIN(SLINES - height, form_height);
552         height += form_height;
553     } else {
554         int thigh = 0;
555         int twide = 0;
556         dlg_auto_size(title, prompt, &thigh, &twide, 0, width);
557         thigh = SLINES - (height - (thigh + 1 + 3 * MARGIN));
558         form_height = MIN(thigh, form_height);
559     }
560
561     dlg_print_size(height, width);
562     dlg_ctl_size(height, width);
563
564     x = dlg_box_x_ordinate(width);
565     y = dlg_box_y_ordinate(height);
566
567     dialog = dlg_new_window(height, width, y, x);
568     dlg_register_window(dialog, "formbox", binding);
569     dlg_register_buttons(dialog, "formbox", buttons);
570
571     dlg_mouse_setbase(x, y);
572
573     dlg_draw_box2(dialog, 0, 0, height, width, dialog_attr, border_attr, border2_attr);
574     dlg_draw_bottom_box2(dialog, border_attr, border2_attr, dialog_attr);
575     dlg_draw_title(dialog, title);
576
577     dlg_attrset(dialog, dialog_attr);
578     dlg_print_autowrap(dialog, prompt, height, width);
579
580     form_width = width - 6;
581     getyx(dialog, cur_y, cur_x);
582     (void) cur_x;
583     box_y = cur_y + 1;
584     box_x = (width - form_width) / 2 - 1;
585
586     /* create new window for the form */
587     form = dlg_sub_window(dialog, form_height, form_width, y + box_y + 1,
588                           x + box_x + 1);
589     dlg_register_window(form, "formfield", binding2);
590
591     /* draw a box around the form items */
592     dlg_draw_box(dialog, box_y, box_x, form_height + 2, form_width + 2,
593                  menubox_border_attr, menubox_border2_attr);
594
595     /* register the new window, along with its borders */
596     dlg_mouse_mkbigregion(getbegy(form) - getbegy(dialog),
597                           getbegx(form) - getbegx(dialog),
598                           getmaxy(form),
599                           getmaxx(form),
600                           KEY_MAX, 1, 1, 3 /* by cells */ );
601
602     show_buttons = TRUE;
603     scroll_changed = TRUE;
604
605     choice = set_choice(items, choice, item_no, &non_editable);
606     current = &items[choice];
607     if (non_editable)
608         state = next_valid_buttonindex(state, sTEXT, non_editable);
609
610     while (result == DLG_EXIT_UNKNOWN) {
611         int edit = FALSE;
612         int key;
613
614         if (scroll_changed) {
615             print_form(form, items, item_no, scrollamt, choice);
616             dlg_draw_scrollbar(dialog,
617                                scrollamt,
618                                scrollamt,
619                                scrollamt + form_height + 1,
620                                min_height,
621                                box_x + 1,
622                                box_x + form_width,
623                                box_y,
624                                box_y + form_height + 1,
625                                menubox_border2_attr,
626                                menubox_border_attr);
627             scroll_changed = FALSE;
628         }
629
630         if (show_buttons) {
631             dlg_item_help("");
632             dlg_draw_buttons(dialog, height - 2, 0, buttons,
633                              ((state < 0)
634                               ? 1000    /* no such button, not highlighted */
635                               : state),
636                              FALSE, width);
637             show_buttons = FALSE;
638         }
639
640         if (first_trace) {
641             first_trace = FALSE;
642             dlg_trace_win(dialog);
643         }
644
645         if (field_changed || state == sTEXT) {
646             if (field_changed)
647                 chr_offset = 0;
648             current = &items[choice];
649             dialog_vars.max_input = current->text_ilen;
650             dlg_item_help(current->help);
651             dlg_show_string(form, current->text, chr_offset,
652                             form_active_text_attr,
653                             current->text_y - scrollamt,
654                             current->text_x,
655                             current->text_len,
656                             is_hidden(current), first);
657             wsyncup(form);
658             wcursyncup(form);
659             field_changed = FALSE;
660         }
661
662         key = dlg_mouse_wgetch((state == sTEXT) ? form : dialog, &fkey);
663         if (dlg_result_key(key, fkey, &result)) {
664             break;
665         }
666
667         /* handle non-functionkeys */
668         if (!fkey) {
669             if (state != sTEXT) {
670                 code = dlg_char_to_button(key, buttons);
671                 if (code >= 0) {
672                     dlg_del_window(dialog);
673                     result = dlg_ok_buttoncode(code);
674                     continue;
675                 }
676             }
677         }
678
679         /* handle functionkeys */
680         if (fkey) {
681             bool do_scroll = FALSE;
682             bool do_tab = FALSE;
683             int move_by = 0;
684
685             switch (key) {
686             case DLGK_MOUSE(KEY_PPAGE):
687             case DLGK_PAGE_PREV:
688                 do_scroll = TRUE;
689                 move_by = -form_height;
690                 break;
691
692             case DLGK_MOUSE(KEY_NPAGE):
693             case DLGK_PAGE_NEXT:
694                 do_scroll = TRUE;
695                 move_by = form_height;
696                 break;
697
698             case DLGK_TOGGLE:
699             case DLGK_ENTER:
700                 dlg_del_window(dialog);
701                 result = (state >= 0) ? dlg_enter_buttoncode(state) : DLG_EXIT_OK;
702                 continue;
703
704             case DLGK_GRID_LEFT:
705                 if (state == sTEXT)
706                     break;
707                 /* FALLTHRU */
708             case DLGK_ITEM_PREV:
709                 if (state == sTEXT) {
710                     do_tab = TRUE;
711                     move_by = -1;
712                     break;
713                 } else {
714                     state = prev_valid_buttonindex(state, 0, non_editable);
715                     show_buttons = TRUE;
716                     continue;
717                 }
718
719             case DLGK_FORM_PREV:
720                 if (state == sTEXT && !is_first_field(items, choice)) {
721                     do_tab = TRUE;
722                     move_by = -1;
723                     break;
724                 } else {
725                     int old_state = state;
726                     state = prev_valid_buttonindex(state, sTEXT, non_editable);
727                     show_buttons = TRUE;
728                     if (old_state >= 0 && state == sTEXT) {
729                         new_choice = item_no - 1;
730                         if (choice != new_choice) {
731                             print_item(form, items + choice, scrollamt, FALSE);
732                             choice = new_choice;
733                         }
734                     }
735                     continue;
736                 }
737
738             case DLGK_FIELD_PREV:
739                 state = prev_valid_buttonindex(state, sTEXT, non_editable);
740                 show_buttons = TRUE;
741                 continue;
742
743             case DLGK_FIELD_NEXT:
744                 state = next_valid_buttonindex(state, sTEXT, non_editable);
745                 show_buttons = TRUE;
746                 continue;
747
748             case DLGK_GRID_RIGHT:
749                 if (state == sTEXT)
750                     break;
751                 /* FALLTHRU */
752
753             case DLGK_ITEM_NEXT:
754                 if (state == sTEXT) {
755                     do_tab = TRUE;
756                     move_by = 1;
757                     break;
758                 } else {
759                     state = next_valid_buttonindex(state, 0, non_editable);
760                     show_buttons = TRUE;
761                     continue;
762                 }
763
764             case DLGK_FORM_NEXT:
765                 if (state == sTEXT && !is_last_field(items, choice, item_no)) {
766                     do_tab = TRUE;
767                     move_by = 1;
768                     break;
769                 } else {
770                     state = next_valid_buttonindex(state, sTEXT, non_editable);
771                     show_buttons = TRUE;
772                     if (state == sTEXT && choice) {
773                         print_item(form, items + choice, scrollamt, FALSE);
774                         choice = 0;
775                     }
776                     continue;
777                 }
778
779 #ifdef KEY_RESIZE
780             case KEY_RESIZE:
781                 dlg_will_resize(dialog);
782                 /* reset data */
783                 height = old_height;
784                 width = old_width;
785                 free(prompt);
786                 _dlg_resize_cleanup(dialog);
787                 dlg_unregister_window(form);
788                 /* repaint */
789                 goto retry;
790 #endif
791             default:
792 #if USE_MOUSE
793                 if (is_DLGK_MOUSE(key)) {
794                     if (key >= DLGK_MOUSE(KEY_MAX)) {
795                         int cell = key - DLGK_MOUSE(KEY_MAX);
796                         int row = (cell / getmaxx(form)) + scrollamt;
797                         int col = (cell % getmaxx(form));
798                         int n;
799
800                         for (n = 0; n < item_no; ++n) {
801                             if (items[n].name_y == row
802                                 && items[n].name_x <= col
803                                 && (items[n].name_x + items[n].name_len > col
804                                     || (items[n].name_y == items[n].text_y
805                                         && items[n].text_x > col))) {
806                                 if (!is_readonly(&(items[n]))) {
807                                     field_changed = TRUE;
808                                     break;
809                                 }
810                             }
811                             if (items[n].text_y == row
812                                 && items[n].text_x <= col
813                                 && items[n].text_x + items[n].text_ilen > col) {
814                                 if (!is_readonly(&(items[n]))) {
815                                     field_changed = TRUE;
816                                     break;
817                                 }
818                             }
819                         }
820                         if (field_changed) {
821                             print_item(form, items + choice, scrollamt, FALSE);
822                             choice = n;
823                             continue;
824                         }
825                         beep();
826                     } else if ((code = dlg_ok_buttoncode(key - M_EVENT)) >= 0) {
827                         result = code;
828                     }
829                     continue;
830                 }
831 #endif
832                 break;
833             }
834
835             new_scroll = scrollamt;
836             new_choice = choice;
837             if (do_scroll) {
838                 if (scroll_next(form, items, move_by, &new_choice, &new_scroll)) {
839                     if (choice != new_choice) {
840                         choice = new_choice;
841                         field_changed = TRUE;
842                     }
843                     if (scrollamt != new_scroll) {
844                         scrollamt = new_scroll;
845                         scroll_changed = TRUE;
846                     }
847                 }
848                 continue;
849             }
850             if (do_tab) {
851                 if (tab_next(form, items, item_no, move_by, &new_choice, &new_scroll)) {
852                     if (choice != new_choice) {
853                         choice = new_choice;
854                         field_changed = TRUE;
855                     }
856                     if (scrollamt != new_scroll) {
857                         scrollamt = new_scroll;
858                         scroll_changed = TRUE;
859                     }
860                 }
861                 continue;
862             }
863         }
864
865         if (state == sTEXT) {   /* Input box selected */
866             if (!is_readonly(current))
867                 edit = dlg_edit_string(current->text, &chr_offset, key,
868                                        fkey, first);
869             if (edit) {
870                 dlg_show_string(form, current->text, chr_offset,
871                                 form_active_text_attr,
872                                 current->text_y - scrollamt,
873                                 current->text_x,
874                                 current->text_len,
875                                 is_hidden(current), first);
876                 continue;
877             }
878         }
879
880     }
881
882     dlg_mouse_free_regions();
883     dlg_unregister_window(form);
884     dlg_del_window(dialog);
885     free(prompt);
886
887     *current_item = choice;
888     return result;
889 }
890
891 /*
892  * Free memory owned by a list of DIALOG_FORMITEM's.
893  */
894 void
895 dlg_free_formitems(DIALOG_FORMITEM * items)
896 {
897     int n;
898     for (n = 0; items[n].name != 0; ++n) {
899         if (items[n].name_free)
900             free(items[n].name);
901         if (items[n].text_free)
902             free(items[n].text);
903         if (items[n].help_free && items[n].help != dlg_strempty())
904             free(items[n].help);
905     }
906     free(items);
907 }
908
909 /*
910  * The script accepts values beginning at 1, while curses starts at 0.
911  */
912 int
913 dlg_ordinate(const char *s)
914 {
915     int result = atoi(s);
916     if (result > 0)
917         --result;
918     else
919         result = 0;
920     return result;
921 }
922
923 int
924 dialog_form(const char *title,
925             const char *cprompt,
926             int height,
927             int width,
928             int form_height,
929             int item_no,
930             char **items)
931 {
932     int result;
933     int choice = 0;
934     int i;
935     DIALOG_FORMITEM *listitems;
936     DIALOG_VARS save_vars;
937     bool show_status = FALSE;
938     char *help_result;
939
940     dlg_save_vars(&save_vars);
941     dialog_vars.separate_output = TRUE;
942
943     listitems = dlg_calloc(DIALOG_FORMITEM, (size_t) item_no + 1);
944     assert_ptr(listitems, "dialog_form");
945
946     for (i = 0; i < item_no; ++i) {
947         listitems[i].type = dialog_vars.formitem_type;
948         listitems[i].name = ItemName(i);
949         listitems[i].name_len = (int) strlen(ItemName(i));
950         listitems[i].name_y = dlg_ordinate(ItemNameY(i));
951         listitems[i].name_x = dlg_ordinate(ItemNameX(i));
952         listitems[i].text = ItemText(i);
953         listitems[i].text_len = (int) strlen(ItemText(i));
954         listitems[i].text_y = dlg_ordinate(ItemTextY(i));
955         listitems[i].text_x = dlg_ordinate(ItemTextX(i));
956         listitems[i].text_flen = atoi(ItemTextFLen(i));
957         listitems[i].text_ilen = atoi(ItemTextILen(i));
958         listitems[i].help = ((dialog_vars.item_help)
959                              ? ItemHelp(i)
960                              : dlg_strempty());
961     }
962
963     result = dlg_form(title,
964                       cprompt,
965                       height,
966                       width,
967                       form_height,
968                       item_no,
969                       listitems,
970                       &choice);
971
972     switch (result) {
973     case DLG_EXIT_OK:           /* FALLTHRU */
974     case DLG_EXIT_EXTRA:
975         show_status = TRUE;
976         break;
977     case DLG_EXIT_HELP:
978         dlg_add_help_formitem(&result, &help_result, &listitems[choice]);
979         show_status = dialog_vars.help_status;
980         dlg_add_string(help_result);
981         if (show_status)
982             dlg_add_separator();
983         break;
984     }
985     if (show_status) {
986         for (i = 0; i < item_no; i++) {
987             if (listitems[i].text_flen > 0) {
988                 dlg_add_string(listitems[i].text);
989                 dlg_add_separator();
990             }
991         }
992         dlg_add_last_key(-1);
993     }
994
995     dlg_free_formitems(listitems);
996     dlg_restore_vars(&save_vars);
997
998     return result;
999 }