source: trunk/gtk/details.c @ 8001

Last change on this file since 8001 was 8001, checked in by charles, 13 years ago

(trunk gtk) back out a little bit of bad code.

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