source: trunk/gtk/add-dialog.c @ 10101

Last change on this file since 10101 was 10101, checked in by charles, 12 years ago

(trunk gtk) #2864 "when adding a non-BitTorrent? magnet link fails, explain why to the user" -- implemented for GTK+ client for 1.90

  • Property svn:keywords set to Date Rev Author Id
File size: 16.6 KB
Line 
1/*
2 * This file Copyright (C) 2008-2010 Mnemosyne LLC
3 *
4 * This file is licensed by the GPL version 2.  Works owned by the
5 * Transmission project are granted a special exemption to clause 2(b)
6 * so that the bulk of its code can remain under the MIT license.
7 * This exemption does not extend to derived works not owned by
8 * the Transmission project.
9 *
10 * $Id: add-dialog.c 10101 2010-02-05 13:59:44Z charles $
11 */
12
13#include <glib/gi18n.h>
14#include <gtk/gtk.h>
15
16#include <string.h>
17
18#include "add-dialog.h"
19#include "conf.h"
20#include "file-list.h"
21#include "hig.h"
22#include "tr-prefs.h"
23
24/****
25*****
26****/
27
28#define N_RECENT 4
29
30static GSList*
31get_recent_destinations( void )
32{
33    int i;
34    GSList * list = NULL;
35
36    for( i=0; i<N_RECENT; ++i )
37    {
38        char key[64];
39        const char * val;
40        g_snprintf( key, sizeof( key ), "recent-download-dir-%d", i+1 );
41        if(( val = pref_string_get( key )))
42            list = g_slist_append( list, (void*)val );
43    }
44    return list;
45}
46
47static void
48save_recent_destination( TrCore * core, const char * dir  )
49{
50    int i;
51    GSList * l;
52    GSList * list = get_recent_destinations( );
53
54    if( dir == NULL )
55        return;
56
57    /* if it was already in the list, remove it */
58    if(( l = g_slist_find_custom( list, dir, (GCompareFunc)strcmp )))
59        list = g_slist_delete_link( list, l );
60
61    /* add it to the front of the list */
62    list = g_slist_prepend( list, (void*)dir );
63
64    /* make local copies of the strings that aren't
65     * invalidated by pref_string_set() */
66    for( l=list; l; l=l->next )
67        l->data = g_strdup( l->data );
68
69    /* save the first N_RECENT directories */
70    for( l=list, i=0; l && ( i<N_RECENT ); ++i, l=l->next ) {
71        char key[64];
72        g_snprintf( key, sizeof( key ), "recent-download-dir-%d", i + 1 );
73        pref_string_set( key, l->data );
74    }
75    pref_save( tr_core_session( core ) );
76
77    /* cleanup */
78    g_slist_foreach( list, (GFunc)g_free, NULL );
79    g_slist_free( list );
80}
81
82/****
83*****
84****/
85
86struct AddData
87{
88    TrCore *     core;
89    GtkWidget *  list;
90    GtkWidget *  run_check;
91    GtkWidget *  trash_check;
92    GtkWidget *  priority_combo;
93    char *       filename;
94    char *       downloadDir;
95    TrTorrent *  gtor;
96    tr_ctor *    ctor;
97};
98
99static void
100removeOldTorrent( struct AddData * data )
101{
102    if( data->gtor )
103    {
104        file_list_clear( data->list );
105        tr_torrent_set_remove_flag( data->gtor, TRUE );
106        g_object_unref( G_OBJECT( data->gtor ) );
107        data->gtor = NULL;
108    }
109}
110
111static void
112addResponseCB( GtkDialog * dialog,
113               gint        response,
114               gpointer    gdata )
115{
116    struct AddData * data = gdata;
117
118    if( data->gtor )
119    {
120        if( response != GTK_RESPONSE_ACCEPT )
121        {
122            removeOldTorrent( data );
123        }
124        else
125        {
126            tr_torrent * tor = tr_torrent_handle( data->gtor );
127
128            tr_torrentSetPriority( tor, gtr_priority_combo_get_value( data->priority_combo ) );
129
130            if( gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON( data->run_check ) ) )
131                tr_torrentStart( tor );
132
133            tr_core_add_torrent( data->core, data->gtor, FALSE );
134
135            if( gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON( data->trash_check ) ) )
136
137                tr_file_trash_or_remove( data->filename );
138            save_recent_destination( data->core, data->downloadDir );
139        }
140    }
141
142    tr_ctorFree( data->ctor );
143    g_free( data->filename );
144    g_free( data->downloadDir );
145    g_free( data );
146    gtk_widget_destroy( GTK_WIDGET( dialog ) );
147}
148
149static void
150updateTorrent( struct AddData * o )
151{
152    if( !o->gtor )
153        file_list_clear( o->list );
154    else {
155        tr_torrent * tor = tr_torrent_handle( o->gtor );
156        tr_torrentSetDownloadDir( tor, o->downloadDir );
157        file_list_set_torrent( o->list, tr_torrentId( tor ) );
158    }
159}
160
161/**
162 * When the source .torrent file is deleted
163 * (such as, if it was a temp file that a web browser passed to us),
164 * gtk invokes this callback and `filename' will be NULL.
165 * The `filename' tests here are to prevent us from losing the current
166 * metadata when that happens.
167 */
168static void
169sourceChanged( GtkFileChooserButton * b,
170               gpointer               gdata )
171{
172    struct AddData * data = gdata;
173    char *           filename = gtk_file_chooser_get_filename(
174         GTK_FILE_CHOOSER( b ) );
175
176    /* maybe instantiate a torrent */
177    if( data->filename || !data->gtor )
178    {
179        int          err = 0;
180        int          new_file = 0;
181        tr_torrent * torrent;
182
183        if( filename
184          && ( !data->filename || strcmp( filename, data->filename ) ) )
185        {
186            g_free( data->filename );
187            data->filename = g_strdup( filename );
188            tr_ctorSetMetainfoFromFile( data->ctor, data->filename );
189            new_file = 1;
190        }
191
192        tr_ctorSetDownloadDir( data->ctor, TR_FORCE, data->downloadDir );
193        tr_ctorSetPaused( data->ctor, TR_FORCE, TRUE );
194        tr_ctorSetDeleteSource( data->ctor, FALSE );
195
196        if( ( torrent = tr_torrentNew( data->ctor, &err ) ) )
197        {
198            removeOldTorrent( data );
199            data->gtor = tr_torrent_new_preexisting( torrent );
200        }
201        else if( new_file )
202        {
203            addTorrentErrorDialog( GTK_WIDGET( b ), err, data->filename );
204        }
205
206        updateTorrent( data );
207    }
208
209    g_free( filename );
210}
211
212static void
213verifyRequested( GtkButton * button UNUSED, gpointer gdata )
214{
215    struct AddData * data = gdata;
216
217    if( data->gtor )
218        tr_torrentVerify( tr_torrent_handle( data->gtor ) );
219}
220
221static void
222downloadDirChanged( GtkFileChooserButton * b,
223                    gpointer               gdata )
224{
225    char *           fname = gtk_file_chooser_get_filename(
226         GTK_FILE_CHOOSER( b ) );
227    struct AddData * data = gdata;
228
229    if( fname && ( !data->downloadDir || strcmp( fname, data->downloadDir ) ) )
230    {
231        g_free( data->downloadDir );
232        data->downloadDir = g_strdup( fname );
233
234        updateTorrent( data );
235        verifyRequested( NULL, data );
236    }
237
238    g_free( fname );
239}
240
241static void
242addTorrentFilters( GtkFileChooser * chooser )
243{
244    GtkFileFilter * filter;
245
246    filter = gtk_file_filter_new( );
247    gtk_file_filter_set_name( filter, _( "Torrent files" ) );
248    gtk_file_filter_add_pattern( filter, "*.torrent" );
249    gtk_file_chooser_add_filter( chooser, filter );
250
251    filter = gtk_file_filter_new( );
252    gtk_file_filter_set_name( filter, _( "All files" ) );
253    gtk_file_filter_add_pattern( filter, "*" );
254    gtk_file_chooser_add_filter( chooser, filter );
255}
256
257/****
258*****
259****/
260
261GtkWidget*
262addSingleTorrentDialog( GtkWindow * parent, TrCore * core, tr_ctor * ctor )
263{
264    int              row;
265    int              col;
266    const char *     str;
267    GtkWidget *      w;
268    GtkWidget *      d;
269    GtkWidget *      t;
270    GtkWidget *      l;
271    GtkWidget *      source_chooser;
272    struct AddData * data;
273    uint8_t          flag;
274    GSList *         list;
275    GSList *         walk;
276
277    /* make the dialog */
278    d = gtk_dialog_new_with_buttons( _( "Torrent Options" ), parent,
279                                     GTK_DIALOG_DESTROY_WITH_PARENT |
280                                     GTK_DIALOG_NO_SEPARATOR,
281                                     GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
282                                     GTK_STOCK_ADD, GTK_RESPONSE_ACCEPT,
283                                     NULL );
284    gtk_dialog_set_default_response( GTK_DIALOG( d ),
285                                     GTK_RESPONSE_ACCEPT );
286    gtk_dialog_set_alternative_button_order( GTK_DIALOG( d ),
287                                             GTK_RESPONSE_ACCEPT,
288                                             GTK_RESPONSE_CANCEL,
289                                             -1 );
290
291    if( tr_ctorGetDownloadDir( ctor, TR_FORCE, &str ) )
292        g_assert_not_reached( );
293    g_assert( str );
294
295    data = g_new0( struct AddData, 1 );
296    data->core = core;
297    data->ctor = ctor;
298    data->filename = g_strdup( tr_ctorGetSourceFile( ctor ) );
299    data->downloadDir = g_strdup( str );
300    data->list = file_list_new( core, 0 );
301    str = _( "Mo_ve .torrent file to the trash" );
302    data->trash_check = gtk_check_button_new_with_mnemonic( str );
303    str = _( "_Start when added" );
304    data->run_check = gtk_check_button_new_with_mnemonic( str );
305
306    data->priority_combo = gtr_priority_combo_new( );
307    gtr_priority_combo_set_value( data->priority_combo, TR_PRI_NORMAL );
308
309    g_signal_connect( G_OBJECT( d ), "response",
310                      G_CALLBACK( addResponseCB ), data );
311
312    t = gtk_table_new( 6, 2, FALSE );
313    gtk_container_set_border_width( GTK_CONTAINER( t ), GUI_PAD_BIG );
314    gtk_table_set_row_spacings( GTK_TABLE( t ), GUI_PAD );
315    gtk_table_set_col_spacings( GTK_TABLE( t ), GUI_PAD_BIG );
316
317    row = col = 0;
318    l = gtk_label_new_with_mnemonic( _( "_Torrent file:" ) );
319    gtk_misc_set_alignment( GTK_MISC( l ), 0.0f, 0.5f );
320    gtk_table_attach( GTK_TABLE( t ), l, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0 );
321    ++col;
322    w = gtk_file_chooser_button_new( _( "Select Source File" ),
323                                     GTK_FILE_CHOOSER_ACTION_OPEN );
324    source_chooser = w;
325    gtk_table_attach( GTK_TABLE( t ), w, col, col + 1, row, row + 1, ~0, 0, 0, 0 );
326    gtk_label_set_mnemonic_widget( GTK_LABEL( l ), w );
327    addTorrentFilters( GTK_FILE_CHOOSER( w ) );
328    g_signal_connect( w, "selection-changed",
329                      G_CALLBACK( sourceChanged ), data );
330
331    ++row;
332    col = 0;
333    l = gtk_label_new_with_mnemonic( _( "_Destination folder:" ) );
334    gtk_misc_set_alignment( GTK_MISC( l ), 0.0f, 0.5f );
335    gtk_table_attach( GTK_TABLE( t ), l, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0 );
336    ++col;
337    w = gtk_file_chooser_button_new( _( "Select Destination Folder" ),
338                                     GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER );
339    if( !gtk_file_chooser_set_filename( GTK_FILE_CHOOSER( w ),
340                                        data->downloadDir ) )
341        g_warning( "couldn't select '%s'", data->downloadDir );
342    list = get_recent_destinations( );
343    for( walk = list; walk; walk = walk->next )
344        gtk_file_chooser_add_shortcut_folder( GTK_FILE_CHOOSER( w ), walk->data, NULL );
345    g_slist_free( list );
346    gtk_table_attach( GTK_TABLE( t ), w, col, col + 1, row, row + 1, ~0, 0, 0, 0 );
347    gtk_label_set_mnemonic_widget( GTK_LABEL( l ), w );
348    g_signal_connect( w, "selection-changed",
349                      G_CALLBACK( downloadDirChanged ), data );
350
351    ++row;
352    col = 0;
353    w = data->list;
354    gtk_widget_set_size_request ( w, 466u, 300u );
355    gtk_table_attach_defaults( GTK_TABLE( t ), w, col, col + 2, row, row + 1 );
356
357    ++row;
358    col = 0;
359    w = gtk_label_new_with_mnemonic( _( "Torrent _priority:" ) );
360    gtk_misc_set_alignment( GTK_MISC( w ), 0.0f, 0.5f );
361    gtk_table_attach_defaults( GTK_TABLE( t ), w, col, col + 1, row, row + 1 );
362    ++col;
363    gtk_table_attach_defaults( GTK_TABLE( t ), data->priority_combo, col, col + 1, row, row + 1 );
364    gtk_label_set_mnemonic_widget( GTK_LABEL( w ), data->priority_combo );
365
366    ++row;
367    col = 0;
368    w = data->run_check;
369    if( tr_ctorGetPaused( ctor, TR_FORCE, &flag ) )
370        g_assert_not_reached( );
371    gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON( w ), !flag );
372    gtk_table_attach( GTK_TABLE( t ), w, col, col + 2, row, row + 1, GTK_FILL, 0, 0, 0 );
373
374    ++row;
375    col = 0;
376    w = data->trash_check;
377    if( tr_ctorGetDeleteSource( ctor, &flag ) )
378        g_assert_not_reached( );
379    gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON( w ), flag );
380    gtk_table_attach( GTK_TABLE( t ), w, col, col + 2, row, row + 1, GTK_FILL, 0, 0, 0 );
381
382    ++row;
383    col = 0;
384    w = data->priority_combo;
385
386    /* trigger sourceChanged, either directly or indirectly,
387     * so that it creates the tor/gtor objects */
388    w = source_chooser;
389    if( data->filename )
390        gtk_file_chooser_set_filename( GTK_FILE_CHOOSER( w ), data->filename );
391    else
392        sourceChanged( GTK_FILE_CHOOSER_BUTTON( w ), data );
393
394    gtk_box_pack_start( GTK_BOX( GTK_DIALOG( d )->vbox ), t, TRUE, TRUE, 0 );
395    gtk_widget_show_all( d );
396    return d;
397}
398
399/****
400*****
401****/
402
403static void
404onAddDialogResponse( GtkDialog * dialog,
405                     int         response,
406                     gpointer    core )
407{
408    char * folder;
409
410    /* remember this folder the next time we use this dialog */
411    folder = gtk_file_chooser_get_current_folder( GTK_FILE_CHOOSER( dialog ) );
412    pref_string_set( PREF_KEY_OPEN_DIALOG_FOLDER, folder );
413    g_free( folder );
414
415    if( response == GTK_RESPONSE_ACCEPT )
416    {
417        GtkFileChooser  * chooser = GTK_FILE_CHOOSER( dialog );
418        GtkWidget       * w = gtk_file_chooser_get_extra_widget( chooser );
419        GtkToggleButton * tb = GTK_TOGGLE_BUTTON( w );
420        const gboolean    showOptions = gtk_toggle_button_get_active( tb );
421        const pref_flag_t start = PREF_FLAG_DEFAULT;
422        const pref_flag_t prompt = showOptions
423                                 ? PREF_FLAG_TRUE
424                                 : PREF_FLAG_FALSE;
425        GSList * l = gtk_file_chooser_get_filenames( chooser );
426
427        tr_core_add_list( core, l, start, prompt, FALSE );
428    }
429
430    gtk_widget_destroy( GTK_WIDGET( dialog ) );
431}
432
433GtkWidget*
434addDialog( GtkWindow * parent,
435           TrCore *    core )
436{
437    GtkWidget *  w;
438    GtkWidget *  c;
439    const char * folder;
440
441    w = gtk_file_chooser_dialog_new( _( "Add a Torrent" ), parent,
442                                     GTK_FILE_CHOOSER_ACTION_OPEN,
443                                     GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
444                                     GTK_STOCK_ADD, GTK_RESPONSE_ACCEPT,
445                                     NULL );
446    gtk_dialog_set_alternative_button_order( GTK_DIALOG( w ),
447                                             GTK_RESPONSE_ACCEPT,
448                                             GTK_RESPONSE_CANCEL,
449                                             -1 );
450    gtk_file_chooser_set_select_multiple( GTK_FILE_CHOOSER( w ), TRUE );
451    addTorrentFilters( GTK_FILE_CHOOSER( w ) );
452    g_signal_connect( w, "response", G_CALLBACK(
453                          onAddDialogResponse ), core );
454
455    if( ( folder = pref_string_get( PREF_KEY_OPEN_DIALOG_FOLDER ) ) )
456        gtk_file_chooser_set_current_folder( GTK_FILE_CHOOSER( w ), folder );
457
458    c = gtk_check_button_new_with_mnemonic( _( "Show _options dialog" ) );
459    gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON( c ),
460                                 pref_flag_get( PREF_KEY_OPTIONS_PROMPT ) );
461    gtk_file_chooser_set_extra_widget( GTK_FILE_CHOOSER( w ), c );
462    gtk_widget_show( c );
463
464    gtk_widget_show( w );
465    return w;
466}
467
468/***
469****
470***/
471
472static void
473onAddURLResponse( GtkDialog * dialog, int response, gpointer user_data )
474{
475    gboolean destroy = TRUE;
476
477    if( response == GTK_RESPONSE_ACCEPT )
478    {
479        GtkWidget * e = GTK_WIDGET( g_object_get_data( G_OBJECT( dialog ), "url-entry" ) );
480        char * url = g_strdup( gtk_entry_get_text( GTK_ENTRY( e ) ) );
481        g_strstrip( url );
482
483        if( url && *url )
484        {
485            TrCore * core = user_data;
486
487            if( gtr_is_supported_url( url ) || gtr_is_magnet_link( url )
488                                            || gtr_is_hex_hashcode( url ) )
489            {
490                tr_core_add_from_url( core, url );
491            }
492            else
493            {
494                gtr_unrecognized_url_dialog( GTK_WIDGET( dialog ), url );
495                destroy = FALSE;
496            }
497        }
498
499        g_free( url );
500    }
501
502    if( destroy )
503        gtk_widget_destroy( GTK_WIDGET( dialog ) );
504}
505
506GtkWidget*
507addURLDialog( GtkWindow * parent, TrCore * core )
508{
509    int row;
510    GtkWidget * e;
511    GtkWidget * t;
512    GtkWidget * w;
513
514    w = gtk_dialog_new_with_buttons( _( "Add URL" ), parent,
515                                     GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR,
516                                     GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
517                                     GTK_STOCK_ADD, GTK_RESPONSE_ACCEPT,
518                                     NULL );
519    gtk_dialog_set_alternative_button_order( GTK_DIALOG( w ),
520                                             GTK_RESPONSE_ACCEPT,
521                                             GTK_RESPONSE_CANCEL,
522                                             -1 );
523    g_signal_connect( w, "response", G_CALLBACK( onAddURLResponse ), core );
524
525    row = 0;
526    t = hig_workarea_create( );
527    hig_workarea_add_section_title( t, &row, _( "Add torrent from URL" ) );
528    e = gtk_entry_new( );
529    g_object_set_data( G_OBJECT( w ), "url-entry", e );
530    hig_workarea_add_row( t, &row, _( "_URL" ), e, NULL );
531
532    gtk_box_pack_start( GTK_BOX( GTK_DIALOG( w )->vbox ), t, TRUE, TRUE, 0 );
533    gtk_widget_show_all( t );
534    gtk_widget_show( w );
535    return w;
536}
Note: See TracBrowser for help on using the repository browser.