pidgin/gtkxfer.c

branch
soc.2013.gobjectification.plugins
changeset 36688
e49025233301
parent 36637
9b0109ae118d
parent 34929
4deeac0840fe
child 36782
64936dae41a3
equal deleted inserted replaced
36687:7a83a1a657e3 36688:e49025233301
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 }

mercurial