transmission/gtk/open-dialog.c

520 lines
16 KiB
C

/*
* This file Copyright (C) Mnemosyne LLC
*
* This file is licensed by the GPL version 2. Works owned by the
* Transmission project are granted a special exemption to clause 2 (b)
* so that the bulk of its code can remain under the MIT license.
* This exemption does not extend to derived works not owned by
* the Transmission project.
*
* $Id$
*/
#include <glib/gi18n.h>
#include <gtk/gtk.h>
#include <string.h>
#include <libtransmission/transmission.h>
#include <libtransmission/utils.h> /* tr_is_same_file () */
#include "conf.h"
#include "file-list.h"
#include "hig.h"
#include "open-dialog.h"
#include "tr-prefs.h"
#include "util.h" /* gtr_priority_combo_get_value () */
/****
*****
****/
#define N_RECENT 4
static GSList*
get_recent_destinations (void)
{
int i;
GSList * list = NULL;
for (i=0; i<N_RECENT; ++i)
{
char key[64];
const char * val;
g_snprintf (key, sizeof (key), "recent-download-dir-%d", i+1);
if ((val = gtr_pref_string_get (tr_quark_new(key,-1))))
list = g_slist_append (list, (void*)val);
}
return list;
}
static void
save_recent_destination (TrCore * core, const char * dir)
{
int i;
GSList * l;
GSList * list = get_recent_destinations ();
if (dir == NULL)
return;
/* if it was already in the list, remove it */
if ((l = g_slist_find_custom (list, dir, (GCompareFunc)strcmp)))
list = g_slist_delete_link (list, l);
/* add it to the front of the list */
list = g_slist_prepend (list, (void*)dir);
/* make local copies of the strings that aren't
* invalidated by gtr_pref_string_set () */
for (l=list; l; l=l->next)
l->data = g_strdup (l->data);
/* save the first N_RECENT directories */
for (l=list, i=0; l && (i<N_RECENT); ++i, l=l->next) {
char key[64];
g_snprintf (key, sizeof (key), "recent-download-dir-%d", i + 1);
gtr_pref_string_set (tr_quark_new(key,-1), l->data);
}
gtr_pref_save (gtr_core_session (core));
/* cleanup */
g_slist_foreach (list, (GFunc)g_free, NULL);
g_slist_free (list);
}
/****
*****
****/
struct OpenData
{
TrCore * core;
GtkWidget * file_list;
GtkWidget * run_check;
GtkWidget * trash_check;
GtkWidget * priority_combo;
char * filename;
char * downloadDir;
tr_torrent * tor;
tr_ctor * ctor;
};
static void
removeOldTorrent (struct OpenData * o)
{
if (o->tor)
{
gtr_file_list_clear (o->file_list);
tr_torrentRemove (o->tor, FALSE, NULL);
o->tor = NULL;
}
}
static void
addResponseCB (GtkDialog * dialog,
gint response,
gpointer gdata)
{
struct OpenData * o = gdata;
if (o->tor)
{
if (response != GTK_RESPONSE_ACCEPT)
{
removeOldTorrent (o);
}
else
{
tr_torrentSetPriority (o->tor, gtr_priority_combo_get_value (GTK_COMBO_BOX (o->priority_combo)));
if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (o->run_check)))
tr_torrentStart (o->tor);
gtr_core_add_torrent (o->core, o->tor, FALSE);
if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (o->trash_check)))
gtr_file_trash_or_remove (o->filename);
save_recent_destination (o->core, o->downloadDir);
}
}
tr_ctorFree (o->ctor);
g_free (o->filename);
g_free (o->downloadDir);
g_free (o);
gtk_widget_destroy (GTK_WIDGET (dialog));
}
static void
updateTorrent (struct OpenData * o)
{
const gboolean isLocalFile = tr_ctorGetSourceFile (o->ctor) != NULL;
gtk_widget_set_sensitive (o->trash_check, isLocalFile);
if (!o->tor)
{
gtr_file_list_clear (o->file_list);
gtk_widget_set_sensitive (o->file_list, FALSE);
}
else
{
tr_torrentSetDownloadDir (o->tor, o->downloadDir);
gtk_widget_set_sensitive (o->file_list, tr_torrentHasMetadata (o->tor));
gtr_file_list_set_torrent (o->file_list, tr_torrentId (o->tor));
tr_torrentVerify (o->tor);
}
}
/**
* When the source .torrent file is deleted
* (such as, if it was a temp file that a web browser passed to us),
* gtk invokes this callback and `filename' will be NULL.
* The `filename' tests here are to prevent us from losing the current
* metadata when that happens.
*/
static void
sourceChanged (GtkFileChooserButton * b, gpointer gdata)
{
struct OpenData * o = gdata;
char * filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (b));
/* maybe instantiate a torrent */
if (filename || !o->tor)
{
int err = 0;
int new_file = 0;
tr_torrent * torrent;
if (filename && (!o->filename || !tr_is_same_file (filename, o->filename)))
{
g_free (o->filename);
o->filename = g_strdup (filename);
tr_ctorSetMetainfoFromFile (o->ctor, o->filename);
new_file = 1;
}
tr_ctorSetDownloadDir (o->ctor, TR_FORCE, o->downloadDir);
tr_ctorSetPaused (o->ctor, TR_FORCE, TRUE);
tr_ctorSetDeleteSource (o->ctor, FALSE);
if ((torrent = tr_torrentNew (o->ctor, &err)))
{
removeOldTorrent (o);
o->tor = torrent;
}
else if (new_file)
{
gtr_add_torrent_error_dialog (GTK_WIDGET (b), err, o->filename);
}
updateTorrent (o);
}
g_free (filename);
}
static void
downloadDirChanged (GtkFileChooserButton * b, gpointer gdata)
{
struct OpenData * data = gdata;
char * fname = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (b));
if (fname && (!data->downloadDir || !tr_is_same_file (fname, data->downloadDir)))
{
g_free (data->downloadDir);
data->downloadDir = g_strdup (fname);
updateTorrent (data);
}
g_free (fname);
}
static void
addTorrentFilters (GtkFileChooser * chooser)
{
GtkFileFilter * filter;
filter = gtk_file_filter_new ();
gtk_file_filter_set_name (filter, _("Torrent files"));
gtk_file_filter_add_pattern (filter, "*.torrent");
gtk_file_chooser_add_filter (chooser, filter);
filter = gtk_file_filter_new ();
gtk_file_filter_set_name (filter, _("All files"));
gtk_file_filter_add_pattern (filter, "*");
gtk_file_chooser_add_filter (chooser, filter);
}
/****
*****
****/
GtkWidget*
gtr_torrent_options_dialog_new (GtkWindow * parent, TrCore * core, tr_ctor * ctor)
{
const char * str;
GtkWidget * w;
GtkWidget * d;
GtkGrid * grid;
int row;
GtkWidget * l;
GtkWidget * source_chooser;
struct OpenData * data;
bool flag;
GSList * list;
GSList * walk;
/* make the dialog */
d = gtk_dialog_new_with_buttons (_("Torrent Options"), parent,
GTK_DIALOG_DESTROY_WITH_PARENT,
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT,
NULL);
gtk_dialog_set_default_response (GTK_DIALOG (d),
GTK_RESPONSE_ACCEPT);
gtk_dialog_set_alternative_button_order (GTK_DIALOG (d),
GTK_RESPONSE_ACCEPT,
GTK_RESPONSE_CANCEL,
-1);
if (tr_ctorGetDownloadDir (ctor, TR_FORCE, &str))
g_assert_not_reached ();
g_assert (str);
data = g_new0 (struct OpenData, 1);
data->core = core;
data->ctor = ctor;
data->filename = g_strdup (tr_ctorGetSourceFile (ctor));
data->downloadDir = g_strdup (str);
data->file_list = gtr_file_list_new (core, 0);
str = _("Mo_ve .torrent file to the trash");
data->trash_check = gtk_check_button_new_with_mnemonic (str);
str = _("_Start when added");
data->run_check = gtk_check_button_new_with_mnemonic (str);
w = data->priority_combo = gtr_priority_combo_new ();
gtr_priority_combo_set_value (GTK_COMBO_BOX (w), TR_PRI_NORMAL);
g_signal_connect (G_OBJECT (d), "response",
G_CALLBACK (addResponseCB), data);
row = 0;
grid = GTK_GRID (gtk_grid_new ());
gtk_container_set_border_width (GTK_CONTAINER (grid), GUI_PAD_BIG);
gtk_grid_set_row_spacing (grid, GUI_PAD);
gtk_grid_set_column_spacing (grid, GUI_PAD_BIG);
// "torrent file" row
l = gtk_label_new_with_mnemonic (_("_Torrent file:"));
gtk_misc_set_alignment (GTK_MISC (l), 0.0f, 0.5f);
gtk_grid_attach (grid, l, 0, row, 1, 1);
w = gtk_file_chooser_button_new (_("Select Source File"),
GTK_FILE_CHOOSER_ACTION_OPEN);
source_chooser = w;
gtk_widget_set_hexpand (w, TRUE);
gtk_grid_attach_next_to (grid, w, l, GTK_POS_RIGHT, 1, 1);
gtk_label_set_mnemonic_widget (GTK_LABEL (l), w);
addTorrentFilters (GTK_FILE_CHOOSER (w));
g_signal_connect (w, "selection-changed",
G_CALLBACK (sourceChanged), data);
// "destination folder" row
row++;
l = gtk_label_new_with_mnemonic (_("_Destination folder:"));
gtk_misc_set_alignment (GTK_MISC (l), 0.0f, 0.5f);
gtk_grid_attach (grid, l, 0, row, 1, 1);
w = gtk_file_chooser_button_new (_("Select Destination Folder"),
GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER);
if (!gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (w),
data->downloadDir))
g_warning ("couldn't select '%s'", data->downloadDir);
list = get_recent_destinations ();
for (walk = list; walk; walk = walk->next)
gtk_file_chooser_add_shortcut_folder (GTK_FILE_CHOOSER (w), walk->data, NULL);
g_slist_free (list);
gtk_grid_attach_next_to (grid, w, l, GTK_POS_RIGHT, 1, 1);
gtk_label_set_mnemonic_widget (GTK_LABEL (l), w);
g_signal_connect (w, "selection-changed",
G_CALLBACK (downloadDirChanged), data);
// file list row
row++;
w = data->file_list;
gtk_widget_set_vexpand (w, TRUE);
gtk_widget_set_size_request (w, 466u, 300u);
gtk_grid_attach (grid, w, 0, row, 2, 1);
// torrent priority row
row++;
l = gtk_label_new_with_mnemonic (_("Torrent _priority:"));
gtk_misc_set_alignment (GTK_MISC (l), 0.0f, 0.5f);
gtk_grid_attach (grid, l, 0, row, 1, 1);
w = data->priority_combo;
gtk_label_set_mnemonic_widget (GTK_LABEL (l), w);
gtk_grid_attach_next_to (grid, w, l, GTK_POS_RIGHT, 1, 1);
// torrent priority row
row++;
w = data->run_check;
if (tr_ctorGetPaused (ctor, TR_FORCE, &flag))
g_assert_not_reached ();
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (w), !flag);
gtk_grid_attach (grid, w, 0, row, 2, 1);
// "trash .torrent file" row
row++;
w = data->trash_check;
if (tr_ctorGetDeleteSource (ctor, &flag))
g_assert_not_reached ();
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (w), flag);
gtk_grid_attach (grid, w, 0, row, 2, 1);
/* trigger sourceChanged, either directly or indirectly,
* so that it creates the tor/gtor objects */
w = source_chooser;
if (data->filename)
gtk_file_chooser_set_filename (GTK_FILE_CHOOSER (w), data->filename);
else
sourceChanged (GTK_FILE_CHOOSER_BUTTON (w), data);
gtr_dialog_set_content (GTK_DIALOG (d), GTK_WIDGET (grid));
w = gtk_dialog_get_widget_for_response (GTK_DIALOG (d), GTK_RESPONSE_ACCEPT);
gtk_widget_grab_focus (w);
return d;
}
/****
*****
****/
static void
onOpenDialogResponse (GtkDialog * dialog, int response, gpointer core)
{
char * folder;
/* remember this folder the next time we use this dialog */
folder = gtk_file_chooser_get_current_folder (GTK_FILE_CHOOSER (dialog));
gtr_pref_string_set (TR_KEY_open_dialog_dir, folder);
g_free (folder);
if (response == GTK_RESPONSE_ACCEPT)
{
GtkFileChooser * chooser = GTK_FILE_CHOOSER (dialog);
GtkWidget * w = gtk_file_chooser_get_extra_widget (chooser);
GtkToggleButton * tb = GTK_TOGGLE_BUTTON (w);
const gboolean do_start = gtr_pref_flag_get (TR_KEY_start_added_torrents);
const gboolean do_prompt = gtk_toggle_button_get_active (tb);
const gboolean do_notify = FALSE;
GSList * files = gtk_file_chooser_get_files (chooser);
gtr_core_add_files (core, files, do_start, do_prompt, do_notify);
g_slist_foreach (files, (GFunc)g_object_unref, NULL);
g_slist_free (files);
}
gtk_widget_destroy (GTK_WIDGET (dialog));
}
GtkWidget*
gtr_torrent_open_from_file_dialog_new (GtkWindow * parent, TrCore * core)
{
GtkWidget * w;
GtkWidget * c;
const char * folder;
w = gtk_file_chooser_dialog_new (_("Open a Torrent"), parent,
GTK_FILE_CHOOSER_ACTION_OPEN,
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT,
NULL);
gtk_dialog_set_alternative_button_order (GTK_DIALOG (w),
GTK_RESPONSE_ACCEPT,
GTK_RESPONSE_CANCEL,
-1);
gtk_file_chooser_set_select_multiple (GTK_FILE_CHOOSER (w), TRUE);
addTorrentFilters (GTK_FILE_CHOOSER (w));
g_signal_connect (w, "response", G_CALLBACK (onOpenDialogResponse), core);
if ((folder = gtr_pref_string_get (TR_KEY_open_dialog_dir)))
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (w), folder);
c = gtk_check_button_new_with_mnemonic (_("Show _options dialog"));
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (c),
gtr_pref_flag_get (TR_KEY_show_options_window));
gtk_file_chooser_set_extra_widget (GTK_FILE_CHOOSER (w), c);
gtk_widget_show (c);
return w;
}
/***
****
***/
static void
onOpenURLResponse (GtkDialog * dialog, int response, gpointer user_data)
{
bool handled = false;
if (response == GTK_RESPONSE_ACCEPT)
{
GtkWidget * e = GTK_WIDGET (g_object_get_data (G_OBJECT (dialog), "url-entry"));
char * url = g_strdup (gtk_entry_get_text (GTK_ENTRY (e)));
g_strstrip (url);
if (url) {
handled = gtr_core_add_from_url (user_data, url);
if (!handled)
gtr_unrecognized_url_dialog (GTK_WIDGET (dialog), url);
g_free (url);
}
}
else if (response == GTK_RESPONSE_CANCEL)
{
handled = TRUE;
}
if (handled)
gtk_widget_destroy (GTK_WIDGET (dialog));
}
GtkWidget*
gtr_torrent_open_from_url_dialog_new (GtkWindow * parent, TrCore * core)
{
guint row;
GtkWidget * e;
GtkWidget * t;
GtkWidget * w;
w = gtk_dialog_new_with_buttons (_("Open URL"), parent,
GTK_DIALOG_DESTROY_WITH_PARENT,
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT,
NULL);
gtk_dialog_set_alternative_button_order (GTK_DIALOG (w),
GTK_RESPONSE_ACCEPT,
GTK_RESPONSE_CANCEL,
-1);
g_signal_connect (w, "response", G_CALLBACK (onOpenURLResponse), core);
row = 0;
t = hig_workarea_create ();
hig_workarea_add_section_title (t, &row, _("Open torrent from URL"));
e = gtk_entry_new ();
gtk_widget_set_size_request (e, 400, -1);
gtr_paste_clipboard_url_into_entry (e);
g_object_set_data (G_OBJECT (w), "url-entry", e);
hig_workarea_add_row (t, &row, _("_URL"), e, NULL);
gtr_dialog_set_content (GTK_DIALOG (w), t);
if (gtk_entry_get_text_length (GTK_ENTRY (e)) == 0)
gtk_widget_grab_focus (e);
else
gtk_widget_grab_focus (gtk_dialog_get_widget_for_response (GTK_DIALOG (w), GTK_RESPONSE_ACCEPT));
return w;
}