2 * Routines for progress-bar (modal) dialog
6 * Wireshark - Network traffic analyzer
7 * By Gerald Combs <gerald@wireshark.org>
8 * Copyright 1998 Gerald Combs
10 * This program is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU General Public License
12 * as published by the Free Software Foundation; either version 2
13 * of the License, or (at your option) any later version.
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
31 #include "ui/progress_dlg.h"
33 #include "ui/gtk/gtkglobals.h"
34 #include "ui/gtk/dlg_utils.h"
35 #include "ui/gtk/gui_utils.h"
38 #define PROG_BAR_KEY "progress_bar"
40 static gboolean
delete_event_cb(GtkWidget
*w
, GdkEvent
*event
, gpointer data
);
41 static void stop_cb(GtkWidget
*w
, gpointer data
);
44 * Define the structure describing a progress dialog.
47 GtkWidget
*dlg_w
; /* top-level window widget */
49 GTimeVal last_time
; /* last time it was updated */
53 GtkLabel
*time_left_lb
;
54 GtkLabel
*percentage_lb
;
59 * Create and pop up the progress dialog; allocate a "progdlg_t"
60 * and initialize it to contain all information the implementation
61 * needs in order to manipulate the dialog, and return a pointer to
64 * The first argument is the task to do, e.g. "Loading".
65 * The second argument is the item to do, e.g. "capture.cap".
66 * The third argument is TRUE if the "terminate this operation" button should
67 * be a "Stop" button (meaning that the operation is stopped, but not undone),
68 * and FALSE if it should be a "Cancel" button (meaning that it's stopped
69 * and anything it's done would be undone)
70 * The fourth argument is a pointer to a Boolean variable that will be
71 * set to TRUE if the user hits that button.
73 * XXX - provide a way to specify the progress in units, with the total
74 * number of units specified as an argument when the progress dialog
75 * is created; updates would be given in units, with the progress dialog
76 * code computing the percentage, and the progress bar would have a
77 * label "0" on the left and <total number of units> on the right, with
78 * a label in the middle giving the number of units we've processed
79 * so far. This could be used when filtering packets, for example; we
80 * wouldn't always use it, as we have no idea how many packets are to
84 create_progress_dlg(const gpointer top_level_window _U_
, const gchar
*task_title
, const gchar
*item_title
,
85 gboolean terminate_is_stop
, gboolean
*stop_flag
)
88 GtkWidget
*dlg_w
, *main_vb
, *title_lb
, *status_lb
, *elapsed_lb
, *time_left_lb
, *percentage_lb
;
89 GtkWidget
*prog_bar
, *bbox
, *cancel_bt
;
90 GtkWidget
*static_vb
, *tmp_lb
, *main_hb
, *dynamic_vb
, *percentage_hb
;
91 gchar
*task_title_dup
;
92 gchar
*item_title_dup
;
94 dlg
= (progdlg_t
*)g_malloc(sizeof (progdlg_t
));
96 /* limit the item_title to some reasonable length */
97 item_title_dup
= g_strdup(item_title
);
98 if (strlen(item_title_dup
) > 110) {
99 g_strlcpy(&item_title_dup
[100], "...", 4);
102 dlg
->title
= g_strdup_printf("%s: %s", task_title
, item_title_dup
);
104 dlg_w
= dlg_window_new(dlg
->title
);
105 gtk_window_set_modal(GTK_WINDOW(dlg_w
), TRUE
);
108 * Container for dialog widgets.
110 main_vb
= ws_gtk_box_new(GTK_ORIENTATION_VERTICAL
, 1, FALSE
);
111 gtk_container_set_border_width(GTK_CONTAINER(main_vb
), 5);
112 gtk_container_add(GTK_CONTAINER(dlg_w
), main_vb
);
115 * Static labels (left dialog side, labels aligned to the right)
117 static_vb
= ws_gtk_box_new(GTK_ORIENTATION_VERTICAL
, 1, FALSE
);
118 task_title_dup
= g_strdup_printf ("%s:", task_title
);
119 tmp_lb
= gtk_label_new(task_title_dup
);
120 gtk_misc_set_alignment(GTK_MISC(tmp_lb
), 1.0f
, 0.0f
);
121 gtk_box_pack_start(GTK_BOX(static_vb
), tmp_lb
, FALSE
, TRUE
, 3);
122 tmp_lb
= gtk_label_new("Status:");
123 gtk_misc_set_alignment(GTK_MISC(tmp_lb
), 1.0f
, 0.0f
);
124 gtk_box_pack_start(GTK_BOX(static_vb
), tmp_lb
, FALSE
, TRUE
, 3);
125 tmp_lb
= gtk_label_new("Elapsed Time:");
126 gtk_misc_set_alignment(GTK_MISC(tmp_lb
), 1.0f
, 0.0f
);
127 gtk_box_pack_start(GTK_BOX(static_vb
), tmp_lb
, FALSE
, TRUE
, 3);
128 tmp_lb
= gtk_label_new("Time Left:");
129 gtk_misc_set_alignment(GTK_MISC(tmp_lb
), 1.0f
, 0.0f
);
130 gtk_box_pack_start(GTK_BOX(static_vb
), tmp_lb
, FALSE
, TRUE
, 3);
131 tmp_lb
= gtk_label_new("Progress:");
132 gtk_misc_set_alignment(GTK_MISC(tmp_lb
), 1.0f
, 0.0f
);
133 gtk_box_pack_start(GTK_BOX(static_vb
), tmp_lb
, FALSE
, TRUE
, 3);
137 * Dynamic labels (right dialog side, labels aligned to the left)
139 dynamic_vb
= ws_gtk_box_new(GTK_ORIENTATION_VERTICAL
, 1, FALSE
);
142 * Put the item_title here as a label indicating what we're
143 * doing; set its alignment and padding so it's aligned on the
146 title_lb
= gtk_label_new(item_title_dup
);
147 gtk_box_pack_start(GTK_BOX(dynamic_vb
), title_lb
, FALSE
, TRUE
, 3);
148 gtk_misc_set_alignment(GTK_MISC(title_lb
), 0.0f
, 0.0f
);
149 gtk_misc_set_padding(GTK_MISC(title_lb
), 0, 0);
151 /* same for "Status" */
152 status_lb
= gtk_label_new("");
153 gtk_box_pack_start(GTK_BOX(dynamic_vb
), status_lb
, FALSE
, TRUE
, 3);
154 gtk_misc_set_alignment(GTK_MISC(status_lb
), 0.0f
, 0.0f
);
155 gtk_misc_set_padding(GTK_MISC(status_lb
), 0, 0);
156 dlg
->status_lb
= (GtkLabel
*) status_lb
;
158 /* same for "Elapsed Time" */
159 elapsed_lb
= gtk_label_new("00:00");
160 gtk_box_pack_start(GTK_BOX(dynamic_vb
), elapsed_lb
, FALSE
, TRUE
, 3);
161 gtk_misc_set_alignment(GTK_MISC(elapsed_lb
), 0.0f
, 0.0f
);
162 gtk_misc_set_padding(GTK_MISC(elapsed_lb
), 0, 0);
163 dlg
->elapsed_lb
= (GtkLabel
*) elapsed_lb
;
165 /* same for "Time Left" */
166 time_left_lb
= gtk_label_new("--:--");
167 gtk_box_pack_start(GTK_BOX(dynamic_vb
), time_left_lb
, FALSE
, TRUE
, 3);
168 gtk_misc_set_alignment(GTK_MISC(time_left_lb
), 0.0f
, 0.0f
);
169 gtk_misc_set_padding(GTK_MISC(time_left_lb
), 0, 0);
170 dlg
->time_left_lb
= (GtkLabel
*) time_left_lb
;
173 * The progress bar (in its own horizontal box, including
176 percentage_hb
= ws_gtk_box_new(GTK_ORIENTATION_HORIZONTAL
, 1, FALSE
);
177 gtk_box_pack_start(GTK_BOX(dynamic_vb
), percentage_hb
, FALSE
, TRUE
, 3);
179 prog_bar
= gtk_progress_bar_new();
180 gtk_box_pack_start(GTK_BOX(percentage_hb
), prog_bar
, FALSE
, TRUE
, 3);
182 percentage_lb
= gtk_label_new(" 0%");
183 gtk_misc_set_alignment(GTK_MISC(percentage_lb
), 0.0f
, 0.0f
);
184 gtk_box_pack_start(GTK_BOX(percentage_hb
), percentage_lb
, FALSE
, TRUE
, 3);
185 dlg
->percentage_lb
= (GtkLabel
*) percentage_lb
;
188 * Attach a pointer to the progress bar widget to the top-level widget.
190 g_object_set_data(G_OBJECT(dlg_w
), PROG_BAR_KEY
, prog_bar
);
193 * Static and dynamic boxes are now complete
195 main_hb
= ws_gtk_box_new(GTK_ORIENTATION_HORIZONTAL
, 1, FALSE
);
196 gtk_box_pack_start(GTK_BOX(main_hb
), static_vb
, FALSE
, TRUE
, 3);
197 gtk_box_pack_start(GTK_BOX(main_hb
), dynamic_vb
, FALSE
, TRUE
, 3);
198 gtk_box_pack_start(GTK_BOX(main_vb
), main_hb
, FALSE
, TRUE
, 3);
201 bbox
= dlg_button_row_new(terminate_is_stop
? GTK_STOCK_STOP
:
202 GTK_STOCK_CANCEL
, NULL
);
203 gtk_container_add(GTK_CONTAINER(main_vb
), bbox
);
204 gtk_widget_show(bbox
);
206 cancel_bt
= (GtkWidget
*)g_object_get_data(G_OBJECT(bbox
), terminate_is_stop
? GTK_STOCK_STOP
:
208 gtk_widget_grab_default(cancel_bt
);
211 * Allow user to either click the "Cancel"/"Stop" button, or
212 * the close button on the window, to stop an operation in
215 g_signal_connect(cancel_bt
, "clicked", G_CALLBACK(stop_cb
), stop_flag
);
216 g_signal_connect(dlg_w
, "delete_event", G_CALLBACK(delete_event_cb
), stop_flag
);
218 gtk_widget_show_all(dlg_w
);
222 g_get_current_time(&dlg
->start_time
);
223 memset(&dlg
->last_time
, 0, sizeof(dlg
->last_time
));
225 g_free(task_title_dup
);
226 g_free(item_title_dup
);
232 delayed_create_progress_dlg(const gpointer top_level_window
, const gchar
*task_title
,
233 const gchar
*item_title
, gboolean terminate_is_stop
,
234 gboolean
*stop_flag
, const GTimeVal
*start_time
, gfloat progress
)
241 #define INIT_DELAY 0.1 * 1e6 /* .1 second = 0.1e6 microseconds */
242 #define MIN_DISPLAY_DEFAULT 2.0 * 1e6
244 /* Create a progress dialog, but only if it's not likely to disappear
245 * immediately, which can be disconcerting for the user.
247 * Arguments are as for create_progress_dlg(), plus:
249 * (a) A pointer to a GTimeVal structure which holds the time at which
250 * the caller started to process the data.
251 * (b) The current progress as a real number between 0 and 1.
254 g_get_current_time(&time_now
);
256 /* Get the time elapsed since the caller started processing the data */
258 delta_time
= (time_now
.tv_sec
- start_time
->tv_sec
) * 1e6
+
259 time_now
.tv_usec
- start_time
->tv_usec
;
261 /* Do nothing for the first INIT_DELAY microseconds */
263 if (delta_time
< INIT_DELAY
)
266 /* If we create the progress dialog we want it to be displayed for a
267 * minimum of MIN_DISPLAY_DEFAULT microseconds. However, if we
268 * previously estimated that the progress dialog didn't need to be
269 * created and the caller's processing is slowing down (perhaps due
270 * to the action of the operating system's scheduler on a compute-
271 * intensive task), we tail off the minimum display time such that
272 * the progress dialog will always be created after
273 * 2*MIN_DISPLAY_DEFAULT microseconds.
276 if (delta_time
<= INIT_DELAY
+ MIN_DISPLAY_DEFAULT
)
277 min_display
= MIN_DISPLAY_DEFAULT
;
279 min_display
= 2 * MIN_DISPLAY_DEFAULT
- delta_time
;
280 /* = MIN_DISPLAY_DEFAULT - (delta_time - MIN_DISPLAY_DEFAULT) */
282 /* Assuming the progress increases linearly, see if the progress
283 * dialog would be displayed for at least min_display microseconds if
287 if (progress
>= (delta_time
/ (delta_time
+ min_display
)))
290 dlg
= create_progress_dlg(top_level_window
, task_title
, item_title
, terminate_is_stop
,
294 * Flush out the dialog so we don't see an "empty" one until first update.
296 while (gtk_events_pending())
297 gtk_main_iteration();
299 /* set dialog start_time to the start of processing, not box creation */
300 dlg
->start_time
= *start_time
;
306 * Called when the dialog box is to be deleted.
307 * Set the "stop" flag to TRUE, and return TRUE - we don't want the dialog
308 * box deleted now, our caller will do so when they see that the
309 * "stop" flag is TRUE and abort the operation.
312 delete_event_cb(GtkWidget
*w _U_
, GdkEvent
*event _U_
, gpointer data
)
314 gboolean
*stop_flag
= (gboolean
*) data
;
321 * Called when the "stop this operation" button is clicked.
322 * Set the "stop" flag to TRUE; we don't have to destroy the dialog
323 * box, as our caller will do so when they see that the "stop" flag is
324 * true and abort the operation.
327 stop_cb(GtkWidget
*w _U_
, gpointer data
)
329 gboolean
*stop_flag
= (gboolean
*) data
;
335 * Update the progress information of the progress dialog box.
338 update_progress_dlg(progdlg_t
*dlg
, gfloat percentage
, const gchar
*status
)
340 GtkWidget
*dlg_w
= dlg
->dlg_w
;
346 gulong ul_percentage
;
350 /* calculate some timing values */
351 g_get_current_time(&time_now
);
353 delta_time
= (time_now
.tv_sec
- dlg
->last_time
.tv_sec
) * 1e6
+
354 time_now
.tv_usec
- dlg
->last_time
.tv_usec
;
356 /* after the first time don't update more than every 100ms */
357 if (dlg
->last_time
.tv_sec
&& delta_time
< 100*1000)
360 dlg
->last_time
= time_now
;
361 delta_time
= (time_now
.tv_sec
- dlg
->start_time
.tv_sec
) * 1e6
+
362 time_now
.tv_usec
- dlg
->start_time
.tv_usec
;
364 ul_percentage
= (gulong
) (percentage
* 100);
365 ul_elapsed
= (gulong
) (delta_time
/ 1000 / 1000);
368 g_snprintf(tmp
, sizeof(tmp
), "%lu%% of %s", ul_percentage
, dlg
->title
);
369 gtk_window_set_title(GTK_WINDOW(dlg_w
), tmp
);
371 gtk_label_set_text(dlg
->status_lb
, status
);
373 g_snprintf(tmp
, sizeof(tmp
), "%lu%%", ul_percentage
);
374 gtk_label_set_text(dlg
->percentage_lb
, tmp
);
376 g_snprintf(tmp
, sizeof(tmp
), "%02lu:%02lu", ul_elapsed
/ 60,
378 gtk_label_set_text(dlg
->elapsed_lb
, tmp
);
380 /* show "Time Left" only,
381 * if at least 5% and 3 seconds running (to get a useful estimation) */
382 if (ul_percentage
>= 5 && delta_time
>= 3 * 1e6
) {
383 ul_left
= (gulong
) ((delta_time
/ percentage
- delta_time
) / 1000 / 1000);
385 g_snprintf(tmp
, sizeof(tmp
), "%02lu:%02lu", ul_left
/ 60,
387 gtk_label_set_text(dlg
->time_left_lb
, tmp
);
390 /* update progress bar */
391 prog_bar
= (GtkWidget
*)g_object_get_data(G_OBJECT(dlg_w
), PROG_BAR_KEY
);
392 gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(prog_bar
), percentage
);
395 * Flush out the update and process any input events.
397 while (gtk_events_pending())
398 gtk_main_iteration();
402 * Destroy the progress dialog.
405 destroy_progress_dlg(progdlg_t
*dlg
)
407 GtkWidget
*dlg_w
= dlg
->dlg_w
;
409 window_destroy(GTK_WIDGET(dlg_w
));