2 * $Id: mixedgauge.c,v 1.34 2018/06/18 22:09:31 tom Exp $
4 * mixedgauge.c -- implements the mixedgauge dialog
6 * Copyright 2007-2012,2018 Thomas E. Dickey
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.
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.
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.
23 * This is inspired by a patch from Kiran Cherupally
24 * (but different interface design).
29 #define LLEN(n) ((n) * MIXEDGAUGE_TAGS)
30 #define ItemData(i) &items[LLEN(i)]
31 #define ItemName(i) items[LLEN(i)]
32 #define ItemText(i) items[LLEN(i) + 1]
35 #define MIN_WIDE (10 + 2 * (2 + MARGIN))
42 int height, old_height, min_height;
43 int width, old_width, min_width;
44 int len_name, len_text;
46 DIALOG_LISTITEM *list;
50 status_string(char *given, char **freeMe)
55 if (isdigit(UCH(*given))) {
58 result = _("Succeeded");
67 result = _("Completed");
70 result = _("Checked");
76 result = _("Skipped");
79 result = _("In Progress");
91 } else if (*given == '-') {
92 size_t need = strlen(++given) + 4;
93 char *temp = dlg_malloc(char, need);
95 sprintf(temp, "%3s%%", given);
97 } else if (!isspace(UCH(*given))) {
105 /* This function displays status messages */
107 myprint_status(DIALOG_MIXEDGAUGE * dlg)
109 WINDOW *win = dlg->dialog;
110 int limit_y = dlg->height;
111 int limit_x = dlg->width;
115 int cells = dlg->len_text - 2;
116 int lm = limit_x - dlg->len_text - 1;
117 int bm = limit_y; /* bottom margin */
118 int last_y = 0, last_x = 0;
121 const char *status = "";
125 getyx(win, last_y, last_x);
126 for (item = 0; item < dlg->item_no; ++item) {
127 chtype attr = A_NORMAL;
129 y = item + MARGIN + 1;
133 status = status_string(dlg->list[item].text, &freeMe);
134 if (status == 0 || *status == 0)
137 (void) wmove(win, y, 2 * MARGIN);
138 dlg_attrset(win, dialog_attr);
139 dlg_print_text(win, dlg->list[item].name, lm, &attr);
141 (void) wmove(win, y, lm);
142 (void) waddch(win, '[');
143 (void) wmove(win, y, lm + (cells - (int) strlen(status)) / 2);
145 (void) wmove(win, y, lm + 1);
146 dlg_attrset(win, title_attr);
147 for (j = 0; j < cells; j++)
148 (void) waddch(win, ' ');
150 (void) wmove(win, y, lm + (cells - (int) strlen(status)) / 2);
151 (void) waddstr(win, status);
153 if ((title_attr & A_REVERSE) != 0) {
154 dlg_attroff(win, A_REVERSE);
156 dlg_attrset(win, A_REVERSE);
158 (void) wmove(win, y, lm + 1);
160 if (sscanf(status, "%f%%", &percent) != 1)
162 xxx = (int) ((cells * (percent + 0.5)) / 100.0);
163 for (j = 0; j < xxx; j++) {
164 chtype ch1 = winch(win);
165 if (title_attr & A_REVERSE) {
168 (void) waddch(win, ch1);
173 (void) wmove(win, y, lm + (cells - (int) strlen(status)) / 2);
174 (void) waddstr(win, status);
176 (void) wmove(win, y, limit_x - 3);
177 dlg_attrset(win, dialog_attr);
178 (void) waddch(win, ']');
179 (void) wnoutrefresh(win);
182 wmove(win, last_y, last_x);
186 mydraw_mixed_box(WINDOW *win, int y, int x, int height, int width,
187 chtype boxchar, chtype borderchar)
189 dlg_draw_box(win, y, x, height, width, boxchar, borderchar);
191 chtype attr = A_NORMAL;
192 const char *message = _("Overall Progress");
193 chtype save2 = dlg_get_attrs(win);
194 dlg_attrset(win, title_attr);
195 (void) wmove(win, y, x + 2);
196 dlg_print_text(win, message, width, &attr);
197 dlg_attrset(win, save2);
202 clean_copy(const char *string)
204 char *result = dlg_strclone(string);
206 dlg_trim_string(result);
207 dlg_tab_correct_str(result);
212 * Update mixed-gauge dialog (may be from pipe, may be via direct calls).
215 dlg_update_mixedgauge(DIALOG_MIXEDGAUGE * dlg, int percent)
220 * Clear the area for the progress bar by filling it with spaces
221 * in the title-attribute, and write the percentage with that
224 (void) wmove(dlg->dialog, dlg->height - 3, 4);
225 dlg_attrset(dlg->dialog, gauge_attr);
227 for (i = 0; i < (dlg->width - 2 * (3 + MARGIN)); i++)
228 (void) waddch(dlg->dialog, ' ');
230 (void) wmove(dlg->dialog, dlg->height - 3, (dlg->width / 2) - 2);
231 (void) wprintw(dlg->dialog, "%3d%%", percent);
234 * Now draw a bar in reverse, relative to the background.
235 * The window attribute was useful for painting the background,
236 * but requires some tweaks to reverse it.
238 x = (percent * (dlg->width - 2 * (3 + MARGIN))) / 100;
239 if ((title_attr & A_REVERSE) != 0) {
240 dlg_attroff(dlg->dialog, A_REVERSE);
242 dlg_attrset(dlg->dialog, A_REVERSE);
244 (void) wmove(dlg->dialog, dlg->height - 3, 4);
245 for (i = 0; i < x; i++) {
246 chtype ch = winch(dlg->dialog);
247 if (title_attr & A_REVERSE) {
250 (void) waddch(dlg->dialog, ch);
253 dlg_trace_win(dlg->dialog);
260 dlg_begin_mixedgauge(DIALOG_MIXEDGAUGE * dlg,
274 memset(dlg, 0, sizeof(*dlg));
276 dlg->prompt = clean_copy(aPrompt);
277 dlg->height = dlg->old_height = aHeight;
278 dlg->width = dlg->old_width = aWidth;
279 dlg->item_no = aItemNo;
281 dlg->list = dlg_calloc(DIALOG_LISTITEM, (size_t) aItemNo);
282 assert_ptr(dlg->list, "dialog_mixedgauge");
287 for (n = 0; n < aItemNo; ++n) {
288 int thisWidth = (int) strlen(ItemName(n));
289 if (dlg->len_name < thisWidth)
290 dlg->len_name = thisWidth;
291 dlg->list[n].name = ItemName(n);
292 dlg->list[n].text = ItemText(n);
295 dlg->min_height = MIN_HIGH + aItemNo;
296 dlg->min_width = MIN_WIDE + dlg->len_name + GUTTER + dlg->len_text;
298 if (dlg->prompt != 0 && *(dlg->prompt) != 0)
299 dlg->min_height += (2 * MARGIN);
301 nodelay(stdscr, TRUE);
306 dlg_del_window(dlg->dialog);
307 dlg->height = dlg->old_height;
308 dlg->width = dlg->old_width;
312 dlg_auto_size(dlg->title, dlg->prompt,
317 dlg_print_size(dlg->height, dlg->width);
318 dlg_ctl_size(dlg->height, dlg->width);
320 /* center dialog box on screen */
321 x = dlg_box_x_ordinate(dlg->width);
322 y = dlg_box_y_ordinate(dlg->height);
324 dlg->dialog = dlg_new_window(dlg->height, dlg->width, y, x);
326 (void) werase(dlg->dialog);
327 dlg_draw_box2(dlg->dialog,
331 dialog_attr, border_attr, border2_attr);
333 dlg_draw_title(dlg->dialog, dlg->title);
334 dlg_draw_helpline(dlg->dialog, FALSE);
336 if ((dlg->prompt != 0 && *(dlg->prompt) != 0)
337 && wmove(dlg->dialog, dlg->item_no, 0) != ERR) {
338 dlg->caption = dlg_sub_window(dlg->dialog,
339 dlg->height - dlg->item_no - (2 * MARGIN),
341 y + dlg->item_no + (2 * MARGIN),
343 dlg_attrset(dlg->caption, dialog_attr);
344 dlg_print_autowrap(dlg->caption, dlg->prompt, dlg->height, dlg->width);
347 mydraw_mixed_box(dlg->dialog,
351 dlg->width - 2 * (2 + MARGIN),
359 * Discard the mixed-gauge dialog.
362 dlg_finish_mixedgauge(DIALOG_MIXEDGAUGE * dlg, int status)
364 (void) wrefresh(dlg->dialog);
366 nodelay(stdscr, FALSE);
369 dlg_del_window(dlg->dialog);
374 * Setup dialog, read mixed-gauge data from pipe.
377 dialog_mixedgauge(const char *title,
385 DIALOG_MIXEDGAUGE dlg;
388 DLG_TRACE(("# mixedgauge args:\n"));
389 DLG_TRACE2S("title", title);
390 DLG_TRACE2S("message", cprompt);
391 DLG_TRACE2N("height", height);
392 DLG_TRACE2N("width", width);
393 DLG_TRACE2N("percent", percent);
394 DLG_TRACE2N("llength", item_no);
395 /* FIXME dump the items[][] too */
397 dlg_begin_mixedgauge(&dlg, &began, title, cprompt, height,
398 width, item_no, items);
400 dlg_update_mixedgauge(&dlg, percent);
402 return dlg_finish_mixedgauge(&dlg, DLG_EXIT_OK);