source: branches/1.3x/gtk/tr-prefs.c @ 6492

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

(gtk 1.3x) #1182: "Untranslatable" strings

  • Property svn:keywords set to Date Rev Author Id
File size: 39.5 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 6492 2008-08-12 00:52:34Z charles $
11 */
12
13#include <ctype.h> /* isspace */
14#include <errno.h>
15#include <stdarg.h>
16#include <stdlib.h> /* free() */
17#include <unistd.h>
18#include <glib/gi18n.h>
19#include <gtk/gtk.h>
20#include <libtransmission/transmission.h>
21#include <libtransmission/utils.h>
22#include <libtransmission/version.h>
23#include <libtransmission/web.h>
24#include "conf.h"
25#include "hig.h"
26#include "tr-core.h"
27#include "tr-prefs.h"
28#include "util.h"
29
30/**
31 * This is where we initialize the preferences file with the default values.
32 * If you add a new preferences key, you /must/ add a default value here.
33 */
34void
35tr_prefs_init_global( void )
36{
37    int i;
38    char pw[32];
39    const char * str;
40    const char * pool = "abcdefghijklmnopqrstuvwxyz"
41                        "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
42                        "1234567890";
43
44    cf_check_older_configs( );
45
46#if HAVE_GIO
47    str = NULL;
48    if( !str ) str = g_get_user_special_dir( G_USER_DIRECTORY_DESKTOP );
49    if( !str ) str = g_get_home_dir( );
50    pref_string_set_default ( PREF_KEY_DIR_WATCH, str );
51    pref_flag_set_default   ( PREF_KEY_DIR_WATCH_ENABLED, FALSE );
52#endif
53
54    pref_int_set_default    ( PREF_KEY_PEER_SOCKET_TOS, TR_DEFAULT_PEER_SOCKET_TOS );
55    pref_flag_set_default   ( PREF_KEY_ALLOW_HIBERNATION, FALSE );
56    pref_flag_set_default   ( PREF_KEY_BLOCKLIST_ENABLED, TR_DEFAULT_BLOCKLIST_ENABLED );
57
58    pref_string_set_default ( PREF_KEY_OPEN_DIALOG_FOLDER, g_get_home_dir( ) );
59
60    pref_int_set_default    ( PREF_KEY_MAX_PEERS_GLOBAL, TR_DEFAULT_GLOBAL_PEER_LIMIT );
61    pref_int_set_default    ( PREF_KEY_MAX_PEERS_PER_TORRENT, 50 );
62
63    pref_flag_set_default   ( PREF_KEY_TOOLBAR, TRUE );
64    pref_flag_set_default   ( PREF_KEY_FILTERBAR, TRUE );
65    pref_flag_set_default   ( PREF_KEY_STATUSBAR, TRUE );
66    pref_flag_set_default   ( PREF_KEY_SHOW_TRAY_ICON, FALSE );
67    pref_string_set_default ( PREF_KEY_STATUSBAR_STATS, "total-ratio" );
68
69    pref_flag_set_default   ( PREF_KEY_DL_LIMIT_ENABLED, FALSE );
70    pref_int_set_default    ( PREF_KEY_DL_LIMIT, 100 );
71    pref_flag_set_default   ( PREF_KEY_UL_LIMIT_ENABLED, FALSE );
72    pref_int_set_default    ( PREF_KEY_UL_LIMIT, 50 );
73    pref_flag_set_default   ( PREF_KEY_OPTIONS_PROMPT, TRUE );
74
75    pref_int_set_default    ( PREF_KEY_MAIN_WINDOW_HEIGHT, 500 );
76    pref_int_set_default    ( PREF_KEY_MAIN_WINDOW_WIDTH, 300 );
77    pref_int_set_default    ( PREF_KEY_MAIN_WINDOW_X, 50 );
78    pref_int_set_default    ( PREF_KEY_MAIN_WINDOW_Y, 50 );
79
80    pref_string_set_default ( PREF_KEY_PROXY_SERVER, "" );
81    pref_int_set_default    ( PREF_KEY_PROXY_PORT, TR_DEFAULT_PROXY_PORT );
82    pref_int_set_default    ( PREF_KEY_PROXY_TYPE, TR_DEFAULT_PROXY_TYPE );
83    pref_flag_set_default   ( PREF_KEY_PROXY_SERVER_ENABLED, TR_DEFAULT_PROXY_ENABLED );
84    pref_flag_set_default   ( PREF_KEY_PROXY_AUTH_ENABLED, TR_DEFAULT_PROXY_AUTH_ENABLED );
85    pref_string_set_default ( PREF_KEY_PROXY_USERNAME, "" );
86    pref_string_set_default ( PREF_KEY_PROXY_PASSWORD, "" );
87
88    str = NULL;
89#if GLIB_CHECK_VERSION(2,14,0)
90    if( !str ) str = g_get_user_special_dir( G_USER_DIRECTORY_DOWNLOAD );
91#endif
92    if( !str ) str = g_get_home_dir( );
93    pref_string_set_default ( PREF_KEY_DOWNLOAD_DIR, str );
94
95    pref_int_set_default    ( PREF_KEY_PORT, TR_DEFAULT_PORT );
96
97    pref_flag_set_default   ( PREF_KEY_PORT_FORWARDING, TRUE );
98    pref_flag_set_default   ( PREF_KEY_PEX, TR_DEFAULT_PEX_ENABLED );
99    pref_flag_set_default   ( PREF_KEY_ASKQUIT, TRUE );
100    pref_flag_set_default   ( PREF_KEY_ENCRYPTION, TR_ENCRYPTION_PREFERRED );
101
102    pref_int_set_default    ( PREF_KEY_MSGLEVEL, TR_MSG_INF );
103
104    pref_string_set_default ( PREF_KEY_SORT_MODE, "sort-by-name" );
105    pref_flag_set_default   ( PREF_KEY_SORT_REVERSED, FALSE );
106    pref_flag_set_default   ( PREF_KEY_MINIMAL_VIEW, FALSE );
107
108    pref_flag_set_default   ( PREF_KEY_START, TRUE );
109    pref_flag_set_default   ( PREF_KEY_TRASH_ORIGINAL, FALSE );
110
111    pref_flag_set_default   ( PREF_KEY_RPC_ENABLED, TR_DEFAULT_RPC_ENABLED );
112    pref_int_set_default    ( PREF_KEY_RPC_PORT, TR_DEFAULT_RPC_PORT );
113    pref_string_set_default ( PREF_KEY_RPC_ACL, TR_DEFAULT_RPC_ACL );
114
115    for( i=0; i<16; ++i )
116        pw[i] = pool[ tr_rand( strlen( pool ) ) ];
117    pw[16] = '\0';
118    pref_string_set_default( PREF_KEY_RPC_USERNAME, "transmission" );
119    pref_string_set_default( PREF_KEY_RPC_PASSWORD, pw );
120    pref_flag_set_default  ( PREF_KEY_RPC_AUTH_ENABLED, FALSE );
121
122    pref_save( );
123}
124
125/**
126***
127**/
128
129#define PREF_KEY "pref-key"
130
131static void
132response_cb( GtkDialog * dialog, int response, gpointer unused UNUSED )
133{
134    if( response == GTK_RESPONSE_HELP ) {
135        char * base = gtr_get_help_url( );
136        char * url = g_strdup_printf( "%s/html/preferences.html", base );
137        gtr_open_file( url );
138        g_free( url );
139        g_free( base );
140    }
141
142    if( response == GTK_RESPONSE_CLOSE )
143        gtk_widget_destroy( GTK_WIDGET(dialog) );
144}
145
146static void
147toggled_cb( GtkToggleButton * w, gpointer core )
148{
149    const char * key = g_object_get_data( G_OBJECT(w), PREF_KEY );
150    const gboolean flag = gtk_toggle_button_get_active( w );
151    tr_core_set_pref_bool( TR_CORE(core), key, flag );
152}
153
154static GtkWidget*
155new_check_button( const char * mnemonic, const char * key, gpointer core )
156{
157    GtkWidget * w = gtk_check_button_new_with_mnemonic( mnemonic );
158    g_object_set_data_full( G_OBJECT(w), PREF_KEY, g_strdup(key), g_free );
159    gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(w), pref_flag_get(key) );
160    g_signal_connect( w, "toggled", G_CALLBACK(toggled_cb), core );
161    return w;
162}
163
164static void
165spun_cb( GtkSpinButton * w, gpointer core )
166{
167    const char * key = g_object_get_data( G_OBJECT(w), PREF_KEY );
168    const int value = gtk_spin_button_get_value_as_int( w );
169    tr_core_set_pref_int( TR_CORE(core), key, value );
170}
171
172static GtkWidget*
173new_spin_button( const char * key, gpointer core, int low, int high, int step )
174{
175    GtkWidget * w = gtk_spin_button_new_with_range( low, high, step );
176    g_object_set_data_full( G_OBJECT(w), PREF_KEY, g_strdup(key), g_free );
177    gtk_spin_button_set_digits( GTK_SPIN_BUTTON(w), 0 );
178    gtk_spin_button_set_value( GTK_SPIN_BUTTON(w), pref_int_get(key) );
179    g_signal_connect( w, "value-changed", G_CALLBACK(spun_cb), core );
180    return w;
181}
182
183static void
184entry_changed_cb( GtkEntry * w, gpointer core )
185{
186    const char * key = g_object_get_data( G_OBJECT( w ), PREF_KEY );
187    const char * value = gtk_entry_get_text( w );
188    tr_core_set_pref( TR_CORE( core ), key, value );
189}
190
191static GtkWidget*
192new_entry( const char * key, gpointer core )
193{
194    GtkWidget * w = gtk_entry_new( );
195    const char * value = pref_string_get( key );
196    if( value )
197        gtk_entry_set_text( GTK_ENTRY( w ), value );
198    g_object_set_data_full( G_OBJECT(w), PREF_KEY, g_strdup(key), g_free );
199    g_signal_connect( w, "changed", G_CALLBACK(entry_changed_cb), core );
200    return w;
201}
202
203static void
204chosen_cb( GtkFileChooser * w, gpointer core )
205{
206    const char * key = g_object_get_data( G_OBJECT(w), PREF_KEY );
207    char * value = gtk_file_chooser_get_filename( GTK_FILE_CHOOSER(w) );
208    tr_core_set_pref( TR_CORE(core), key, value );
209    g_free( value );
210}
211
212static GtkWidget*
213new_path_chooser_button( const char * key, gpointer core )
214{
215    GtkWidget * w = gtk_file_chooser_button_new( NULL,
216                                    GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER );
217    const char * path = pref_string_get( key );
218    g_object_set_data_full( G_OBJECT(w), PREF_KEY, g_strdup(key), g_free );
219    g_signal_connect( w, "selection-changed", G_CALLBACK(chosen_cb), core );
220    gtk_file_chooser_set_current_folder( GTK_FILE_CHOOSER(w), path );
221    return w;
222}
223
224static void
225target_cb( GtkWidget * tb, gpointer target )
226{
227    const gboolean b = gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON( tb ) );
228    gtk_widget_set_sensitive( GTK_WIDGET(target), b );
229}
230
231struct test_port_data
232{
233    GtkWidget * label;
234    gboolean * alive;
235};
236
237static void
238testing_port_done( tr_handle   * handle         UNUSED,
239                   long          response_code  UNUSED,
240                   const void  * response,
241                   size_t        response_len,
242                   void        * gdata )
243{
244    struct test_port_data * data = gdata;
245    if( *data->alive )
246    {
247        const int isOpen = response_len && *(char*)response=='1';
248        gtk_label_set_markup( GTK_LABEL( data->label ), isOpen
249                              ? _("Port is <b>open</b>")
250                              : _("Port is <b>closed</b>") );
251    }
252}
253
254static gboolean
255testing_port_begin( gpointer gdata )
256{
257    struct test_port_data * data = gdata;
258    if( *data->alive )
259    {
260        GtkSpinButton * spin = g_object_get_data( G_OBJECT( data->label ), "tr-port-spin" );
261        tr_handle * handle = g_object_get_data( G_OBJECT( data->label ), "handle" );
262        const int port = gtk_spin_button_get_value_as_int( spin );
263        char url[256];
264        g_snprintf( url, sizeof(url), "http://portcheck.transmissionbt.com/%d", port );
265        tr_webRun( handle, url, NULL, testing_port_done, data );
266    }
267    return FALSE;
268}
269
270static void
271testing_port_cb( GtkWidget * unused UNUSED, gpointer l )
272{
273    struct test_port_data * data;
274
275    gtk_label_set_markup( GTK_LABEL( l ), _( "<i>Testing port...</i>" ) );
276
277    /* wait three seconds to give the port forwarding time to kick in */
278    data = g_new0( struct test_port_data, 1 );
279    data->label = l;
280    data->alive = g_object_get_data( G_OBJECT( l ), "alive" );
281    g_timeout_add( 3000, testing_port_begin, data );
282}
283
284static void
285dialogDestroyed( gpointer alive, GObject * dialog UNUSED )
286{
287    *(gboolean*)alive = FALSE;
288}
289
290static GtkWidget*
291torrentPage( GObject * core )
292{
293    int row = 0;
294    const char * s;
295    GtkWidget * t;
296    GtkWidget * w;
297#ifdef HAVE_GIO
298    GtkWidget * l;
299#endif
300
301    t = hig_workarea_create( );
302    hig_workarea_add_section_title( t, &row, _( "Adding Torrents" ) );
303
304#ifdef HAVE_GIO
305        s = _( "Automatically _add torrents from:" );
306        l = new_check_button( s, PREF_KEY_DIR_WATCH_ENABLED, core );
307        w = new_path_chooser_button( PREF_KEY_DIR_WATCH, core );
308        gtk_widget_set_sensitive( GTK_WIDGET(w), pref_flag_get( PREF_KEY_DIR_WATCH_ENABLED ) );
309        g_signal_connect( l, "toggled", G_CALLBACK(target_cb), w );
310        hig_workarea_add_row_w( t, &row, l, w, NULL );
311#endif
312
313        s = _( "Display _options dialog" );
314        w = new_check_button( s, PREF_KEY_OPTIONS_PROMPT, core );
315        hig_workarea_add_wide_control( t, &row, w );
316
317        s = _( "_Start when added" );
318        w = new_check_button( s, PREF_KEY_START, core );
319        hig_workarea_add_wide_control( t, &row, w );
320
321        s = _( "Mo_ve source files to Trash" );
322        w = new_check_button( s, PREF_KEY_TRASH_ORIGINAL, core ); 
323        hig_workarea_add_wide_control( t, &row, w );
324
325        w = new_path_chooser_button( PREF_KEY_DOWNLOAD_DIR, core );
326        hig_workarea_add_row( t, &row, _( "_Destination folder:" ), w, NULL );
327
328    hig_workarea_finish( t, &row );
329    return t;
330}
331
332/***
333****
334***/
335
336struct blocklist_data
337{
338    GtkWidget * check;
339    GtkWidget * dialog;
340    TrCore * core;
341    int abortFlag;
342    char secondary[256];
343};
344
345static void
346updateBlocklistText( GtkWidget * w, TrCore * core )
347{
348    const int n = tr_blocklistGetRuleCount( tr_core_handle( core ) );
349    char buf[512];
350    g_snprintf( buf, sizeof( buf ),
351                ngettext( "Ignore the %'d _blocklisted peer",
352                          "Ignore the %'d _blocklisted peers", n ), n );
353    gtk_button_set_label( GTK_BUTTON( w ), buf );
354}
355static gboolean
356updateBlocklistTextFromData( gpointer gdata )
357{
358    struct blocklist_data * data = gdata;
359    updateBlocklistText( data->check, data->core );
360    return FALSE;
361}
362
363static gboolean
364blocklistDialogSetSecondary( gpointer gdata )
365{
366    struct blocklist_data * data = gdata;
367    GtkMessageDialog * md = GTK_MESSAGE_DIALOG( data->dialog );
368    gtk_message_dialog_format_secondary_text( md, data->secondary );
369    return FALSE;
370}
371
372static gboolean
373blocklistDialogAllowClose( gpointer dialog )
374{
375    GtkDialog * d = GTK_DIALOG( dialog );
376    gtk_dialog_set_response_sensitive( GTK_DIALOG( d ), GTK_RESPONSE_CANCEL, FALSE );
377    gtk_dialog_set_response_sensitive( GTK_DIALOG( d ), GTK_RESPONSE_CLOSE, TRUE );
378    return FALSE;
379}
380
381static void
382got_blocklist( tr_handle   * handle         UNUSED,
383               long          response_code  UNUSED,
384               const void  * response,
385               size_t        response_len,
386               void        * gdata )
387{
388    struct blocklist_data * data = gdata;
389    const char * text = response;
390    int size = response_len;
391    int rules = 0;
392    gchar * filename = NULL;
393    gchar * filename2 = NULL;
394    int fd = -1;
395    int ok = 1;
396
397    if( !data->abortFlag && ( !text || !size ) )
398    {
399        ok = FALSE;
400        g_snprintf( data->secondary, sizeof( data->secondary ),
401                    _( "Unable to get blocklist." ) );
402        g_message( data->secondary );
403        g_idle_add( blocklistDialogSetSecondary, data );
404    }     
405
406    if( ok && !data->abortFlag )
407    {
408        GError * err = NULL;
409        fd = g_file_open_tmp( "transmission-blockfile-XXXXXX", &filename, &err );
410        if( err ) {
411            g_snprintf( data->secondary, sizeof( data->secondary ),
412                        _( "Unable to get blocklist: %s" ), err->message );
413            g_warning( data->secondary );
414            g_idle_add( blocklistDialogSetSecondary, data );
415            g_clear_error( &err );
416            ok = FALSE;
417        } else {
418            write( fd, text, size );
419            close( fd );
420        }
421    }
422    if( ok && !data->abortFlag )
423    {
424        char * cmd;
425        filename2 = g_strdup_printf( "%s.txt", filename );
426        g_snprintf( data->secondary, sizeof( data->secondary ),
427                    _( "Uncompressing blocklist..." ) );
428        g_idle_add( blocklistDialogSetSecondary, data );
429        cmd = g_strdup_printf( "zcat %s > %s ", filename, filename2 );
430        tr_dbg( "%s", cmd );
431        system( cmd );
432        g_free( cmd );
433    }
434    if( ok && !data->abortFlag )
435    {
436        g_snprintf( data->secondary, sizeof( data->secondary ),
437                    _( "Parsing blocklist..." ) );
438        g_idle_add( blocklistDialogSetSecondary, data );
439        rules = tr_blocklistSetContent( tr_core_handle( data->core ), filename2 );
440    }
441    if( ok && !data->abortFlag )
442    {
443        g_snprintf( data->secondary, sizeof( data->secondary ),
444                    _( "Blocklist updated with %'d entries" ), rules );
445        g_idle_add( blocklistDialogSetSecondary, data );
446        g_idle_add( blocklistDialogAllowClose, data->dialog );
447        g_idle_add( updateBlocklistTextFromData, data );
448    }
449
450    /* g_free( data ); */
451    if( filename2 ) {
452        unlink( filename2 );
453        g_free( filename2 );
454    }
455    if( filename ) {
456        unlink( filename );
457        g_free( filename );
458    }
459}
460
461static void
462onUpdateBlocklistResponseCB( GtkDialog * dialog, int response, gpointer vdata )
463{
464    struct blocklist_data * data = vdata;
465
466    if( response == GTK_RESPONSE_CANCEL )
467        data->abortFlag = 1;
468
469    data->dialog = NULL;
470    gtk_widget_destroy( GTK_WIDGET( dialog ) );
471}
472
473static void
474onUpdateBlocklistCB( GtkButton * w, gpointer gdata )
475{
476    GtkWidget * d;
477    struct blocklist_data * data = gdata;
478    tr_handle * handle = g_object_get_data( G_OBJECT( w ), "handle" );
479    const char * url = "http://download.m0k.org/transmission/files/level1.gz";
480   
481    d = gtk_message_dialog_new( GTK_WINDOW( gtk_widget_get_toplevel( GTK_WIDGET( w ) ) ),
482                                GTK_DIALOG_DESTROY_WITH_PARENT,
483                                GTK_MESSAGE_INFO,
484                                GTK_BUTTONS_NONE,
485                                _( "Updating Blocklist" ) );
486    gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG( d ),
487                                              _( "Retrieving blocklist..." ) );
488    gtk_dialog_add_buttons( GTK_DIALOG( d ),
489                            GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
490                            GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
491                            NULL );
492    gtk_dialog_set_response_sensitive( GTK_DIALOG( d ), GTK_RESPONSE_CLOSE, FALSE );
493
494    data->dialog = d;
495
496    g_signal_connect( d, "response", G_CALLBACK(onUpdateBlocklistResponseCB), data );
497    gtk_widget_show( d );
498
499    tr_webRun( handle, url, NULL, got_blocklist, data );
500}
501
502static void
503onEncryptionToggled( GtkToggleButton * w, gpointer core )
504{
505    const int val = gtk_toggle_button_get_active( w )
506                  ? TR_ENCRYPTION_REQUIRED
507                  : TR_ENCRYPTION_PREFERRED;
508    tr_core_set_pref_int( TR_CORE( core ), PREF_KEY_ENCRYPTION, val );
509}
510
511static GtkWidget*
512peerPage( GObject * core, gboolean * alive )
513{
514    int row = 0;
515    const char * s;
516    GtkWidget * t;
517    GtkWidget * w;
518    GtkWidget * w2;
519    GtkWidget * b;
520    GtkWidget * h;
521    GtkWidget * l;
522    struct blocklist_data * data;
523
524    t = hig_workarea_create( );
525    hig_workarea_add_section_title (t, &row, _("Options"));
526
527        w = new_check_button( "", PREF_KEY_BLOCKLIST_ENABLED, core );
528        updateBlocklistText( w, TR_CORE( core ) );
529        h = gtk_hbox_new( FALSE, GUI_PAD_BIG );
530        gtk_box_pack_start_defaults( GTK_BOX(h), w );
531        b = gtk_button_new_with_mnemonic( _( "_Update Blocklist" ) );
532
533        data = g_new0( struct blocklist_data, 1 );
534        data->core = TR_CORE( core );
535        data->check = w;
536
537        g_object_set_data( G_OBJECT( b ), "handle", tr_core_handle( TR_CORE( core ) ) );
538        g_signal_connect( b, "clicked", G_CALLBACK(onUpdateBlocklistCB), data );
539        gtk_box_pack_start( GTK_BOX(h), b, FALSE, FALSE, 0 );
540        g_signal_connect( w, "toggled", G_CALLBACK(target_cb), b );
541        target_cb( w, b );
542        hig_workarea_add_wide_control( t, &row, h );
543       
544        s = _("_Ignore unencrypted peers");
545        w = gtk_check_button_new_with_mnemonic( s );
546        gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(w),
547                                      pref_int_get(PREF_KEY_ENCRYPTION)==TR_ENCRYPTION_REQUIRED );
548        g_signal_connect( w, "toggled", G_CALLBACK(onEncryptionToggled), core );
549        hig_workarea_add_wide_control( t, &row, w );
550
551        s = _("Use peer e_xchange");
552        w = new_check_button( s, PREF_KEY_PEX, core );
553        hig_workarea_add_wide_control( t, &row, w );
554
555        h = gtk_hbox_new( FALSE, GUI_PAD_BIG );
556        w2 = new_spin_button( PREF_KEY_PORT, core, 1, INT_MAX, 1 );
557        gtk_box_pack_start( GTK_BOX(h), w2, FALSE, FALSE, 0 );
558        l = gtk_label_new( NULL );
559        gtk_misc_set_alignment( GTK_MISC(l), 0.0f, 0.5f );
560        gtk_box_pack_start( GTK_BOX(h), l, FALSE, FALSE, 0 );
561        hig_workarea_add_row( t, &row, _("Listening _port:"), h, w2 );
562
563        g_object_set_data( G_OBJECT(l), "tr-port-spin", w2 );
564        g_object_set_data( G_OBJECT(l), "alive", alive );
565        g_object_set_data( G_OBJECT(l), "handle", tr_core_handle( TR_CORE( core ) ) );
566        testing_port_cb( NULL, l );
567        g_signal_connect( w2, "value-changed", G_CALLBACK(testing_port_cb), l );
568       
569    hig_workarea_add_section_divider( t, &row );
570    hig_workarea_add_section_title( t, &row, _( "Limits" ) );
571 
572        w = new_spin_button( PREF_KEY_MAX_PEERS_GLOBAL, core, 1, 3000, 5 );
573        hig_workarea_add_row( t, &row, _( "Maximum peers _overall:" ), w, NULL );
574        w = new_spin_button( PREF_KEY_MAX_PEERS_PER_TORRENT, core, 1, 300, 5 );
575        hig_workarea_add_row( t, &row, _( "Maximum peers per _torrent:" ), w, NULL );
576
577    hig_workarea_finish( t, &row );
578    return t;
579}
580
581static GtkTreeModel*
582allow_deny_model_new( void )
583{
584    GtkTreeIter iter;
585    GtkListStore * store = gtk_list_store_new( 2, G_TYPE_STRING, G_TYPE_CHAR );
586    gtk_list_store_append( store, &iter );
587    gtk_list_store_set( store, &iter, 0, _( "Allow" ), 1, '+', -1 );
588    gtk_list_store_append( store, &iter );
589    gtk_list_store_set( store, &iter, 0, _( "Deny" ), 1, '-', -1 );
590    return GTK_TREE_MODEL( store );
591}
592
593enum
594{
595    COL_ADDRESS,
596    COL_PERMISSION,
597    N_COLS
598};
599
600static GtkTreeModel*
601acl_tree_model_new( const char * acl )
602{
603    int i;
604    char ** rules;
605    GtkListStore * store = gtk_list_store_new( N_COLS,
606                                               G_TYPE_STRING,
607                                               G_TYPE_STRING );
608    rules = g_strsplit( acl, ",", 0 );
609
610    for( i=0; rules && rules[i]; ++i )
611    {
612        const char * s = rules[i];
613        while( isspace( *s ) ) ++s;
614        if( *s=='+' || *s=='-' )
615        {
616            GtkTreeIter iter;
617            gtk_list_store_append( store, &iter );
618            gtk_list_store_set( store, &iter,
619                COL_PERMISSION, *s=='+' ? _( "Allow" ) : _( "Deny" ) ,
620                COL_ADDRESS, s+1,
621                -1 );
622        }
623    }
624
625    g_strfreev( rules );
626    return GTK_TREE_MODEL( store );
627}
628
629struct remote_page
630{
631    TrCore * core;
632    GtkTreeView * view;
633    GtkListStore * store;
634    GtkWidget * remove_button;
635    GSList * widgets;
636    GSList * auth_widgets;
637    GtkToggleButton * rpc_tb;
638    GtkToggleButton * auth_tb;
639};
640
641static void
642refreshACL( struct remote_page * page )
643{
644    GtkTreeIter iter;
645    GtkTreeModel * model = GTK_TREE_MODEL( page->store );
646    GString * gstr = g_string_new( NULL );
647
648    if( gtk_tree_model_get_iter_first( model, &iter ) ) do
649    {
650        char * permission;
651        char * address;
652        gtk_tree_model_get( model, &iter, COL_PERMISSION, &permission,
653                                          COL_ADDRESS, &address,
654                                          -1 );
655        g_string_append_c( gstr, strcmp(permission,_("Allow")) ? '-' : '+' );
656        g_string_append( gstr, address );
657        g_string_append( gstr, ", " );
658        g_free( address );
659        g_free( permission );
660    }
661    while( gtk_tree_model_iter_next( model, &iter ) );
662
663    g_string_truncate( gstr, gstr->len-2 ); /* remove the trailing ", " */
664
665    tr_core_set_pref( page->core, PREF_KEY_RPC_ACL, gstr->str );
666
667    g_string_free( gstr, TRUE );
668}
669
670static void
671onPermissionEdited( GtkCellRendererText  * renderer UNUSED,
672                    gchar                * path_string,
673                    gchar                * new_text,
674                    gpointer               gpage )
675{
676    GtkTreeIter iter;
677    GtkTreePath * path = gtk_tree_path_new_from_string( path_string );
678    struct remote_page * page = gpage;
679    GtkTreeModel * model = GTK_TREE_MODEL( page->store );
680    if( gtk_tree_model_get_iter( model, &iter, path ) )
681        gtk_list_store_set( page->store, &iter, COL_PERMISSION, new_text, -1 );
682    gtk_tree_path_free( path );
683    refreshACL( page );
684}
685
686static void
687onAddressEdited( GtkCellRendererText  * r UNUSED,
688                 gchar                * path_string,
689                 gchar                * new_text,
690                 gpointer               gpage )
691{
692    char * acl;
693    GtkTreeIter iter;
694    struct remote_page * page = gpage;
695    GtkTreeModel * model = GTK_TREE_MODEL( page->store );
696    tr_handle * session = tr_core_handle( page->core );
697    GtkTreePath * path = gtk_tree_path_new_from_string( path_string );
698
699    acl = g_strdup_printf( "+%s", new_text );
700    if( !tr_sessionTestRPCACL( session, acl, NULL ) )
701        if( gtk_tree_model_get_iter( model, &iter, path ) )
702            gtk_list_store_set( page->store, &iter, COL_ADDRESS, new_text, -1 );
703
704    g_free( acl );
705    gtk_tree_path_free( path );
706    refreshACL( page );
707}
708
709static void
710onAddACLClicked( GtkButton * b UNUSED, gpointer gpage )
711{
712    GtkTreeIter iter;
713    GtkTreePath * path;
714    struct remote_page * page = gpage;
715    gtk_list_store_append( page->store, &iter );
716    gtk_list_store_set( page->store, &iter,
717                        COL_PERMISSION, _( "Allow" ),
718                        COL_ADDRESS,  "0.0.0.0",
719                        -1 );
720
721    path = gtk_tree_model_get_path( GTK_TREE_MODEL( page->store ), &iter );
722    gtk_tree_view_set_cursor(
723        page->view, path,
724        gtk_tree_view_get_column( page->view, COL_ADDRESS ),
725        TRUE );
726    gtk_tree_path_free( path );
727}
728
729static void
730onRemoveACLClicked( GtkButton * b UNUSED, gpointer gpage )
731{
732    struct remote_page * page = gpage;
733    GtkTreeSelection * sel = gtk_tree_view_get_selection( page->view );
734    GtkTreeIter iter;
735    if( gtk_tree_selection_get_selected( sel, NULL, &iter ) )
736    {
737        gtk_list_store_remove( page->store, &iter );
738        refreshACL( page );
739    }
740}
741
742static void
743refreshRPCSensitivity( struct remote_page * page )
744{
745    GSList * l;
746    const int rpc_active = gtk_toggle_button_get_active( page->rpc_tb );
747    const int auth_active = gtk_toggle_button_get_active( page->auth_tb );
748    GtkTreeSelection * sel = gtk_tree_view_get_selection( page->view );
749    const int have_addr = gtk_tree_selection_get_selected( sel, NULL, NULL );
750    const int n_rules = gtk_tree_model_iter_n_children(
751                                       GTK_TREE_MODEL( page->store ), NULL );
752
753    for( l=page->widgets; l!=NULL; l=l->next )
754        gtk_widget_set_sensitive( GTK_WIDGET( l->data ), rpc_active );
755
756    for( l=page->auth_widgets; l!=NULL; l=l->next )
757        gtk_widget_set_sensitive( GTK_WIDGET( l->data ), rpc_active && auth_active);
758
759    gtk_widget_set_sensitive( page->remove_button,
760                              rpc_active && have_addr && n_rules>1 );
761}
762
763static void
764onRPCToggled( GtkToggleButton * tb UNUSED, gpointer page )
765{
766    refreshRPCSensitivity( page );
767}
768static void
769onACLSelectionChanged( GtkTreeSelection * sel UNUSED, gpointer page )
770{
771    refreshRPCSensitivity( page );
772}
773
774static void
775onLaunchClutchCB( GtkButton * w UNUSED, gpointer data UNUSED )
776{
777    int port = pref_int_get( PREF_KEY_RPC_PORT );
778    char * url = g_strdup_printf( "http://localhost:%d/transmission/web", port );
779    gtr_open_file( url );
780    g_free( url );
781}
782
783static GtkWidget*
784webPage( GObject * core )
785{
786    const char  * s;
787    int row = 0;
788    GtkWidget * t;
789    GtkWidget * w;
790    GtkWidget * h;
791    struct remote_page * page = g_new0( struct remote_page, 1 );
792
793    page->core = TR_CORE( core );
794
795    t = hig_workarea_create( );
796    g_object_set_data_full( G_OBJECT( t ), "page", page, g_free );
797
798    hig_workarea_add_section_title( t, &row, _( "Web Interface" ) );
799
800        /* "enabled" checkbutton */
801        s = _( "_Enable web interface" );
802        w = new_check_button( s, PREF_KEY_RPC_ENABLED, core );
803        page->rpc_tb = GTK_TOGGLE_BUTTON( w );
804        g_signal_connect( w, "clicked", G_CALLBACK(onRPCToggled), page );
805        h = gtk_hbox_new( FALSE, GUI_PAD_BIG );
806        gtk_box_pack_start_defaults( GTK_BOX(h), w );
807        w = gtk_button_new_from_stock( GTK_STOCK_OPEN );
808        page->widgets = g_slist_append( page->widgets, w );
809        g_signal_connect( w, "clicked", G_CALLBACK(onLaunchClutchCB), NULL );
810        gtk_box_pack_start( GTK_BOX(h), w, FALSE, FALSE, 0 );
811        hig_workarea_add_wide_control( t, &row, h );
812
813        /* require authentication */
814        s = _( "_Require username" );
815        w = new_check_button( s, PREF_KEY_RPC_AUTH_ENABLED, core );
816        hig_workarea_add_wide_control( t, &row, w );
817        page->auth_tb = GTK_TOGGLE_BUTTON( w );
818        page->widgets = g_slist_append( page->widgets, w );
819        g_signal_connect( w, "clicked", G_CALLBACK(onRPCToggled), page );
820
821        /* username */
822        s = _( "_Username:" );
823        w = new_entry( PREF_KEY_RPC_USERNAME, core );
824        page->auth_widgets = g_slist_append( page->auth_widgets, w );
825        w = hig_workarea_add_row( t, &row, s, w, NULL );
826        page->auth_widgets = g_slist_append( page->auth_widgets, w );
827
828        /* password */
829        s = _( "Pass_word:" );
830        w = new_entry( PREF_KEY_RPC_PASSWORD, core );
831        gtk_entry_set_visibility( GTK_ENTRY( w ), FALSE );
832        page->auth_widgets = g_slist_append( page->auth_widgets, w );
833        w = hig_workarea_add_row( t, &row, s, w, NULL );
834        page->auth_widgets = g_slist_append( page->auth_widgets, w );
835
836        /* port */
837        w = new_spin_button( PREF_KEY_RPC_PORT, core, 0, 65535, 1 );
838        page->widgets = g_slist_append( page->widgets, w );
839        w = hig_workarea_add_row( t, &row, _( "Listening _port:" ), w, NULL );
840        page->widgets = g_slist_append( page->widgets, w );
841
842        /* access control list */
843        {
844        const char * val = pref_string_get( PREF_KEY_RPC_ACL );
845        GtkTreeModel * m = acl_tree_model_new( val );
846        GtkTreeViewColumn * c;
847        GtkCellRenderer * r;
848        GtkTreeSelection * sel;
849        GtkTreeView * v;
850        GtkWidget * w;
851        GtkWidget * h;
852        GtkTooltips * tips = gtk_tooltips_new( );
853
854        s = _( "Access control list:" );
855        page->store = GTK_LIST_STORE( m );
856        w = gtk_tree_view_new_with_model( m );
857
858        page->widgets = g_slist_append( page->widgets, w );
859        v = page->view = GTK_TREE_VIEW( w );
860        gtk_tooltips_set_tip( tips, w,
861            _( "IP addresses may use wildcards, such as 192.168.*.*" ),
862            NULL );
863        sel = gtk_tree_view_get_selection( v );
864        g_signal_connect( sel, "changed",
865                          G_CALLBACK( onACLSelectionChanged ), page );
866        g_object_unref( G_OBJECT( m ) );
867        gtk_tree_view_set_headers_visible( v, TRUE );
868        w = gtk_frame_new( NULL );
869        gtk_frame_set_shadow_type( GTK_FRAME( w ), GTK_SHADOW_IN );
870        gtk_container_add( GTK_CONTAINER( w ), GTK_WIDGET( v ) );
871
872        /* ip address column */
873        r = gtk_cell_renderer_text_new( );
874        g_signal_connect( r, "edited",
875                          G_CALLBACK( onAddressEdited ), page );
876        g_object_set( G_OBJECT( r ), "editable", TRUE, NULL );
877        c = gtk_tree_view_column_new_with_attributes( _( "IP Address" ), r,
878                "text", COL_ADDRESS,
879                NULL );
880        gtk_tree_view_column_set_expand( c, TRUE );
881        gtk_tree_view_append_column( v, c );
882
883        w = hig_workarea_add_row( t, &row, s, w, NULL );
884        gtk_misc_set_alignment( GTK_MISC( w ), 0.0f, 0.1f );
885        page->widgets = g_slist_append( page->widgets, w );
886
887        /* permission column */
888        m = allow_deny_model_new( );
889        r = gtk_cell_renderer_combo_new( );
890        g_object_set( G_OBJECT( r ), "model", m,
891                                     "editable", TRUE,
892                                     "has-entry", FALSE,
893                                     "text-column", 0,
894                                     NULL );
895        c = gtk_tree_view_column_new_with_attributes( _( "Permission" ), r,
896                "text", COL_PERMISSION,
897                NULL );
898        g_signal_connect( r, "edited",
899                          G_CALLBACK( onPermissionEdited ), page );
900        gtk_tree_view_append_column( v, c );
901
902        h = gtk_hbox_new( TRUE, GUI_PAD );
903        w = gtk_button_new_from_stock( GTK_STOCK_REMOVE );
904        g_signal_connect( w, "clicked", G_CALLBACK(onRemoveACLClicked), page );
905        page->remove_button = w;
906        onACLSelectionChanged( sel, page );
907        gtk_box_pack_start_defaults( GTK_BOX( h ), w );
908        w = gtk_button_new_from_stock( GTK_STOCK_ADD );
909        page->widgets = g_slist_append( page->widgets, w );
910        g_signal_connect( w, "clicked", G_CALLBACK(onAddACLClicked), page );
911        gtk_box_pack_start_defaults( GTK_BOX( h ), w );
912        w = gtk_hbox_new( FALSE, 0 );
913        gtk_box_pack_start_defaults( GTK_BOX( w ), gtk_alignment_new( 0, 0, 0, 0 ) );
914        gtk_box_pack_start( GTK_BOX( w ), h, FALSE, FALSE, 0 );
915        hig_workarea_add_wide_control( t, &row, w );
916        }
917
918    refreshRPCSensitivity( page );
919    hig_workarea_finish( t, &row );
920    return t;
921}
922
923struct ProxyPage
924{
925    TrCore * core;
926    GSList * proxy_widgets;
927    GSList * proxy_auth_widgets;
928};
929
930static void
931refreshProxySensitivity( struct ProxyPage * p )
932{
933    GSList * l;
934    const gboolean proxy_enabled = pref_flag_get( PREF_KEY_PROXY_SERVER_ENABLED );
935    const gboolean proxy_auth_enabled = pref_flag_get( PREF_KEY_PROXY_AUTH_ENABLED );
936
937    for( l=p->proxy_widgets; l!=NULL; l=l->next )
938        gtk_widget_set_sensitive( GTK_WIDGET( l->data ), proxy_enabled );
939
940    for( l=p->proxy_auth_widgets; l!=NULL; l=l->next )
941        gtk_widget_set_sensitive( GTK_WIDGET( l->data ),
942                                  proxy_enabled && proxy_auth_enabled);
943}
944
945static void
946onProxyToggled( GtkToggleButton * tb UNUSED, gpointer user_data )
947{
948    refreshProxySensitivity( user_data );
949}
950
951static void
952proxyPageFree( gpointer gpage )
953{
954    struct ProxyPage * page = gpage;
955    g_slist_free( page->proxy_widgets );
956    g_slist_free( page->proxy_auth_widgets );
957    g_free( page );
958}
959
960static GtkTreeModel*
961proxyTypeModelNew( void )
962{
963    GtkTreeIter iter;
964    GtkListStore * store = gtk_list_store_new( 2, G_TYPE_STRING, G_TYPE_INT );
965    gtk_list_store_append( store, &iter );
966    gtk_list_store_set( store, &iter, 0, "HTTP", 1, TR_PROXY_HTTP, -1 );
967    gtk_list_store_append( store, &iter );
968    gtk_list_store_set( store, &iter, 0, "SOCKS4", 1, TR_PROXY_SOCKS4, -1 );
969    gtk_list_store_append( store, &iter );
970    gtk_list_store_set( store, &iter, 0, "SOCKS5", 1, TR_PROXY_SOCKS5, -1 );
971    return GTK_TREE_MODEL( store );
972}
973
974static void
975onProxyTypeChanged( GtkComboBox * w, gpointer gpage )
976{
977    GtkTreeIter iter;
978    if( gtk_combo_box_get_active_iter( w, &iter ) )
979    {
980        struct ProxyPage * page = gpage;
981        int type = TR_PROXY_HTTP;
982        gtk_tree_model_get( gtk_combo_box_get_model( w ), &iter, 1, &type, -1 );
983        tr_core_set_pref_int( TR_CORE( page->core ), PREF_KEY_PROXY_TYPE, type );
984    }
985}
986
987static GtkWidget*
988trackerPage( GObject * core )
989{
990    int row = 0;
991    const char * s;
992    GtkWidget * t;
993    GtkWidget * w;
994    GtkTreeModel * m;
995    GtkCellRenderer * r;
996    struct ProxyPage * page = tr_new0( struct ProxyPage, 1 );
997
998    page->core = TR_CORE( core );
999
1000    t = hig_workarea_create( );
1001    hig_workarea_add_section_title (t, &row, _( "Tracker Proxy" ) );
1002
1003        s = _( "Connect to tracker via a pro_xy" );
1004        w = new_check_button( s, PREF_KEY_PROXY_SERVER_ENABLED, core );
1005        g_signal_connect( w, "toggled", G_CALLBACK(onProxyToggled), page );
1006        hig_workarea_add_wide_control( t, &row, w );
1007
1008        s = _( "Proxy _server:" );
1009        w = new_entry( PREF_KEY_PROXY_SERVER, core );
1010        page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
1011        w = hig_workarea_add_row( t, &row, s, w, NULL );
1012        page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
1013
1014        w = new_spin_button( PREF_KEY_PROXY_PORT, core, 0, 65536, 1 );
1015        page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
1016        w = hig_workarea_add_row( t, &row, _( "Proxy _port:" ), w, NULL );
1017        page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
1018
1019        s = _( "Proxy _type:" );
1020        m = proxyTypeModelNew( );
1021        w = gtk_combo_box_new_with_model( m );
1022        r = gtk_cell_renderer_text_new( );
1023        gtk_cell_layout_pack_start( GTK_CELL_LAYOUT( w ), r, TRUE );
1024        gtk_cell_layout_set_attributes( GTK_CELL_LAYOUT( w ), r, "text", 0, NULL );
1025        gtk_combo_box_set_active( GTK_COMBO_BOX( w ), pref_int_get( PREF_KEY_PROXY_TYPE ) );
1026        g_signal_connect( w, "changed", G_CALLBACK(onProxyTypeChanged), page );
1027        g_object_unref( G_OBJECT( m ) );
1028        page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
1029        w = hig_workarea_add_row( t, &row, s, w, NULL );
1030        page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
1031
1032        s = _( "_Authentication is required" );
1033        w = new_check_button( s, PREF_KEY_PROXY_AUTH_ENABLED, core );
1034        g_signal_connect( w, "toggled", G_CALLBACK(onProxyToggled), page );
1035        hig_workarea_add_wide_control( t, &row, w );
1036        page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
1037
1038        s = _( "_Username:" );
1039        w = new_entry( PREF_KEY_PROXY_USERNAME, core );
1040        page->proxy_auth_widgets = g_slist_append( page->proxy_auth_widgets, w );
1041        w = hig_workarea_add_row( t, &row, s, w, NULL );
1042        page->proxy_auth_widgets = g_slist_append( page->proxy_auth_widgets, w );
1043
1044        s = _( "Pass_word:" );
1045        w = new_entry( PREF_KEY_PROXY_PASSWORD, core );
1046        gtk_entry_set_visibility( GTK_ENTRY( w ), FALSE );
1047        page->proxy_auth_widgets = g_slist_append( page->proxy_auth_widgets, w );
1048        w = hig_workarea_add_row( t, &row, s, w, NULL );
1049        page->proxy_auth_widgets = g_slist_append( page->proxy_auth_widgets, w );
1050
1051    hig_workarea_finish( t, &row );
1052    g_object_set_data_full( G_OBJECT( t ), "page", page, proxyPageFree );
1053
1054    refreshProxySensitivity( page );
1055    return t;
1056}
1057
1058static GtkWidget*
1059networkPage( GObject * core )
1060{
1061    int row = 0;
1062    const char * s;
1063    GtkWidget * t;
1064    GtkWidget * w, * w2;
1065
1066    t = hig_workarea_create( );
1067    hig_workarea_add_section_title (t, &row, _( "Router" ) );
1068
1069        s = _("Use UPnP or NAT-PMP port _forwarding from my router" );
1070        w = new_check_button( s, PREF_KEY_PORT_FORWARDING, core );
1071        hig_workarea_add_wide_control( t, &row, w );
1072
1073    hig_workarea_add_section_divider( t, &row );
1074    hig_workarea_add_section_title (t, &row, _("Bandwidth"));
1075
1076        s = _("Limit _download speed (KB/s):");
1077        w = new_check_button( s, PREF_KEY_DL_LIMIT_ENABLED, core );
1078        w2 = new_spin_button( PREF_KEY_DL_LIMIT, core, 0, INT_MAX, 5 );
1079        gtk_widget_set_sensitive( GTK_WIDGET(w2), pref_flag_get( PREF_KEY_DL_LIMIT_ENABLED ) );
1080        g_signal_connect( w, "toggled", G_CALLBACK(target_cb), w2 );
1081        hig_workarea_add_row_w( t, &row, w, w2, NULL );
1082
1083        s = _("Limit _upload speed (KB/s):");
1084        w = new_check_button( s, PREF_KEY_UL_LIMIT_ENABLED, core );
1085        w2 = new_spin_button( PREF_KEY_UL_LIMIT, core, 0, INT_MAX, 5 );
1086        gtk_widget_set_sensitive( GTK_WIDGET(w2), pref_flag_get( PREF_KEY_UL_LIMIT_ENABLED ) );
1087        g_signal_connect( w, "toggled", G_CALLBACK(target_cb), w2 );
1088        hig_workarea_add_row_w( t, &row, w, w2, NULL );
1089
1090    hig_workarea_finish( t, &row );
1091    return t;
1092}
1093
1094GtkWidget *
1095tr_prefs_dialog_new( GObject * core, GtkWindow * parent )
1096{
1097    GtkWidget * d;
1098    GtkWidget * n;
1099    gboolean * alive;
1100
1101    alive = g_new( gboolean, 1 );
1102    *alive = TRUE;
1103
1104    d = gtk_dialog_new_with_buttons( _( "Transmission Preferences" ), parent,
1105                                     GTK_DIALOG_DESTROY_WITH_PARENT,
1106                                     GTK_STOCK_HELP, GTK_RESPONSE_HELP,
1107                                     GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
1108                                     NULL );
1109    gtk_window_set_role( GTK_WINDOW(d), "transmission-preferences-dialog" );
1110    gtk_dialog_set_has_separator( GTK_DIALOG( d ), FALSE );
1111    gtk_container_set_border_width( GTK_CONTAINER( d ), GUI_PAD );
1112    g_object_weak_ref( G_OBJECT( d ), dialogDestroyed, alive );
1113
1114    n = gtk_notebook_new( );
1115    gtk_container_set_border_width ( GTK_CONTAINER ( n ), GUI_PAD );
1116
1117    gtk_notebook_append_page( GTK_NOTEBOOK( n ),
1118                              torrentPage( core ),
1119                              gtk_label_new (_("Torrents")) );
1120    gtk_notebook_append_page( GTK_NOTEBOOK( n ),
1121                              peerPage( core, alive ),
1122                              gtk_label_new (_("Peers")) );
1123    gtk_notebook_append_page( GTK_NOTEBOOK( n ),
1124                              trackerPage( core ),
1125                              gtk_label_new (_("Trackers")) );
1126    gtk_notebook_append_page( GTK_NOTEBOOK( n ),
1127                              networkPage( core ),
1128                              gtk_label_new (_("Network")) );
1129    gtk_notebook_append_page( GTK_NOTEBOOK( n ),
1130                              webPage( core ),
1131                              gtk_label_new (_("Web")) );
1132
1133    g_signal_connect( d, "response", G_CALLBACK(response_cb), core );
1134    gtk_box_pack_start_defaults( GTK_BOX(GTK_DIALOG(d)->vbox), n );
1135    gtk_widget_show_all( GTK_DIALOG(d)->vbox );
1136    return d;
1137}
Note: See TracBrowser for help on using the repository browser.