| |
1 /** |
| |
2 * @file gtkxfer.c GTK+ File Transfer UI |
| |
3 * @ingroup pidgin |
| |
4 */ |
| |
5 |
| |
6 /* pidgin |
| |
7 * |
| |
8 * Pidgin is the legal property of its developers, whose names are too numerous |
| |
9 * to list here. Please refer to the COPYRIGHT file distributed with this |
| |
10 * source distribution. |
| |
11 * |
| |
12 * This program is free software; you can redistribute it and/or modify |
| |
13 * it under the terms of the GNU General Public License as published by |
| |
14 * the Free Software Foundation; either version 2 of the License, or |
| |
15 * (at your option) any later version. |
| |
16 * |
| |
17 * This program is distributed in the hope that it will be useful, |
| |
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| |
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| |
20 * GNU General Public License for more details. |
| |
21 * |
| |
22 * You should have received a copy of the GNU General Public License |
| |
23 * along with this program; if not, write to the Free Software |
| |
24 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA |
| |
25 */ |
| |
26 #include "internal.h" |
| |
27 #include "pidgin.h" |
| |
28 |
| |
29 #include "debug.h" |
| |
30 #include "notify.h" |
| |
31 #include "xfer.h" |
| |
32 #include "protocol.h" |
| |
33 #include "util.h" |
| |
34 |
| |
35 #include "gtkxfer.h" |
| |
36 #include "prefs.h" |
| |
37 #include "pidginstock.h" |
| |
38 #include "gtkutils.h" |
| |
39 |
| |
40 /* the maximum size of files we will try to make a thumbnail for */ |
| |
41 #define PIDGIN_XFER_MAX_SIZE_IMAGE_THUMBNAIL 10 * 1024 * 1024 |
| |
42 |
| |
43 struct _PidginXferDialog |
| |
44 { |
| |
45 gboolean keep_open; |
| |
46 gboolean auto_clear; |
| |
47 |
| |
48 gint num_transfers; |
| |
49 |
| |
50 PurpleXfer *selected_xfer; |
| |
51 |
| |
52 GtkWidget *window; |
| |
53 GtkWidget *tree; |
| |
54 GtkListStore *model; |
| |
55 |
| |
56 GtkWidget *expander; |
| |
57 |
| |
58 GtkWidget *table; |
| |
59 |
| |
60 GtkWidget *local_user_desc_label; |
| |
61 GtkWidget *local_user_label; |
| |
62 GtkWidget *remote_user_desc_label; |
| |
63 GtkWidget *remote_user_label; |
| |
64 GtkWidget *protocol_label; |
| |
65 GtkWidget *filename_label; |
| |
66 GtkWidget *localfile_label; |
| |
67 GtkWidget *status_label; |
| |
68 GtkWidget *speed_label; |
| |
69 GtkWidget *time_elapsed_label; |
| |
70 GtkWidget *time_remaining_label; |
| |
71 |
| |
72 GtkWidget *progress; |
| |
73 |
| |
74 /* Buttons */ |
| |
75 GtkWidget *open_button; |
| |
76 GtkWidget *remove_button; |
| |
77 GtkWidget *stop_button; |
| |
78 GtkWidget *close_button; |
| |
79 }; |
| |
80 |
| |
81 typedef struct |
| |
82 { |
| |
83 GtkTreeIter iter; |
| |
84 time_t last_updated_time; |
| |
85 gboolean in_list; |
| |
86 |
| |
87 char *name; |
| |
88 |
| |
89 } PidginXferUiData; |
| |
90 |
| |
91 static PidginXferDialog *xfer_dialog = NULL; |
| |
92 |
| |
93 enum |
| |
94 { |
| |
95 COLUMN_STATUS = 0, |
| |
96 COLUMN_PROGRESS, |
| |
97 COLUMN_FILENAME, |
| |
98 COLUMN_SIZE, |
| |
99 COLUMN_REMAINING, |
| |
100 COLUMN_DATA, |
| |
101 NUM_COLUMNS |
| |
102 }; |
| |
103 |
| |
104 |
| |
105 /************************************************************************** |
| |
106 * Utility Functions |
| |
107 **************************************************************************/ |
| |
108 static void |
| |
109 get_xfer_info_strings(PurpleXfer *xfer, char **kbsec, char **time_elapsed, |
| |
110 char **time_remaining) |
| |
111 { |
| |
112 double kb_sent, kb_rem; |
| |
113 double kbps = 0.0; |
| |
114 time_t elapsed, now; |
| |
115 |
| |
116 now = purple_xfer_get_end_time(xfer); |
| |
117 if (now == 0) |
| |
118 now = time(NULL); |
| |
119 |
| |
120 kb_sent = purple_xfer_get_bytes_sent(xfer) / 1024.0; |
| |
121 kb_rem = purple_xfer_get_bytes_remaining(xfer) / 1024.0; |
| |
122 elapsed = purple_xfer_get_start_time(xfer); |
| |
123 if (elapsed > 0) |
| |
124 elapsed = now - elapsed; |
| |
125 else |
| |
126 elapsed = 0; |
| |
127 kbps = (elapsed > 0 ? (kb_sent / elapsed) : 0); |
| |
128 |
| |
129 if (kbsec != NULL) { |
| |
130 *kbsec = g_strdup_printf(_("%.2f KiB/s"), kbps); |
| |
131 } |
| |
132 |
| |
133 if (time_elapsed != NULL) |
| |
134 { |
| |
135 int h, m, s; |
| |
136 int secs_elapsed; |
| |
137 |
| |
138 if (purple_xfer_get_start_time(xfer) > 0) |
| |
139 { |
| |
140 secs_elapsed = now - purple_xfer_get_start_time(xfer); |
| |
141 |
| |
142 h = secs_elapsed / 3600; |
| |
143 m = (secs_elapsed % 3600) / 60; |
| |
144 s = secs_elapsed % 60; |
| |
145 |
| |
146 *time_elapsed = g_strdup_printf("%d:%02d:%02d", h, m, s); |
| |
147 } |
| |
148 else |
| |
149 { |
| |
150 *time_elapsed = g_strdup(_("Not started")); |
| |
151 } |
| |
152 } |
| |
153 |
| |
154 if (time_remaining != NULL) { |
| |
155 if (purple_xfer_is_completed(xfer)) { |
| |
156 *time_remaining = g_strdup(_("Finished")); |
| |
157 } |
| |
158 else if (purple_xfer_is_cancelled(xfer)) { |
| |
159 *time_remaining = g_strdup(_("Cancelled")); |
| |
160 } |
| |
161 else if (purple_xfer_get_size(xfer) == 0 || (kb_sent > 0 && kbps < 0.001)) { |
| |
162 *time_remaining = g_strdup(_("Unknown")); |
| |
163 } |
| |
164 else if (kb_sent <= 0) { |
| |
165 *time_remaining = g_strdup(_("Waiting for transfer to begin")); |
| |
166 } |
| |
167 else { |
| |
168 int h, m, s; |
| |
169 int secs_remaining; |
| |
170 |
| |
171 secs_remaining = (int)(kb_rem / kbps); |
| |
172 |
| |
173 h = secs_remaining / 3600; |
| |
174 m = (secs_remaining % 3600) / 60; |
| |
175 s = secs_remaining % 60; |
| |
176 |
| |
177 *time_remaining = g_strdup_printf("%d:%02d:%02d", h, m, s); |
| |
178 } |
| |
179 } |
| |
180 } |
| |
181 |
| |
182 static void |
| |
183 update_title_progress(PidginXferDialog *dialog) |
| |
184 { |
| |
185 gboolean valid; |
| |
186 GtkTreeIter iter; |
| |
187 int num_active_xfers = 0; |
| |
188 guint64 total_bytes_xferred = 0; |
| |
189 guint64 total_file_size = 0; |
| |
190 |
| |
191 if (dialog->window == NULL) |
| |
192 return; |
| |
193 |
| |
194 valid = gtk_tree_model_get_iter_first(GTK_TREE_MODEL(dialog->model), &iter); |
| |
195 |
| |
196 /* Find all active transfers */ |
| |
197 while (valid) { |
| |
198 GValue val; |
| |
199 PurpleXfer *xfer = NULL; |
| |
200 |
| |
201 val.g_type = 0; |
| |
202 gtk_tree_model_get_value(GTK_TREE_MODEL(dialog->model), |
| |
203 &iter, COLUMN_DATA, &val); |
| |
204 |
| |
205 xfer = g_value_get_pointer(&val); |
| |
206 if (purple_xfer_get_status(xfer) == PURPLE_XFER_STATUS_STARTED) { |
| |
207 num_active_xfers++; |
| |
208 total_bytes_xferred += purple_xfer_get_bytes_sent(xfer); |
| |
209 total_file_size += purple_xfer_get_size(xfer); |
| |
210 } |
| |
211 |
| |
212 valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(dialog->model), &iter); |
| |
213 } |
| |
214 |
| |
215 /* Update the title */ |
| |
216 if (num_active_xfers > 0) |
| |
217 { |
| |
218 gchar *title; |
| |
219 int total_pct = 0; |
| |
220 |
| |
221 if (total_file_size > 0) { |
| |
222 total_pct = 100 * total_bytes_xferred / total_file_size; |
| |
223 } |
| |
224 |
| |
225 title = g_strdup_printf(ngettext("File Transfers - %d%% of %d file", |
| |
226 "File Transfers - %d%% of %d files", |
| |
227 num_active_xfers), |
| |
228 total_pct, num_active_xfers); |
| |
229 gtk_window_set_title(GTK_WINDOW(dialog->window), title); |
| |
230 g_free(title); |
| |
231 } else { |
| |
232 gtk_window_set_title(GTK_WINDOW(dialog->window), _("File Transfers")); |
| |
233 } |
| |
234 } |
| |
235 |
| |
236 static void |
| |
237 update_detailed_info(PidginXferDialog *dialog, PurpleXfer *xfer) |
| |
238 { |
| |
239 PidginXferUiData *data; |
| |
240 char *kbsec, *time_elapsed, *time_remaining; |
| |
241 char *status, *utf8; |
| |
242 |
| |
243 if (dialog == NULL || xfer == NULL) |
| |
244 return; |
| |
245 |
| |
246 data = purple_xfer_get_ui_data(xfer); |
| |
247 |
| |
248 get_xfer_info_strings(xfer, &kbsec, &time_elapsed, &time_remaining); |
| |
249 |
| |
250 status = g_strdup_printf("%d%% (%" G_GOFFSET_FORMAT " of %" G_GOFFSET_FORMAT " bytes)", |
| |
251 (int)(purple_xfer_get_progress(xfer)*100), |
| |
252 purple_xfer_get_bytes_sent(xfer), |
| |
253 purple_xfer_get_size(xfer)); |
| |
254 |
| |
255 if (purple_xfer_is_completed(xfer)) { |
| |
256 |
| |
257 GdkPixbuf *pixbuf = NULL; |
| |
258 |
| |
259 pixbuf = gtk_widget_render_icon(xfer_dialog->window, |
| |
260 PIDGIN_STOCK_FILE_DONE, |
| |
261 GTK_ICON_SIZE_MENU, NULL); |
| |
262 |
| |
263 gtk_list_store_set(GTK_LIST_STORE(xfer_dialog->model), &data->iter, |
| |
264 COLUMN_STATUS, pixbuf, |
| |
265 -1); |
| |
266 |
| |
267 g_object_unref(pixbuf); |
| |
268 } |
| |
269 |
| |
270 if (purple_xfer_get_xfer_type(xfer) == PURPLE_XFER_TYPE_RECEIVE) { |
| |
271 gtk_label_set_markup(GTK_LABEL(dialog->local_user_desc_label), |
| |
272 _("<b>Receiving As:</b>")); |
| |
273 gtk_label_set_markup(GTK_LABEL(dialog->remote_user_desc_label), |
| |
274 _("<b>Receiving From:</b>")); |
| |
275 } |
| |
276 else { |
| |
277 gtk_label_set_markup(GTK_LABEL(dialog->remote_user_desc_label), |
| |
278 _("<b>Sending To:</b>")); |
| |
279 gtk_label_set_markup(GTK_LABEL(dialog->local_user_desc_label), |
| |
280 _("<b>Sending As:</b>")); |
| |
281 } |
| |
282 |
| |
283 gtk_label_set_text(GTK_LABEL(dialog->local_user_label), |
| |
284 purple_account_get_username(purple_xfer_get_account(xfer))); |
| |
285 gtk_label_set_text(GTK_LABEL(dialog->remote_user_label), purple_xfer_get_remote_user(xfer)); |
| |
286 gtk_label_set_text(GTK_LABEL(dialog->protocol_label), |
| |
287 purple_account_get_protocol_name(purple_xfer_get_account(xfer))); |
| |
288 |
| |
289 if (purple_xfer_get_xfer_type(xfer) == PURPLE_XFER_TYPE_RECEIVE) { |
| |
290 gtk_label_set_text(GTK_LABEL(dialog->filename_label), |
| |
291 purple_xfer_get_filename(xfer)); |
| |
292 } else { |
| |
293 char *tmp; |
| |
294 |
| |
295 tmp = g_path_get_basename(purple_xfer_get_local_filename(xfer)); |
| |
296 utf8 = g_filename_to_utf8(tmp, -1, NULL, NULL, NULL); |
| |
297 g_free(tmp); |
| |
298 |
| |
299 gtk_label_set_text(GTK_LABEL(dialog->filename_label), utf8); |
| |
300 g_free(utf8); |
| |
301 } |
| |
302 |
| |
303 utf8 = g_filename_to_utf8((purple_xfer_get_local_filename(xfer)), -1, NULL, NULL, NULL); |
| |
304 gtk_label_set_text(GTK_LABEL(dialog->localfile_label), utf8); |
| |
305 g_free(utf8); |
| |
306 |
| |
307 gtk_label_set_text(GTK_LABEL(dialog->status_label), status); |
| |
308 |
| |
309 gtk_label_set_text(GTK_LABEL(dialog->speed_label), kbsec); |
| |
310 gtk_label_set_text(GTK_LABEL(dialog->time_elapsed_label), time_elapsed); |
| |
311 gtk_label_set_text(GTK_LABEL(dialog->time_remaining_label), |
| |
312 time_remaining); |
| |
313 |
| |
314 gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(dialog->progress), |
| |
315 purple_xfer_get_progress(xfer)); |
| |
316 |
| |
317 g_free(kbsec); |
| |
318 g_free(time_elapsed); |
| |
319 g_free(time_remaining); |
| |
320 g_free(status); |
| |
321 } |
| |
322 |
| |
323 static void |
| |
324 update_buttons(PidginXferDialog *dialog, PurpleXfer *xfer) |
| |
325 { |
| |
326 if (dialog->selected_xfer == NULL) { |
| |
327 gtk_widget_set_sensitive(dialog->expander, FALSE); |
| |
328 gtk_widget_set_sensitive(dialog->open_button, FALSE); |
| |
329 gtk_widget_set_sensitive(dialog->stop_button, FALSE); |
| |
330 |
| |
331 gtk_widget_show(dialog->stop_button); |
| |
332 gtk_widget_hide(dialog->remove_button); |
| |
333 |
| |
334 return; |
| |
335 } |
| |
336 |
| |
337 if (dialog->selected_xfer != xfer) |
| |
338 return; |
| |
339 |
| |
340 if (purple_xfer_is_completed(xfer)) { |
| |
341 gtk_widget_hide(dialog->stop_button); |
| |
342 gtk_widget_show(dialog->remove_button); |
| |
343 |
| |
344 #ifdef _WIN32 |
| |
345 /* If using Win32... */ |
| |
346 if (purple_xfer_get_xfer_type(xfer) == PURPLE_XFER_TYPE_RECEIVE) { |
| |
347 gtk_widget_set_sensitive(dialog->open_button, TRUE); |
| |
348 } else { |
| |
349 gtk_widget_set_sensitive(dialog->open_button, FALSE); |
| |
350 } |
| |
351 #else |
| |
352 if (purple_xfer_get_xfer_type(xfer) == PURPLE_XFER_TYPE_RECEIVE) { |
| |
353 gtk_widget_set_sensitive(dialog->open_button, TRUE); |
| |
354 } else { |
| |
355 gtk_widget_set_sensitive (dialog->open_button, FALSE); |
| |
356 } |
| |
357 #endif |
| |
358 |
| |
359 gtk_widget_set_sensitive(dialog->remove_button, TRUE); |
| |
360 } else if (purple_xfer_is_cancelled(xfer)) { |
| |
361 gtk_widget_hide(dialog->stop_button); |
| |
362 gtk_widget_show(dialog->remove_button); |
| |
363 |
| |
364 gtk_widget_set_sensitive(dialog->open_button, FALSE); |
| |
365 |
| |
366 gtk_widget_set_sensitive(dialog->remove_button, TRUE); |
| |
367 } else { |
| |
368 gtk_widget_show(dialog->stop_button); |
| |
369 gtk_widget_hide(dialog->remove_button); |
| |
370 |
| |
371 gtk_widget_set_sensitive(dialog->open_button, FALSE); |
| |
372 gtk_widget_set_sensitive(dialog->stop_button, TRUE); |
| |
373 } |
| |
374 } |
| |
375 |
| |
376 static void |
| |
377 ensure_row_selected(PidginXferDialog *dialog) |
| |
378 { |
| |
379 GtkTreeIter iter; |
| |
380 GtkTreeSelection *selection; |
| |
381 |
| |
382 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(dialog->tree)); |
| |
383 |
| |
384 if (gtk_tree_selection_get_selected(selection, NULL, &iter)) |
| |
385 return; |
| |
386 |
| |
387 if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(dialog->model), &iter)) |
| |
388 gtk_tree_selection_select_iter(selection, &iter); |
| |
389 } |
| |
390 |
| |
391 /************************************************************************** |
| |
392 * Callbacks |
| |
393 **************************************************************************/ |
| |
394 static gint |
| |
395 delete_win_cb(GtkWidget *w, GdkEventAny *e, gpointer d) |
| |
396 { |
| |
397 PidginXferDialog *dialog; |
| |
398 |
| |
399 dialog = (PidginXferDialog *)d; |
| |
400 |
| |
401 pidgin_xfer_dialog_hide(dialog); |
| |
402 |
| |
403 return TRUE; |
| |
404 } |
| |
405 |
| |
406 static void |
| |
407 toggle_keep_open_cb(GtkWidget *w, PidginXferDialog *dialog) |
| |
408 { |
| |
409 dialog->keep_open = !dialog->keep_open; |
| |
410 purple_prefs_set_bool(PIDGIN_PREFS_ROOT "/filetransfer/keep_open", |
| |
411 dialog->keep_open); |
| |
412 } |
| |
413 |
| |
414 static void |
| |
415 toggle_clear_finished_cb(GtkWidget *w, PidginXferDialog *dialog) |
| |
416 { |
| |
417 dialog->auto_clear = !dialog->auto_clear; |
| |
418 purple_prefs_set_bool(PIDGIN_PREFS_ROOT "/filetransfer/clear_finished", |
| |
419 dialog->auto_clear); |
| |
420 } |
| |
421 |
| |
422 static void |
| |
423 selection_changed_cb(GtkTreeSelection *selection, PidginXferDialog *dialog) |
| |
424 { |
| |
425 GtkTreeIter iter; |
| |
426 PurpleXfer *xfer = NULL; |
| |
427 |
| |
428 if (gtk_tree_selection_get_selected(selection, NULL, &iter)) { |
| |
429 GValue val; |
| |
430 |
| |
431 gtk_widget_set_sensitive(dialog->expander, TRUE); |
| |
432 |
| |
433 val.g_type = 0; |
| |
434 gtk_tree_model_get_value(GTK_TREE_MODEL(dialog->model), |
| |
435 &iter, COLUMN_DATA, &val); |
| |
436 |
| |
437 xfer = g_value_get_pointer(&val); |
| |
438 |
| |
439 update_detailed_info(dialog, xfer); |
| |
440 |
| |
441 dialog->selected_xfer = xfer; |
| |
442 } |
| |
443 else { |
| |
444 gtk_expander_set_expanded(GTK_EXPANDER(dialog->expander), |
| |
445 FALSE); |
| |
446 |
| |
447 gtk_widget_set_sensitive(dialog->expander, FALSE); |
| |
448 |
| |
449 dialog->selected_xfer = NULL; |
| |
450 } |
| |
451 |
| |
452 update_buttons(dialog, xfer); |
| |
453 } |
| |
454 |
| |
455 static void |
| |
456 open_button_cb(GtkButton *button, PidginXferDialog *dialog) |
| |
457 { |
| |
458 #ifdef _WIN32 |
| |
459 /* If using Win32... */ |
| |
460 int code; |
| |
461 wchar_t *wc_filename = g_utf8_to_utf16( |
| |
462 purple_xfer_get_local_filename( |
| |
463 dialog->selected_xfer), |
| |
464 -1, NULL, NULL, NULL); |
| |
465 |
| |
466 code = (int) ShellExecuteW(NULL, NULL, wc_filename, NULL, NULL, |
| |
467 SW_SHOW); |
| |
468 |
| |
469 g_free(wc_filename); |
| |
470 |
| |
471 if (code == SE_ERR_ASSOCINCOMPLETE || code == SE_ERR_NOASSOC) |
| |
472 { |
| |
473 purple_notify_error(dialog, NULL, |
| |
474 _("There is no application configured to open this type of file."), NULL); |
| |
475 } |
| |
476 else if (code < 32) |
| |
477 { |
| |
478 purple_notify_error(dialog, NULL, |
| |
479 _("An error occurred while opening the file."), NULL); |
| |
480 purple_debug_warning("ft", "filename: %s; code: %d\n", |
| |
481 purple_xfer_get_local_filename(dialog->selected_xfer), code); |
| |
482 } |
| |
483 #else |
| |
484 const char *filename = purple_xfer_get_local_filename(dialog->selected_xfer); |
| |
485 char *command = NULL; |
| |
486 char *tmp = NULL; |
| |
487 GError *error = NULL; |
| |
488 |
| |
489 if (purple_running_gnome()) |
| |
490 { |
| |
491 char *escaped = g_shell_quote(filename); |
| |
492 command = g_strdup_printf("gnome-open %s", escaped); |
| |
493 g_free(escaped); |
| |
494 } |
| |
495 else if (purple_running_kde()) |
| |
496 { |
| |
497 char *escaped = g_shell_quote(filename); |
| |
498 |
| |
499 if (purple_str_has_suffix(filename, ".desktop")) |
| |
500 command = g_strdup_printf("kfmclient openURL %s 'text/plain'", escaped); |
| |
501 else |
| |
502 command = g_strdup_printf("kfmclient openURL %s", escaped); |
| |
503 g_free(escaped); |
| |
504 } |
| |
505 else |
| |
506 { |
| |
507 purple_notify_uri(NULL, filename); |
| |
508 return; |
| |
509 } |
| |
510 |
| |
511 if (purple_program_is_valid(command)) |
| |
512 { |
| |
513 gint exit_status; |
| |
514 if (!g_spawn_command_line_sync(command, NULL, NULL, &exit_status, &error)) |
| |
515 { |
| |
516 tmp = g_strdup_printf(_("Error launching %s: %s"), |
| |
517 purple_xfer_get_local_filename(dialog->selected_xfer), |
| |
518 error->message); |
| |
519 purple_notify_error(dialog, NULL, _("Unable to open file."), tmp); |
| |
520 g_free(tmp); |
| |
521 g_error_free(error); |
| |
522 } |
| |
523 if (exit_status != 0) |
| |
524 { |
| |
525 char *primary = g_strdup_printf(_("Error running %s"), command); |
| |
526 char *secondary = g_strdup_printf(_("Process returned error code %d"), |
| |
527 exit_status); |
| |
528 purple_notify_error(dialog, NULL, primary, secondary); |
| |
529 g_free(tmp); |
| |
530 } |
| |
531 } |
| |
532 #endif |
| |
533 } |
| |
534 |
| |
535 static void |
| |
536 remove_button_cb(GtkButton *button, PidginXferDialog *dialog) |
| |
537 { |
| |
538 pidgin_xfer_dialog_remove_xfer(dialog, dialog->selected_xfer); |
| |
539 } |
| |
540 |
| |
541 static void |
| |
542 stop_button_cb(GtkButton *button, PidginXferDialog *dialog) |
| |
543 { |
| |
544 purple_xfer_cancel_local(dialog->selected_xfer); |
| |
545 } |
| |
546 |
| |
547 static void |
| |
548 close_button_cb(GtkButton *button, PidginXferDialog *dialog) |
| |
549 { |
| |
550 pidgin_xfer_dialog_hide(dialog); |
| |
551 } |
| |
552 |
| |
553 |
| |
554 /************************************************************************** |
| |
555 * Dialog Building Functions |
| |
556 **************************************************************************/ |
| |
557 static GtkWidget * |
| |
558 setup_tree(PidginXferDialog *dialog) |
| |
559 { |
| |
560 GtkWidget *tree; |
| |
561 GtkListStore *model; |
| |
562 GtkCellRenderer *renderer; |
| |
563 GtkTreeViewColumn *column; |
| |
564 GtkTreeSelection *selection; |
| |
565 |
| |
566 /* Build the tree model */ |
| |
567 /* Transfer type, Progress Bar, Filename, Size, Remaining */ |
| |
568 model = gtk_list_store_new(NUM_COLUMNS, GDK_TYPE_PIXBUF, G_TYPE_INT, |
| |
569 G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING, |
| |
570 G_TYPE_POINTER); |
| |
571 dialog->model = model; |
| |
572 |
| |
573 /* Create the treeview */ |
| |
574 dialog->tree = tree = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model)); |
| |
575 gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(tree), TRUE); |
| |
576 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree)); |
| |
577 /* gtk_tree_selection_set_mode(selection, GTK_SELECTION_MULTIPLE); */ |
| |
578 |
| |
579 gtk_widget_show(tree); |
| |
580 |
| |
581 g_signal_connect(G_OBJECT(selection), "changed", |
| |
582 G_CALLBACK(selection_changed_cb), dialog); |
| |
583 |
| |
584 g_object_unref(G_OBJECT(model)); |
| |
585 |
| |
586 |
| |
587 /* Columns */ |
| |
588 |
| |
589 /* Transfer Type column */ |
| |
590 renderer = gtk_cell_renderer_pixbuf_new(); |
| |
591 column = gtk_tree_view_column_new_with_attributes(NULL, renderer, |
| |
592 "pixbuf", COLUMN_STATUS, NULL); |
| |
593 gtk_tree_view_column_set_sizing(GTK_TREE_VIEW_COLUMN(column), |
| |
594 GTK_TREE_VIEW_COLUMN_FIXED); |
| |
595 gtk_tree_view_column_set_fixed_width(GTK_TREE_VIEW_COLUMN(column), 25); |
| |
596 gtk_tree_view_column_set_resizable(GTK_TREE_VIEW_COLUMN(column), TRUE); |
| |
597 gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column); |
| |
598 |
| |
599 /* Progress bar column */ |
| |
600 renderer = gtk_cell_renderer_progress_new(); |
| |
601 column = gtk_tree_view_column_new_with_attributes(_("Progress"), renderer, |
| |
602 "value", COLUMN_PROGRESS, NULL); |
| |
603 gtk_tree_view_column_set_resizable(GTK_TREE_VIEW_COLUMN(column), TRUE); |
| |
604 gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column); |
| |
605 |
| |
606 /* Filename column */ |
| |
607 renderer = gtk_cell_renderer_text_new(); |
| |
608 column = gtk_tree_view_column_new_with_attributes(_("Filename"), renderer, |
| |
609 "text", COLUMN_FILENAME, NULL); |
| |
610 gtk_tree_view_column_set_resizable(GTK_TREE_VIEW_COLUMN(column), TRUE); |
| |
611 gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column); |
| |
612 |
| |
613 /* File Size column */ |
| |
614 renderer = gtk_cell_renderer_text_new(); |
| |
615 column = gtk_tree_view_column_new_with_attributes(_("Size"), renderer, |
| |
616 "text", COLUMN_SIZE, NULL); |
| |
617 gtk_tree_view_column_set_resizable(GTK_TREE_VIEW_COLUMN(column), TRUE); |
| |
618 gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column); |
| |
619 |
| |
620 /* Bytes Remaining column */ |
| |
621 renderer = gtk_cell_renderer_text_new(); |
| |
622 column = gtk_tree_view_column_new_with_attributes(_("Remaining"), |
| |
623 renderer, "text", COLUMN_REMAINING, NULL); |
| |
624 gtk_tree_view_column_set_resizable(GTK_TREE_VIEW_COLUMN(column), TRUE); |
| |
625 gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column); |
| |
626 |
| |
627 gtk_tree_view_columns_autosize(GTK_TREE_VIEW(tree)); |
| |
628 |
| |
629 gtk_widget_show(tree); |
| |
630 |
| |
631 return tree; |
| |
632 } |
| |
633 |
| |
634 static GtkWidget * |
| |
635 make_info_table(PidginXferDialog *dialog) |
| |
636 { |
| |
637 GtkWidget *table; |
| |
638 GtkWidget *label; |
| |
639 gsize i; |
| |
640 |
| |
641 struct |
| |
642 { |
| |
643 GtkWidget **desc_label; |
| |
644 GtkWidget **val_label; |
| |
645 const char *desc; |
| |
646 |
| |
647 } labels[] = |
| |
648 { |
| |
649 { &dialog->local_user_desc_label, &dialog->local_user_label, NULL }, |
| |
650 { &dialog->remote_user_desc_label, &dialog->remote_user_label, NULL }, |
| |
651 { &label, &dialog->protocol_label, _("Protocol:") }, |
| |
652 { &label, &dialog->filename_label, _("Filename:") }, |
| |
653 { &label, &dialog->localfile_label, _("Local File:") }, |
| |
654 { &label, &dialog->status_label, _("Status:") }, |
| |
655 { &label, &dialog->speed_label, _("Speed:") }, |
| |
656 { &label, &dialog->time_elapsed_label, _("Time Elapsed:") }, |
| |
657 { &label, &dialog->time_remaining_label, _("Time Remaining:") } |
| |
658 }; |
| |
659 |
| |
660 /* Setup the initial table */ |
| |
661 dialog->table = table = gtk_table_new(G_N_ELEMENTS(labels) + 1, 2, FALSE); |
| |
662 gtk_table_set_row_spacings(GTK_TABLE(table), PIDGIN_HIG_BOX_SPACE); |
| |
663 gtk_table_set_col_spacings(GTK_TABLE(table), PIDGIN_HIG_BOX_SPACE); |
| |
664 |
| |
665 /* Setup the labels */ |
| |
666 for (i = 0; i < G_N_ELEMENTS(labels); i++) { |
| |
667 GtkWidget *label; |
| |
668 char buf[256]; |
| |
669 |
| |
670 g_snprintf(buf, sizeof(buf), "<b>%s</b>", |
| |
671 labels[i].desc != NULL ? labels[i].desc : ""); |
| |
672 |
| |
673 *labels[i].desc_label = label = gtk_label_new(NULL); |
| |
674 gtk_label_set_markup(GTK_LABEL(label), buf); |
| |
675 gtk_label_set_justify(GTK_LABEL(label), GTK_JUSTIFY_RIGHT); |
| |
676 gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); |
| |
677 gtk_table_attach(GTK_TABLE(table), label, 0, 1, i, i + 1, |
| |
678 GTK_FILL, 0, 0, 0); |
| |
679 gtk_widget_show(label); |
| |
680 |
| |
681 *labels[i].val_label = label = gtk_label_new(NULL); |
| |
682 gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); |
| |
683 gtk_table_attach(GTK_TABLE(table), label, 1, 2, i, i + 1, |
| |
684 GTK_FILL | GTK_EXPAND, 0, 0, 0); |
| |
685 gtk_widget_show(label); |
| |
686 } |
| |
687 |
| |
688 /* Setup the progress bar */ |
| |
689 dialog->progress = gtk_progress_bar_new(); |
| |
690 gtk_table_attach(GTK_TABLE(table), dialog->progress, |
| |
691 0, 2, |
| |
692 G_N_ELEMENTS(labels), G_N_ELEMENTS(labels) + 1, |
| |
693 GTK_FILL, GTK_FILL, 0, 0); |
| |
694 gtk_widget_show(dialog->progress); |
| |
695 |
| |
696 return table; |
| |
697 } |
| |
698 |
| |
699 PidginXferDialog * |
| |
700 pidgin_xfer_dialog_new(void) |
| |
701 { |
| |
702 PidginXferDialog *dialog; |
| |
703 GtkWidget *window; |
| |
704 GtkWidget *vbox; |
| |
705 GtkWidget *expander; |
| |
706 GtkWidget *alignment; |
| |
707 GtkWidget *table; |
| |
708 GtkWidget *checkbox; |
| |
709 GtkWidget *bbox; |
| |
710 |
| |
711 dialog = g_new0(PidginXferDialog, 1); |
| |
712 dialog->keep_open = |
| |
713 purple_prefs_get_bool(PIDGIN_PREFS_ROOT "/filetransfer/keep_open"); |
| |
714 dialog->auto_clear = |
| |
715 purple_prefs_get_bool(PIDGIN_PREFS_ROOT "/filetransfer/clear_finished"); |
| |
716 |
| |
717 /* Create the window. */ |
| |
718 #if GTK_CHECK_VERSION(3,0,0) |
| |
719 dialog->window = window = pidgin_create_dialog(_("File Transfers"), 0, "file transfer", TRUE); |
| |
720 #else |
| |
721 dialog->window = window = pidgin_create_dialog(_("File Transfers"), PIDGIN_HIG_BORDER, "file transfer", TRUE); |
| |
722 #endif |
| |
723 gtk_window_set_default_size(GTK_WINDOW(window), 450, 250); |
| |
724 |
| |
725 g_signal_connect(G_OBJECT(window), "delete_event", |
| |
726 G_CALLBACK(delete_win_cb), dialog); |
| |
727 |
| |
728 /* Create the main vbox for top half of the window. */ |
| |
729 vbox = pidgin_dialog_get_vbox_with_properties(GTK_DIALOG(window), FALSE, PIDGIN_HIG_BORDER); |
| |
730 |
| |
731 /* Setup the listbox */ |
| |
732 gtk_box_pack_start(GTK_BOX(vbox), |
| |
733 pidgin_make_scrollable(setup_tree(dialog), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC, GTK_SHADOW_IN, -1, 140), |
| |
734 TRUE, TRUE, 0); |
| |
735 |
| |
736 /* "Close this window when all transfers finish" */ |
| |
737 checkbox = gtk_check_button_new_with_mnemonic( |
| |
738 _("Close this window when all transfers _finish")); |
| |
739 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbox), |
| |
740 !dialog->keep_open); |
| |
741 g_signal_connect(G_OBJECT(checkbox), "toggled", |
| |
742 G_CALLBACK(toggle_keep_open_cb), dialog); |
| |
743 gtk_box_pack_start(GTK_BOX(vbox), checkbox, FALSE, FALSE, 0); |
| |
744 gtk_widget_show(checkbox); |
| |
745 |
| |
746 /* "Clear finished transfers" */ |
| |
747 checkbox = gtk_check_button_new_with_mnemonic( |
| |
748 _("C_lear finished transfers")); |
| |
749 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbox), |
| |
750 dialog->auto_clear); |
| |
751 g_signal_connect(G_OBJECT(checkbox), "toggled", |
| |
752 G_CALLBACK(toggle_clear_finished_cb), dialog); |
| |
753 gtk_box_pack_start(GTK_BOX(vbox), checkbox, FALSE, FALSE, 0); |
| |
754 gtk_widget_show(checkbox); |
| |
755 |
| |
756 /* "Download Details" arrow */ |
| |
757 expander = gtk_expander_new_with_mnemonic(_("File transfer _details")); |
| |
758 dialog->expander = expander; |
| |
759 gtk_box_pack_start(GTK_BOX(vbox), expander, FALSE, FALSE, 0); |
| |
760 gtk_widget_show(expander); |
| |
761 |
| |
762 gtk_widget_set_sensitive(expander, FALSE); |
| |
763 |
| |
764 /* Small indent make table fall under GtkExpander's label */ |
| |
765 alignment = gtk_alignment_new(1, 0, 1, 1); |
| |
766 gtk_alignment_set_padding(GTK_ALIGNMENT(alignment), 0, 0, 20, 0); |
| |
767 gtk_container_add(GTK_CONTAINER(expander), alignment); |
| |
768 gtk_widget_show(alignment); |
| |
769 |
| |
770 /* The table of information. */ |
| |
771 table = make_info_table(dialog); |
| |
772 gtk_container_add(GTK_CONTAINER(alignment), table); |
| |
773 gtk_widget_show(table); |
| |
774 |
| |
775 bbox = pidgin_dialog_get_action_area(GTK_DIALOG(window)); |
| |
776 |
| |
777 #define ADD_BUTTON(b, label, callback, callbackdata) do { \ |
| |
778 GtkWidget *button = gtk_button_new_from_stock(label); \ |
| |
779 gtk_box_pack_start(GTK_BOX(bbox), button, FALSE, FALSE, 0); \ |
| |
780 g_signal_connect(G_OBJECT(button), "clicked", callback, callbackdata); \ |
| |
781 gtk_widget_show(button); \ |
| |
782 b = button; \ |
| |
783 } while (0) |
| |
784 |
| |
785 /* Open button */ |
| |
786 ADD_BUTTON(dialog->open_button, GTK_STOCK_OPEN, G_CALLBACK(open_button_cb), dialog); |
| |
787 gtk_widget_set_sensitive(dialog->open_button, FALSE); |
| |
788 |
| |
789 /* Remove button */ |
| |
790 ADD_BUTTON(dialog->remove_button, GTK_STOCK_REMOVE, G_CALLBACK(remove_button_cb), dialog); |
| |
791 gtk_widget_hide(dialog->remove_button); |
| |
792 |
| |
793 /* Stop button */ |
| |
794 ADD_BUTTON(dialog->stop_button, GTK_STOCK_STOP, G_CALLBACK(stop_button_cb), dialog); |
| |
795 gtk_widget_set_sensitive(dialog->stop_button, FALSE); |
| |
796 |
| |
797 /* Close button */ |
| |
798 ADD_BUTTON(dialog->close_button, GTK_STOCK_CLOSE, G_CALLBACK(close_button_cb), dialog); |
| |
799 |
| |
800 #undef ADD_BUTTON |
| |
801 |
| |
802 #ifdef _WIN32 |
| |
803 g_signal_connect(G_OBJECT(dialog->window), "show", |
| |
804 G_CALLBACK(winpidgin_ensure_onscreen), dialog->window); |
| |
805 #endif |
| |
806 |
| |
807 return dialog; |
| |
808 } |
| |
809 |
| |
810 void |
| |
811 pidgin_xfer_dialog_destroy(PidginXferDialog *dialog) |
| |
812 { |
| |
813 g_return_if_fail(dialog != NULL); |
| |
814 |
| |
815 purple_notify_close_with_handle(dialog); |
| |
816 |
| |
817 gtk_widget_destroy(dialog->window); |
| |
818 |
| |
819 g_free(dialog); |
| |
820 } |
| |
821 |
| |
822 void |
| |
823 pidgin_xfer_dialog_show(PidginXferDialog *dialog) |
| |
824 { |
| |
825 PidginXferDialog *tmp; |
| |
826 |
| |
827 if (dialog == NULL) { |
| |
828 tmp = pidgin_get_xfer_dialog(); |
| |
829 |
| |
830 if (tmp == NULL) { |
| |
831 tmp = pidgin_xfer_dialog_new(); |
| |
832 pidgin_set_xfer_dialog(tmp); |
| |
833 } |
| |
834 |
| |
835 gtk_widget_show(tmp->window); |
| |
836 } else { |
| |
837 gtk_window_present(GTK_WINDOW(dialog->window)); |
| |
838 } |
| |
839 } |
| |
840 |
| |
841 void |
| |
842 pidgin_xfer_dialog_hide(PidginXferDialog *dialog) |
| |
843 { |
| |
844 g_return_if_fail(dialog != NULL); |
| |
845 |
| |
846 purple_notify_close_with_handle(dialog); |
| |
847 |
| |
848 gtk_widget_hide(dialog->window); |
| |
849 } |
| |
850 |
| |
851 void |
| |
852 pidgin_xfer_dialog_add_xfer(PidginXferDialog *dialog, PurpleXfer *xfer) |
| |
853 { |
| |
854 PidginXferUiData *data; |
| |
855 PurpleXferType type; |
| |
856 GdkPixbuf *pixbuf; |
| |
857 char *size_str, *remaining_str; |
| |
858 char *lfilename, *utf8; |
| |
859 |
| |
860 g_return_if_fail(dialog != NULL); |
| |
861 g_return_if_fail(xfer != NULL); |
| |
862 |
| |
863 g_object_ref(xfer); |
| |
864 |
| |
865 data = purple_xfer_get_ui_data(xfer); |
| |
866 data->in_list = TRUE; |
| |
867 |
| |
868 pidgin_xfer_dialog_show(dialog); |
| |
869 |
| |
870 data->last_updated_time = 0; |
| |
871 |
| |
872 type = purple_xfer_get_xfer_type(xfer); |
| |
873 |
| |
874 size_str = purple_str_size_to_units(purple_xfer_get_size(xfer)); |
| |
875 remaining_str = purple_str_size_to_units(purple_xfer_get_bytes_remaining(xfer)); |
| |
876 |
| |
877 pixbuf = gtk_widget_render_icon(dialog->window, |
| |
878 (type == PURPLE_XFER_TYPE_RECEIVE |
| |
879 ? PIDGIN_STOCK_DOWNLOAD |
| |
880 : PIDGIN_STOCK_UPLOAD), |
| |
881 GTK_ICON_SIZE_MENU, NULL); |
| |
882 |
| |
883 gtk_list_store_append(dialog->model, &data->iter); |
| |
884 lfilename = g_path_get_basename(purple_xfer_get_local_filename(xfer)); |
| |
885 utf8 = g_filename_to_utf8(lfilename, -1, NULL, NULL, NULL); |
| |
886 g_free(lfilename); |
| |
887 lfilename = utf8; |
| |
888 gtk_list_store_set(dialog->model, &data->iter, |
| |
889 COLUMN_STATUS, pixbuf, |
| |
890 COLUMN_PROGRESS, 0, |
| |
891 COLUMN_FILENAME, (type == PURPLE_XFER_TYPE_RECEIVE) |
| |
892 ? purple_xfer_get_filename(xfer) |
| |
893 : lfilename, |
| |
894 COLUMN_SIZE, size_str, |
| |
895 COLUMN_REMAINING, _("Waiting for transfer to begin"), |
| |
896 COLUMN_DATA, xfer, |
| |
897 -1); |
| |
898 g_free(lfilename); |
| |
899 |
| |
900 gtk_tree_view_columns_autosize(GTK_TREE_VIEW(dialog->tree)); |
| |
901 |
| |
902 g_object_unref(pixbuf); |
| |
903 |
| |
904 g_free(size_str); |
| |
905 g_free(remaining_str); |
| |
906 |
| |
907 dialog->num_transfers++; |
| |
908 |
| |
909 ensure_row_selected(dialog); |
| |
910 update_title_progress(dialog); |
| |
911 } |
| |
912 |
| |
913 void |
| |
914 pidgin_xfer_dialog_remove_xfer(PidginXferDialog *dialog, |
| |
915 PurpleXfer *xfer) |
| |
916 { |
| |
917 PidginXferUiData *data; |
| |
918 |
| |
919 g_return_if_fail(dialog != NULL); |
| |
920 g_return_if_fail(xfer != NULL); |
| |
921 |
| |
922 data = purple_xfer_get_ui_data(xfer); |
| |
923 |
| |
924 if (data == NULL) |
| |
925 return; |
| |
926 |
| |
927 if (!data->in_list) |
| |
928 return; |
| |
929 |
| |
930 data->in_list = FALSE; |
| |
931 |
| |
932 gtk_list_store_remove(GTK_LIST_STORE(dialog->model), &data->iter); |
| |
933 |
| |
934 dialog->num_transfers--; |
| |
935 |
| |
936 ensure_row_selected(dialog); |
| |
937 |
| |
938 update_title_progress(dialog); |
| |
939 g_object_unref(xfer); |
| |
940 } |
| |
941 |
| |
942 void |
| |
943 pidgin_xfer_dialog_cancel_xfer(PidginXferDialog *dialog, |
| |
944 PurpleXfer *xfer) |
| |
945 { |
| |
946 PidginXferUiData *data; |
| |
947 GdkPixbuf *pixbuf; |
| |
948 const gchar *status; |
| |
949 |
| |
950 g_return_if_fail(dialog != NULL); |
| |
951 g_return_if_fail(xfer != NULL); |
| |
952 |
| |
953 data = purple_xfer_get_ui_data(xfer); |
| |
954 |
| |
955 if (data == NULL) |
| |
956 return; |
| |
957 |
| |
958 if (!data->in_list) |
| |
959 return; |
| |
960 |
| |
961 if ((purple_xfer_get_status(xfer) == PURPLE_XFER_STATUS_CANCEL_LOCAL) && (dialog->auto_clear)) { |
| |
962 pidgin_xfer_dialog_remove_xfer(dialog, xfer); |
| |
963 return; |
| |
964 } |
| |
965 |
| |
966 data = purple_xfer_get_ui_data(xfer); |
| |
967 |
| |
968 update_detailed_info(dialog, xfer); |
| |
969 update_title_progress(dialog); |
| |
970 |
| |
971 pixbuf = gtk_widget_render_icon(dialog->window, |
| |
972 PIDGIN_STOCK_FILE_CANCELED, |
| |
973 GTK_ICON_SIZE_MENU, NULL); |
| |
974 |
| |
975 if (purple_xfer_is_cancelled(xfer)) |
| |
976 status = _("Cancelled"); |
| |
977 else |
| |
978 status = _("Failed"); |
| |
979 |
| |
980 gtk_list_store_set(dialog->model, &data->iter, |
| |
981 COLUMN_STATUS, pixbuf, |
| |
982 COLUMN_REMAINING, status, |
| |
983 -1); |
| |
984 |
| |
985 g_object_unref(pixbuf); |
| |
986 |
| |
987 update_buttons(dialog, xfer); |
| |
988 } |
| |
989 |
| |
990 void |
| |
991 pidgin_xfer_dialog_update_xfer(PidginXferDialog *dialog, |
| |
992 PurpleXfer *xfer) |
| |
993 { |
| |
994 PidginXferUiData *data; |
| |
995 char *size_str, *remaining_str; |
| |
996 time_t current_time; |
| |
997 GtkTreeIter iter; |
| |
998 gboolean valid; |
| |
999 |
| |
1000 g_return_if_fail(dialog != NULL); |
| |
1001 g_return_if_fail(xfer != NULL); |
| |
1002 |
| |
1003 if ((data = purple_xfer_get_ui_data(xfer)) == NULL) |
| |
1004 return; |
| |
1005 |
| |
1006 if (data->in_list == FALSE) |
| |
1007 return; |
| |
1008 |
| |
1009 current_time = time(NULL); |
| |
1010 if (((current_time - data->last_updated_time) == 0) && |
| |
1011 (!purple_xfer_is_completed(xfer))) |
| |
1012 { |
| |
1013 /* Don't update the window more than once per second */ |
| |
1014 return; |
| |
1015 } |
| |
1016 data->last_updated_time = current_time; |
| |
1017 |
| |
1018 size_str = purple_str_size_to_units(purple_xfer_get_size(xfer)); |
| |
1019 remaining_str = purple_str_size_to_units(purple_xfer_get_bytes_remaining(xfer)); |
| |
1020 |
| |
1021 gtk_list_store_set(xfer_dialog->model, &data->iter, |
| |
1022 COLUMN_PROGRESS, (gint)(purple_xfer_get_progress(xfer) * 100), |
| |
1023 COLUMN_SIZE, size_str, |
| |
1024 COLUMN_REMAINING, remaining_str, |
| |
1025 -1); |
| |
1026 |
| |
1027 g_free(size_str); |
| |
1028 g_free(remaining_str); |
| |
1029 |
| |
1030 if (purple_xfer_is_completed(xfer)) |
| |
1031 { |
| |
1032 GdkPixbuf *pixbuf; |
| |
1033 |
| |
1034 pixbuf = gtk_widget_render_icon(dialog->window, |
| |
1035 PIDGIN_STOCK_FILE_DONE, |
| |
1036 GTK_ICON_SIZE_MENU, NULL); |
| |
1037 |
| |
1038 gtk_list_store_set(GTK_LIST_STORE(xfer_dialog->model), &data->iter, |
| |
1039 COLUMN_STATUS, pixbuf, |
| |
1040 COLUMN_REMAINING, _("Finished"), |
| |
1041 -1); |
| |
1042 |
| |
1043 g_object_unref(pixbuf); |
| |
1044 } |
| |
1045 |
| |
1046 update_title_progress(dialog); |
| |
1047 if (xfer == dialog->selected_xfer) |
| |
1048 update_detailed_info(xfer_dialog, xfer); |
| |
1049 |
| |
1050 if (purple_xfer_is_completed(xfer) && dialog->auto_clear) |
| |
1051 pidgin_xfer_dialog_remove_xfer(dialog, xfer); |
| |
1052 else |
| |
1053 update_buttons(dialog, xfer); |
| |
1054 |
| |
1055 /* |
| |
1056 * If all transfers are finished, and the pref is set, then |
| |
1057 * close the dialog. Otherwise just exit this function. |
| |
1058 */ |
| |
1059 if (dialog->keep_open) |
| |
1060 return; |
| |
1061 |
| |
1062 valid = gtk_tree_model_get_iter_first(GTK_TREE_MODEL(dialog->model), &iter); |
| |
1063 while (valid) |
| |
1064 { |
| |
1065 GValue val; |
| |
1066 PurpleXfer *next; |
| |
1067 |
| |
1068 val.g_type = 0; |
| |
1069 gtk_tree_model_get_value(GTK_TREE_MODEL(dialog->model), |
| |
1070 &iter, COLUMN_DATA, &val); |
| |
1071 |
| |
1072 next = g_value_get_pointer(&val); |
| |
1073 if (!purple_xfer_is_completed(next)) |
| |
1074 return; |
| |
1075 |
| |
1076 valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(dialog->model), &iter); |
| |
1077 } |
| |
1078 |
| |
1079 /* If we got to this point then we know everything is finished */ |
| |
1080 pidgin_xfer_dialog_hide(dialog); |
| |
1081 } |
| |
1082 |
| |
1083 /************************************************************************** |
| |
1084 * File Transfer UI Ops |
| |
1085 **************************************************************************/ |
| |
1086 static void |
| |
1087 pidgin_xfer_new_xfer(PurpleXfer *xfer) |
| |
1088 { |
| |
1089 PidginXferUiData *data; |
| |
1090 |
| |
1091 /* This is where we're setting xfer's "ui_data" for the first time. */ |
| |
1092 data = g_new0(PidginXferUiData, 1); |
| |
1093 purple_xfer_set_ui_data(xfer, data); |
| |
1094 } |
| |
1095 |
| |
1096 static void |
| |
1097 pidgin_xfer_destroy(PurpleXfer *xfer) |
| |
1098 { |
| |
1099 PidginXferUiData *data; |
| |
1100 |
| |
1101 data = purple_xfer_get_ui_data(xfer); |
| |
1102 if (data) { |
| |
1103 g_free(data->name); |
| |
1104 g_free(data); |
| |
1105 purple_xfer_set_ui_data(xfer, NULL); |
| |
1106 } |
| |
1107 } |
| |
1108 |
| |
1109 static void |
| |
1110 pidgin_xfer_add_xfer(PurpleXfer *xfer) |
| |
1111 { |
| |
1112 if (xfer_dialog == NULL) |
| |
1113 xfer_dialog = pidgin_xfer_dialog_new(); |
| |
1114 |
| |
1115 pidgin_xfer_dialog_add_xfer(xfer_dialog, xfer); |
| |
1116 } |
| |
1117 |
| |
1118 static void |
| |
1119 pidgin_xfer_update_progress(PurpleXfer *xfer, double percent) |
| |
1120 { |
| |
1121 pidgin_xfer_dialog_update_xfer(xfer_dialog, xfer); |
| |
1122 } |
| |
1123 |
| |
1124 static void |
| |
1125 pidgin_xfer_cancel_local(PurpleXfer *xfer) |
| |
1126 { |
| |
1127 if (xfer_dialog) |
| |
1128 pidgin_xfer_dialog_cancel_xfer(xfer_dialog, xfer); |
| |
1129 } |
| |
1130 |
| |
1131 static void |
| |
1132 pidgin_xfer_cancel_remote(PurpleXfer *xfer) |
| |
1133 { |
| |
1134 if (xfer_dialog) |
| |
1135 pidgin_xfer_dialog_cancel_xfer(xfer_dialog, xfer); |
| |
1136 } |
| |
1137 |
| |
1138 static void |
| |
1139 pidgin_xfer_add_thumbnail(PurpleXfer *xfer, const gchar *formats) |
| |
1140 { |
| |
1141 purple_debug_info("ft", "creating thumbnail for transfer\n"); |
| |
1142 |
| |
1143 if (purple_xfer_get_size(xfer) <= PIDGIN_XFER_MAX_SIZE_IMAGE_THUMBNAIL) { |
| |
1144 GdkPixbuf *thumbnail = |
| |
1145 pidgin_pixbuf_new_from_file_at_size( |
| |
1146 purple_xfer_get_local_filename(xfer), 128, 128); |
| |
1147 |
| |
1148 if (thumbnail) { |
| |
1149 gchar **formats_split = g_strsplit(formats, ",", 0); |
| |
1150 gchar *buffer = NULL; |
| |
1151 gsize size; |
| |
1152 char *option_keys[2] = {NULL, NULL}; |
| |
1153 char *option_values[2] = {NULL, NULL}; |
| |
1154 int i; |
| |
1155 gchar *format = NULL; |
| |
1156 |
| |
1157 for (i = 0; formats_split[i]; i++) { |
| |
1158 if (purple_strequal(formats_split[i], "jpeg")) { |
| |
1159 purple_debug_info("ft", "creating JPEG thumbnail\n"); |
| |
1160 option_keys[0] = "quality"; |
| |
1161 option_values[0] = "90"; |
| |
1162 format = "jpeg"; |
| |
1163 break; |
| |
1164 } else if (purple_strequal(formats_split[i], "png")) { |
| |
1165 purple_debug_info("ft", "creating PNG thumbnail\n"); |
| |
1166 option_keys[0] = "compression"; |
| |
1167 option_values[0] = "9"; |
| |
1168 format = "png"; |
| |
1169 break; |
| |
1170 } |
| |
1171 } |
| |
1172 |
| |
1173 /* Try the first format given by the protocol without options */ |
| |
1174 if (format == NULL) { |
| |
1175 purple_debug_info("ft", |
| |
1176 "creating thumbnail of format %s as demanded by protocol\n", |
| |
1177 formats_split[0]); |
| |
1178 format = formats_split[0]; |
| |
1179 } |
| |
1180 |
| |
1181 gdk_pixbuf_save_to_bufferv(thumbnail, &buffer, &size, format, |
| |
1182 option_keys, option_values, NULL); |
| |
1183 |
| |
1184 if (buffer) { |
| |
1185 gchar *mimetype = g_strdup_printf("image/%s", format); |
| |
1186 purple_debug_info("ft", |
| |
1187 "created thumbnail of %" G_GSIZE_FORMAT " bytes\n", |
| |
1188 size); |
| |
1189 purple_xfer_set_thumbnail(xfer, buffer, size, mimetype); |
| |
1190 g_free(buffer); |
| |
1191 g_free(mimetype); |
| |
1192 } |
| |
1193 g_object_unref(thumbnail); |
| |
1194 g_strfreev(formats_split); |
| |
1195 } |
| |
1196 } |
| |
1197 } |
| |
1198 |
| |
1199 static PurpleXferUiOps ops = |
| |
1200 { |
| |
1201 pidgin_xfer_new_xfer, |
| |
1202 pidgin_xfer_destroy, |
| |
1203 pidgin_xfer_add_xfer, |
| |
1204 pidgin_xfer_update_progress, |
| |
1205 pidgin_xfer_cancel_local, |
| |
1206 pidgin_xfer_cancel_remote, |
| |
1207 NULL, |
| |
1208 NULL, |
| |
1209 NULL, |
| |
1210 pidgin_xfer_add_thumbnail |
| |
1211 }; |
| |
1212 |
| |
1213 /************************************************************************** |
| |
1214 * GTK+ File Transfer API |
| |
1215 **************************************************************************/ |
| |
1216 void |
| |
1217 pidgin_xfers_init(void) |
| |
1218 { |
| |
1219 purple_prefs_add_none(PIDGIN_PREFS_ROOT "/filetransfer"); |
| |
1220 purple_prefs_add_bool(PIDGIN_PREFS_ROOT "/filetransfer/clear_finished", TRUE); |
| |
1221 purple_prefs_add_bool(PIDGIN_PREFS_ROOT "/filetransfer/keep_open", FALSE); |
| |
1222 } |
| |
1223 |
| |
1224 void |
| |
1225 pidgin_xfers_uninit(void) |
| |
1226 { |
| |
1227 if (xfer_dialog != NULL) |
| |
1228 pidgin_xfer_dialog_destroy(xfer_dialog); |
| |
1229 } |
| |
1230 |
| |
1231 void |
| |
1232 pidgin_set_xfer_dialog(PidginXferDialog *dialog) |
| |
1233 { |
| |
1234 xfer_dialog = dialog; |
| |
1235 } |
| |
1236 |
| |
1237 PidginXferDialog * |
| |
1238 pidgin_get_xfer_dialog(void) |
| |
1239 { |
| |
1240 return xfer_dialog; |
| |
1241 } |
| |
1242 |
| |
1243 PurpleXferUiOps * |
| |
1244 pidgin_xfers_get_ui_ops(void) |
| |
1245 { |
| |
1246 return &ops; |
| |
1247 } |