source: branches/1.4x/gtk/details.c @ 7141

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

(1.4x gtk) backport #1504: when the details dialog's peers list is sorted, some rows are duplicated

  • Property svn:keywords set to Date Rev Author Id
File size: 50.3 KB
Line 
1/*
2 * This file Copyright (C) 2007-2008 Charles Kerr <charles@rebelbase.com>
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: details.c 7141 2008-11-23 17:18:48Z charles $
11 */
12
13#include <errno.h>
14#include <stddef.h>
15#include <stdio.h>
16#include <stdlib.h>
17#include <glib/gi18n.h>
18#include <gtk/gtk.h>
19
20#include <libtransmission/transmission.h>
21#include <libtransmission/utils.h> /* tr_httpIsValidURL */
22
23#include "actions.h"
24#include "details.h"
25#include "file-list.h"
26#include "tr-torrent.h"
27#include "tracker-list.h"
28#include "hig.h"
29#include "util.h"
30
31#define UPDATE_INTERVAL_MSEC 2000
32
33/****
34*****  PIECES VIEW
35****/
36
37/* define SHOW_PIECES */
38
39#ifdef SHOW_PIECES
40static int
41getGridSize( int   pieceCount,
42             int * n_rows,
43             int * n_cols )
44{
45    const int MAX_ACROSS = 16;
46
47    if( pieceCount >= ( MAX_ACROSS * MAX_ACROSS ) )
48    {
49        *n_rows = *n_cols = MAX_ACROSS;
50        return MAX_ACROSS * MAX_ACROSS;
51    }
52    else
53    {
54        int i;
55        for( i = 0; ( i * i ) < pieceCount; ++i ) ;
56        *n_rows = *n_cols = i;
57        return pieceCount;
58    }
59}
60
61 #define TO16( a ) ( (guint16)( ( a << 8 ) | ( a ) ) )
62 #define RGB_2_GDK( R, G, B ) { 0, TO16( R ), TO16( G ), TO16( B ) }
63
64enum { DRAW_AVAIL, DRAW_PROG };
65
66static void
67release_gobject_array( gpointer data )
68{
69    int       i;
70    GObject **objects = (GObject**) data;
71
72    for( i = 0; objects[i] != NULL; ++i )
73        g_object_unref ( G_OBJECT( objects[i] ) );
74    g_free ( objects );
75}
76
77static gboolean
78refresh_pieces( GtkWidget *            da,
79                GdkEventExpose * event UNUSED,
80                gpointer               gtor )
81{
82    tr_torrent *       tor = tr_torrent_handle( TR_TORRENT( gtor ) );
83    const tr_info *    info = tr_torrent_info( TR_TORRENT( gtor ) );
84    int                mode =
85        GPOINTER_TO_INT ( g_object_get_data ( G_OBJECT( da ), "draw-mode" ) );
86
87    GdkColormap *      colormap = gtk_widget_get_colormap ( da );
88    const int          widget_w = da->allocation.width;
89    const int          widget_h = da->allocation.height;
90    int                n_rows, n_cols;
91    const int          n_cells = getGridSize ( info->pieceCount,  &n_rows,
92                                               &n_cols );
93    const GdkRectangle grid_bounds = { 0, 0, widget_w, widget_h };
94    const double       piece_w = grid_bounds.width / (double)n_cols;
95    const double       piece_h = grid_bounds.height / (double)n_rows;
96    const int          piece_w_int = (int) ( piece_w + 1 ); /* pad for roundoff
97                                                              */
98    const int          piece_h_int = (int) ( piece_h + 1 ); /* pad for roundoff
99                                                              */
100    const gboolean     rtl = gtk_widget_get_direction( da ) ==
101                             GTK_TEXT_DIR_RTL;
102
103    guint8 *           prev_color = NULL;
104    gboolean           first_time = FALSE;
105
106    int                i, x, y;
107    int8_t *           pieces = NULL;
108    float *            completeness = NULL;
109
110    /**
111    ***  Get the Graphics Contexts...
112    **/
113
114    enum { ALL, LOTS, SOME, FEW, NONE,
115           BLACK, GRAY, BLINK,
116           N_COLORS };
117    GdkGC **           gcs = (GdkGC**) g_object_get_data ( G_OBJECT(
118                                                               da ),
119                                                           "graphics-contexts" );
120    if( gcs == NULL )
121    {
122        const GdkColor colors[N_COLORS] = {
123            RGB_2_GDK ( 114, 159, 207 ), /* all */
124            RGB_2_GDK (  52, 101, 164 ), /* lots */
125            RGB_2_GDK (  32,  74, 135 ), /* some */
126            RGB_2_GDK (  85,  87, 83 ), /* few */
127            RGB_2_GDK ( 238, 238, 236 ), /* none - tango aluminum highlight */
128            RGB_2_GDK (  46,  52, 54 ), /* black - tango slate shadow */
129            RGB_2_GDK ( 186, 189, 182 ), /* gray - tango aluminum shadow */
130            RGB_2_GDK ( 252, 233, 79 ), /* blink - tango butter highlight */
131        };
132
133        gcs = g_new ( GdkGC *, N_COLORS + 1 );
134
135        for( i = 0; i < N_COLORS; ++i )
136        {
137            gcs[i] = gdk_gc_new ( da->window );
138            gdk_gc_set_colormap ( gcs[i], colormap );
139            gdk_gc_set_rgb_fg_color ( gcs[i], &colors[i] );
140            gdk_gc_set_rgb_bg_color ( gcs[i], &colors[i] );
141        }
142
143        gcs[N_COLORS] = NULL; /* a sentinel in the release function */
144        g_object_set_data_full ( G_OBJECT( da ), "graphics-contexts",
145                                 gcs, release_gobject_array );
146    }
147
148    /**
149    ***  Get the cells' previous colors...
150    ***  (this is used for blinking when the color changes)
151    **/
152
153    prev_color = (guint8*) g_object_get_data ( G_OBJECT( da ), "prev-color" );
154    if( prev_color == NULL )
155    {
156        first_time = TRUE;
157        prev_color = g_new0 ( guint8, n_cells );
158        g_object_set_data_full ( G_OBJECT(
159                                     da ), "prev-color", prev_color, g_free );
160    }
161
162    /**
163    ***  Get the piece data values...
164    **/
165
166    switch( mode )
167    {
168        case DRAW_AVAIL:
169            pieces = g_new ( int8_t, n_cells );
170            tr_torrentAvailability ( tor, pieces, n_cells );
171            break;
172
173        case DRAW_PROG:
174            completeness = g_new ( float, n_cells );
175            tr_torrentAmountFinished ( tor, completeness, n_cells );
176            break;
177
178        default:
179            g_error( "no mode defined!" );
180    }
181
182    /**
183    ***  Draw...
184    **/
185
186    i = 0;
187    for( y = 0; y < n_rows; ++y )
188    {
189        for( x = 0; x < n_cols; ++x )
190        {
191            int draw_x = grid_bounds.x + (int)( x * piece_w );
192            int draw_y = grid_bounds.y + (int)( y * piece_h );
193            int color = BLACK;
194            int border = BLACK;
195
196            if( rtl )
197                draw_x = grid_bounds.x + grid_bounds.width -
198                         (int)( ( x + 1 ) * piece_w );
199            else
200                draw_x = grid_bounds.x + (int)( x * piece_w );
201            draw_y = grid_bounds.y + (int)( y * piece_h );
202
203            if( i < n_cells )
204            {
205                border = GRAY;
206
207                if( mode == DRAW_AVAIL )
208                {
209                    const int8_t val = pieces[i];
210                    if( val <  0 ) color = ALL;
211                    else if( val == 0 ) color = NONE;
212                    else if( val <= 4 ) color = FEW;
213                    else if( val <= 8 ) color = SOME;
214                    else color = LOTS;
215                }
216                else /* completeness */
217                {
218                    const float val = completeness[i];
219                    if( val >= 1.00 ) color = ALL;
220                    else if( val >= 0.66 ) color = LOTS;
221                    else if( val >= 0.33 ) color = SOME;
222                    else if( val >= 0.01 ) color = FEW;
223                    else color = NONE;
224                }
225
226                /* draw a "blink" for one interval when a piece changes */
227                if( first_time )
228                    prev_color[i] = color;
229                else if( color != prev_color[i] )
230                {
231                    prev_color[i] = color;
232                    color = border = BLINK;
233                }
234            }
235
236            gdk_draw_rectangle ( da->window, gcs[color], TRUE,
237                                 draw_x, draw_y,
238                                 piece_w_int, piece_h_int );
239
240            if( i < n_cells )
241                gdk_draw_rectangle ( da->window, gcs[border], FALSE,
242                                     draw_x, draw_y,
243                                     piece_w_int, piece_h_int );
244
245            ++i;
246        }
247    }
248
249    gdk_draw_rectangle ( da->window, gcs[GRAY], FALSE,
250                         grid_bounds.x, grid_bounds.y,
251                         grid_bounds.width - 1, grid_bounds.height - 1 );
252
253    g_free ( pieces );
254    g_free ( completeness );
255    return FALSE;
256}
257
258#endif
259
260/****
261*****  PEERS TAB
262****/
263
264enum
265{
266    WEBSEED_COL_URL,
267    WEBSEED_COL_DOWNLOAD_RATE,
268    N_WEBSEED_COLS
269};
270
271static const char * webseed_column_names[N_WEBSEED_COLS] =
272{
273    N_( "Web Seeds" ),
274    /* 'download speed' column header. terse to keep the column narrow. */
275    N_( "Down" )
276};
277
278static GtkTreeModel*
279webseed_model_new( const tr_torrent * tor )
280{
281    int             i;
282    const tr_info * inf = tr_torrentInfo( tor );
283    float *         speeds = tr_torrentWebSpeeds( tor );
284    GtkListStore *  store = gtk_list_store_new( N_WEBSEED_COLS,
285                                                G_TYPE_STRING,
286                                                G_TYPE_FLOAT );
287
288    for( i = 0; i < inf->webseedCount; ++i )
289    {
290        GtkTreeIter iter;
291        gtk_list_store_append( store, &iter );
292        gtk_list_store_set( store, &iter, WEBSEED_COL_URL, inf->webseeds[i],
293                            WEBSEED_COL_DOWNLOAD_RATE, speeds[i],
294                            -1 );
295    }
296
297    tr_free( speeds );
298    return GTK_TREE_MODEL( store );
299}
300
301enum
302{
303    PEER_COL_ADDRESS,
304    PEER_COL_DOWNLOAD_RATE,
305    PEER_COL_UPLOAD_RATE,
306    PEER_COL_CLIENT,
307    PEER_COL_PROGRESS,
308    PEER_COL_IS_ENCRYPTED,
309    PEER_COL_STATUS,
310    N_PEER_COLS
311};
312
313static const char* peer_column_names[N_PEER_COLS] =
314{
315    N_( "Address" ),
316    /* 'download speed' column header. terse to keep the column narrow. */
317    N_( "Down" ),
318    /* 'upload speed' column header.  terse to keep the column narrow. */
319    N_( "Up" ),
320    N_( "Client" ),
321    /* 'percent done' column header. terse to keep the column narrow. */
322    N_( "%" ),
323    " ",
324    N_( "Status" )
325};
326
327static int
328compare_peers( const void * a, const void * b )
329{
330    const tr_peer_stat * pa = a;
331    const tr_peer_stat * pb = b;
332
333    return strcmp( pa->addr, pb->addr );
334}
335
336static int
337compare_addr_to_peer( const void * addr, const void * b )
338{
339    const tr_peer_stat * peer = b;
340
341    return strcmp( addr, peer->addr );
342}
343
344static void
345peer_row_set( GtkListStore        * store,
346              GtkTreeIter         * iter,
347              const tr_peer_stat  * peer )
348{
349    const char * client = peer->client;
350
351    if( !client || !strcmp( client, "Unknown Client" ) )
352        client = " ";
353
354    gtk_list_store_set( store, iter,
355                        PEER_COL_ADDRESS, peer->addr,
356                        PEER_COL_CLIENT, client,
357                        PEER_COL_IS_ENCRYPTED, peer->isEncrypted,
358                        PEER_COL_PROGRESS, (int)( 100.0 * peer->progress ),
359                        PEER_COL_DOWNLOAD_RATE, peer->rateToClient,
360                        PEER_COL_UPLOAD_RATE, peer->rateToPeer,
361                        PEER_COL_STATUS, peer->flagStr,
362                        -1 );
363}
364
365static void
366append_peers_to_model( GtkListStore *       store,
367                       const tr_peer_stat * peers,
368                       int                  n_peers )
369{
370    int i;
371
372    for( i = 0; i < n_peers; ++i )
373    {
374        GtkTreeIter iter;
375        gtk_list_store_append( store, &iter );
376        peer_row_set ( store, &iter, &peers[i] );
377    }
378}
379
380static GtkTreeModel*
381peer_model_new( tr_torrent * tor )
382{
383    GtkListStore * m = gtk_list_store_new ( N_PEER_COLS,
384                                            G_TYPE_STRING, /* addr */
385                                            G_TYPE_FLOAT, /* downloadFromRate */
386                                            G_TYPE_FLOAT, /* uploadToRate */
387                                            G_TYPE_STRING, /* client */
388                                            G_TYPE_INT,   /* progress [0..100] */
389                                            G_TYPE_BOOLEAN, /* isEncrypted */
390                                            G_TYPE_STRING ); /* flagString */
391
392    int            n_peers = 0;
393    tr_peer_stat * peers = tr_torrentPeers ( tor, &n_peers );
394
395    qsort ( peers, n_peers, sizeof( tr_peer_stat ), compare_peers );
396    append_peers_to_model ( m, peers, n_peers );
397    tr_torrentPeersFree( peers, 0 );
398    return GTK_TREE_MODEL ( m );
399}
400
401static void
402render_encrypted( GtkTreeViewColumn  * column UNUSED,
403                  GtkCellRenderer *           renderer,
404                  GtkTreeModel *              tree_model,
405                  GtkTreeIter *               iter,
406                  gpointer             data   UNUSED )
407{
408    gboolean is_encrypted = FALSE;
409
410    gtk_tree_model_get ( tree_model, iter, PEER_COL_IS_ENCRYPTED,
411                         &is_encrypted,
412                         -1 );
413    g_object_set ( renderer, "xalign", (gfloat)0.0,
414                   "yalign", (gfloat)0.5,
415                   "stock-id", ( is_encrypted ? "transmission-lock" : NULL ),
416                   NULL );
417}
418
419static void
420render_ul_rate( GtkTreeViewColumn  * column UNUSED,
421                GtkCellRenderer *           renderer,
422                GtkTreeModel *              tree_model,
423                GtkTreeIter *               iter,
424                gpointer             data   UNUSED )
425{
426    float rate = 0.0;
427
428    gtk_tree_model_get ( tree_model, iter, PEER_COL_UPLOAD_RATE, &rate, -1 );
429    if( rate < 0.01 )
430        g_object_set ( renderer, "text", "", NULL );
431    else
432    {
433        char speedStr[64];
434        tr_strlspeed( speedStr, rate, sizeof( speedStr ) );
435        g_object_set( renderer, "text", speedStr, NULL );
436    }
437}
438
439static void
440render_dl_rate( GtkTreeViewColumn  * column UNUSED,
441                GtkCellRenderer *           renderer,
442                GtkTreeModel *              tree_model,
443                GtkTreeIter *               iter,
444                gpointer             data   UNUSED )
445{
446    float rate = 0.0;
447
448    gtk_tree_model_get ( tree_model, iter, PEER_COL_DOWNLOAD_RATE, &rate,
449                         -1 );
450    if( rate < 0.01 )
451        g_object_set ( renderer, "text", "", NULL );
452    else
453    {
454        char speedStr[64];
455        tr_strlspeed( speedStr, rate, sizeof( speedStr ) );
456        g_object_set( renderer, "text", speedStr, NULL );
457    }
458}
459
460static void
461render_client( GtkTreeViewColumn   * column UNUSED,
462               GtkCellRenderer *            renderer,
463               GtkTreeModel *               tree_model,
464               GtkTreeIter *                iter,
465               gpointer              data   UNUSED )
466{
467    char * client = NULL;
468
469    gtk_tree_model_get ( tree_model, iter, PEER_COL_CLIENT, &client,
470                         -1 );
471    g_object_set ( renderer, "text", ( client ? client : "" ), NULL );
472    g_free ( client );
473}
474
475typedef struct
476{
477    TrTorrent *     gtor;
478    GtkTreeModel *  model; /* same object as store, but recast */
479    GtkListStore *  store; /* same object as model, but recast */
480    GtkListStore *  webseeds;
481    GtkWidget *     completeness;
482    GtkWidget *     seeders_lb;
483    GtkWidget *     leechers_lb;
484    GtkWidget *     completed_lb;
485    GtkWidget *     peer_tree_view;
486}
487PeerData;
488
489static void
490fmtpeercount( GtkWidget * l,
491              int         count )
492{
493    if( 0 > count )
494    {
495        gtk_label_set_text( GTK_LABEL( l ), "?" );
496    }
497    else
498    {
499        char str[16];
500        g_snprintf( str, sizeof str, "%'d", count );
501        gtk_label_set_text( GTK_LABEL( l ), str );
502    }
503}
504
505static void
506refresh_peers( GtkWidget * top )
507{
508    int             i;
509    int             n_peers;
510    GtkTreeIter     iter;
511    PeerData *      p = (PeerData*) g_object_get_data ( G_OBJECT(
512                                                            top ),
513                                                        "peer-data" );
514    tr_torrent *    tor = tr_torrent_handle ( p->gtor );
515    GtkTreeModel *  model = p->model;
516    GtkListStore *  store = p->store;
517    tr_peer_stat *  peers;
518    const tr_stat * stat = tr_torrent_stat( p->gtor );
519    const tr_info * inf = tr_torrent_info( p->gtor );
520
521    if( inf->webseedCount )
522    {
523        float * speeds = tr_torrentWebSpeeds( tor );
524        for( i = 0; i < inf->webseedCount; ++i )
525        {
526            GtkTreeIter iter;
527            gtk_tree_model_iter_nth_child( GTK_TREE_MODEL(
528                                               p->webseeds ), &iter, NULL,
529                                           i );
530            gtk_list_store_set( p->webseeds, &iter,
531                                WEBSEED_COL_DOWNLOAD_RATE, speeds[i],
532                                -1 );
533        }
534        tr_free( speeds );
535    }
536
537    /**
538    ***  merge the peer diffs into the tree model.
539    ***
540    ***  this is more complicated than creating a new model,
541    ***  but is also (a) more efficient and (b) doesn't undo
542    ***  the view's visible area and sorting on every refresh.
543    **/
544
545    n_peers = 0;
546    peers = tr_torrentPeers ( tor, &n_peers );
547    qsort ( peers, n_peers, sizeof( tr_peer_stat ), compare_peers );
548
549    if( gtk_tree_model_get_iter_first ( model, &iter ) ) do
550        {
551            char *         addr = NULL;
552            tr_peer_stat * peer = NULL;
553            gtk_tree_model_get ( model, &iter, PEER_COL_ADDRESS, &addr, -1 );
554            peer = bsearch ( addr, peers, n_peers, sizeof( tr_peer_stat ),
555                             compare_addr_to_peer );
556            g_free ( addr );
557
558            if( peer ) /* update a pre-existing row */
559            {
560                const int pos = peer - peers;
561                const int n_rhs = n_peers - ( pos + 1 );
562                g_assert ( n_rhs >= 0 );
563
564                peer_row_set ( store, &iter, peer );
565
566                /* remove it from the tr_peer_stat list */
567                g_memmove ( peer, peer + 1, sizeof( tr_peer_stat ) * n_rhs );
568                --n_peers;
569            }
570            else if( !gtk_list_store_remove ( store, &iter ) )
571                break; /* we removed the model's last item */
572        }
573        while( gtk_tree_model_iter_next ( model, &iter ) );
574
575    append_peers_to_model ( store, peers, n_peers ); /* all these are new */
576
577#ifdef SHOW_PIECES
578    if( GDK_IS_DRAWABLE ( p->completeness->window ) )
579        refresh_pieces ( p->completeness, NULL, p->gtor );
580#endif
581
582    fmtpeercount ( p->seeders_lb, stat->seeders );
583    fmtpeercount ( p->leechers_lb, stat->leechers );
584    fmtpeercount ( p->completed_lb, stat->timesCompleted );
585
586    free( peers );
587
588    gtk_widget_queue_draw( p->peer_tree_view );
589}
590
591#if GTK_CHECK_VERSION( 2, 12, 0 )
592static gboolean
593onPeerViewQueryTooltip( GtkWidget *            widget,
594                        gint                   x,
595                        gint                   y,
596                        gboolean               keyboard_tip,
597                        GtkTooltip *           tooltip,
598                        gpointer     user_data UNUSED )
599{
600    gboolean       show_tip = FALSE;
601    GtkTreeModel * model;
602    GtkTreeIter    iter;
603
604    if( gtk_tree_view_get_tooltip_context( GTK_TREE_VIEW( widget ),
605                                           &x, &y, keyboard_tip,
606                                           &model, NULL, &iter ) )
607    {
608        const char * pch;
609        char *       str = NULL;
610        GString *    gstr = g_string_new( NULL );
611        gtk_tree_model_get( model, &iter, PEER_COL_STATUS, &str, -1 );
612        for( pch = str; pch && *pch; ++pch )
613        {
614            const char * txt = NULL;
615            switch( *pch )
616            {
617                case 'O':
618                    txt = _( "Optimistic unchoke" ); break;
619
620                case 'D':
621                    txt = _( "Downloading from this peer" ); break;
622
623                case 'd':
624                    txt = _(
625                        "We would download from this peer if they would let us" );
626                    break;
627
628                case 'U':
629                    txt = _( "Uploading to peer" ); break;
630
631                case 'u':
632                    txt = _( "We would upload to this peer if they asked" );
633                    break;
634
635                case 'K':
636                    txt = _(
637                        "Peer has unchoked us, but we're not interested" );
638                    break;
639
640                case '?':
641                    txt = _(
642                        "We unchoked this peer, but they're not interested" );
643                    break;
644
645                case 'E':
646                    txt = _( "Encrypted connection" ); break;
647
648                case 'X':
649                    txt = _(
650                        "Peer was discovered through Peer Exchange (PEX)" );
651                    break;
652
653                case 'I':
654                    txt = _( "Peer is an incoming connection" ); break;
655            }
656            if( txt )
657                g_string_append_printf( gstr, "%c: %s\n", *pch, txt );
658        }
659        if( gstr->len ) /* remove the last linefeed */
660            g_string_set_size( gstr, gstr->len - 1 );
661        gtk_tooltip_set_text( tooltip, gstr->str );
662        g_string_free( gstr, TRUE );
663        g_free( str );
664        show_tip = TRUE;
665    }
666
667    return show_tip;
668}
669
670#endif
671
672static GtkWidget*
673peer_page_new( TrTorrent * gtor )
674{
675    guint           i;
676    GtkTreeModel *  m;
677    GtkWidget *     v, *w, *ret, *sw, *l, *vbox, *hbox;
678    GtkWidget *     webtree = NULL;
679    tr_torrent *    tor = tr_torrent_handle ( gtor );
680    PeerData *      p = g_new ( PeerData, 1 );
681    const tr_info * inf = tr_torrent_info( gtor );
682
683    /* TODO: make this configurable? */
684    int view_columns[] = { PEER_COL_IS_ENCRYPTED,
685                           PEER_COL_UPLOAD_RATE,
686                           PEER_COL_DOWNLOAD_RATE,
687                           PEER_COL_PROGRESS,
688                           PEER_COL_STATUS,
689                           PEER_COL_ADDRESS,
690                           PEER_COL_CLIENT };
691
692
693    if( inf->webseedCount )
694    {
695        GtkTreeViewColumn * c;
696        GtkCellRenderer *   r;
697        const char *        t;
698        GtkWidget *         fr;
699
700        m = webseed_model_new( tr_torrent_handle( gtor ) );
701        webtree = gtk_tree_view_new_with_model( m );
702        g_signal_connect( webtree, "button-release-event",
703                          G_CALLBACK( on_tree_view_button_released ), NULL );
704        gtk_tree_view_set_rules_hint( GTK_TREE_VIEW( webtree ), TRUE );
705        p->webseeds = GTK_LIST_STORE( m );
706        g_object_unref( G_OBJECT( m ) );
707
708        t = _( webseed_column_names[WEBSEED_COL_URL] );
709        r = gtk_cell_renderer_text_new ( );
710        g_object_set( G_OBJECT( r ), "ellipsize", PANGO_ELLIPSIZE_END, NULL );
711        c =
712            gtk_tree_view_column_new_with_attributes( t, r, "text",
713                                                      WEBSEED_COL_URL,
714                                                      NULL );
715        g_object_set( G_OBJECT( c ), "expand", TRUE, NULL );
716        gtk_tree_view_column_set_sort_column_id( c, WEBSEED_COL_URL );
717        gtk_tree_view_append_column( GTK_TREE_VIEW( webtree ), c );
718
719        t = _( webseed_column_names[WEBSEED_COL_DOWNLOAD_RATE] );
720        r = gtk_cell_renderer_text_new ( );
721        c = gtk_tree_view_column_new_with_attributes (
722            t, r, "text", WEBSEED_COL_DOWNLOAD_RATE, NULL );
723        gtk_tree_view_column_set_cell_data_func ( c, r, render_dl_rate,
724                                                  NULL, NULL );
725        gtk_tree_view_column_set_sort_column_id( c,
726                                                 WEBSEED_COL_DOWNLOAD_RATE );
727        gtk_tree_view_append_column( GTK_TREE_VIEW( webtree ), c );
728
729        fr = gtk_frame_new( NULL );
730        gtk_frame_set_shadow_type( GTK_FRAME( fr ), GTK_SHADOW_IN );
731        gtk_container_add( GTK_CONTAINER( fr ), webtree );
732        webtree = fr;
733    }
734
735    m  = peer_model_new ( tor );
736    v = GTK_WIDGET( g_object_new( GTK_TYPE_TREE_VIEW,
737                                  "model",  gtk_tree_model_sort_new_with_model( m ),
738                                  "rules-hint", TRUE,
739#if GTK_CHECK_VERSION( 2, 12, 0 )
740                                  "has-tooltip", TRUE,
741#endif
742                                  NULL ) );
743    p->peer_tree_view = v;
744
745#if GTK_CHECK_VERSION( 2, 12, 0 )
746    g_signal_connect( v, "query-tooltip",
747                      G_CALLBACK( onPeerViewQueryTooltip ), NULL );
748#endif
749    gtk_widget_set_size_request( v, 550, 0 );
750    g_object_unref ( G_OBJECT( m ) );
751    g_signal_connect( v, "button-release-event",
752                      G_CALLBACK( on_tree_view_button_released ), NULL );
753
754    for( i = 0; i < G_N_ELEMENTS( view_columns ); ++i )
755    {
756        const int           col = view_columns[i];
757        const char *        t = _( peer_column_names[col] );
758        GtkTreeViewColumn * c;
759        GtkCellRenderer *   r;
760
761        switch( col )
762        {
763            case PEER_COL_ADDRESS:
764                r = gtk_cell_renderer_text_new ( );
765                c = gtk_tree_view_column_new_with_attributes ( t, r, "text", col, NULL );
766                break;
767
768            case PEER_COL_CLIENT:
769                r = gtk_cell_renderer_text_new ( );
770                c = gtk_tree_view_column_new_with_attributes ( t, r, "text", col, NULL );
771                gtk_tree_view_column_set_cell_data_func ( c, r, render_client, NULL, NULL );
772                break;
773
774            case PEER_COL_PROGRESS:
775                r = gtk_cell_renderer_progress_new ( );
776                c = gtk_tree_view_column_new_with_attributes ( t, r, "value", PEER_COL_PROGRESS, NULL );
777                break;
778
779            case PEER_COL_IS_ENCRYPTED:
780                r = gtk_cell_renderer_pixbuf_new ( );
781                c = gtk_tree_view_column_new_with_attributes ( t, r, NULL );
782                gtk_tree_view_column_set_sizing ( c, GTK_TREE_VIEW_COLUMN_FIXED );
783                gtk_tree_view_column_set_fixed_width ( c, 20 );
784                gtk_tree_view_column_set_cell_data_func ( c, r, render_encrypted, NULL, NULL );
785                break;
786
787            case PEER_COL_DOWNLOAD_RATE:
788                r = gtk_cell_renderer_text_new ( );
789                c = gtk_tree_view_column_new_with_attributes ( t, r, "text", col, NULL );
790                gtk_tree_view_column_set_cell_data_func ( c, r, render_dl_rate, NULL, NULL );
791                break;
792
793            case PEER_COL_UPLOAD_RATE:
794                r = gtk_cell_renderer_text_new ( );
795                c = gtk_tree_view_column_new_with_attributes ( t, r, "text", col, NULL );
796                gtk_tree_view_column_set_cell_data_func ( c, r, render_ul_rate, NULL, NULL );
797                break;
798
799            case PEER_COL_STATUS:
800                r = gtk_cell_renderer_text_new( );
801                c = gtk_tree_view_column_new_with_attributes ( t, r, "text", col, NULL );
802                break;
803
804            default:
805                abort ( );
806        }
807
808        gtk_tree_view_column_set_resizable ( c, FALSE );
809        gtk_tree_view_column_set_sort_column_id ( c, col );
810        gtk_tree_view_append_column ( GTK_TREE_VIEW( v ), c );
811    }
812
813    /* the 'expander' column has a 10-pixel margin on the left
814       that doesn't look quite correct in any of these columns...
815       so create a non-visible column and assign it as the
816       'expander column. */
817    {
818        GtkTreeViewColumn *c = gtk_tree_view_column_new ( );
819        gtk_tree_view_column_set_visible ( c, FALSE );
820        gtk_tree_view_append_column ( GTK_TREE_VIEW( v ), c );
821        gtk_tree_view_set_expander_column ( GTK_TREE_VIEW( v ), c );
822    }
823
824    w = sw = gtk_scrolled_window_new ( NULL, NULL );
825    gtk_scrolled_window_set_policy ( GTK_SCROLLED_WINDOW( w ),
826                                     GTK_POLICY_AUTOMATIC,
827                                     GTK_POLICY_AUTOMATIC );
828    gtk_scrolled_window_set_shadow_type ( GTK_SCROLLED_WINDOW( w ),
829                                          GTK_SHADOW_IN );
830    gtk_container_add ( GTK_CONTAINER( w ), v );
831
832
833    vbox = gtk_vbox_new ( FALSE, GUI_PAD );
834    gtk_container_set_border_width ( GTK_CONTAINER( vbox ), GUI_PAD_BIG );
835
836    if( webtree )
837        gtk_box_pack_start( GTK_BOX( vbox ), webtree, FALSE, FALSE, 0 );
838
839    /* h = gtk_hbox_new (FALSE, GUI_PAD); */
840    /* gtk_box_pack_start_defaults (GTK_BOX(h), sw); */
841    gtk_box_pack_start( GTK_BOX( vbox ), sw, TRUE, TRUE, 0 );
842
843    hbox = gtk_hbox_new ( FALSE, GUI_PAD );
844    l = gtk_label_new ( NULL );
845    gtk_label_set_markup ( GTK_LABEL( l ), _( "<b>Seeders:</b>" ) );
846    gtk_box_pack_start ( GTK_BOX( hbox ), l, FALSE, FALSE, 0 );
847    l = p->seeders_lb = gtk_label_new ( NULL );
848    gtk_box_pack_start ( GTK_BOX( hbox ), l, FALSE, FALSE, 0 );
849    gtk_box_pack_start( GTK_BOX( hbox ),
850                        gtk_alignment_new ( 0.0f, 0.0f, 0.0f, 0.0f ),
851                        TRUE, TRUE, 0 );
852    l = gtk_label_new ( NULL );
853    gtk_label_set_markup ( GTK_LABEL( l ), _( "<b>Leechers:</b>" ) );
854    gtk_box_pack_start ( GTK_BOX( hbox ), l, FALSE, FALSE, 0 );
855    l = p->leechers_lb = gtk_label_new ( NULL );
856    gtk_box_pack_start ( GTK_BOX( hbox ), l, FALSE, FALSE, 0 );
857    gtk_box_pack_start( GTK_BOX( hbox ),
858                        gtk_alignment_new ( 0.0f, 0.0f, 0.0f, 0.0f ),
859                        TRUE, TRUE, 0 );
860    l = gtk_label_new ( NULL );
861    gtk_label_set_markup ( GTK_LABEL( l ), _( "<b>Times Completed:</b>" ) );
862    gtk_box_pack_start ( GTK_BOX( hbox ), l, FALSE, FALSE, 0 );
863    l = p->completed_lb = gtk_label_new ( NULL );
864    gtk_box_pack_start ( GTK_BOX( hbox ), l, FALSE, FALSE, 0 );
865    gtk_box_pack_start ( GTK_BOX( vbox ), hbox, FALSE, FALSE, 0 );
866
867    ret = vbox;
868    p->gtor = gtor;
869    p->model = m;
870    p->store = GTK_LIST_STORE( m );
871    g_object_set_data_full ( G_OBJECT( ret ), "peer-data", p, g_free );
872    return ret;
873}
874
875/****
876*****  INFO TAB
877****/
878
879static void
880refresh_time_lb( GtkWidget * l,
881                 time_t      t )
882{
883    const char * never = _( "Never" );
884
885    if( !t )
886        gtk_label_set_text( GTK_LABEL( l ), never );
887    else
888    {
889        char * str = gtr_localtime( t );
890        gtk_label_set_text( GTK_LABEL( l ), str );
891        g_free( str );
892    }
893}
894
895static GtkWidget*
896info_page_new( tr_torrent * tor )
897{
898    int             row = 0;
899    GtkWidget *     t = hig_workarea_create ( );
900    GtkWidget *     l, *w, *fr;
901    char *          pch;
902    char            sizeStr[128];
903    char            countStr[128];
904    char            buf[256];
905    GtkTextBuffer * b;
906    const tr_info * info = tr_torrentInfo( tor );
907
908    hig_workarea_add_section_title ( t, &row, _( "Details" ) );
909
910    g_snprintf( countStr, sizeof( countStr ),
911                ngettext( "%'d Piece", "%'d Pieces", info->pieceCount ),
912                info->pieceCount );
913    tr_strlsize( sizeStr, info->pieceSize, sizeof( sizeStr ) );
914    g_snprintf( buf, sizeof( buf ),
915                /* %1$s is number of pieces;
916                   %2$s is how big each piece is */
917                _( "%1$s @ %2$s" ),
918                countStr, sizeStr );
919
920    l = gtk_label_new ( buf );
921    hig_workarea_add_row ( t, &row, _( "Pieces:" ), l, NULL );
922
923    l =
924        g_object_new( GTK_TYPE_LABEL, "label", info->hashString,
925                      "selectable",
926                      TRUE,
927                      "ellipsize", PANGO_ELLIPSIZE_END,
928                      NULL );
929    hig_workarea_add_row ( t, &row, _( "Hash:" ), l, NULL );
930
931    pch = ( info->isPrivate )
932          ? _( "Private to this tracker -- PEX disabled" )
933          : _( "Public torrent" );
934    l = gtk_label_new ( pch );
935    hig_workarea_add_row ( t, &row, _( "Privacy:" ), l, NULL );
936
937    b = gtk_text_buffer_new ( NULL );
938    if( info->comment )
939        gtk_text_buffer_set_text ( b, info->comment, -1 );
940    w = gtk_text_view_new_with_buffer ( b );
941    gtk_widget_set_size_request ( w, 0u, 100u );
942    gtk_text_view_set_wrap_mode ( GTK_TEXT_VIEW( w ), GTK_WRAP_WORD );
943    gtk_text_view_set_editable ( GTK_TEXT_VIEW( w ), FALSE );
944    fr = gtk_frame_new ( NULL );
945    gtk_frame_set_shadow_type ( GTK_FRAME( fr ), GTK_SHADOW_IN );
946    gtk_container_add ( GTK_CONTAINER( fr ), w );
947    w = hig_workarea_add_row ( t, &row, _( "Comment:" ), fr, NULL );
948    gtk_misc_set_alignment ( GTK_MISC( w ), 0.0f, 0.0f );
949
950    hig_workarea_add_section_divider ( t, &row );
951    hig_workarea_add_section_title ( t, &row, _( "Origins" ) );
952
953    l = gtk_label_new ( *info->creator ? info->creator : _( "Unknown" ) );
954    gtk_label_set_ellipsize( GTK_LABEL( l ), PANGO_ELLIPSIZE_END );
955    hig_workarea_add_row ( t, &row, _( "Creator:" ), l, NULL );
956
957    l = gtk_label_new( NULL );
958    refresh_time_lb( l, info->dateCreated );
959    hig_workarea_add_row ( t, &row, _( "Date:" ), l, NULL );
960
961    hig_workarea_add_section_divider ( t, &row );
962    hig_workarea_add_section_title ( t, &row, _( "Location" ) );
963
964    l =
965        g_object_new( GTK_TYPE_LABEL, "label", tr_torrentGetDownloadDir(
966                          tor ), "selectable", TRUE,
967                      "ellipsize", PANGO_ELLIPSIZE_END, NULL );
968    hig_workarea_add_row ( t, &row, _( "Destination folder:" ), l, NULL );
969
970    l =
971        g_object_new( GTK_TYPE_LABEL, "label", info->torrent, "selectable",
972                      TRUE,
973                      "ellipsize", PANGO_ELLIPSIZE_END,
974                      NULL );
975    hig_workarea_add_row ( t, &row, _( "Torrent file:" ), l, NULL );
976
977    hig_workarea_finish ( t, &row );
978    return t;
979}
980
981/****
982*****  ACTIVITY TAB
983****/
984
985typedef struct
986{
987    GtkWidget *  state_lb;
988    GtkWidget *  progress_lb;
989    GtkWidget *  have_lb;
990    GtkWidget *  dl_lb;
991    GtkWidget *  ul_lb;
992    GtkWidget *  failed_lb;
993    GtkWidget *  ratio_lb;
994    GtkWidget *  err_lb;
995    GtkWidget *  swarm_lb;
996    GtkWidget *  date_added_lb;
997    GtkWidget *  last_activity_lb;
998    GtkWidget *  availability_da;
999    TrTorrent *  gtor;
1000}
1001Activity;
1002
1003static void
1004refresh_activity( GtkWidget * top )
1005{
1006    Activity *      a = g_object_get_data ( G_OBJECT(
1007                                                top ), "activity-data" );
1008    char *          pch;
1009    char            sizeStr[64];
1010    char            sizeStr2[64];
1011    char            buf[128];
1012    const tr_stat * stat = tr_torrent_stat( a->gtor );
1013    const double    complete = stat->percentComplete * 100.0;
1014    const double    done = stat->percentDone * 100.0;
1015
1016    pch = tr_torrent_status_str( a->gtor );
1017    gtk_label_set_text ( GTK_LABEL( a->state_lb ), pch );
1018    g_free ( pch );
1019
1020    if( (int)complete == (int)done )
1021        pch = g_strdup_printf( _( "%.1f%%" ), complete );
1022    else
1023        /* %1$.1f is percent of how much of what we want's been downloaded,
1024           %2$.1f is percent of how much of the whole torrent we've downloaded
1025           */
1026        pch = g_strdup_printf( _(
1027                                   "%1$.1f%% (%2$.1f%% selected)" ),
1028                               complete, done );
1029    gtk_label_set_text ( GTK_LABEL( a->progress_lb ), pch );
1030    g_free ( pch );
1031
1032    tr_strlsize( sizeStr,  stat->haveValid + stat->haveUnchecked,
1033                sizeof( sizeStr ) );
1034    tr_strlsize( sizeStr2, stat->haveValid,
1035                sizeof( sizeStr2 ) );
1036    /* %1$s is total size of what we've saved to disk
1037       %2$s is how much of it's passed the checksum test */
1038    g_snprintf( buf, sizeof( buf ), _(
1039                    "%1$s (%2$s verified)" ), sizeStr, sizeStr2 );
1040    gtk_label_set_text( GTK_LABEL( a->have_lb ), buf );
1041
1042    tr_strlsize( sizeStr, stat->downloadedEver, sizeof( sizeStr ) );
1043    gtk_label_set_text( GTK_LABEL( a->dl_lb ), sizeStr );
1044
1045    tr_strlsize( sizeStr, stat->uploadedEver, sizeof( sizeStr ) );
1046    gtk_label_set_text( GTK_LABEL( a->ul_lb ), sizeStr );
1047
1048    tr_strlsize( sizeStr, stat->corruptEver, sizeof( sizeStr ) );
1049    gtk_label_set_text( GTK_LABEL( a->failed_lb ), sizeStr );
1050
1051    tr_strlratio( buf, stat->ratio, sizeof( buf ) );
1052    gtk_label_set_text( GTK_LABEL( a->ratio_lb ), buf );
1053
1054    tr_strlspeed( buf, stat->swarmSpeed, sizeof( buf ) );
1055    gtk_label_set_text ( GTK_LABEL( a->swarm_lb ), buf );
1056
1057    gtk_label_set_text ( GTK_LABEL( a->err_lb ),
1058                        *stat->errorString ? stat->errorString : _( "None" ) );
1059
1060    refresh_time_lb( a->date_added_lb, stat->addedDate );
1061
1062    refresh_time_lb( a->last_activity_lb, stat->activityDate );
1063
1064#ifdef SHOW_PIECES
1065    if( GDK_IS_DRAWABLE ( a->availability_da->window ) )
1066        refresh_pieces ( a->availability_da, NULL, a->gtor );
1067#endif
1068}
1069
1070static GtkWidget*
1071activity_page_new( TrTorrent * gtor )
1072{
1073    Activity * a = g_new ( Activity, 1 );
1074    int        row = 0;
1075    GtkWidget *t = hig_workarea_create ( );
1076    GtkWidget *l;
1077
1078    a->gtor = gtor;
1079
1080    hig_workarea_add_section_title ( t, &row, _( "Transfer" ) );
1081
1082    l = a->state_lb = gtk_label_new ( NULL );
1083    hig_workarea_add_row ( t, &row, _( "State:" ), l, NULL );
1084
1085    l = a->progress_lb = gtk_label_new ( NULL );
1086    hig_workarea_add_row ( t, &row, _( "Progress:" ), l, NULL );
1087
1088    l = a->have_lb = gtk_label_new ( NULL );
1089    /* "Have" refers to how much of the torrent we have */
1090    hig_workarea_add_row ( t, &row, _( "Have:" ), l, NULL );
1091
1092    l = a->dl_lb = gtk_label_new ( NULL );
1093    hig_workarea_add_row ( t, &row, _( "Downloaded:" ), l, NULL );
1094
1095    l = a->ul_lb = gtk_label_new ( NULL );
1096    hig_workarea_add_row ( t, &row, _( "Uploaded:" ), l, NULL );
1097
1098    /* how much downloaded data was corrupt */
1099    l = a->failed_lb = gtk_label_new ( NULL );
1100    hig_workarea_add_row ( t, &row, _( "Failed DL:" ), l, NULL );
1101
1102    l = a->ratio_lb = gtk_label_new ( NULL );
1103    hig_workarea_add_row ( t, &row, _( "Ratio:" ), l, NULL );
1104
1105    l = a->swarm_lb = gtk_label_new ( NULL );
1106    hig_workarea_add_row ( t, &row, _( "Swarm rate:" ), l, NULL );
1107
1108    l = a->err_lb = gtk_label_new ( NULL );
1109    hig_workarea_add_row ( t, &row, _( "Error:" ), l, NULL );
1110
1111#ifdef SHOW_PIECES
1112    hig_workarea_add_section_divider ( t, &row );
1113    hig_workarea_add_section_title ( t, &row, _( "Completion" ) );
1114
1115    w = a->availability_da = gtk_drawing_area_new ( );
1116    gtk_widget_set_size_request ( w, 0u, 100u );
1117    g_object_set_data ( G_OBJECT( w ), "draw-mode",
1118                       GINT_TO_POINTER( DRAW_PROG ) );
1119    g_signal_connect ( w, "expose-event", G_CALLBACK(
1120                           refresh_pieces ), gtor );
1121    hig_workarea_add_wide_control( t, &row, w );
1122#endif
1123
1124    hig_workarea_add_section_divider ( t, &row );
1125    hig_workarea_add_section_title ( t, &row, _( "Dates" ) );
1126
1127    l = a->date_added_lb = gtk_label_new ( NULL );
1128    hig_workarea_add_row ( t, &row, _( "Started at:" ), l, NULL );
1129
1130    l = a->last_activity_lb = gtk_label_new ( NULL );
1131    hig_workarea_add_row ( t, &row, _( "Last activity at:" ), l, NULL );
1132
1133    hig_workarea_add_section_divider ( t, &row );
1134    hig_workarea_finish ( t, &row );
1135    g_object_set_data_full ( G_OBJECT( t ), "activity-data", a, g_free );
1136    return t;
1137}
1138
1139/****
1140*****  OPTIONS
1141****/
1142
1143static void
1144speed_toggled_cb( GtkToggleButton * tb,
1145                  gpointer          gtor,
1146                  int               up_or_down )
1147{
1148    tr_torrent * tor = tr_torrent_handle ( gtor );
1149    gboolean     b = gtk_toggle_button_get_active( tb );
1150
1151    tr_torrentSetSpeedMode( tor, up_or_down, b ? TR_SPEEDLIMIT_SINGLE
1152                            : TR_SPEEDLIMIT_GLOBAL );
1153}
1154
1155static void
1156ul_speed_toggled_cb( GtkToggleButton *tb,
1157                     gpointer         gtor )
1158{
1159    speed_toggled_cb( tb, gtor, TR_UP );
1160}
1161
1162static void
1163dl_speed_toggled_cb( GtkToggleButton *tb,
1164                     gpointer         gtor )
1165{
1166    speed_toggled_cb( tb, gtor, TR_DOWN );
1167}
1168
1169static void
1170sensitize_from_check_cb( GtkToggleButton *toggle,
1171                         gpointer         w )
1172{
1173    gtk_widget_set_sensitive ( GTK_WIDGET( w ),
1174                              gtk_toggle_button_get_active( toggle ) );
1175}
1176
1177static void
1178setSpeedLimit( GtkSpinButton* spin,
1179               gpointer       gtor,
1180               int            up_or_down )
1181{
1182    tr_torrent * tor = tr_torrent_handle ( gtor );
1183    int          kb_sec = gtk_spin_button_get_value_as_int ( spin );
1184
1185    tr_torrentSetSpeedLimit( tor, up_or_down, kb_sec );
1186}
1187
1188static void
1189ul_speed_spun_cb( GtkSpinButton *spin,
1190                  gpointer       gtor )
1191{
1192    setSpeedLimit( spin, gtor, TR_UP );
1193}
1194
1195static void
1196dl_speed_spun_cb( GtkSpinButton *spin,
1197                  gpointer       gtor )
1198{
1199    setSpeedLimit( spin, gtor, TR_DOWN );
1200}
1201
1202static void
1203max_peers_spun_cb( GtkSpinButton * spin,
1204                   gpointer        gtor )
1205{
1206    const uint16_t n = gtk_spin_button_get_value( spin );
1207
1208    tr_torrentSetPeerLimit( tr_torrent_handle( gtor ), n );
1209}
1210
1211static GtkWidget*
1212options_page_new( TrTorrent * gtor )
1213{
1214    uint16_t     maxConnectedPeers;
1215    int          i, row;
1216    gboolean     b;
1217    GtkWidget *  t, *w, *tb;
1218    tr_torrent * tor = tr_torrent_handle ( gtor );
1219
1220    row = 0;
1221    t = hig_workarea_create ( );
1222    hig_workarea_add_section_title ( t, &row, _( "Limits" ) );
1223
1224    tb =
1225        gtk_check_button_new_with_mnemonic ( _(
1226                                                "Limit _download speed (KB/s):" ) );
1227    b = tr_torrentGetSpeedMode( tor, TR_DOWN ) == TR_SPEEDLIMIT_SINGLE;
1228    gtk_toggle_button_set_active ( GTK_TOGGLE_BUTTON( tb ), b );
1229    g_signal_connect ( tb, "toggled", G_CALLBACK(
1230                           dl_speed_toggled_cb ), gtor );
1231
1232    i = tr_torrentGetSpeedLimit( tor, TR_DOWN );
1233    w = gtk_spin_button_new_with_range( 1, INT_MAX, 5 );
1234    gtk_spin_button_set_value( GTK_SPIN_BUTTON( w ), i );
1235
1236    g_signal_connect ( w, "value-changed", G_CALLBACK(
1237                           dl_speed_spun_cb ), gtor );
1238    g_signal_connect ( tb, "toggled", G_CALLBACK(
1239                           sensitize_from_check_cb ), w );
1240    sensitize_from_check_cb ( GTK_TOGGLE_BUTTON( tb ), w );
1241    hig_workarea_add_row_w ( t, &row, tb, w, NULL );
1242
1243    tb =
1244        gtk_check_button_new_with_mnemonic ( _(
1245                                                "Limit _upload speed (KB/s):" ) );
1246    b = tr_torrentGetSpeedMode( tor, TR_UP ) == TR_SPEEDLIMIT_SINGLE;
1247    gtk_toggle_button_set_active ( GTK_TOGGLE_BUTTON( tb ), b );
1248    g_signal_connect ( tb, "toggled", G_CALLBACK(
1249                           ul_speed_toggled_cb ), gtor );
1250
1251    i = tr_torrentGetSpeedLimit( tor, TR_UP );
1252    w = gtk_spin_button_new_with_range( 1, INT_MAX, 5 );
1253    gtk_spin_button_set_value( GTK_SPIN_BUTTON( w ), i );
1254
1255    g_signal_connect ( w, "value-changed", G_CALLBACK(
1256                           ul_speed_spun_cb ), gtor );
1257    g_signal_connect ( tb, "toggled", G_CALLBACK(
1258                           sensitize_from_check_cb ), w );
1259    sensitize_from_check_cb ( GTK_TOGGLE_BUTTON( tb ), w );
1260    hig_workarea_add_row_w ( t, &row, tb, w, NULL );
1261
1262    hig_workarea_add_section_divider ( t, &row );
1263    hig_workarea_add_section_title ( t, &row, _( "Peer Connections" ) );
1264
1265    maxConnectedPeers = tr_torrentGetPeerLimit( tor );
1266    w = gtk_spin_button_new_with_range( 1, 3000, 5 );
1267    gtk_spin_button_set_value( GTK_SPIN_BUTTON( w ), maxConnectedPeers );
1268    hig_workarea_add_row( t, &row, _( "_Maximum peers:" ), w, w );
1269    g_signal_connect( w, "value-changed", G_CALLBACK(
1270                          max_peers_spun_cb ), gtor );
1271
1272    hig_workarea_finish ( t, &row );
1273    return t;
1274}
1275
1276static void
1277refresh_options( GtkWidget * top UNUSED )
1278{}
1279
1280/****
1281*****  TRACKER
1282****/
1283
1284#define TRACKER_PAGE "tracker-page"
1285
1286struct tracker_page
1287{
1288    TrTorrent *         gtor;
1289
1290    GtkTreeView *       view;
1291    GtkListStore *      store;
1292    GtkTreeSelection *  sel;
1293
1294    GtkWidget *         add_button;
1295    GtkWidget *         remove_button;
1296    GtkWidget *         save_button;
1297    GtkWidget *         revert_button;
1298
1299    GtkWidget *         last_scrape_time_lb;
1300    GtkWidget *         last_scrape_response_lb;
1301    GtkWidget *         next_scrape_countdown_lb;
1302
1303    GtkWidget *         last_announce_time_lb;
1304    GtkWidget *         last_announce_response_lb;
1305    GtkWidget *         next_announce_countdown_lb;
1306    GtkWidget *         manual_announce_countdown_lb;
1307};
1308
1309static GtkWidget*
1310tracker_page_new( TrTorrent * gtor )
1311{
1312    GtkWidget *           t;
1313    GtkWidget *           l;
1314    GtkWidget *           w;
1315    int                   row = 0;
1316    const char *          s;
1317    struct tracker_page * page = g_new0( struct tracker_page, 1 );
1318    const tr_info *       info = tr_torrent_info ( gtor );
1319
1320    page->gtor = gtor;
1321
1322    t = hig_workarea_create( );
1323    hig_workarea_add_section_title( t, &row, _( "Trackers" ) );
1324
1325    w = tracker_list_new( gtor );
1326    hig_workarea_add_wide_control( t, &row, w );
1327
1328    hig_workarea_add_section_divider( t, &row );
1329    hig_workarea_add_section_title( t, &row, _( "Scrape" ) );
1330
1331    s = _( "Last scrape at:" );
1332    l = gtk_label_new( NULL );
1333    page->last_scrape_time_lb = l;
1334    hig_workarea_add_row( t, &row, s, l, NULL );
1335
1336    s = _( "Tracker responded:" );
1337    l = gtk_label_new( NULL );
1338    page->last_scrape_response_lb = l;
1339    hig_workarea_add_row( t, &row, s, l, NULL );
1340
1341    s = _( "Next scrape in:" );
1342    l = gtk_label_new( NULL );
1343    page->next_scrape_countdown_lb = l;
1344    hig_workarea_add_row( t, &row, s, l, NULL );
1345
1346    hig_workarea_add_section_divider( t, &row );
1347    hig_workarea_add_section_title( t, &row, _( "Announce" ) );
1348
1349    l = gtk_label_new( info->trackers[0].announce );
1350    gtk_label_set_ellipsize( GTK_LABEL( l ), PANGO_ELLIPSIZE_END );
1351    hig_workarea_add_row ( t, &row, _( "Tracker:" ), l, NULL );
1352
1353    s = _( "Last announce at:" );
1354    l = gtk_label_new( NULL );
1355    page->last_announce_time_lb = l;
1356    hig_workarea_add_row( t, &row, s, l, NULL );
1357
1358    s = _( "Tracker responded:" );
1359    l = gtk_label_new( NULL );
1360    page->last_announce_response_lb = l;
1361    hig_workarea_add_row( t, &row, s, l, NULL );
1362
1363    s = _( "Next announce in:" );
1364    l = gtk_label_new( NULL );
1365    page->next_announce_countdown_lb = l;
1366    hig_workarea_add_row( t, &row, s, l, NULL );
1367
1368    /* how long until the tracker will honor user
1369    * pressing the "ask for more peers" button */
1370    s = _( "Manual announce allowed in:" );
1371    l = gtk_label_new( NULL );
1372    page->manual_announce_countdown_lb = l;
1373    hig_workarea_add_row( t, &row, s, l, NULL );
1374
1375    hig_workarea_finish( t, &row );
1376    g_object_set_data_full( G_OBJECT( t ), TRACKER_PAGE, page, g_free );
1377    return t;
1378}
1379
1380static void
1381refresh_countdown_lb( GtkWidget *  w,
1382                      time_t       t,
1383                      const char * countdown_done )
1384{
1385    const time_t now = time( NULL );
1386    GtkLabel *   l = GTK_LABEL( w );
1387
1388    if( t == 1 )
1389        gtk_label_set_text( l, _( "In progress" ) );
1390    else if( t < now )
1391        gtk_label_set_text( l, countdown_done );
1392    else
1393    {
1394        char      buf[1024];
1395        const int seconds = t - now;
1396        tr_strltime( buf, seconds, sizeof( buf ) );
1397        gtk_label_set_text( l, buf );
1398    }
1399}
1400
1401static void
1402refresh_tracker( GtkWidget * w )
1403{
1404    GtkWidget *           l;
1405    time_t                t;
1406    struct tracker_page * page = g_object_get_data( G_OBJECT(
1407                                                        w ), TRACKER_PAGE );
1408    const tr_stat *       torStat = tr_torrent_stat( page->gtor );
1409
1410    l = page->last_scrape_time_lb;
1411    t = torStat->lastScrapeTime;
1412    refresh_time_lb( l, t );
1413
1414    l = page->last_scrape_response_lb;
1415    gtk_label_set_text( GTK_LABEL( l ), torStat->scrapeResponse );
1416
1417    l = page->next_scrape_countdown_lb;
1418    t = torStat->nextScrapeTime;
1419    refresh_countdown_lb( l, t, _( "Never" ) );
1420
1421    l = page->last_announce_time_lb;
1422    t = torStat->lastAnnounceTime;
1423    refresh_time_lb( l, t );
1424
1425    l = page->last_announce_response_lb;
1426    gtk_label_set_text( GTK_LABEL( l ), torStat->announceResponse );
1427
1428    l = page->next_announce_countdown_lb;
1429    t = torStat->nextAnnounceTime;
1430    refresh_countdown_lb( l, t, _( "Never" ) );
1431
1432    l = page->manual_announce_countdown_lb;
1433    t = torStat->manualAnnounceTime;
1434    refresh_countdown_lb( l, t, _( "Now" ) );
1435}
1436
1437/****
1438*****  DIALOG
1439****/
1440
1441static void
1442torrent_destroyed( gpointer               dialog,
1443                   GObject * dead_torrent UNUSED )
1444{
1445    gtk_widget_destroy ( GTK_WIDGET( dialog ) );
1446}
1447
1448static void
1449remove_tag( gpointer tag )
1450{
1451    g_source_remove ( GPOINTER_TO_UINT( tag ) ); /* stop the periodic refresh */
1452}
1453
1454static void
1455response_cb( GtkDialog *  dialog,
1456             int response UNUSED,
1457             gpointer     gtor )
1458{
1459    g_object_weak_unref ( G_OBJECT( gtor ), torrent_destroyed, dialog );
1460    gtk_widget_destroy ( GTK_WIDGET( dialog ) );
1461}
1462
1463static gboolean
1464periodic_refresh( gpointer data )
1465{
1466    refresh_tracker   ( g_object_get_data ( G_OBJECT( data ), "tracker-top" ) );
1467    refresh_peers     ( g_object_get_data ( G_OBJECT( data ), "peers-top" ) );
1468    refresh_activity  ( g_object_get_data ( G_OBJECT( data ),
1469                                            "activity-top" ) );
1470    refresh_options   ( g_object_get_data ( G_OBJECT( data ), "options-top" ) );
1471    return TRUE;
1472}
1473
1474GtkWidget*
1475torrent_inspector_new( GtkWindow * parent,
1476                       TrTorrent * gtor )
1477{
1478    guint           tag;
1479    GtkWidget *     d, *n, *w;
1480    tr_torrent *    tor = tr_torrent_handle ( gtor );
1481    char            sizeStr[64];
1482    char            title[512];
1483    const tr_info * info = tr_torrent_info ( gtor );
1484
1485    /* create the dialog */
1486    tr_strlsize( sizeStr, info->totalSize, sizeof( sizeStr ) );
1487    /* %1$s is torrent name
1488       %2$s its file size */
1489    g_snprintf( title, sizeof( title ), _(
1490                    "Details for %1$s (%2$s)" ), info->name, sizeStr );
1491    d = gtk_dialog_new_with_buttons ( title, parent, 0,
1492                                      GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
1493                                      NULL );
1494    gtk_window_set_role ( GTK_WINDOW( d ), "tr-info" );
1495    g_signal_connect ( d, "response", G_CALLBACK ( response_cb ), gtor );
1496    gtk_dialog_set_has_separator( GTK_DIALOG( d ), FALSE );
1497    gtk_container_set_border_width( GTK_CONTAINER( d ), GUI_PAD );
1498    g_object_weak_ref ( G_OBJECT( gtor ), torrent_destroyed, d );
1499
1500
1501    /* add the notebook */
1502    n = gtk_notebook_new ( );
1503    gtk_container_set_border_width ( GTK_CONTAINER ( n ), GUI_PAD );
1504
1505    w = activity_page_new ( gtor );
1506    g_object_set_data ( G_OBJECT( d ), "activity-top", w );
1507    gtk_notebook_append_page ( GTK_NOTEBOOK( n ), w,
1508                              gtk_label_new ( _( "Activity" ) ) );
1509
1510    w = peer_page_new ( gtor );
1511    g_object_set_data ( G_OBJECT( d ), "peers-top", w );
1512    gtk_notebook_append_page ( GTK_NOTEBOOK( n ),  w,
1513                              gtk_label_new ( _( "Peers" ) ) );
1514
1515    w = tracker_page_new( gtor );
1516    g_object_set_data( G_OBJECT( d ), "tracker-top", w );
1517    gtk_notebook_append_page( GTK_NOTEBOOK( n ), w,
1518                             gtk_label_new( _( "Tracker" ) ) );
1519
1520    gtk_notebook_append_page ( GTK_NOTEBOOK( n ),
1521                              info_page_new ( tor ),
1522                              gtk_label_new ( _( "Information" ) ) );
1523
1524    w = file_list_new( gtor );
1525    gtk_container_set_border_width( GTK_CONTAINER( w ), GUI_PAD_BIG );
1526    g_object_set_data ( G_OBJECT( d ), "files-top", w );
1527    gtk_notebook_append_page ( GTK_NOTEBOOK( n ), w,
1528                              gtk_label_new ( _( "Files" ) ) );
1529
1530    w = options_page_new ( gtor );
1531    g_object_set_data ( G_OBJECT( d ), "options-top", w );
1532    gtk_notebook_append_page ( GTK_NOTEBOOK( n ), w,
1533                              gtk_label_new ( _( "Options" ) ) );
1534
1535    gtk_box_pack_start( GTK_BOX( GTK_DIALOG( d )->vbox ), n, TRUE, TRUE, 0 );
1536
1537    tag = g_timeout_add ( UPDATE_INTERVAL_MSEC, periodic_refresh, d );
1538    g_object_set_data_full ( G_OBJECT( d ), "tag",
1539                             GUINT_TO_POINTER( tag ), remove_tag );
1540
1541    /* return the results */
1542    periodic_refresh ( d );
1543    gtk_widget_show_all ( GTK_DIALOG( d )->vbox );
1544    return d;
1545}
1546
Note: See TracBrowser for help on using the repository browser.