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

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

(gtk) better error recording in the prefs dialog when updating the blocklist

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