(gtk) #769: when removing a torrent, prompt for confirmation if (a) the torrent is incomplete, (b) the torrent is connected to peers, (c) the user is also deleting the downloaded files.
This commit is contained in:
parent
1df3d8d844
commit
0cebab1247
|
@ -110,8 +110,8 @@ static GtkActionEntry entries[] =
|
|||
N_("_Verify Local Data"), NULL, NULL, G_CALLBACK(action_cb) },
|
||||
{ "pause-torrent", GTK_STOCK_MEDIA_PAUSE,
|
||||
N_("_Pause"), "<control>P", NULL, G_CALLBACK(action_cb) },
|
||||
{ "close-torrent", GTK_STOCK_CLOSE, NULL, "Delete", NULL, G_CALLBACK(action_cb) },
|
||||
{ "delete-torrent", GTK_STOCK_DELETE, N_("_Delete Files and Close"), "<control>Delete", NULL, G_CALLBACK(action_cb) },
|
||||
{ "remove-torrent", GTK_STOCK_REMOVE, NULL, "Delete", NULL, G_CALLBACK(action_cb) },
|
||||
{ "delete-torrent", GTK_STOCK_DELETE, N_("_Delete Files and Remove"), "<control>Delete", NULL, G_CALLBACK(action_cb) },
|
||||
{ "new-torrent", GTK_STOCK_NEW, N_("_New..."), NULL,
|
||||
N_("Create a new torrent"),
|
||||
G_CALLBACK(action_cb) },
|
||||
|
|
|
@ -292,58 +292,98 @@ struct DeleteData
|
|||
gboolean delete_files;
|
||||
GList * torrents;
|
||||
TrCore * core;
|
||||
int busyCount;
|
||||
};
|
||||
|
||||
static void
|
||||
removeTorrents( struct DeleteData * data )
|
||||
{
|
||||
GList * l;
|
||||
for( l=data->torrents; l!=NULL; l=l->next )
|
||||
tr_core_remove_torrent( data->core, l->data, data->delete_files );
|
||||
g_list_free( data->torrents );
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
removeResponse( GtkDialog * dialog, gint response, gpointer gdata )
|
||||
{
|
||||
struct DeleteData * data = gdata;
|
||||
const int doRemove = response == GTK_RESPONSE_ACCEPT;
|
||||
const int doDelete = data->delete_files;
|
||||
GList * l;
|
||||
|
||||
for( l=data->torrents; l!=NULL; l=l->next )
|
||||
{
|
||||
TrTorrent * gtor = TR_TORRENT( l->data );
|
||||
|
||||
if( doRemove )
|
||||
tr_core_remove_torrent( data->core, gtor, doDelete );
|
||||
if( response == GTK_RESPONSE_ACCEPT )
|
||||
removeTorrents( data );
|
||||
else
|
||||
g_object_unref( G_OBJECT( gtor ) );
|
||||
}
|
||||
g_list_foreach( data->torrents, (GFunc)g_object_unref, NULL );
|
||||
|
||||
gtk_widget_destroy( GTK_WIDGET( dialog ) );
|
||||
g_list_free( data->torrents );
|
||||
g_free( data );
|
||||
}
|
||||
|
||||
static void
|
||||
tabulateTorrents( gpointer gtor, gpointer gdata )
|
||||
{
|
||||
struct DeleteData * data = gdata;
|
||||
const tr_stat * stat = tr_torrent_stat( gtor );
|
||||
|
||||
if( stat->leftUntilDone || stat->peersConnected )
|
||||
++data->busyCount;
|
||||
}
|
||||
|
||||
void
|
||||
confirmDelete( GtkWindow * parent,
|
||||
confirmRemove( GtkWindow * parent,
|
||||
TrCore * core,
|
||||
GList * torrents )
|
||||
GList * torrents,
|
||||
gboolean delete_files )
|
||||
{
|
||||
GtkWidget * d;
|
||||
char text[128];
|
||||
struct DeleteData * dd = g_new0( struct DeleteData, 1 );
|
||||
struct DeleteData * dd;
|
||||
const int count = g_list_length( torrents );
|
||||
const char * primary_text;
|
||||
const char * secondary_text;
|
||||
|
||||
if( !count )
|
||||
return;
|
||||
|
||||
dd = g_new0( struct DeleteData, 1 );
|
||||
dd->core = core;
|
||||
dd->torrents = torrents;
|
||||
dd->delete_files = TRUE;
|
||||
dd->delete_files = delete_files;
|
||||
|
||||
g_list_foreach( torrents, tabulateTorrents, dd );
|
||||
|
||||
if( !dd->busyCount && !delete_files ) /* don't prompt boring torrents */
|
||||
{
|
||||
removeTorrents( dd );
|
||||
g_free( dd );
|
||||
return;
|
||||
}
|
||||
|
||||
if( !delete_files )
|
||||
primary_text = ngettext( "Remove torrent?", "Remove torrents?", count );
|
||||
else
|
||||
primary_text = ngettext( "Delete this torrent's downloaded files?",
|
||||
"Delete these torrents' downloaded files?",
|
||||
count );
|
||||
|
||||
if( dd->busyCount > 1 )
|
||||
secondary_text = _( "Some of these torrents are incomplete or connected to peers." );
|
||||
else if( dd->busyCount == 0 )
|
||||
secondary_text = NULL;
|
||||
else
|
||||
secondary_text = ngettext( "This torrent is incomplete or connected to peers.",
|
||||
"One of these torrents is incomplete or connected to peers.",
|
||||
count );
|
||||
|
||||
g_snprintf( text, sizeof( text ),
|
||||
ngettext( "Delete torrent?",
|
||||
"Delete torrents?",
|
||||
g_list_length( torrents ) ) );
|
||||
d = gtk_message_dialog_new_with_markup( parent,
|
||||
GTK_DIALOG_DESTROY_WITH_PARENT,
|
||||
GTK_MESSAGE_WARNING,
|
||||
( delete_files ? GTK_MESSAGE_WARNING : GTK_MESSAGE_QUESTION ),
|
||||
GTK_BUTTONS_NONE,
|
||||
"<b>%s</b>", text );
|
||||
gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG( d ),
|
||||
_( "This removes the torrent and deletes the downloaded files!" ) );
|
||||
"<b>%s</b>", primary_text );
|
||||
if( secondary_text )
|
||||
gtk_message_dialog_format_secondary_markup( GTK_MESSAGE_DIALOG( d ), secondary_text );
|
||||
gtk_dialog_add_buttons( GTK_DIALOG( d ),
|
||||
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
|
||||
GTK_STOCK_DELETE, GTK_RESPONSE_ACCEPT,
|
||||
(delete_files ? GTK_STOCK_DELETE : GTK_STOCK_REMOVE), GTK_RESPONSE_ACCEPT,
|
||||
NULL );
|
||||
gtk_dialog_set_default_response( GTK_DIALOG ( d ),
|
||||
GTK_RESPONSE_CANCEL );
|
||||
|
|
|
@ -38,6 +38,9 @@ GtkWidget* promptfordir( GtkWindow * parent,
|
|||
/* prompt if the user wants to quit, calls func with cbdata if they do */
|
||||
void askquit( TrCore*, GtkWindow* parent, callbackfunc_t func, void* cbdata );
|
||||
|
||||
void confirmDelete( GtkWindow * parent, TrCore * core, GList * torrents );
|
||||
void confirmRemove( GtkWindow * parent,
|
||||
TrCore * core,
|
||||
GList * gtorrents,
|
||||
gboolean doDelete );
|
||||
|
||||
#endif /* TG_PREFS_H */
|
||||
|
|
27
gtk/main.c
27
gtk/main.c
|
@ -199,7 +199,7 @@ refreshTorrentActions( GtkTreeSelection * s )
|
|||
gtk_tree_selection_selected_foreach( s, accumulateStatusForeach, &counts );
|
||||
action_sensitize( "pause-torrent", counts.activeCount!=0 );
|
||||
action_sensitize( "start-torrent", counts.inactiveCount!=0 );
|
||||
action_sensitize( "close-torrent", counts.totalCount!=0 );
|
||||
action_sensitize( "remove-torrent", counts.totalCount!=0 );
|
||||
action_sensitize( "delete-torrent", counts.totalCount!=0 );
|
||||
action_sensitize( "verify-torrent", counts.totalCount!=0 );
|
||||
action_sensitize( "show-torrent-details", counts.totalCount==1 );
|
||||
|
@ -1037,27 +1037,14 @@ accumulateSelectedTorrents( GtkTreeModel * model,
|
|||
}
|
||||
|
||||
static void
|
||||
closeSelectedForeach( gpointer gtor, gpointer gdata )
|
||||
{
|
||||
struct cbdata * data = gdata;
|
||||
tr_core_remove_torrent( data->core, gtor, FALSE );
|
||||
}
|
||||
|
||||
static void
|
||||
closeSelected( struct cbdata * data, gboolean doDelete )
|
||||
removeSelected( struct cbdata * data, gboolean delete_files )
|
||||
{
|
||||
GList * l = NULL;
|
||||
GtkTreeSelection * s = tr_window_get_selection( data->wind );
|
||||
gtk_tree_selection_selected_foreach( s, accumulateSelectedTorrents, &l );
|
||||
gtk_tree_selection_unselect_all( s );
|
||||
if( l ) {
|
||||
if( doDelete )
|
||||
confirmDelete( data->wind, data->core, l );
|
||||
else {
|
||||
g_list_foreach( l, closeSelectedForeach, data );
|
||||
g_list_free( l );
|
||||
}
|
||||
}
|
||||
if( l )
|
||||
confirmRemove( data->wind, data->core, l, delete_files );
|
||||
}
|
||||
|
||||
void
|
||||
|
@ -1111,13 +1098,13 @@ doAction ( const char * action_name, gpointer user_data )
|
|||
tr_core_handle( data->core ) );
|
||||
gtk_widget_show_all( w );
|
||||
}
|
||||
else if( !strcmp( action_name, "close-torrent" ) )
|
||||
else if( !strcmp( action_name, "remove-torrent" ) )
|
||||
{
|
||||
closeSelected( data, FALSE );
|
||||
removeSelected( data, FALSE );
|
||||
}
|
||||
else if( !strcmp( action_name, "delete-torrent" ) )
|
||||
{
|
||||
closeSelected( data, TRUE );
|
||||
removeSelected( data, TRUE );
|
||||
}
|
||||
else if (!strcmp (action_name, "close"))
|
||||
{
|
||||
|
|
6
gtk/ui.h
6
gtk/ui.h
|
@ -10,7 +10,7 @@ const char * fallback_ui_file =
|
|||
" <menuitem action='pause-torrent'/>\n"
|
||||
" <menuitem action='verify-torrent'/>\n"
|
||||
" <menuitem action='show-torrent-details'/>\n"
|
||||
" <menuitem action='close-torrent'/>\n"
|
||||
" <menuitem action='remove-torrent'/>\n"
|
||||
" <menuitem action='delete-torrent'/>\n"
|
||||
" <separator/>\n"
|
||||
" <menuitem action='close'/>\n"
|
||||
|
@ -49,7 +49,7 @@ const char * fallback_ui_file =
|
|||
" <toolitem action='open-torrent-toolbar'/>\n"
|
||||
" <toolitem action='start-torrent'/>\n"
|
||||
" <toolitem action='pause-torrent'/>\n"
|
||||
" <toolitem action='close-torrent'/>\n"
|
||||
" <toolitem action='remove-torrent'/>\n"
|
||||
" <separator/>\n"
|
||||
" <toolitem action='show-torrent-details'/>\n"
|
||||
" </toolbar>\n"
|
||||
|
@ -62,7 +62,7 @@ const char * fallback_ui_file =
|
|||
" <menuitem action='verify-torrent'/>\n"
|
||||
" <menuitem action='update-tracker'/>\n"
|
||||
" <separator/>\n"
|
||||
" <menuitem action='close-torrent'/>\n"
|
||||
" <menuitem action='remove-torrent'/>\n"
|
||||
" <menuitem action='delete-torrent'/>\n"
|
||||
" </popup>\n"
|
||||
"\n"
|
||||
|
|
Loading…
Reference in New Issue