/* * This file Copyright (C) 2008 Charles Kerr * * 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 #include #include "add-dialog.h" #include "conf.h" #include "file-list.h" #include "hig.h" #include "tr-prefs.h" /**** ***** ****/ #define N_RECENT 4 static GSList* get_recent_destinations( void ) { int i; GSList * list = NULL; for( i=0; inext ) l->data = g_strdup( l->data ); /* save the first N_RECENT directories */ for( l=list, i=0; l && (inext ) { char key[64]; g_snprintf( key, sizeof(key), "recent-download-dir-%d", i+1 ); pref_string_set( key, l->data ); } pref_save( ); /* cleanup */ g_slist_foreach( list, (GFunc)g_free, NULL ); g_slist_free( list ); } /**** ***** ****/ struct AddData { TrCore * core; GtkWidget * list; GtkWidget * run_check; GtkWidget * trash_check; char * filename; char * downloadDir; TrTorrent * gtor; tr_ctor * ctor; }; static void removeOldTorrent( struct AddData * data ) { if( data->gtor ) { file_list_set_torrent( data->list, NULL ); tr_torrent_set_remove_flag( data->gtor, TRUE ); g_object_unref( G_OBJECT( data->gtor ) ); data->gtor = NULL; } } static void addResponseCB( GtkDialog * dialog, gint response, gpointer gdata ) { struct AddData * data = gdata; if( data->gtor ) { if( response != GTK_RESPONSE_ACCEPT ) removeOldTorrent( data ); else { if( gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON( data->run_check ) ) ) tr_torrentStart( tr_torrent_handle( data->gtor ) ); tr_core_add_torrent( data->core, data->gtor ); if( gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON( data->trash_check ) ) ) tr_file_trash_or_unlink( data->filename ); save_recent_destination( data->downloadDir ); } } tr_ctorFree( data->ctor ); g_free( data->filename ); g_free( data->downloadDir ); g_free( data ); gtk_widget_destroy( GTK_WIDGET( dialog ) ); } static void updateTorrent( struct AddData * o ) { if( o->gtor ) tr_torrentSetDownloadDir( tr_torrent_handle( o->gtor ), o->downloadDir ); file_list_set_torrent( o->list, o->gtor ); } static void sourceChanged( GtkFileChooserButton * b, gpointer gdata ) { struct AddData * data = gdata; removeOldTorrent( data ); g_free( data->filename ); data->filename = gtk_file_chooser_get_filename( GTK_FILE_CHOOSER( b ) ); /* maybe instantiate a torrent */ if( data->filename ) { int err = 0; tr_torrent * torrent; tr_handle * handle = tr_core_handle( data->core ); tr_ctorSetMetainfoFromFile( data->ctor, data->filename ); tr_ctorSetDownloadDir( data->ctor, TR_FORCE, data->downloadDir ); tr_ctorSetPaused( data->ctor, TR_FORCE, TRUE ); tr_ctorSetDeleteSource( data->ctor, FALSE ); if(( torrent = tr_torrentNew( handle, data->ctor, &err ))) data->gtor = tr_torrent_new_preexisting( torrent ); else addTorrentErrorDialog( GTK_WIDGET(b), err, data->filename ); } updateTorrent( data ); } static void verifyRequested( GtkButton * button UNUSED, gpointer gdata ) { struct AddData * data = gdata; if( data->gtor ) tr_torrentVerify( tr_torrent_handle( data->gtor ) ); } static void downloadDirChanged( GtkFileChooserButton * b, gpointer gdata ) { char * fname = gtk_file_chooser_get_filename( GTK_FILE_CHOOSER( b ) ); if( fname ) { struct AddData * data = gdata; g_free( data->downloadDir ); data->downloadDir = fname; updateTorrent( data ); verifyRequested( NULL, data ); } } 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* addSingleTorrentDialog( GtkWindow * parent, TrCore * core, tr_ctor * ctor ) { int row; int col; const char * str; GtkWidget * w; GtkWidget * d; GtkWidget * t; GtkWidget * l; struct AddData * data; uint8_t flag; GSList * list; GSList * walk; /* make the dialog */ d = gtk_dialog_new_with_buttons( _( "Torrent Options" ), parent, GTK_DIALOG_DESTROY_WITH_PARENT|GTK_DIALOG_NO_SEPARATOR, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_ADD, 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 AddData, 1 ); data->core = core; data->ctor = ctor; data->filename = g_strdup( tr_ctorGetSourceFile( ctor ) ); data->downloadDir = g_strdup( str ); data->list = file_list_new( NULL ); data->trash_check = gtk_check_button_new_with_mnemonic( _( "_Move source file to Trash" ) ); data->run_check = gtk_check_button_new_with_mnemonic( _( "_Start when added" ) ); g_signal_connect( G_OBJECT( d ), "response", G_CALLBACK( addResponseCB ), data ); t = gtk_table_new( 6, 2, FALSE ); gtk_container_set_border_width( GTK_CONTAINER( t ), GUI_PAD_BIG ); gtk_table_set_row_spacings( GTK_TABLE( t ), GUI_PAD ); gtk_table_set_col_spacings( GTK_TABLE( t ), GUI_PAD_BIG ); row = col = 0; l = gtk_label_new_with_mnemonic( _( "_Torrent file:" ) ); gtk_misc_set_alignment( GTK_MISC( l ), 0.0f, 0.5f ); gtk_table_attach( GTK_TABLE( t ), l, col, col+1, row, row+1, GTK_FILL, 0, 0, 0 ); ++col; w = gtk_file_chooser_button_new( _( "Select Source File" ), GTK_FILE_CHOOSER_ACTION_OPEN ); gtk_table_attach( GTK_TABLE( t ), w, col, col+1, row, row+1, ~0, 0, 0, 0 ); gtk_label_set_mnemonic_widget( GTK_LABEL( l ), w ); addTorrentFilters( GTK_FILE_CHOOSER( w ) ); if( data->filename ) if( !gtk_file_chooser_set_filename( GTK_FILE_CHOOSER( w ), data->filename ) ) g_warning( "couldn't select '%s'", data->filename ); g_signal_connect( w, "selection-changed", G_CALLBACK( sourceChanged ), data ); ++row; col = 0; l = gtk_label_new_with_mnemonic( _( "_Destination folder:" ) ); gtk_misc_set_alignment( GTK_MISC( l ), 0.0f, 0.5f ); gtk_table_attach( GTK_TABLE( t ), l, col, col+1, row, row+1, GTK_FILL, 0, 0, 0 ); ++col; w = gtk_file_chooser_button_new( _( "Select Destination Folder" ), GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER ); if( !gtk_file_chooser_set_filename( 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_table_attach( GTK_TABLE( t ), w, col, col+1, row, row+1, ~0, 0, 0, 0 ); gtk_label_set_mnemonic_widget( GTK_LABEL( l ), w ); g_signal_connect( w, "selection-changed", G_CALLBACK( downloadDirChanged ), data ); ++row; col = 0; w = data->list; gtk_widget_set_size_request ( w, 466u, 300u ); gtk_table_attach_defaults( GTK_TABLE( t ), w, col, col+2, row, row+1 ); ++row; col = 0; w = gtk_button_new_with_mnemonic( _( "_Verify Local Data" ) ); gtk_table_attach( GTK_TABLE( t ), w, col, col+1, row, row+1, GTK_FILL, 0, 0, 0 ); g_signal_connect( w, "clicked", G_CALLBACK( verifyRequested ), data ); ++row; col = 0; 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_table_attach( GTK_TABLE( t ), w, col, col+2, row, row+1, GTK_FILL, 0, 0, 0 ); ++row; col = 0; w = data->trash_check; if( tr_ctorGetDeleteSource( ctor, &flag ) ) g_assert_not_reached( ); gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON( w ), flag ); gtk_table_attach( GTK_TABLE( t ), w, col, col+2, row, row+1, GTK_FILL, 0, 0, 0 ); gtk_box_pack_start_defaults( GTK_BOX( GTK_DIALOG( d )->vbox ), t ); gtk_widget_show_all( d ); return d; } /**** ***** ****/ static void onAddDialogResponse( 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 ) ); pref_string_set( PREF_KEY_OPEN_DIALOG_FOLDER, folder ); g_free( folder ); if( response == GTK_RESPONSE_ACCEPT ) { GtkWidget * w = gtk_file_chooser_get_extra_widget( GTK_FILE_CHOOSER( dialog ) ); const gboolean showOptions = gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON ( w ) ); const pref_flag_t start = PREF_FLAG_DEFAULT; const pref_flag_t prompt = showOptions ? PREF_FLAG_TRUE : PREF_FLAG_FALSE; GSList * l = gtk_file_chooser_get_filenames( GTK_FILE_CHOOSER( dialog ) ); tr_core_add_list( core, l, start, prompt ); } gtk_widget_destroy( GTK_WIDGET( dialog ) ); } GtkWidget* addDialog( GtkWindow * parent, TrCore * core ) { GtkWidget * w; GtkWidget * c; const char * folder; w = gtk_file_chooser_dialog_new( _( "Add a Torrent" ), parent, GTK_FILE_CHOOSER_ACTION_OPEN, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_ADD, 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(onAddDialogResponse), core ); if(( folder = pref_string_get( PREF_KEY_OPEN_DIALOG_FOLDER ))) gtk_file_chooser_set_current_folder( GTK_FILE_CHOOSER( w ), folder ); c = gtk_check_button_new_with_mnemonic( _( "Display _options dialog" ) ); gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON( c ), pref_flag_get( PREF_KEY_OPTIONS_PROMPT ) ); gtk_file_chooser_set_extra_widget( GTK_FILE_CHOOSER( w ), c ); gtk_widget_show( c ); gtk_widget_show( w ); return w; }