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

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

(trunk gtk) #3143 "remote RPC calls cause gtk+ API calls to be made from the wrong thread" -- fixed in trunk for 2.00

  • Property svn:keywords set to Date Rev Author Id
File size: 17.0 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 10505 2010-04-21 04:38:54Z 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                gtr_file_trash_or_remove( data->filename );
137
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        tr_torrentVerify( tor );
159    }
160}
161
162/**
163 * When the source .torrent file is deleted
164 * (such as, if it was a temp file that a web browser passed to us),
165 * gtk invokes this callback and `filename' will be NULL.
166 * The `filename' tests here are to prevent us from losing the current
167 * metadata when that happens.
168 */
169static void
170sourceChanged( GtkFileChooserButton * b, gpointer gdata )
171{
172    struct AddData * data = gdata;
173    char * filename = gtk_file_chooser_get_filename( GTK_FILE_CHOOSER( b ) );
174
175    /* maybe instantiate a torrent */
176    if( data->filename || !data->gtor )
177    {
178        int          err = 0;
179        int          new_file = 0;
180        tr_torrent * torrent;
181
182        if( filename
183          && ( !data->filename || strcmp( filename, data->filename ) ) )
184        {
185            g_free( data->filename );
186            data->filename = g_strdup( filename );
187            tr_ctorSetMetainfoFromFile( data->ctor, data->filename );
188            new_file = 1;
189        }
190
191        tr_ctorSetDownloadDir( data->ctor, TR_FORCE, data->downloadDir );
192        tr_ctorSetPaused( data->ctor, TR_FORCE, TRUE );
193        tr_ctorSetDeleteSource( data->ctor, FALSE );
194
195        if( ( torrent = tr_torrentNew( data->ctor, &err ) ) )
196        {
197            removeOldTorrent( data );
198            data->gtor = tr_torrent_new_preexisting( torrent );
199        }
200        else if( new_file )
201        {
202            addTorrentErrorDialog( GTK_WIDGET( b ), err, data->filename );
203        }
204
205        updateTorrent( data );
206    }
207
208    g_free( filename );
209}
210
211static void
212downloadDirChanged( GtkFileChooserButton * b, gpointer gdata )
213{
214    struct AddData * data = gdata;
215    char * fname = gtk_file_chooser_get_filename( GTK_FILE_CHOOSER( b ) );
216
217    if( fname && ( !data->downloadDir || strcmp( fname, data->downloadDir ) ) )
218    {
219        g_free( data->downloadDir );
220        data->downloadDir = g_strdup( fname );
221        updateTorrent( data );
222    }
223
224    g_free( fname );
225}
226
227static void
228addTorrentFilters( GtkFileChooser * chooser )
229{
230    GtkFileFilter * filter;
231
232    filter = gtk_file_filter_new( );
233    gtk_file_filter_set_name( filter, _( "Torrent files" ) );
234    gtk_file_filter_add_pattern( filter, "*.torrent" );
235    gtk_file_chooser_add_filter( chooser, filter );
236
237    filter = gtk_file_filter_new( );
238    gtk_file_filter_set_name( filter, _( "All files" ) );
239    gtk_file_filter_add_pattern( filter, "*" );
240    gtk_file_chooser_add_filter( chooser, filter );
241}
242
243/****
244*****
245****/
246
247GtkWidget*
248addSingleTorrentDialog( GtkWindow * parent, TrCore * core, tr_ctor * ctor )
249{
250    int              row;
251    int              col;
252    const char *     str;
253    GtkWidget *      w;
254    GtkWidget *      d;
255    GtkWidget *      t;
256    GtkWidget *      l;
257    GtkWidget *      grab;
258    GtkWidget *      source_chooser;
259    struct AddData * data;
260    uint8_t          flag;
261    GSList *         list;
262    GSList *         walk;
263
264    /* make the dialog */
265    d = gtk_dialog_new_with_buttons( _( "Torrent Options" ), parent,
266                                     GTK_DIALOG_DESTROY_WITH_PARENT |
267                                     GTK_DIALOG_NO_SEPARATOR,
268                                     NULL );
269    gtk_dialog_add_button( GTK_DIALOG( d ), GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL );
270    grab = gtk_dialog_add_button( GTK_DIALOG( d ), GTK_STOCK_ADD, GTK_RESPONSE_ACCEPT );
271    gtk_dialog_set_default_response( GTK_DIALOG( d ),
272                                     GTK_RESPONSE_ACCEPT );
273    gtk_dialog_set_alternative_button_order( GTK_DIALOG( d ),
274                                             GTK_RESPONSE_ACCEPT,
275                                             GTK_RESPONSE_CANCEL,
276                                             -1 );
277
278    if( tr_ctorGetDownloadDir( ctor, TR_FORCE, &str ) )
279        g_assert_not_reached( );
280    g_assert( str );
281
282    data = g_new0( struct AddData, 1 );
283    data->core = core;
284    data->ctor = ctor;
285    data->filename = g_strdup( tr_ctorGetSourceFile( ctor ) );
286    data->downloadDir = g_strdup( str );
287    data->list = file_list_new( core, 0 );
288    str = _( "Mo_ve .torrent file to the trash" );
289    data->trash_check = gtk_check_button_new_with_mnemonic( str );
290    str = _( "_Start when added" );
291    data->run_check = gtk_check_button_new_with_mnemonic( str );
292
293    data->priority_combo = gtr_priority_combo_new( );
294    gtr_priority_combo_set_value( data->priority_combo, TR_PRI_NORMAL );
295
296    g_signal_connect( G_OBJECT( d ), "response",
297                      G_CALLBACK( addResponseCB ), data );
298
299    t = gtk_table_new( 6, 2, FALSE );
300    gtk_container_set_border_width( GTK_CONTAINER( t ), GUI_PAD_BIG );
301    gtk_table_set_row_spacings( GTK_TABLE( t ), GUI_PAD );
302    gtk_table_set_col_spacings( GTK_TABLE( t ), GUI_PAD_BIG );
303
304    row = col = 0;
305    l = gtk_label_new_with_mnemonic( _( "_Torrent file:" ) );
306    gtk_misc_set_alignment( GTK_MISC( l ), 0.0f, 0.5f );
307    gtk_table_attach( GTK_TABLE( t ), l, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0 );
308    ++col;
309    w = gtk_file_chooser_button_new( _( "Select Source File" ),
310                                     GTK_FILE_CHOOSER_ACTION_OPEN );
311    source_chooser = w;
312    gtk_table_attach( GTK_TABLE( t ), w, col, col + 1, row, row + 1, ~0, 0, 0, 0 );
313    gtk_label_set_mnemonic_widget( GTK_LABEL( l ), w );
314    addTorrentFilters( GTK_FILE_CHOOSER( w ) );
315    g_signal_connect( w, "selection-changed",
316                      G_CALLBACK( sourceChanged ), data );
317
318    ++row;
319    col = 0;
320    l = gtk_label_new_with_mnemonic( _( "_Destination folder:" ) );
321    gtk_misc_set_alignment( GTK_MISC( l ), 0.0f, 0.5f );
322    gtk_table_attach( GTK_TABLE( t ), l, col, col + 1, row, row + 1, GTK_FILL, 0, 0, 0 );
323    ++col;
324    w = gtk_file_chooser_button_new( _( "Select Destination Folder" ),
325                                     GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER );
326    if( !gtk_file_chooser_set_current_folder( GTK_FILE_CHOOSER( w ),
327                                              data->downloadDir ) )
328        g_warning( "couldn't select '%s'", data->downloadDir );
329    list = get_recent_destinations( );
330    for( walk = list; walk; walk = walk->next )
331        gtk_file_chooser_add_shortcut_folder( GTK_FILE_CHOOSER( w ), walk->data, NULL );
332    g_slist_free( list );
333    gtk_table_attach( GTK_TABLE( t ), w, col, col + 1, row, row + 1, ~0, 0, 0, 0 );
334    gtk_label_set_mnemonic_widget( GTK_LABEL( l ), w );
335    g_signal_connect( w, "selection-changed",
336                      G_CALLBACK( downloadDirChanged ), data );
337
338    ++row;
339    col = 0;
340    w = data->list;
341    gtk_widget_set_size_request ( w, 466u, 300u );
342    gtk_table_attach_defaults( GTK_TABLE( t ), w, col, col + 2, row, row + 1 );
343
344    ++row;
345    col = 0;
346    w = gtk_label_new_with_mnemonic( _( "Torrent _priority:" ) );
347    gtk_misc_set_alignment( GTK_MISC( w ), 0.0f, 0.5f );
348    gtk_table_attach( GTK_TABLE( t ), w, col, col + 1, row, row + 1, ~0, 0, 0, 0 );
349    ++col;
350    gtk_table_attach( GTK_TABLE( t ), data->priority_combo, col, col + 1, row, row + 1, ~0, 0, 0, 0 );
351    gtk_label_set_mnemonic_widget( GTK_LABEL( w ), data->priority_combo );
352
353    ++row;
354    col = 0;
355    w = data->run_check;
356    if( tr_ctorGetPaused( ctor, TR_FORCE, &flag ) )
357        g_assert_not_reached( );
358    gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON( w ), !flag );
359    gtk_table_attach( GTK_TABLE( t ), w, col, col + 2, row, row + 1, GTK_FILL, 0, 0, 0 );
360
361    ++row;
362    col = 0;
363    w = data->trash_check;
364    if( tr_ctorGetDeleteSource( ctor, &flag ) )
365        g_assert_not_reached( );
366    gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON( w ), flag );
367    gtk_table_attach( GTK_TABLE( t ), w, col, col + 2, row, row + 1, GTK_FILL, 0, 0, 0 );
368
369    ++row;
370    col = 0;
371    w = data->priority_combo;
372
373    /* trigger sourceChanged, either directly or indirectly,
374     * so that it creates the tor/gtor objects */
375    w = source_chooser;
376    if( data->filename )
377        gtk_file_chooser_set_filename( GTK_FILE_CHOOSER( w ), data->filename );
378    else
379        sourceChanged( GTK_FILE_CHOOSER_BUTTON( w ), data );
380
381    gtk_box_pack_start( GTK_BOX( GTK_DIALOG( d )->vbox ), t, TRUE, TRUE, 0 );
382
383    gtk_widget_grab_focus( grab );
384    gtk_widget_show_all( d );
385    return d;
386}
387
388/****
389*****
390****/
391
392static void
393onAddDialogResponse( GtkDialog * dialog,
394                     int         response,
395                     gpointer    core )
396{
397    char * folder;
398
399    /* remember this folder the next time we use this dialog */
400    folder = gtk_file_chooser_get_current_folder( GTK_FILE_CHOOSER( dialog ) );
401    pref_string_set( PREF_KEY_OPEN_DIALOG_FOLDER, folder );
402    g_free( folder );
403
404    if( response == GTK_RESPONSE_ACCEPT )
405    {
406        GtkFileChooser  * chooser = GTK_FILE_CHOOSER( dialog );
407        GtkWidget       * w = gtk_file_chooser_get_extra_widget( chooser );
408        GtkToggleButton * tb = GTK_TOGGLE_BUTTON( w );
409        const gboolean    showOptions = gtk_toggle_button_get_active( tb );
410        const pref_flag_t start = PREF_FLAG_DEFAULT;
411        const pref_flag_t prompt = showOptions
412                                 ? PREF_FLAG_TRUE
413                                 : PREF_FLAG_FALSE;
414        GSList * l = gtk_file_chooser_get_filenames( chooser );
415
416        tr_core_add_list( core, l, start, prompt, FALSE );
417    }
418
419    gtk_widget_destroy( GTK_WIDGET( dialog ) );
420}
421
422GtkWidget*
423addDialog( GtkWindow * parent,
424           TrCore *    core )
425{
426    GtkWidget *  w;
427    GtkWidget *  c;
428    const char * folder;
429
430    w = gtk_file_chooser_dialog_new( _( "Add a Torrent" ), parent,
431                                     GTK_FILE_CHOOSER_ACTION_OPEN,
432                                     GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
433                                     GTK_STOCK_ADD, GTK_RESPONSE_ACCEPT,
434                                     NULL );
435    gtk_dialog_set_alternative_button_order( GTK_DIALOG( w ),
436                                             GTK_RESPONSE_ACCEPT,
437                                             GTK_RESPONSE_CANCEL,
438                                             -1 );
439    gtk_file_chooser_set_select_multiple( GTK_FILE_CHOOSER( w ), TRUE );
440    addTorrentFilters( GTK_FILE_CHOOSER( w ) );
441    g_signal_connect( w, "response", G_CALLBACK(
442                          onAddDialogResponse ), core );
443
444    if( ( folder = pref_string_get( PREF_KEY_OPEN_DIALOG_FOLDER ) ) )
445        gtk_file_chooser_set_current_folder( GTK_FILE_CHOOSER( w ), folder );
446
447    c = gtk_check_button_new_with_mnemonic( _( "Show _options dialog" ) );
448    gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON( c ),
449                                 pref_flag_get( PREF_KEY_OPTIONS_PROMPT ) );
450    gtk_file_chooser_set_extra_widget( GTK_FILE_CHOOSER( w ), c );
451    gtk_widget_show( c );
452
453    gtk_widget_show( w );
454    return w;
455}
456
457/***
458****
459***/
460
461static void
462onAddURLResponse( GtkDialog * dialog, int response, gpointer user_data )
463{
464    gboolean destroy = TRUE;
465
466    if( response == GTK_RESPONSE_ACCEPT )
467    {
468        GtkWidget * e = GTK_WIDGET( g_object_get_data( G_OBJECT( dialog ), "url-entry" ) );
469        char * url = g_strdup( gtk_entry_get_text( GTK_ENTRY( e ) ) );
470        g_strstrip( url );
471
472        if( url && *url )
473        {
474            TrCore * core = user_data;
475
476            if( gtr_is_supported_url( url ) || gtr_is_magnet_link( url )
477                                            || gtr_is_hex_hashcode( url ) )
478            {
479                tr_core_add_from_url( core, url );
480            }
481            else
482            {
483                gtr_unrecognized_url_dialog( GTK_WIDGET( dialog ), url );
484                destroy = FALSE;
485            }
486        }
487
488        g_free( url );
489    }
490
491    if( destroy )
492        gtk_widget_destroy( GTK_WIDGET( dialog ) );
493}
494
495static void
496paste_clipboard_url_into_entry( GtkWidget * e )
497{
498  size_t i;
499
500  char * text[] = {
501    gtk_clipboard_wait_for_text( gtk_clipboard_get( GDK_SELECTION_PRIMARY ) ),
502    gtk_clipboard_wait_for_text( gtk_clipboard_get( GDK_SELECTION_CLIPBOARD ) )
503  };
504
505  for( i=0; i<G_N_ELEMENTS(text); ++i ) {
506      char * s = text[i];
507      if( s && ( gtr_is_supported_url( s ) || gtr_is_magnet_link( s ) ) ) {
508          gtk_entry_set_text( GTK_ENTRY( e ), s );
509          break;
510      }
511  }
512
513  for( i=0; i<G_N_ELEMENTS(text); ++i )
514    g_free( text[i] );
515}
516
517GtkWidget*
518addURLDialog( GtkWindow * parent, TrCore * core )
519{
520    int row;
521    GtkWidget * e;
522    GtkWidget * t;
523    GtkWidget * w;
524
525    w = gtk_dialog_new_with_buttons( _( "Add URL" ), parent,
526                                     GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR,
527                                     GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
528                                     GTK_STOCK_ADD, GTK_RESPONSE_ACCEPT,
529                                     NULL );
530    gtk_dialog_set_alternative_button_order( GTK_DIALOG( w ),
531                                             GTK_RESPONSE_ACCEPT,
532                                             GTK_RESPONSE_CANCEL,
533                                             -1 );
534    g_signal_connect( w, "response", G_CALLBACK( onAddURLResponse ), core );
535
536    row = 0;
537    t = hig_workarea_create( );
538    hig_workarea_add_section_title( t, &row, _( "Add torrent from URL" ) );
539    e = gtk_entry_new( );
540    paste_clipboard_url_into_entry( e );
541    g_object_set_data( G_OBJECT( w ), "url-entry", e );
542    hig_workarea_add_row( t, &row, _( "_URL" ), e, NULL );
543
544    gtk_box_pack_start( GTK_BOX( GTK_DIALOG( w )->vbox ), t, TRUE, TRUE, 0 );
545    gtk_widget_show_all( t );
546    gtk_widget_show( w );
547    return w;
548}
Note: See TracBrowser for help on using the repository browser.