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

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

(gtk) #927: "help" button in the preferences dialog

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