source: trunk/gtk/tr-prefs.c @ 5702

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

update NEWS

  • Property svn:keywords set to Date Rev Author Id
File size: 19.7 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: tr-prefs.c 5702 2008-04-25 20:05:24Z charles $
11 */
12
13#include <stdlib.h> /* free() */
14#include <unistd.h>
15#include <glib/gi18n.h>
16#include <gtk/gtk.h>
17#include <libtransmission/transmission.h>
18#include <libtransmission/utils.h>
19#include <libtransmission/web.h>
20#include "conf.h"
21#include "hig.h"
22#include "tr-core.h"
23#include "tr-prefs.h"
24#include "util.h"
25
26/**
27 * This is where we initialize the preferences file with the default values.
28 * If you add a new preferences key, you /must/ add a default value here.
29 */
30void
31tr_prefs_init_global( void )
32{
33    const char * str;
34
35    cf_check_older_configs( );
36
37#if HAVE_GIO
38    str = NULL;
39    if( !str ) str = g_get_user_special_dir( G_USER_DIRECTORY_DESKTOP );
40    if( !str ) str = g_get_home_dir( );
41    pref_string_set_default ( PREF_KEY_DIR_WATCH, str );
42    pref_flag_set_default   ( PREF_KEY_DIR_WATCH_ENABLED, FALSE );
43#endif
44
45    pref_int_set_default    ( PREF_KEY_PEER_SOCKET_TOS, TR_DEFAULT_PEER_SOCKET_TOS );
46    pref_flag_set_default   ( PREF_KEY_INHIBIT_HIBERNATION, TRUE );
47    pref_flag_set_default   ( PREF_KEY_BLOCKLIST_ENABLED, FALSE );
48
49    pref_string_set_default ( PREF_KEY_OPEN_DIALOG_FOLDER, g_get_home_dir( ) );
50
51    pref_int_set_default    ( PREF_KEY_MAX_PEERS_GLOBAL, TR_DEFAULT_GLOBAL_PEER_LIMIT );
52    pref_int_set_default    ( PREF_KEY_MAX_PEERS_PER_TORRENT, 50 );
53
54    pref_flag_set_default   ( PREF_KEY_TOOLBAR, TRUE );
55    pref_flag_set_default   ( PREF_KEY_FILTERBAR, TRUE );
56    pref_flag_set_default   ( PREF_KEY_STATUSBAR, TRUE );
57    pref_string_set_default ( PREF_KEY_STATUSBAR_STATS, "total-ratio" );
58
59    pref_flag_set_default   ( PREF_KEY_DL_LIMIT_ENABLED, FALSE );
60    pref_int_set_default    ( PREF_KEY_DL_LIMIT, 100 );
61    pref_flag_set_default   ( PREF_KEY_UL_LIMIT_ENABLED, FALSE );
62    pref_int_set_default    ( PREF_KEY_UL_LIMIT, 50 );
63    pref_flag_set_default   ( PREF_KEY_OPTIONS_PROMPT, TRUE );
64
65    pref_int_set_default    ( PREF_KEY_MAIN_WINDOW_HEIGHT, 500 );
66    pref_int_set_default    ( PREF_KEY_MAIN_WINDOW_WIDTH, 300 );
67    pref_int_set_default    ( PREF_KEY_MAIN_WINDOW_X, 50 );
68    pref_int_set_default    ( PREF_KEY_MAIN_WINDOW_Y, 50 );
69
70    str = NULL;
71#if GLIB_CHECK_VERSION(2,14,0)
72    if( !str ) str = g_get_user_special_dir( G_USER_DIRECTORY_DOWNLOAD );
73#endif
74    if( !str ) str = g_get_home_dir( );
75    pref_string_set_default ( PREF_KEY_DIR_DEFAULT, str );
76
77    pref_int_set_default    ( PREF_KEY_PORT, TR_DEFAULT_PORT );
78
79    pref_flag_set_default   ( PREF_KEY_NAT, TRUE );
80    pref_flag_set_default   ( PREF_KEY_PEX, TR_DEFAULT_PEX_ENABLED );
81    pref_flag_set_default   ( PREF_KEY_ASKQUIT, TRUE );
82    pref_flag_set_default   ( PREF_KEY_ENCRYPTED_ONLY, FALSE );
83
84    pref_int_set_default    ( PREF_KEY_MSGLEVEL, TR_MSG_INF );
85
86    pref_string_set_default ( PREF_KEY_SORT_MODE, "sort-by-name" );
87    pref_flag_set_default   ( PREF_KEY_SORT_REVERSED, FALSE );
88    pref_flag_set_default   ( PREF_KEY_MINIMAL_VIEW, FALSE );
89
90    pref_flag_set_default   ( PREF_KEY_START, TRUE );
91    pref_flag_set_default   ( PREF_KEY_TRASH_ORIGINAL, FALSE );
92
93    pref_save( NULL );
94}
95
96/**
97***
98**/
99
100#define PREF_KEY "pref-key"
101
102static void
103response_cb( GtkDialog * dialog, int response UNUSED, gpointer unused UNUSED )
104{
105    gtk_widget_destroy( GTK_WIDGET(dialog) );
106}
107
108static void
109toggled_cb( GtkToggleButton * w, gpointer core )
110{
111    const char * key = g_object_get_data( G_OBJECT(w), PREF_KEY );
112    const gboolean flag = gtk_toggle_button_get_active( w );
113    tr_core_set_pref_bool( TR_CORE(core), key, flag );
114}
115
116static GtkWidget*
117new_check_button( const char * mnemonic, const char * key, gpointer core )
118{
119    GtkWidget * w = gtk_check_button_new_with_mnemonic( mnemonic );
120    g_object_set_data_full( G_OBJECT(w), PREF_KEY, g_strdup(key), g_free );
121    gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(w), pref_flag_get(key) );
122    g_signal_connect( w, "toggled", G_CALLBACK(toggled_cb), core );
123    return w;
124}
125
126static void
127spun_cb( GtkSpinButton * w, gpointer core )
128{
129    const char * key = g_object_get_data( G_OBJECT(w), PREF_KEY );
130    const int value = gtk_spin_button_get_value_as_int( w );
131    tr_core_set_pref_int( TR_CORE(core), key, value );
132}
133
134static GtkWidget*
135new_spin_button( const char * key, gpointer core, int low, int high, int step )
136{
137    GtkWidget * w = gtk_spin_button_new_with_range( low, high, step );
138    g_object_set_data_full( G_OBJECT(w), PREF_KEY, g_strdup(key), g_free );
139    gtk_spin_button_set_digits( GTK_SPIN_BUTTON(w), 0 );
140    gtk_spin_button_set_value( GTK_SPIN_BUTTON(w), pref_int_get(key) );
141    g_signal_connect( w, "value-changed", G_CALLBACK(spun_cb), core );
142    return w;
143}
144
145static void
146chosen_cb( GtkFileChooser * w, gpointer core )
147{
148    const char * key = g_object_get_data( G_OBJECT(w), PREF_KEY );
149    char * value = gtk_file_chooser_get_filename( GTK_FILE_CHOOSER(w) );
150    tr_core_set_pref( TR_CORE(core), key, value );
151    g_free( value );
152}
153
154static GtkWidget*
155new_path_chooser_button( const char * key, gpointer core )
156{
157    GtkWidget * w = gtk_file_chooser_button_new( NULL,
158                                    GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER );
159    char * path = pref_string_get( key );
160    g_object_set_data_full( G_OBJECT(w), PREF_KEY, g_strdup(key), g_free );
161    g_signal_connect( w, "selection-changed", G_CALLBACK(chosen_cb), core );
162    gtk_file_chooser_set_current_folder( GTK_FILE_CHOOSER(w), path );
163    g_free( path );
164    return w;
165}
166
167static void
168target_cb( GtkWidget * tb, gpointer target )
169{
170    const gboolean b = gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON( tb ) );
171    gtk_widget_set_sensitive( GTK_WIDGET(target), b );
172}
173
174struct test_port_data
175{
176    GtkWidget * label;
177    gboolean * alive;
178};
179
180static void
181testing_port_done( tr_handle   * handle         UNUSED,
182                   long          response_code  UNUSED,
183                   const void  * response,
184                   size_t        response_len,
185                   void        * gdata )
186{
187    struct test_port_data * data = gdata;
188    if( *data->alive )
189    {
190        const int isOpen = response_len && *(char*)response=='1';
191        gtk_label_set_markup( GTK_LABEL( data->label ), isOpen
192                              ? _("Port is <b>open</b>")
193                              : _("Port is <b>closed</b>") );
194    }
195}
196
197static gboolean
198testing_port_begin( gpointer gdata )
199{
200    struct test_port_data * data = gdata;
201    if( *data->alive )
202    {
203        GtkSpinButton * spin = g_object_get_data( G_OBJECT( data->label ), "tr-port-spin" );
204        tr_handle * handle = g_object_get_data( G_OBJECT( data->label ), "handle" );
205        const int port = gtk_spin_button_get_value_as_int( spin );
206        char url[256];
207        snprintf( url, sizeof(url), "http://portcheck.transmissionbt.com/%d", port );
208        tr_webRun( handle, url, testing_port_done, data );
209    }
210    return FALSE;
211}
212
213static void
214testing_port_cb( GtkWidget * unused UNUSED, gpointer l )
215{
216    gtk_label_set_markup( GTK_LABEL( l ), _( "<i>Testing port...</i>" ) );
217    /* wait three seconds to give the port forwarding time to kick in */
218    struct test_port_data * data = g_new0( struct test_port_data, 1 );
219    data->label = l;
220    data->alive = g_object_get_data( G_OBJECT( l ), "alive" );
221    g_timeout_add( 3000, testing_port_begin, data );
222}
223
224static void
225dialogDestroyed( gpointer alive, GObject * dialog UNUSED )
226{
227    *(gboolean*)alive = FALSE;
228}
229
230static GtkWidget*
231torrentPage( GObject * core )
232{
233    int row = 0;
234    const char * s;
235    GtkWidget * t;
236    GtkWidget * w;
237#ifdef HAVE_GIO
238    GtkWidget * l;
239#endif
240
241    t = hig_workarea_create( );
242    hig_workarea_add_section_title( t, &row, _( "Adding Torrents" ) );
243
244#ifdef HAVE_GIO
245        s = _( "Automatically add torrents from:" );
246        l = new_check_button( s, PREF_KEY_DIR_WATCH_ENABLED, core );
247        w = new_path_chooser_button( PREF_KEY_DIR_WATCH, core );
248        gtk_widget_set_sensitive( GTK_WIDGET(w), pref_flag_get( PREF_KEY_DIR_WATCH_ENABLED ) );
249        g_signal_connect( l, "toggled", G_CALLBACK(target_cb), w );
250        hig_workarea_add_row_w( t, &row, l, w, NULL );
251#endif
252
253        s = _( "Display _options dialog" );
254        w = new_check_button( s, PREF_KEY_OPTIONS_PROMPT, core );
255        hig_workarea_add_wide_control( t, &row, w );
256
257        s = _( "_Start when added" );
258        w = new_check_button( s, PREF_KEY_START, core );
259        hig_workarea_add_wide_control( t, &row, w );
260
261        s = _( "Mo_ve source files to Trash" );
262        w = new_check_button( s, PREF_KEY_TRASH_ORIGINAL, core ); 
263        hig_workarea_add_wide_control( t, &row, w );
264
265        w = new_path_chooser_button( PREF_KEY_DIR_DEFAULT, core );
266        hig_workarea_add_row( t, &row, _( "_Destination folder:" ), w, NULL );
267
268    hig_workarea_finish( t, &row );
269    return t;
270}
271
272/***
273****
274***/
275
276struct blocklist_data
277{
278    GtkWidget * check;
279    GtkWidget * dialog;
280    TrCore * core;
281    int abortFlag;
282    char secondary[256];
283};
284
285static void
286updateBlocklistText( GtkWidget * w, TrCore * core )
287{
288    const int n = tr_blocklistGetRuleCount( tr_core_handle( core ) );
289    char buf[512];
290    g_snprintf( buf, sizeof( buf ),
291                ngettext( "Ignore the %'d _blocklisted peer",
292                          "Ignore the %'d _blocklisted peers", n ), n );
293    gtk_button_set_label( GTK_BUTTON( w ), buf );
294}
295static gboolean
296updateBlocklistTextFromData( gpointer gdata )
297{
298    struct blocklist_data * data = gdata;
299    updateBlocklistText( data->check, data->core );
300    return FALSE;
301}
302
303static gboolean
304blocklistDialogSetSecondary( gpointer gdata )
305{
306    struct blocklist_data * data = gdata;
307    GtkMessageDialog * md = GTK_MESSAGE_DIALOG( data->dialog );
308    gtk_message_dialog_format_secondary_text( md, data->secondary );
309    return FALSE;
310}
311
312static gboolean
313blocklistDialogAllowClose( gpointer dialog )
314{
315    GtkDialog * d = GTK_DIALOG( dialog );
316    gtk_dialog_set_response_sensitive( GTK_DIALOG( d ), GTK_RESPONSE_CANCEL, FALSE );
317    gtk_dialog_set_response_sensitive( GTK_DIALOG( d ), GTK_RESPONSE_CLOSE, TRUE );
318    return FALSE;
319}
320
321static void
322got_blocklist( tr_handle   * handle         UNUSED,
323               long          response_code  UNUSED,
324               const void  * response,
325               size_t        response_len,
326               void        * gdata )
327{
328    struct blocklist_data * data = gdata;
329    const char * text = response;
330    int size = response_len;
331    int rules = 0;
332    gchar * filename = NULL;
333    gchar * filename2 = NULL;
334    int fd = -1;
335    int ok = 1;
336
337    if( !data->abortFlag && ( !text || !size ) )
338    {
339        ok = FALSE;
340        g_snprintf( data->secondary, sizeof( data->secondary ),
341                    _( "Unable to get blocklist." ) );
342        g_message( data->secondary );
343        g_idle_add( blocklistDialogSetSecondary, data );
344    }     
345
346    if( ok && !data->abortFlag )
347    {
348        GError * err = NULL;
349        fd = g_file_open_tmp( "transmission-blockfile-XXXXXX", &filename, &err );
350        if( err ) {
351            g_snprintf( data->secondary, sizeof( data->secondary ),
352                        _( "Unable to get blocklist: %s" ), err->message );
353            g_warning( data->secondary );
354            g_idle_add( blocklistDialogSetSecondary, data );
355            g_clear_error( &err );
356            ok = FALSE;
357        } else {
358            write( fd, text, size );
359            close( fd );
360        }
361    }
362    if( ok && !data->abortFlag )
363    {
364        filename2 = g_strdup_printf( "%s.txt", filename );
365        g_snprintf( data->secondary, sizeof( data->secondary ),
366                    _( "Uncompressing blocklist..." ) );
367        g_idle_add( blocklistDialogSetSecondary, data );
368        char * cmd = g_strdup_printf( "zcat %s > %s ", filename, filename2 );
369        tr_dbg( "%s", cmd );
370        system( cmd );
371        g_free( cmd );
372    }
373    if( ok && !data->abortFlag )
374    {
375        g_snprintf( data->secondary, sizeof( data->secondary ),
376                    _( "Parsing blocklist..." ) );
377        g_idle_add( blocklistDialogSetSecondary, data );
378        rules = tr_blocklistSetContent( tr_core_handle( data->core ), filename2 );
379    }
380    if( ok && !data->abortFlag )
381    {
382        g_snprintf( data->secondary, sizeof( data->secondary ),
383                    _( "Blocklist updated with %'d entries" ), rules );
384        g_idle_add( blocklistDialogSetSecondary, data );
385        g_idle_add( blocklistDialogAllowClose, data->dialog );
386        g_idle_add( updateBlocklistTextFromData, data );
387    }
388
389    /* g_free( data ); */
390    if( filename2 ) {
391        unlink( filename2 );
392        g_free( filename2 );
393    }
394    if( filename ) {
395        unlink( filename );
396        g_free( filename );
397    }
398}
399
400static void
401onUpdateBlocklistResponseCB( GtkDialog * dialog, int response, gpointer vdata )
402{
403    struct blocklist_data * data = vdata;
404
405    if( response == GTK_RESPONSE_CANCEL )
406        data->abortFlag = 1;
407
408    data->dialog = NULL;
409    gtk_widget_destroy( GTK_WIDGET( dialog ) );
410}
411
412static void
413onUpdateBlocklistCB( GtkButton * w, gpointer gdata )
414{
415    GtkWidget * d;
416    struct blocklist_data * data = gdata;
417    tr_handle * handle = g_object_get_data( G_OBJECT( w ), "handle" );
418    const char * url = "http://download.m0k.org/transmission/files/level1.gz";
419   
420    d = gtk_message_dialog_new( GTK_WINDOW( gtk_widget_get_toplevel( GTK_WIDGET( w ) ) ),
421                                GTK_DIALOG_DESTROY_WITH_PARENT,
422                                GTK_MESSAGE_INFO,
423                                GTK_BUTTONS_NONE,
424                                _( "Updating Blocklist" ) );
425    gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG( d ),
426                                              _( "Retrieving blocklist..." ) );
427    gtk_dialog_add_buttons( GTK_DIALOG( d ),
428                            GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
429                            GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
430                            NULL );
431    gtk_dialog_set_response_sensitive( GTK_DIALOG( d ), GTK_RESPONSE_CLOSE, FALSE );
432
433    data->dialog = d;
434
435    g_signal_connect( d, "response", G_CALLBACK(onUpdateBlocklistResponseCB), data );
436    gtk_widget_show( d );
437
438    tr_webRun( handle, url, got_blocklist, data );
439}
440
441static GtkWidget*
442peerPage( GObject * core )
443{
444    int row = 0;
445    const char * s;
446    GtkWidget * t;
447    GtkWidget * w;
448    GtkWidget * b;
449    GtkWidget * h;
450    struct blocklist_data * data;
451
452    t = hig_workarea_create( );
453    hig_workarea_add_section_title (t, &row, _("Options"));
454
455        w = new_check_button( "", PREF_KEY_BLOCKLIST_ENABLED, core );
456        updateBlocklistText( w, TR_CORE( core ) );
457        h = gtk_hbox_new( FALSE, GUI_PAD_BIG );
458        gtk_box_pack_start_defaults( GTK_BOX(h), w );
459        b = gtk_button_new_with_mnemonic( _( "_Update Blocklist" ) );
460
461        data = g_new0( struct blocklist_data, 1 );
462        data->core = TR_CORE( core );
463        data->check = w;
464
465        g_object_set_data( G_OBJECT( b ), "handle", tr_core_handle( TR_CORE( core ) ) );
466        g_signal_connect( b, "clicked", G_CALLBACK(onUpdateBlocklistCB), data );
467        gtk_box_pack_start( GTK_BOX(h), b, FALSE, FALSE, 0 );
468        g_signal_connect( w, "toggled", G_CALLBACK(target_cb), b );
469        target_cb( w, b );
470        hig_workarea_add_wide_control( t, &row, h );
471       
472        s = _("_Ignore unencrypted peers");
473        w = new_check_button( s, PREF_KEY_ENCRYPTED_ONLY, core );
474        hig_workarea_add_wide_control( t, &row, w );
475
476        s = _("Use peer e_xchange");
477        w = new_check_button( s, PREF_KEY_PEX, core );
478        hig_workarea_add_wide_control( t, &row, w );
479       
480    hig_workarea_add_section_divider( t, &row );
481    /* section header for the "maximum number of peers" section */
482    hig_workarea_add_section_title( t, &row, _( "Limits" ) );
483 
484        w = new_spin_button( PREF_KEY_MAX_PEERS_GLOBAL, core, 1, 3000, 5 );
485        hig_workarea_add_row( t, &row, _( "Maximum peers _overall:" ), w, NULL );
486        w = new_spin_button( PREF_KEY_MAX_PEERS_PER_TORRENT, core, 1, 300, 5 );
487        hig_workarea_add_row( t, &row, _( "Maximum peers per _torrent:" ), w, NULL );
488
489    hig_workarea_finish( t, &row );
490    return t;
491}
492
493static GtkWidget*
494networkPage( GObject * core, gpointer alive )
495{
496    int row = 0;
497    const char * s;
498    GtkWidget * t;
499    GtkWidget * w, * w2;
500    GtkWidget * l;
501    GtkWidget * h;
502
503    t = hig_workarea_create( );
504
505    hig_workarea_add_section_title (t, &row, _("Bandwidth"));
506
507        s = _("Limit _download speed (KB/s):");
508        w = new_check_button( s, PREF_KEY_DL_LIMIT_ENABLED, core );
509        w2 = new_spin_button( PREF_KEY_DL_LIMIT, core, 0, INT_MAX, 5 );
510        gtk_widget_set_sensitive( GTK_WIDGET(w2), pref_flag_get( PREF_KEY_DL_LIMIT_ENABLED ) );
511        g_signal_connect( w, "toggled", G_CALLBACK(target_cb), w2 );
512        hig_workarea_add_row_w( t, &row, w, w2, NULL );
513
514        s = _("Limit _upload speed (KB/s):");
515        w = new_check_button( s, PREF_KEY_UL_LIMIT_ENABLED, core );
516        w2 = new_spin_button( PREF_KEY_UL_LIMIT, core, 0, INT_MAX, 5 );
517        gtk_widget_set_sensitive( GTK_WIDGET(w2), pref_flag_get( PREF_KEY_UL_LIMIT_ENABLED ) );
518        g_signal_connect( w, "toggled", G_CALLBACK(target_cb), w2 );
519        hig_workarea_add_row_w( t, &row, w, w2, NULL );
520
521    hig_workarea_add_section_title (t, &row, _("Ports"));
522       
523        s = _("_Forward port from router" );
524        w = new_check_button( s, PREF_KEY_NAT, core );
525        hig_workarea_add_wide_control( t, &row, w );
526
527        h = gtk_hbox_new( FALSE, GUI_PAD_BIG );
528        w2 = new_spin_button( PREF_KEY_PORT, core, 1, INT_MAX, 1 );
529        gtk_box_pack_start( GTK_BOX(h), w2, FALSE, FALSE, 0 );
530        l = gtk_label_new( NULL );
531        gtk_misc_set_alignment( GTK_MISC(l), 0.0f, 0.5f );
532        gtk_box_pack_start( GTK_BOX(h), l, FALSE, FALSE, 0 );
533        hig_workarea_add_row( t, &row, _("Incoming _port:"), h, w );
534
535        g_object_set_data( G_OBJECT(l), "tr-port-spin", w2 );
536        g_object_set_data( G_OBJECT(l), "alive", alive );
537        g_object_set_data( G_OBJECT(l), "handle", tr_core_handle( TR_CORE( core ) ) );
538        testing_port_cb( NULL, l );
539
540        g_signal_connect( w, "toggled", G_CALLBACK(testing_port_cb), l );
541        g_signal_connect( w2, "value-changed", G_CALLBACK(testing_port_cb), l );
542
543    hig_workarea_finish( t, &row );
544    return t;
545}
546
547GtkWidget *
548tr_prefs_dialog_new( GObject * core, GtkWindow * parent )
549{
550    GtkWidget * d;
551    GtkWidget * n;
552    gboolean * alive;
553
554    alive = g_new( gboolean, 1 );
555    *alive = TRUE;
556
557    d = gtk_dialog_new_with_buttons( _( "Transmission Preferences" ), parent,
558                                     GTK_DIALOG_DESTROY_WITH_PARENT,
559                                     GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
560                                     NULL );
561    gtk_window_set_role( GTK_WINDOW(d), "transmission-preferences-dialog" );
562    gtk_dialog_set_has_separator( GTK_DIALOG( d ), FALSE );
563    gtk_container_set_border_width( GTK_CONTAINER( d ), GUI_PAD );
564    g_object_weak_ref( G_OBJECT( d ), dialogDestroyed, alive );
565
566    n = gtk_notebook_new( );
567    gtk_container_set_border_width ( GTK_CONTAINER ( n ), GUI_PAD );
568
569    gtk_notebook_append_page( GTK_NOTEBOOK( n ),
570                              torrentPage( core ),
571                              gtk_label_new (_("Torrents")) );
572    gtk_notebook_append_page( GTK_NOTEBOOK( n ),
573                              peerPage( core ),
574                              gtk_label_new (_("Peers")) );
575    gtk_notebook_append_page( GTK_NOTEBOOK( n ),
576                              networkPage( core, alive ),
577                              gtk_label_new (_("Network")) );
578
579    g_signal_connect( d, "response", G_CALLBACK(response_cb), core );
580    gtk_box_pack_start_defaults( GTK_BOX(GTK_DIALOG(d)->vbox), n );
581    gtk_widget_show_all( GTK_DIALOG(d)->vbox );
582    return d;
583}
Note: See TracBrowser for help on using the repository browser.