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

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

#857: DSCP support for Transmission

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