source: trunk/gtk/main.c @ 6327

Last change on this file since 6327 was 6327, checked in by charles, 14 years ago

(gtk) #1081: Manual Announce grayed out when allowed

  • Property svn:keywords set to Date Rev Author Id
File size: 42.8 KB
Line 
1/******************************************************************************
2 * $Id: main.c 6327 2008-07-13 18:14:24Z charles $
3 *
4 * Copyright (c) 2005-2008 Transmission authors and contributors
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a
7 * copy of this software and associated documentation files (the "Software"),
8 * to deal in the Software without restriction, including without limitation
9 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
10 * and/or sell copies of the Software, and to permit persons to whom the
11 * Software is furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22 * DEALINGS IN THE SOFTWARE.
23 *****************************************************************************/
24
25#include <sys/param.h>
26#include <signal.h>
27#include <string.h>
28#include <stdio.h>
29#include <stdlib.h>
30#include <time.h>
31#include <unistd.h>
32
33#include <gtk/gtk.h>
34#include <glib/gi18n.h>
35#include <glib/gstdio.h>
36
37#include <gdk/gdk.h>
38#ifdef GDK_WINDOWING_X11
39#include <X11/Xatom.h>
40#include <gdk/gdkx.h>
41#endif
42
43#include <libtransmission/version.h>
44
45#include "actions.h"
46#include "add-dialog.h"
47#include "conf.h"
48#include "details.h"
49#include "dialogs.h"
50#include "hig.h"
51#include "makemeta-ui.h"
52#include "msgwin.h"
53#include "notify.h"
54#include "stats.h"
55#include "tr-core.h"
56#include "tr-icon.h"
57#include "tr-prefs.h"
58#include "tr-torrent.h"
59#include "tr-window.h"
60#include "util.h"
61#include "ui.h"
62
63#include <libtransmission/transmission.h>
64#include <libtransmission/version.h>
65
66/* interval in milliseconds to update the torrent list display */
67#define UPDATE_INTERVAL         1666
68
69/* interval in milliseconds to check for stopped torrents and update display */
70#define EXIT_CHECK_INTERVAL     500
71
72#if GTK_CHECK_VERSION(2,8,0)
73#define SHOW_LICENSE
74static const char * LICENSE = 
75"The Transmission binaries and most of its source code is distributed "
76"license. "
77"\n\n"
78"Some files are copyrighted by Charles Kerr and are covered by "
79"the GPL version 2.  Works owned by the Transmission project "
80"are granted a special exemption to clause 2(b) so that the bulk "
81"of its code can remain under the MIT license.  This exemption does "
82"not extend to original or derived works not owned by the "
83"Transmission project. "
84"\n\n"
85"Permission is hereby granted, free of charge, to any person obtaining "
86"a copy of this software and associated documentation files (the "
87"'Software'), to deal in the Software without restriction, including "
88"without limitation the rights to use, copy, modify, merge, publish, "
89"distribute, sublicense, and/or sell copies of the Software, and to "
90"permit persons to whom the Software is furnished to do so, subject to "
91"the following conditions: "
92"\n\n"
93"The above copyright notice and this permission notice shall be included "
94"in all copies or substantial portions of the Software. "
95"\n\n"
96"THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, "
97"EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF "
98"MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. "
99"IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY "
100"CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, "
101"TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE "
102"SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.";
103#endif
104
105struct cbdata
106{
107    gboolean       minimized;
108    gboolean       closing;
109    guint          timer;
110    guint          idle_hide_mainwindow_tag;
111    gpointer       icon;
112    GtkWindow    * wind;
113    TrCore       * core;
114    GtkWidget    * msgwin;
115    GtkWidget    * prefs;
116    GSList       * errqueue;
117    GSList       * dupqueue;
118    GHashTable   * tor2details;
119    GHashTable   * details2tor;
120};
121
122#define CBDATA_PTR "callback-data-pointer"
123
124static GtkUIManager * myUIManager = NULL;
125
126static void
127appsetup( TrWindow * wind, GSList * args,
128          struct cbdata *,
129          gboolean paused, gboolean minimized );
130static void
131winsetup( struct cbdata * cbdata, TrWindow * wind );
132static void
133wannaquit( void * vdata );
134static void
135setupdrag(GtkWidget *widget, struct cbdata *data);
136static void
137gotdrag(GtkWidget *widget, GdkDragContext *dc, gint x, gint y,
138        GtkSelectionData *sel, guint info, guint time, gpointer gdata);
139
140static void
141coreerr( TrCore * core, enum tr_core_err code, const char * msg,
142         gpointer gdata );
143static void
144onAddTorrent( TrCore *, tr_ctor *, gpointer );
145static void
146prefschanged( TrCore * core, const char * key, gpointer data );
147static gboolean
148updatemodel(gpointer gdata);
149
150struct counts_data
151{
152    int totalCount;
153    int activeCount;
154    int inactiveCount;
155};
156
157static void
158accumulateStatusForeach( GtkTreeModel * model,
159                         GtkTreePath  * path UNUSED,
160                         GtkTreeIter  * iter,
161                         gpointer       user_data )
162{
163    int status = 0;
164    struct counts_data * counts = user_data;
165
166    ++counts->totalCount;
167
168    gtk_tree_model_get( model, iter, MC_STATUS, &status, -1 );
169
170    if( TR_STATUS_IS_ACTIVE( status ) )
171        ++counts->activeCount;
172    else
173        ++counts->inactiveCount;
174}
175
176static void
177accumulateCanUpdateForeach (GtkTreeModel * model,
178                            GtkTreePath  * path UNUSED,
179                            GtkTreeIter  * iter,
180                            gpointer       accumulated_status)
181{
182    tr_torrent * tor;
183    gtk_tree_model_get( model, iter, MC_TORRENT_RAW, &tor, -1 );
184    *(int*)accumulated_status |= tr_torrentCanManualUpdate( tor );
185}
186
187static void
188refreshTorrentActions( GtkTreeSelection * s )
189{
190    int canUpdate;
191    struct counts_data counts;
192
193    counts.activeCount = 0;
194    counts.inactiveCount = 0;
195    counts.totalCount = 0;
196    gtk_tree_selection_selected_foreach( s, accumulateStatusForeach, &counts );
197    action_sensitize( "pause-torrent", counts.activeCount!=0 );
198    action_sensitize( "start-torrent", counts.inactiveCount!=0 );
199    action_sensitize( "remove-torrent", counts.totalCount!=0 );
200    action_sensitize( "delete-torrent", counts.totalCount!=0 );
201    action_sensitize( "verify-torrent", counts.totalCount!=0 );
202    action_sensitize( "open-torrent-folder", counts.totalCount==1 );
203    action_sensitize( "show-torrent-details", counts.totalCount==1 );
204
205    canUpdate = 0;
206    gtk_tree_selection_selected_foreach( s, accumulateCanUpdateForeach, &canUpdate );
207    action_sensitize( "update-tracker", canUpdate!=0 );
208
209    {
210        GtkTreeView * view = gtk_tree_selection_get_tree_view( s );
211        GtkTreeModel * model = gtk_tree_view_get_model( view );
212        const int torrentCount = gtk_tree_model_iter_n_children( model, NULL ) != 0;
213        action_sensitize( "select-all", torrentCount!=0 );
214        action_sensitize( "deselect-all", torrentCount!=0 );
215    }
216}
217
218static void
219selectionChangedCB( GtkTreeSelection * s, gpointer unused UNUSED )
220{
221    refreshTorrentActions( s );
222}
223
224static void
225onMainWindowSizeAllocated( GtkWidget      * window,
226                           GtkAllocation  * alloc UNUSED,
227                           gpointer         gdata UNUSED )
228{
229    const gboolean isMaximized = window->window
230        && ( gdk_window_get_state( window->window ) & GDK_WINDOW_STATE_MAXIMIZED );
231
232    if( !isMaximized )
233    {
234        int x, y, w, h;
235        gtk_window_get_position( GTK_WINDOW( window ), &x, &y );
236        gtk_window_get_size( GTK_WINDOW( window ), &w, &h );
237        pref_int_set( PREF_KEY_MAIN_WINDOW_X, x );
238        pref_int_set( PREF_KEY_MAIN_WINDOW_Y, y );
239        pref_int_set( PREF_KEY_MAIN_WINDOW_WIDTH, w );
240        pref_int_set( PREF_KEY_MAIN_WINDOW_HEIGHT, h );
241    }
242}
243
244static void
245windowStateChanged( GtkWidget * widget UNUSED, GdkEventWindowState * event, gpointer gdata )
246{
247    if( event->changed_mask & GDK_WINDOW_STATE_ICONIFIED )
248    {
249        struct cbdata * cbdata = gdata;
250        cbdata->minimized = ( event->new_window_state & GDK_WINDOW_STATE_ICONIFIED ) ? 1 : 0;
251    }
252}
253
254static sig_atomic_t global_sigcount = 0;
255
256static void
257fatalsig( int sig )
258{
259    static const int SIGCOUNT_MAX = 3; /* revert to default handler after this many */
260
261    if( ++global_sigcount >= SIGCOUNT_MAX )
262    {
263        signal( sig, SIG_DFL );
264        raise( sig );
265    }
266}
267
268
269static void
270setupsighandlers( void )
271{
272#ifdef G_OS_WIN32
273  const int sigs[] = { SIGINT, SIGTERM };
274#else
275  const int sigs[] = { SIGHUP, SIGINT, SIGQUIT, SIGTERM };
276#endif
277  guint i;
278
279  for( i=0; i<G_N_ELEMENTS(sigs); ++i )
280      signal( sigs[i], fatalsig );
281}
282
283struct rpc_data
284{
285    int type;
286    int torrentId;
287    tr_torrent * tor;
288    struct cbdata * cbdata;
289};
290
291static int
292onRPCIdle( void * vdata )
293{
294    struct rpc_data * data = vdata;
295    switch( data->type )
296    {
297        case TR_RPC_TORRENT_ADDED:
298            tr_core_add_torrent( data->cbdata->core, tr_torrent_new_preexisting( data->tor ) );
299            break;
300        case TR_RPC_TORRENT_STARTED:
301            /* this should be automatic */
302            break;
303        case TR_RPC_TORRENT_STOPPED:
304            /* this should be automatic */
305            break;
306        case TR_RPC_TORRENT_REMOVING:
307            /* FIXME */
308            break;
309        case TR_RPC_TORRENT_CHANGED:
310        case TR_RPC_SESSION_CHANGED:
311            /* nothing interesting to do here */
312            break;
313    }
314    g_free( data );
315    return FALSE;
316}
317
318static void
319onRPCChanged( tr_handle            * handle UNUSED,
320              tr_rpc_callback_type   type,
321              struct tr_torrent    * tor,
322              void                 * cbdata )
323{
324    /* this callback is being invoked from the libtransmission thread,
325     * so let's push the information over to the gtk+ thread where
326     * it's safe to update the gui */
327    struct rpc_data * data = g_new0( struct rpc_data, 1 );
328    data->type = type;
329    data->torrentId = tor ? tr_torrentId( tor ) : -1;
330    data->tor = type == TR_RPC_TORRENT_REMOVING ? NULL : tor;
331    data->cbdata = cbdata;
332    g_idle_add( onRPCIdle, data );
333}
334
335int
336main( int argc, char ** argv )
337{
338    char * err = NULL;
339    struct cbdata * cbdata;
340    GSList * argfiles;
341    GError * gerr;
342    gboolean didinit = FALSE;
343    gboolean didlock = FALSE;
344    gboolean startpaused = FALSE;
345    gboolean startminimized = FALSE;
346    char * domain = "transmission";
347    char * configDir = NULL;
348
349    GOptionEntry entries[] = {
350        { "paused", 'p', 0, G_OPTION_ARG_NONE, &startpaused,
351          _("Start with all torrents paused"), NULL },
352#ifdef STATUS_ICON_SUPPORTED
353        { "minimized", 'm', 0, G_OPTION_ARG_NONE, &startminimized,
354          _( "Start minimized in system tray"), NULL },
355#endif
356        { "config-dir", 'g', 0, G_OPTION_ARG_FILENAME, &configDir,
357          _( "Where to look for configuration files" ), NULL },
358        { NULL, 0, 0, 0, NULL, NULL, NULL }
359    };
360
361    cbdata = g_new0( struct cbdata, 1 );
362    cbdata->tor2details = g_hash_table_new( g_str_hash, g_str_equal );
363    cbdata->details2tor = g_hash_table_new( g_direct_hash, g_direct_equal );
364
365    /* bind the gettext domain */
366    bindtextdomain( domain, TRANSMISSIONLOCALEDIR );
367    bind_textdomain_codeset( domain, "UTF-8" );
368    textdomain( domain );
369    g_set_application_name( _( "Transmission" ) );
370
371    /* initialize gtk */
372    g_thread_init( NULL );
373    gerr = NULL;
374    if( !gtk_init_with_args( &argc, &argv, _("[torrent files]"), entries, domain, &gerr ) ) {
375        g_message( "%s", gerr->message );
376        g_clear_error( &gerr );
377        return 0;
378    }
379    if( configDir == NULL )
380        configDir = (char*) tr_getDefaultConfigDir( );
381
382    tr_notify_init( );
383
384    didinit = cf_init( configDir, NULL ); /* must come before actions_init */
385    tr_prefs_init_global( );
386    myUIManager = gtk_ui_manager_new ();
387    actions_init ( myUIManager, cbdata );
388    gtk_ui_manager_add_ui_from_string (myUIManager, fallback_ui_file, -1, NULL);
389    gtk_ui_manager_ensure_update (myUIManager);
390    gtk_window_set_default_icon_name ( "transmission" );
391
392    setupsighandlers( ); /* set up handlers for fatal signals */
393
394    /* either get a lockfile s.t. this is the one instance of
395     * transmission that's running, OR if there are files to
396     * be added, delegate that to the running instance via dbus */
397    didlock = cf_lock( &err );
398    argfiles = checkfilenames( argc-1, argv+1 );
399    if( !didlock && argfiles )
400    {
401        GSList * l;
402        gboolean delegated = FALSE;
403        for( l=argfiles; l; l=l->next )
404            delegated |= gtr_dbus_add_torrent( l->data );
405        if( delegated )
406            err = NULL;
407    }
408
409    if( didlock && ( didinit || cf_init( configDir, &err ) ) )
410    {
411        tr_handle * h = tr_sessionInitFull(
412                            configDir,
413                            "gtk",
414                            pref_string_get( PREF_KEY_DOWNLOAD_DIR ),
415                            pref_flag_get( PREF_KEY_PEX ),
416                            pref_flag_get( PREF_KEY_PORT_FORWARDING ),
417                            pref_int_get( PREF_KEY_PORT ),
418                            pref_int_get( PREF_KEY_ENCRYPTION ),
419                            pref_flag_get( PREF_KEY_UL_LIMIT_ENABLED ),
420                            pref_int_get( PREF_KEY_UL_LIMIT ),
421                            pref_flag_get( PREF_KEY_DL_LIMIT_ENABLED ),
422                            pref_int_get( PREF_KEY_DL_LIMIT ),
423                            pref_int_get( PREF_KEY_MAX_PEERS_GLOBAL ),
424                            pref_int_get( PREF_KEY_MSGLEVEL ),
425                            TRUE, /* message queueing */
426                            pref_flag_get( PREF_KEY_BLOCKLIST_ENABLED ),
427                            pref_int_get( PREF_KEY_PEER_SOCKET_TOS ),
428                            pref_flag_get( PREF_KEY_RPC_ENABLED ),
429                            pref_int_get( PREF_KEY_RPC_PORT ),
430                            pref_string_get( PREF_KEY_RPC_ACL ),
431                            pref_flag_get( PREF_KEY_RPC_AUTH_ENABLED ),
432                            pref_string_get( PREF_KEY_RPC_USERNAME ),
433                            pref_string_get( PREF_KEY_RPC_PASSWORD ),
434                            pref_flag_get( PREF_KEY_PROXY_SERVER_ENABLED ),
435                            pref_string_get( PREF_KEY_PROXY_SERVER ),
436                            pref_int_get( PREF_KEY_PROXY_TYPE ),
437                            pref_flag_get( PREF_KEY_PROXY_AUTH_ENABLED ),
438                            pref_string_get( PREF_KEY_PROXY_USERNAME ),
439                            pref_string_get( PREF_KEY_PROXY_PASSWORD ) );
440
441        cbdata->core = tr_core_new( h );
442
443        /* create main window now to be a parent to any error dialogs */
444        GtkWindow * win = GTK_WINDOW( tr_window_new( myUIManager, cbdata->core ) );
445        g_signal_connect( win, "window-state-event", G_CALLBACK(windowStateChanged), cbdata );
446        g_signal_connect( win, "size-allocate", G_CALLBACK(onMainWindowSizeAllocated), cbdata );
447
448        appsetup( win, argfiles, cbdata, startpaused, startminimized );
449        tr_sessionSetRPCCallback( h, onRPCChanged, cbdata );
450
451        gtk_main();
452    }
453    else if( err )
454    {
455        gtk_widget_show( errmsg_full( NULL, (callbackfunc_t)gtk_main_quit,
456                                      NULL, "%s", err ) );
457        g_free( err );
458        gtk_main();
459    }
460
461    return 0;
462}
463
464static void
465appsetup( TrWindow * wind, GSList * torrentFiles,
466          struct cbdata * cbdata,
467          gboolean forcepause, gboolean minimized )
468{
469    const pref_flag_t start = forcepause ? PREF_FLAG_FALSE : PREF_FLAG_DEFAULT;
470    const pref_flag_t prompt = PREF_FLAG_DEFAULT;
471
472    /* fill out cbdata */
473    cbdata->wind       = NULL;
474    cbdata->icon       = NULL;
475    cbdata->msgwin     = NULL;
476    cbdata->prefs      = NULL;
477    cbdata->timer      = 0;
478    cbdata->closing    = FALSE;
479    cbdata->errqueue   = NULL;
480    cbdata->dupqueue   = NULL;
481    cbdata->minimized  = minimized;
482
483    if( minimized )
484        pref_flag_set( PREF_KEY_SHOW_TRAY_ICON, TRUE );
485
486    actions_set_core( cbdata->core );
487
488    /* set up core handlers */
489    g_signal_connect( cbdata->core, "error", G_CALLBACK( coreerr ), cbdata );
490    g_signal_connect( cbdata->core, "add-torrent-prompt",
491                      G_CALLBACK( onAddTorrent ), cbdata );
492    g_signal_connect_swapped( cbdata->core, "quit",
493                              G_CALLBACK( wannaquit ), cbdata );
494    g_signal_connect( cbdata->core, "prefs-changed",
495                      G_CALLBACK( prefschanged ), cbdata );
496
497    /* add torrents from command-line and saved state */
498    tr_core_load( cbdata->core, forcepause );
499    tr_core_add_list( cbdata->core, torrentFiles, start, prompt );
500    torrentFiles = NULL;
501    tr_core_torrents_added( cbdata->core );
502
503    /* set up main window */
504    winsetup( cbdata, wind );
505
506    /* set up the icon */
507    prefschanged( cbdata->core, PREF_KEY_SHOW_TRAY_ICON, cbdata );
508
509    /* start model update timer */
510    cbdata->timer = g_timeout_add( UPDATE_INTERVAL, updatemodel, cbdata );
511    updatemodel( cbdata );
512
513    /* either show the window or iconify it */
514    if( !minimized )
515        gtk_widget_show( GTK_WIDGET( wind ) );
516    else {
517        gtk_window_iconify( wind );
518        gtk_window_set_skip_taskbar_hint( cbdata->wind, cbdata->icon != NULL );
519    }
520}
521
522
523/**
524 * hideMainWindow, and the timeout hack in toggleMainWindow,
525 * are loosely cribbed from Colin Walters' tr-shell.c in Rhythmbox
526 */
527static gboolean
528idle_hide_mainwindow( gpointer window )
529{
530    gtk_widget_hide( window );
531    return FALSE;
532}
533static void
534hideMainWindow( struct cbdata * cbdata )
535{
536#if defined(STATUS_ICON_SUPPORTED) && defined(GDK_WINDOWING_X11)
537    GdkRectangle  bounds;
538    gulong        data[4];
539    Display      *dpy;
540    GdkWindow    *gdk_window;
541
542    gtk_status_icon_get_geometry( GTK_STATUS_ICON( cbdata->icon ), NULL, &bounds, NULL );
543    gdk_window = GTK_WIDGET (cbdata->wind)->window;
544    dpy = gdk_x11_drawable_get_xdisplay (gdk_window);
545
546    data[0] = bounds.x;
547    data[1] = bounds.y;
548    data[2] = bounds.width;
549    data[3] = bounds.height;
550
551    XChangeProperty (dpy,
552                     GDK_WINDOW_XID (gdk_window),
553                     gdk_x11_get_xatom_by_name_for_display (gdk_drawable_get_display (gdk_window),
554                     "_NET_WM_ICON_GEOMETRY"),
555                     XA_CARDINAL, 32, PropModeReplace,
556                     (guchar*)&data, 4);
557
558    gtk_window_set_skip_taskbar_hint( cbdata->wind, TRUE );
559#endif
560    gtk_window_iconify( cbdata->wind );
561}
562
563static void
564clearTag( guint * tag )
565{
566    if( *tag )
567        g_source_remove( *tag );
568    *tag = 0;
569}
570
571static void
572toggleMainWindow( struct cbdata * cbdata )
573{
574    GtkWindow * window = GTK_WINDOW( cbdata->wind );
575    const int hide = !cbdata->minimized;
576    static int x=0, y=0;
577
578    if( hide )
579    {
580        gtk_window_get_position( window, &x, &y );
581        clearTag( &cbdata->idle_hide_mainwindow_tag );
582        hideMainWindow( cbdata );
583        cbdata->idle_hide_mainwindow_tag = g_timeout_add( 250, idle_hide_mainwindow, window );
584    }
585    else
586    {
587        gtk_window_set_skip_taskbar_hint( window, FALSE );
588        gtk_window_move( window, x, y );
589        gtk_widget_show( GTK_WIDGET( window ) );
590        gtk_window_deiconify( window );
591#if GTK_CHECK_VERSION(2,8,0)
592        gtk_window_present_with_time( window, gtk_get_current_event_time( ) );
593#else
594        gtk_window_present( window );
595#endif
596    }
597}
598
599static gboolean
600winclose( GtkWidget * w UNUSED, GdkEvent * event UNUSED, gpointer gdata )
601{
602    struct cbdata * cbdata = gdata;
603
604    if( cbdata->icon != NULL )
605        action_activate ("toggle-main-window");
606    else
607        askquit( cbdata->core, cbdata->wind, wannaquit, cbdata );
608
609    return TRUE; /* don't propagate event further */
610}
611
612static void
613rowChangedCB( GtkTreeModel  * model UNUSED,
614              GtkTreePath   * path,
615              GtkTreeIter   * iter UNUSED,
616              gpointer        sel)
617{
618    if( gtk_tree_selection_path_is_selected ( sel, path ) )
619        refreshTorrentActions( GTK_TREE_SELECTION(sel) );
620}
621
622static void
623winsetup( struct cbdata * cbdata, TrWindow * wind )
624{
625    GtkTreeModel * model;
626    GtkTreeSelection * sel;
627
628    g_assert( NULL == cbdata->wind );
629    cbdata->wind = GTK_WINDOW( wind );
630
631    sel = tr_window_get_selection( cbdata->wind );
632    g_signal_connect( sel, "changed", G_CALLBACK(selectionChangedCB), NULL );
633    selectionChangedCB( sel, NULL );
634    model = tr_core_model( cbdata->core );
635    g_signal_connect( model, "row-changed", G_CALLBACK(rowChangedCB), sel );
636    g_signal_connect( wind, "delete-event", G_CALLBACK( winclose ), cbdata );
637    refreshTorrentActions( sel );
638   
639    setupdrag( GTK_WIDGET(wind), cbdata );
640}
641
642static gpointer
643quitThreadFunc( gpointer gdata )
644{
645    struct cbdata * cbdata = gdata;
646
647    tr_core_close( cbdata->core );
648
649    /* shutdown the gui */
650    if( cbdata->prefs )
651        gtk_widget_destroy( GTK_WIDGET( cbdata->prefs ) );
652    if( cbdata->wind )
653        gtk_widget_destroy( GTK_WIDGET( cbdata->wind ) );
654    g_object_unref( cbdata->core );
655    if( cbdata->icon )
656        g_object_unref( cbdata->icon );
657    if( cbdata->errqueue ) {
658        g_slist_foreach( cbdata->errqueue, (GFunc)g_free, NULL );
659        g_slist_free( cbdata->errqueue );
660    }
661    if( cbdata->dupqueue ) {
662        g_slist_foreach( cbdata->dupqueue, (GFunc)g_free, NULL );
663        g_slist_free( cbdata->dupqueue );
664    }
665
666    g_hash_table_destroy( cbdata->details2tor );
667    g_hash_table_destroy( cbdata->tor2details );
668    g_free( cbdata );
669
670    /* exit the gtk main loop */
671    gtk_main_quit( );
672    return NULL;
673}
674
675static void
676do_exit_cb( GtkWidget *w UNUSED, gpointer data UNUSED )
677{
678    exit( 0 );
679}
680
681static void
682wannaquit( void * vdata )
683{
684    GtkWidget * r, * p, * b, * w, *c;
685    struct cbdata * cbdata = vdata;
686
687    /* stop the update timer */
688    if( cbdata->timer ) {
689        g_source_remove( cbdata->timer );
690        cbdata->timer = 0;
691    }
692
693    c = GTK_WIDGET( cbdata->wind );
694    gtk_container_remove( GTK_CONTAINER( c ), gtk_bin_get_child( GTK_BIN( c ) ) );
695
696    r = gtk_alignment_new(0.5, 0.5, 0.01, 0.01);
697    gtk_container_add(GTK_CONTAINER(c), r);
698
699    p = gtk_table_new(3, 2, FALSE);
700    gtk_table_set_col_spacings( GTK_TABLE( p ), GUI_PAD_BIG );
701    gtk_container_add( GTK_CONTAINER( r ), p );
702
703    w = gtk_image_new_from_stock( GTK_STOCK_NETWORK, GTK_ICON_SIZE_DIALOG );
704    gtk_table_attach_defaults(GTK_TABLE(p), w, 0, 1, 0, 2 );
705
706    w = gtk_label_new( NULL );
707    gtk_label_set_markup( GTK_LABEL( w ), _( "<b>Closing Connections</b>" ) );
708    gtk_misc_set_alignment( GTK_MISC( w ), 0.0, 0.5 );
709    gtk_table_attach_defaults( GTK_TABLE( p ), w, 1, 2, 0, 1 );
710
711    w = gtk_label_new( _( "Sending upload/download totals to tracker..." ) );
712    gtk_misc_set_alignment( GTK_MISC( w ), 0.0, 0.5 );
713    gtk_table_attach_defaults( GTK_TABLE( p ), w, 1, 2, 1, 2 );
714
715    b = gtk_alignment_new(0.0, 1.0, 0.01, 0.01);
716    w = tr_button_new_from_stock( GTK_STOCK_QUIT, _( "_Quit Now" ) );
717    g_signal_connect(w, "clicked", G_CALLBACK(do_exit_cb), NULL);
718    gtk_container_add(GTK_CONTAINER(b), w);
719    gtk_table_attach(GTK_TABLE(p), b, 1, 2, 2, 3, GTK_FILL, GTK_FILL, 0, 10 );
720
721    gtk_widget_show_all(r);
722
723    /* clear the UI */
724    gtk_list_store_clear( GTK_LIST_STORE( tr_core_model( cbdata->core ) ) );
725
726    /* shut down libT */
727    g_thread_create( quitThreadFunc, vdata, TRUE, NULL );
728}
729
730static void
731gotdrag( GtkWidget         * widget UNUSED,
732         GdkDragContext    * dc,
733         gint                x UNUSED,
734         gint                y UNUSED,
735         GtkSelectionData  * sel,
736         guint               info UNUSED,
737         guint               time,
738         gpointer            gdata )
739{
740    struct cbdata * data = gdata;
741    GSList * paths = NULL;
742    GSList * freeme = NULL;
743
744#if 0
745    int i;
746    char *sele = gdk_atom_name(sel->selection);
747    char *targ = gdk_atom_name(sel->target);
748    char *type = gdk_atom_name(sel->type);
749
750    g_message( "dropped file: sel=%s targ=%s type=%s fmt=%i len=%i",
751               sele, targ, type, sel->format, sel->length );
752    g_free(sele);
753    g_free(targ);
754    g_free(type);
755    if( sel->format == 8 ) {
756        for( i=0; i<sel->length; ++i )
757            fprintf(stderr, "%02X ", sel->data[i]);
758        fprintf(stderr, "\n");
759    }
760#endif
761
762    if( ( sel->format == 8 ) &&
763        ( sel->selection == gdk_atom_intern( "XdndSelection", FALSE ) ) )
764    {
765        int i;
766        char * str = g_strndup( (char*)sel->data, sel->length );
767        gchar ** files = g_strsplit_set( str, "\r\n", -1 );
768        for( i=0; files && files[i]; ++i )
769        {
770            char * filename;
771            if( !*files[i] ) /* empty filename... */
772                continue;
773
774            /* decode the filename */
775            filename = decode_uri( files[i] );
776            freeme = g_slist_prepend( freeme, filename );
777            if( !g_utf8_validate( filename, -1, NULL ) )
778                continue;
779
780            /* walk past "file://", if present */
781            if( g_str_has_prefix( filename, "file:" ) ) {
782                filename += 5;
783                while( g_str_has_prefix( filename, "//" ) )
784                    ++filename;
785            }
786
787            /* if the file doesn't exist, the first part
788               might be a hostname ... walk past it. */
789            if( !g_file_test( filename, G_FILE_TEST_EXISTS ) ) {
790                char * pch = strchr( filename + 1, '/' );
791                if( pch != NULL )
792                    filename = pch;
793            }
794
795            /* finally, add it to the list of torrents to try adding */
796            if( g_file_test( filename, G_FILE_TEST_EXISTS ) )
797                paths = g_slist_prepend( paths, g_strdup( filename ) );
798        }
799
800        /* try to add any torrents we found */
801        if( paths )
802        {
803            paths = g_slist_reverse( paths );
804            tr_core_add_list_defaults( data->core, paths );
805            tr_core_torrents_added( data->core );
806        }
807
808        freestrlist( freeme );
809        g_strfreev( files );
810        g_free( str );
811    }
812
813    gtk_drag_finish(dc, (NULL != paths), FALSE, time);
814}
815
816static void
817setupdrag(GtkWidget *widget, struct cbdata *data) {
818  GtkTargetEntry targets[] = {
819    { "STRING",     0, 0 },
820    { "text/plain", 0, 0 },
821    { "text/uri-list", 0, 0 },
822  };
823
824  g_signal_connect(widget, "drag_data_received", G_CALLBACK(gotdrag), data);
825
826  gtk_drag_dest_set(widget, GTK_DEST_DEFAULT_ALL, targets,
827                    ALEN(targets), GDK_ACTION_COPY | GDK_ACTION_MOVE);
828}
829
830static void
831flushAddTorrentErrors( GtkWindow * window, const char * primary, GSList ** files )
832{
833    GString * s = g_string_new( NULL );
834    GSList * l;
835    GtkWidget * w;
836    for( l=*files; l; l=l->next )
837        g_string_append_printf( s, "%s\n", (const char*)l->data );
838    w = gtk_message_dialog_new( window,
839                                GTK_DIALOG_DESTROY_WITH_PARENT,
840                                GTK_MESSAGE_ERROR,
841                                GTK_BUTTONS_CLOSE,
842                                primary );
843    gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG( w ), s->str );
844    g_signal_connect_swapped( w, "response",
845                              G_CALLBACK( gtk_widget_destroy ), w );
846    gtk_widget_show_all( w );
847    g_string_free( s, TRUE );
848
849    g_slist_foreach( *files, (GFunc)g_free, NULL );
850    g_slist_free( *files );
851    *files = NULL;
852}
853
854static void
855showTorrentErrors( struct cbdata * cbdata )
856{
857    if( cbdata->errqueue )
858        flushAddTorrentErrors( GTK_WINDOW( cbdata->wind ),
859                               ngettext( "Couldn't add corrupt torrent",
860                                         "Couldn't add corrupt torrents",
861                                         g_slist_length( cbdata->errqueue ) ),
862                               &cbdata->errqueue );
863
864    if( cbdata->dupqueue )
865        flushAddTorrentErrors( GTK_WINDOW( cbdata->wind ),
866                               ngettext( "Couldn't add duplicate torrent",
867                                         "Couldn't add duplicate torrents",
868                                         g_slist_length( cbdata->dupqueue ) ),
869                               &cbdata->dupqueue );
870}
871
872static void
873coreerr( TrCore * core UNUSED, enum tr_core_err code, const char * msg,
874         gpointer gdata )
875{
876    struct cbdata * c = gdata;
877
878    switch( code )
879    {
880        case TR_EINVALID:
881            c->errqueue = g_slist_append( c->errqueue, g_path_get_basename( msg ) );
882            break;
883
884        case TR_EDUPLICATE:
885            c->dupqueue = g_slist_append( c->dupqueue, g_path_get_basename( msg ) );
886            break;
887
888        case TR_CORE_ERR_NO_MORE_TORRENTS:
889            showTorrentErrors( c );
890            break;
891
892        case TR_CORE_ERR_SAVE_STATE:
893            errmsg( c->wind, "%s", msg );
894            break;
895
896        default:
897            g_assert_not_reached();
898            break;
899    }
900
901}
902
903#if GTK_CHECK_VERSION(2,8,0)
904static void
905on_main_window_focus_in( GtkWidget      * widget UNUSED,
906                         GdkEventFocus  * event UNUSED,
907                         gpointer         gdata )
908{
909    struct cbdata * cbdata = gdata;
910    gtk_window_set_urgency_hint( GTK_WINDOW( cbdata->wind ), FALSE );
911}
912#endif
913
914static void
915onAddTorrent( TrCore * core, tr_ctor * ctor, gpointer gdata )
916{
917    struct cbdata * cbdata = gdata;
918    GtkWidget * w = addSingleTorrentDialog( cbdata->wind, core, ctor );
919#if GTK_CHECK_VERSION(2,8,0)
920    g_signal_connect( w, "focus-in-event", G_CALLBACK(on_main_window_focus_in),  cbdata );
921    gtk_window_set_urgency_hint( cbdata->wind, TRUE );
922#endif
923}
924
925static void
926prefschanged( TrCore * core UNUSED, const char * key, gpointer data )
927{
928    struct cbdata * cbdata = data;
929    tr_handle     * tr     = tr_core_handle( cbdata->core );
930
931    if( !strcmp( key, PREF_KEY_ENCRYPTION ) )
932    {
933        const int encryption = pref_int_get( key );
934g_message( "setting encryption to %d", encryption );
935        tr_sessionSetEncryption( tr, encryption );
936    }
937    else if( !strcmp( key, PREF_KEY_PORT ) )
938    {
939        const int port = pref_int_get( key );
940        tr_sessionSetPeerPort( tr, port );
941    }
942    else if( !strcmp( key, PREF_KEY_SHOW_TRAY_ICON ) )
943    {
944        const int show = pref_flag_get( key );
945        if( show && !cbdata->icon )
946            cbdata->icon = tr_icon_new( cbdata->core );
947        else if( !show && cbdata->icon ) {
948            g_object_unref( cbdata->icon );
949            cbdata->icon = NULL;
950        }
951    }
952    else if( !strcmp( key, PREF_KEY_DL_LIMIT_ENABLED ) )
953    {
954        const gboolean b = pref_flag_get( key );
955        tr_sessionSetSpeedLimitEnabled( tr, TR_DOWN, b );
956    }
957    else if( !strcmp( key, PREF_KEY_DL_LIMIT ) )
958    {
959        const int limit = pref_int_get( key );
960        tr_sessionSetSpeedLimit( tr, TR_DOWN, limit );
961    }
962    else if( !strcmp( key, PREF_KEY_UL_LIMIT_ENABLED ) )
963    {
964        const gboolean b = pref_flag_get( key );
965        tr_sessionSetSpeedLimitEnabled( tr, TR_UP, b );
966    }
967    else if( !strcmp( key, PREF_KEY_UL_LIMIT ) )
968    {
969        const int limit = pref_int_get( key );
970        tr_sessionSetSpeedLimit( tr, TR_UP, limit );
971    }
972    else if( !strcmp( key, PREF_KEY_PORT_FORWARDING ) )
973    {
974        const gboolean enabled = pref_flag_get( key );
975        tr_sessionSetPortForwardingEnabled( tr, enabled );
976    }
977    else if( !strcmp( key, PREF_KEY_PEX ) )
978    {
979        const gboolean b = pref_flag_get( key );
980        tr_sessionSetPortForwardingEnabled( tr, b );
981    }
982    else if( !strcmp( key, PREF_KEY_RPC_ENABLED ) )
983    {
984        tr_sessionSetRPCEnabled( tr, pref_flag_get( key ) );
985    }
986    else if( !strcmp( key, PREF_KEY_RPC_PORT ) )
987    {
988        tr_sessionSetRPCPort( tr, pref_int_get( key ) );
989    }
990    else if( !strcmp( key, PREF_KEY_RPC_ENABLED ) )
991    {
992        tr_sessionSetRPCEnabled( tr, pref_flag_get( key ) );
993    }
994    else if( !strcmp( key, PREF_KEY_RPC_ACL ) )
995    {
996        char * err = NULL;
997        const char * s = pref_string_get( key );
998        tr_sessionSetRPCACL( tr, s, &err );
999    }
1000    else if( !strcmp( key, PREF_KEY_RPC_USERNAME ) )
1001    {
1002        const char * s = pref_string_get( key );
1003        tr_sessionSetRPCUsername( tr, s );
1004    }
1005    else if( !strcmp( key, PREF_KEY_RPC_PASSWORD ) )
1006    {
1007        const char * s = pref_string_get( key );
1008        tr_sessionSetRPCPassword( tr, s );
1009    }
1010    else if( !strcmp( key, PREF_KEY_RPC_AUTH_ENABLED ) )
1011    {
1012        const gboolean enabled = pref_flag_get( key );
1013        tr_sessionSetRPCPasswordEnabled( tr, enabled );
1014    }
1015    else if( !strcmp( key, PREF_KEY_PROXY_SERVER ) )
1016    {
1017        const char * s = pref_string_get( key );
1018        tr_sessionSetProxy( tr, s );
1019    }
1020    else if( !strcmp( key, PREF_KEY_PROXY_TYPE ) )
1021    {
1022        const int i = pref_int_get( key );
1023        tr_sessionSetProxyType( tr, i );
1024    }
1025    else if( !strcmp( key, PREF_KEY_PROXY_SERVER_ENABLED ) )
1026    {
1027        const gboolean enabled = pref_flag_get( key );
1028        tr_sessionSetProxyEnabled( tr, enabled );
1029    }
1030    else if( !strcmp( key, PREF_KEY_PROXY_AUTH_ENABLED ) )
1031    {
1032        const gboolean enabled = pref_flag_get( key );
1033        tr_sessionSetProxyAuthEnabled( tr, enabled );
1034    }
1035    else if( !strcmp( key, PREF_KEY_PROXY_USERNAME ) )
1036    {
1037        const char * s = pref_string_get( key );
1038        tr_sessionSetProxyUsername( tr, s );
1039    }
1040    else if( !strcmp( key, PREF_KEY_PROXY_PASSWORD ) )
1041    {
1042        const char * s = pref_string_get( key );
1043        tr_sessionSetProxyPassword( tr, s );
1044    }
1045}
1046
1047gboolean
1048updatemodel( gpointer gdata )
1049{
1050    struct cbdata *data = gdata;
1051    const gboolean done = data->closing || global_sigcount;
1052
1053    if( !done )
1054    {
1055        /* update the torrent data in the model */
1056        tr_core_update( data->core );
1057
1058        /* update the main window's statusbar and toolbar buttons */
1059        if( data->wind )
1060            tr_window_update( data->wind );
1061
1062        /* update the actions */
1063        refreshTorrentActions( tr_window_get_selection( data->wind ) );
1064    }
1065
1066    return !done;
1067}
1068
1069static void
1070aboutDialogActivateLink( GtkAboutDialog * dialog UNUSED,
1071                         const gchar    * link_,
1072                         gpointer         user_data UNUSED )
1073{
1074    gtr_open_file( link_ );
1075}
1076
1077static void
1078about ( GtkWindow * parent )
1079{
1080    const char *authors[] =
1081    {
1082        "Charles Kerr (Backend; GTK+)",
1083        "Mitchell Livingston (Backend; OS X)",
1084        "Eric Petit (Backend; OS X)",
1085        "Josh Elsasser (Daemon; Backend; GTK+)",
1086        "Bryan Varner (BeOS)", 
1087        NULL
1088    };
1089
1090    gtk_about_dialog_set_url_hook( aboutDialogActivateLink, NULL, NULL );
1091
1092    gtk_show_about_dialog( parent,
1093        "name", g_get_application_name(),
1094        "comments", _("A fast and easy BitTorrent client"),
1095        "version", LONG_VERSION_STRING,
1096        "website", "http://www.transmissionbt.com/",
1097        "copyright",_("Copyright 2005-2008 The Transmission Project"),
1098        "logo-icon-name", "transmission",
1099#ifdef SHOW_LICENSE
1100        "license", LICENSE,
1101        "wrap-license", TRUE,
1102#endif
1103        "authors", authors,
1104        /* Translators: translate "translator-credits" as your name
1105           to have it appear in the credits in the "About" dialog */
1106        "translator-credits", _("translator-credits"),
1107        NULL );
1108}
1109
1110static void
1111startTorrentForeach (GtkTreeModel * model,
1112                     GtkTreePath  * path UNUSED,
1113                     GtkTreeIter  * iter,
1114                     gpointer       data UNUSED)
1115{
1116    tr_torrent * tor = NULL;
1117    gtk_tree_model_get( model, iter, MC_TORRENT_RAW, &tor, -1 );
1118    tr_torrentStart( tor );
1119}
1120
1121static void
1122stopTorrentForeach (GtkTreeModel * model,
1123                    GtkTreePath  * path UNUSED,
1124                    GtkTreeIter  * iter,
1125                    gpointer       data UNUSED)
1126{
1127    tr_torrent * tor = NULL;
1128    gtk_tree_model_get( model, iter, MC_TORRENT_RAW, &tor, -1 );
1129    tr_torrentStop( tor );
1130}
1131
1132static void
1133updateTrackerForeach (GtkTreeModel * model,
1134                      GtkTreePath  * path UNUSED,
1135                      GtkTreeIter  * iter,
1136                      gpointer       data UNUSED)
1137{
1138    tr_torrent * tor = NULL;
1139    gtk_tree_model_get( model, iter, MC_TORRENT_RAW, &tor, -1 );
1140    tr_torrentManualUpdate( tor );
1141}
1142
1143static void
1144detailsClosed( gpointer user_data, GObject * details )
1145{
1146    struct cbdata * data = user_data;
1147    gpointer hashString = g_hash_table_lookup( data->details2tor, details );
1148    g_hash_table_remove( data->details2tor, details );
1149    g_hash_table_remove( data->tor2details, hashString );
1150}
1151
1152static void
1153openFolderForeach( GtkTreeModel * model,
1154                   GtkTreePath  * path UNUSED,
1155                   GtkTreeIter  * iter,
1156                   gpointer       user_data UNUSED )
1157{
1158    TrTorrent * gtor = NULL;
1159    gtk_tree_model_get( model, iter, MC_TORRENT, &gtor, -1 );
1160    tr_torrent_open_folder( gtor );
1161    g_object_unref( G_OBJECT( gtor ) );
1162}
1163
1164static void
1165showInfoForeach (GtkTreeModel * model,
1166                 GtkTreePath  * path UNUSED,
1167                 GtkTreeIter  * iter,
1168                 gpointer       user_data )
1169{
1170    const char * hashString;
1171    struct cbdata * data = user_data;
1172    TrTorrent * tor = NULL;
1173    GtkWidget * w;
1174
1175    gtk_tree_model_get( model, iter, MC_TORRENT, &tor, -1 );
1176    hashString = tr_torrent_info(tor)->hashString;
1177    w = g_hash_table_lookup( data->tor2details, hashString );
1178    if( w != NULL )
1179        gtk_window_present( GTK_WINDOW( w ) );
1180    else {
1181        w = torrent_inspector_new( GTK_WINDOW( data->wind ), tor );
1182        gtk_widget_show( w );
1183        g_hash_table_insert( data->tor2details, (gpointer)hashString, w );
1184        g_hash_table_insert( data->details2tor, w, (gpointer)hashString );
1185        g_object_weak_ref( G_OBJECT( w ), detailsClosed, data );
1186    }
1187
1188    g_object_unref( G_OBJECT( tor ) );
1189}
1190
1191static void
1192recheckTorrentForeach (GtkTreeModel * model,
1193                       GtkTreePath  * path UNUSED,
1194                       GtkTreeIter  * iter,
1195                       gpointer       data UNUSED)
1196{
1197    TrTorrent * gtor = NULL;
1198    gtk_tree_model_get( model, iter, MC_TORRENT, &gtor, -1 );
1199    tr_torrentVerify( tr_torrent_handle( gtor ) );
1200    g_object_unref( G_OBJECT( gtor ) );
1201}
1202
1203static gboolean
1204msgwinclosed( void )
1205{
1206  action_toggle( "toggle-message-log", FALSE );
1207  return FALSE;
1208}
1209
1210static void
1211accumulateSelectedTorrents( GtkTreeModel * model,
1212                            GtkTreePath  * path UNUSED,
1213                            GtkTreeIter  * iter,
1214                            gpointer       gdata )
1215{
1216    GSList ** data = ( GSList** ) gdata;
1217    TrTorrent * tor = NULL;
1218    gtk_tree_model_get( model, iter, MC_TORRENT, &tor, -1 );
1219    *data = g_slist_prepend( *data, tor );
1220}
1221
1222static void
1223removeSelected( struct cbdata * data, gboolean delete_files )
1224{
1225    GSList * l = NULL;
1226    GtkTreeSelection * s = tr_window_get_selection( data->wind );
1227    gtk_tree_selection_selected_foreach( s, accumulateSelectedTorrents, &l );
1228    gtk_tree_selection_unselect_all( s );
1229    if( l ) {
1230        l = g_slist_reverse( l );
1231        confirmRemove( data->wind, data->core, l, delete_files );
1232    }
1233}
1234
1235void
1236doAction ( const char * action_name, gpointer user_data )
1237{
1238    struct cbdata * data = user_data;
1239    gboolean changed = FALSE;
1240
1241    if ( !strcmp (action_name, "add-torrent-menu") ||
1242         !strcmp( action_name, "add-torrent-toolbar" ))
1243    {
1244        addDialog( data->wind, data->core );
1245    }
1246    else if (!strcmp (action_name, "show-stats"))
1247    {
1248        GtkWidget * dialog = stats_dialog_create( data->wind,
1249                                                  data->core );
1250        gtk_widget_show( dialog );
1251    }
1252    else if (!strcmp (action_name, "start-torrent"))
1253    {
1254        GtkTreeSelection * s = tr_window_get_selection(data->wind);
1255        gtk_tree_selection_selected_foreach( s, startTorrentForeach, NULL );
1256        changed |= gtk_tree_selection_count_selected_rows( s ) != 0;
1257    }
1258    else if (!strcmp (action_name, "pause-torrent"))
1259    {
1260        GtkTreeSelection * s = tr_window_get_selection(data->wind);
1261        gtk_tree_selection_selected_foreach( s, stopTorrentForeach, NULL );
1262        changed |= gtk_tree_selection_count_selected_rows( s ) != 0;
1263    }
1264    else if (!strcmp (action_name, "verify-torrent"))
1265    {
1266        GtkTreeSelection * s = tr_window_get_selection(data->wind);
1267        gtk_tree_selection_selected_foreach( s, recheckTorrentForeach, NULL );
1268        changed |= gtk_tree_selection_count_selected_rows( s ) != 0;
1269    }
1270    else if (!strcmp (action_name, "open-torrent-folder"))
1271    {
1272        GtkTreeSelection * s = tr_window_get_selection(data->wind);
1273        gtk_tree_selection_selected_foreach( s, openFolderForeach, data );
1274    }
1275    else if (!strcmp (action_name, "show-torrent-details"))
1276    {
1277        GtkTreeSelection * s = tr_window_get_selection(data->wind);
1278        gtk_tree_selection_selected_foreach( s, showInfoForeach, data );
1279    }
1280    else if (!strcmp( action_name, "update-tracker"))
1281    {
1282        GtkTreeSelection * s = tr_window_get_selection(data->wind);
1283        gtk_tree_selection_selected_foreach( s, updateTrackerForeach, data->wind );
1284    }
1285    else if (!strcmp (action_name, "new-torrent"))
1286    {
1287        GtkWidget * w = make_meta_ui( GTK_WINDOW( data->wind ),
1288                                      tr_core_handle( data->core ) );
1289        gtk_widget_show_all( w );
1290    }
1291    else if( !strcmp( action_name, "remove-torrent" ) )
1292    {
1293        removeSelected( data, FALSE );
1294    }
1295    else if( !strcmp( action_name, "delete-torrent" ) )
1296    {
1297        removeSelected( data, TRUE );
1298    }
1299    else if (!strcmp (action_name, "quit"))
1300    {
1301        askquit( data->core, data->wind, wannaquit, data );
1302    }
1303    else if (!strcmp (action_name, "select-all"))
1304    {
1305        GtkTreeSelection * s = tr_window_get_selection(data->wind);
1306        gtk_tree_selection_select_all( s );
1307    }
1308    else if (!strcmp (action_name, "deselect-all"))
1309    {
1310        GtkTreeSelection * s = tr_window_get_selection(data->wind);
1311        gtk_tree_selection_unselect_all( s );
1312    }
1313    else if (!strcmp (action_name, "edit-preferences"))
1314    {
1315        if( NULL == data->prefs )
1316        {
1317            data->prefs = tr_prefs_dialog_new( G_OBJECT(data->core), data->wind );
1318            g_signal_connect( data->prefs, "destroy",
1319                             G_CALLBACK( gtk_widget_destroyed ), &data->prefs );
1320            gtk_widget_show( GTK_WIDGET( data->prefs ) );
1321        }
1322    }
1323    else if (!strcmp (action_name, "toggle-message-log"))
1324    {
1325        if( !data->msgwin )
1326        {
1327            GtkWidget * win = msgwin_new( data->core );
1328            g_signal_connect( win, "destroy", G_CALLBACK( msgwinclosed ), 
1329                             NULL );
1330            data->msgwin = win;
1331        }
1332        else
1333        {
1334            action_toggle("toggle-message-log", FALSE);
1335            gtk_widget_destroy( data->msgwin );
1336            data->msgwin = NULL;
1337        }
1338    }
1339    else if (!strcmp (action_name, "show-about-dialog"))
1340    {
1341        about( data->wind );
1342    }
1343    else if (!strcmp (action_name, "help"))
1344    {
1345        char * url = gtr_get_help_url( );
1346        gtr_open_file( url );
1347        g_free( url );
1348    }
1349    else if (!strcmp (action_name, "toggle-main-window"))
1350    {
1351        toggleMainWindow( data );
1352    }
1353    else g_error ("Unhandled action: %s", action_name );
1354
1355    if( changed )
1356        updatemodel( data );
1357}
Note: See TracBrowser for help on using the repository browser.