1427 lines
48 KiB
C
1427 lines
48 KiB
C
/*
|
|
* This file Copyright (C) 2007-2009 Mnemosyne LLC
|
|
*
|
|
* This file is licensed by the GPL version 2. Works owned by the
|
|
* Transmission project are granted a special exemption to clause 2(b)
|
|
* so that the bulk of its code can remain under the MIT license.
|
|
* This exemption does not extend to derived works not owned by
|
|
* the Transmission project.
|
|
*
|
|
* $Id$
|
|
*/
|
|
|
|
#include <ctype.h> /* isspace */
|
|
#include <errno.h>
|
|
#include <stdarg.h>
|
|
#include <limits.h> /* USHRT_MAX */
|
|
#include <stdlib.h> /* free() */
|
|
#include <unistd.h>
|
|
#include <glib/gi18n.h>
|
|
#include <gtk/gtk.h>
|
|
#include <libtransmission/transmission.h>
|
|
#include <libtransmission/utils.h>
|
|
#include <libtransmission/version.h>
|
|
#include <libtransmission/web.h>
|
|
#include "conf.h"
|
|
#include "hig.h"
|
|
#include "tr-core.h"
|
|
#include "tr-prefs.h"
|
|
#include "util.h"
|
|
|
|
/**
|
|
***
|
|
**/
|
|
|
|
#define PREF_KEY "pref-key"
|
|
|
|
static void
|
|
response_cb( GtkDialog * dialog,
|
|
int response,
|
|
gpointer unused UNUSED )
|
|
{
|
|
if( response == GTK_RESPONSE_HELP )
|
|
{
|
|
char * base = gtr_get_help_url( );
|
|
char * url = g_strdup_printf( "%s/html/preferences.html", base );
|
|
gtr_open_file( url );
|
|
g_free( url );
|
|
g_free( base );
|
|
}
|
|
|
|
if( response == GTK_RESPONSE_CLOSE )
|
|
gtk_widget_destroy( GTK_WIDGET( dialog ) );
|
|
}
|
|
|
|
static void
|
|
toggled_cb( GtkToggleButton * w,
|
|
gpointer core )
|
|
{
|
|
const char * key = g_object_get_data( G_OBJECT( w ), PREF_KEY );
|
|
const gboolean flag = gtk_toggle_button_get_active( w );
|
|
|
|
tr_core_set_pref_bool( TR_CORE( core ), key, flag );
|
|
}
|
|
|
|
static GtkWidget*
|
|
new_check_button( const char * mnemonic,
|
|
const char * key,
|
|
gpointer core )
|
|
{
|
|
GtkWidget * w = gtk_check_button_new_with_mnemonic( mnemonic );
|
|
|
|
g_object_set_data_full( G_OBJECT( w ), PREF_KEY, g_strdup(
|
|
key ), g_free );
|
|
gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON( w ),
|
|
pref_flag_get( key ) );
|
|
g_signal_connect( w, "toggled", G_CALLBACK( toggled_cb ), core );
|
|
return w;
|
|
}
|
|
|
|
#define IDLE_DATA "idle-data"
|
|
|
|
struct spin_idle_data
|
|
{
|
|
gpointer core;
|
|
GTimer * last_change;
|
|
gboolean isDouble;
|
|
};
|
|
|
|
static void
|
|
spin_idle_data_free( gpointer gdata )
|
|
{
|
|
struct spin_idle_data * data = gdata;
|
|
|
|
g_timer_destroy( data->last_change );
|
|
g_free( data );
|
|
}
|
|
|
|
static gboolean
|
|
spun_cb_idle( gpointer spin )
|
|
{
|
|
gboolean keep_waiting = TRUE;
|
|
GObject * o = G_OBJECT( spin );
|
|
struct spin_idle_data * data = g_object_get_data( o, IDLE_DATA );
|
|
|
|
/* has the user stopped making changes? */
|
|
if( g_timer_elapsed( data->last_change, NULL ) > 0.33f )
|
|
{
|
|
/* update the core */
|
|
const char * key = g_object_get_data( o, PREF_KEY );
|
|
if (data->isDouble)
|
|
{
|
|
const double value = gtk_spin_button_get_value( GTK_SPIN_BUTTON( spin ) );
|
|
tr_core_set_pref_double( TR_CORE( data->core ), key, value );
|
|
}
|
|
else
|
|
{
|
|
const int value = gtk_spin_button_get_value_as_int(
|
|
GTK_SPIN_BUTTON( spin ) );
|
|
tr_core_set_pref_int( TR_CORE( data->core ), key, value );
|
|
}
|
|
|
|
/* cleanup */
|
|
g_object_set_data( o, IDLE_DATA, NULL );
|
|
keep_waiting = FALSE;
|
|
g_object_unref( G_OBJECT( o ) );
|
|
}
|
|
|
|
return keep_waiting;
|
|
}
|
|
|
|
static void
|
|
spun_cb( GtkSpinButton * w,
|
|
gpointer core,
|
|
gboolean isDouble )
|
|
{
|
|
/* user may be spinning through many values, so let's hold off
|
|
for a moment to keep from flooding the core with changes */
|
|
GObject * o = G_OBJECT( w );
|
|
struct spin_idle_data * data = g_object_get_data( o, IDLE_DATA );
|
|
|
|
if( data == NULL )
|
|
{
|
|
data = g_new( struct spin_idle_data, 1 );
|
|
data->core = core;
|
|
data->last_change = g_timer_new( );
|
|
data->isDouble = isDouble;
|
|
g_object_set_data_full( o, IDLE_DATA, data, spin_idle_data_free );
|
|
g_object_ref( G_OBJECT( o ) );
|
|
gtr_timeout_add_seconds( 1, spun_cb_idle, w );
|
|
}
|
|
g_timer_start( data->last_change );
|
|
}
|
|
|
|
static void
|
|
spun_cb_int( GtkSpinButton * w,
|
|
gpointer core )
|
|
{
|
|
spun_cb( w, core, FALSE );
|
|
}
|
|
|
|
static void
|
|
spun_cb_double( GtkSpinButton * w,
|
|
gpointer core )
|
|
{
|
|
spun_cb( w, core, TRUE );
|
|
}
|
|
|
|
static GtkWidget*
|
|
new_spin_button( const char * key,
|
|
gpointer core,
|
|
int low,
|
|
int high,
|
|
int step )
|
|
{
|
|
GtkWidget * w = gtk_spin_button_new_with_range( low, high, step );
|
|
|
|
g_object_set_data_full( G_OBJECT( w ), PREF_KEY, g_strdup(
|
|
key ), g_free );
|
|
gtk_spin_button_set_digits( GTK_SPIN_BUTTON( w ), 0 );
|
|
gtk_spin_button_set_value( GTK_SPIN_BUTTON( w ), pref_int_get( key ) );
|
|
g_signal_connect( w, "value-changed", G_CALLBACK( spun_cb_int ), core );
|
|
return w;
|
|
}
|
|
|
|
static GtkWidget*
|
|
new_spin_button_double( const char * key,
|
|
gpointer core,
|
|
double low,
|
|
double high,
|
|
double step )
|
|
{
|
|
GtkWidget * w = gtk_spin_button_new_with_range( low, high, step );
|
|
|
|
g_object_set_data_full( G_OBJECT( w ), PREF_KEY, g_strdup(
|
|
key ), g_free );
|
|
gtk_spin_button_set_digits( GTK_SPIN_BUTTON( w ), 2 );
|
|
gtk_spin_button_set_value( GTK_SPIN_BUTTON( w ), pref_double_get( key ) );
|
|
g_signal_connect( w, "value-changed", G_CALLBACK( spun_cb_double ), core );
|
|
return w;
|
|
}
|
|
|
|
static void
|
|
entry_changed_cb( GtkEntry * w,
|
|
gpointer core )
|
|
{
|
|
const char * key = g_object_get_data( G_OBJECT( w ), PREF_KEY );
|
|
const char * value = gtk_entry_get_text( w );
|
|
|
|
tr_core_set_pref( TR_CORE( core ), key, value );
|
|
}
|
|
|
|
static GtkWidget*
|
|
new_entry( const char * key,
|
|
gpointer core )
|
|
{
|
|
GtkWidget * w = gtk_entry_new( );
|
|
const char * value = pref_string_get( key );
|
|
|
|
if( value )
|
|
gtk_entry_set_text( GTK_ENTRY( w ), value );
|
|
g_object_set_data_full( G_OBJECT( w ), PREF_KEY, g_strdup(
|
|
key ), g_free );
|
|
g_signal_connect( w, "changed", G_CALLBACK( entry_changed_cb ), core );
|
|
return w;
|
|
}
|
|
|
|
static void
|
|
chosen_cb( GtkFileChooser * w,
|
|
gpointer core )
|
|
{
|
|
const char * key = g_object_get_data( G_OBJECT( w ), PREF_KEY );
|
|
char * value = gtk_file_chooser_get_filename( GTK_FILE_CHOOSER( w ) );
|
|
|
|
tr_core_set_pref( TR_CORE( core ), key, value );
|
|
g_free( value );
|
|
}
|
|
|
|
static GtkWidget*
|
|
new_path_chooser_button( const char * key,
|
|
gpointer core )
|
|
{
|
|
GtkWidget * w = gtk_file_chooser_button_new(
|
|
NULL,
|
|
GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER );
|
|
const char * path = pref_string_get( key );
|
|
|
|
g_object_set_data_full( G_OBJECT( w ), PREF_KEY, g_strdup(
|
|
key ), g_free );
|
|
g_signal_connect( w, "selection-changed", G_CALLBACK( chosen_cb ), core );
|
|
gtk_file_chooser_set_current_folder( GTK_FILE_CHOOSER( w ), path );
|
|
return w;
|
|
}
|
|
|
|
static void
|
|
target_cb( GtkWidget * tb,
|
|
gpointer target )
|
|
{
|
|
const gboolean b = gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON( tb ) );
|
|
|
|
gtk_widget_set_sensitive( GTK_WIDGET( target ), b );
|
|
}
|
|
|
|
/****
|
|
***** Torrent Tab
|
|
****/
|
|
|
|
static GtkWidget*
|
|
torrentPage( GObject * core )
|
|
{
|
|
int row = 0;
|
|
const char * s;
|
|
GtkWidget * t;
|
|
GtkWidget * w;
|
|
GtkWidget * w2;
|
|
GtkWidget * l;
|
|
|
|
t = hig_workarea_create( );
|
|
hig_workarea_add_section_title( t, &row, _( "Adding Torrents" ) );
|
|
|
|
#ifdef HAVE_GIO
|
|
s = _( "Automatically _add torrents from:" );
|
|
l = new_check_button( s, PREF_KEY_DIR_WATCH_ENABLED, core );
|
|
w = new_path_chooser_button( PREF_KEY_DIR_WATCH, core );
|
|
gtk_widget_set_sensitive( GTK_WIDGET( w ),
|
|
pref_flag_get( PREF_KEY_DIR_WATCH_ENABLED ) );
|
|
g_signal_connect( l, "toggled", G_CALLBACK( target_cb ), w );
|
|
hig_workarea_add_row_w( t, &row, l, w, NULL );
|
|
#endif
|
|
|
|
s = _( "Show _options dialog" );
|
|
w = new_check_button( s, PREF_KEY_OPTIONS_PROMPT, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
s = _( "_Start when added" );
|
|
w = new_check_button( s, PREF_KEY_START, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
s = _( "Mo_ve .torrent file to the trash" );
|
|
w = new_check_button( s, PREF_KEY_TRASH_ORIGINAL, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
s = _( "Keep _incomplete torrents in:" );
|
|
l = new_check_button( s, TR_PREFS_KEY_INCOMPLETE_DIR_ENABLED, core );
|
|
w = new_path_chooser_button( TR_PREFS_KEY_INCOMPLETE_DIR, core );
|
|
gtk_widget_set_sensitive( GTK_WIDGET( w ), pref_flag_get( TR_PREFS_KEY_INCOMPLETE_DIR_ENABLED ) );
|
|
g_signal_connect( l, "toggled", G_CALLBACK( target_cb ), w );
|
|
hig_workarea_add_row_w( t, &row, l, w, NULL );
|
|
|
|
w = new_path_chooser_button( TR_PREFS_KEY_DOWNLOAD_DIR, core );
|
|
hig_workarea_add_row( t, &row, _( "Save to _Location:" ), w, NULL );
|
|
|
|
hig_workarea_add_section_divider( t, &row );
|
|
hig_workarea_add_section_title( t, &row, _( "Limits" ) );
|
|
|
|
s = _( "_Seed torrent until its ratio reaches:" );
|
|
w = new_check_button( s, TR_PREFS_KEY_RATIO_ENABLED, core );
|
|
w2 = new_spin_button_double( TR_PREFS_KEY_RATIO, core, 0, INT_MAX, .05 );
|
|
gtk_widget_set_sensitive( GTK_WIDGET( w2 ), pref_flag_get( TR_PREFS_KEY_RATIO_ENABLED ) );
|
|
g_signal_connect( w, "toggled", G_CALLBACK( target_cb ), w2 );
|
|
hig_workarea_add_row_w( t, &row, w, w2, NULL );
|
|
|
|
hig_workarea_finish( t, &row );
|
|
return t;
|
|
}
|
|
|
|
/****
|
|
***** Desktop Tab
|
|
****/
|
|
|
|
static GtkWidget*
|
|
desktopPage( GObject * core )
|
|
{
|
|
int row = 0;
|
|
const char * s;
|
|
GtkWidget * t;
|
|
GtkWidget * w;
|
|
|
|
t = hig_workarea_create( );
|
|
hig_workarea_add_section_title( t, &row, _( "Desktop" ) );
|
|
|
|
s = _( "Inhibit _hibernation when torrents are active" );
|
|
w = new_check_button( s, PREF_KEY_INHIBIT_HIBERNATION, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
s = _( "Show Transmission icon in the _notification area" );
|
|
w = new_check_button( s, PREF_KEY_SHOW_TRAY_ICON, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
s = _( "Show _popup notifications" );
|
|
w = new_check_button( s, PREF_KEY_SHOW_DESKTOP_NOTIFICATION, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
#ifdef HAVE_LIBCANBERRA
|
|
s = _( "Play _sound when downloads are complete" );
|
|
w = new_check_button( s, PREF_KEY_PLAY_DOWNLOAD_COMPLETE_SOUND, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
#endif
|
|
|
|
hig_workarea_finish( t, &row );
|
|
return t;
|
|
}
|
|
|
|
/****
|
|
***** Peer Tab
|
|
****/
|
|
|
|
struct blocklist_data
|
|
{
|
|
gulong updateBlocklistTag;
|
|
GtkWidget * updateBlocklistButton;
|
|
GtkWidget * updateBlocklistDialog;
|
|
GtkWidget * check;
|
|
TrCore * core;
|
|
};
|
|
|
|
static void
|
|
updateBlocklistText( GtkWidget * w, TrCore * core )
|
|
{
|
|
const int n = tr_blocklistGetRuleCount( tr_core_session( core ) );
|
|
char buf[512];
|
|
g_snprintf( buf, sizeof( buf ),
|
|
ngettext( "Enable _blocklist (contains %'d rule)",
|
|
"Enable _blocklist (contains %'d rules)", n ), n );
|
|
gtk_button_set_label( GTK_BUTTON( w ), buf );
|
|
}
|
|
|
|
/* prefs dialog is being destroyed, so stop listening to blocklist updates */
|
|
static void
|
|
privacyPageDestroyed( gpointer gdata, GObject * dead UNUSED )
|
|
{
|
|
struct blocklist_data * data = gdata;
|
|
if( data->updateBlocklistTag > 0 )
|
|
g_signal_handler_disconnect( data->core, data->updateBlocklistTag );
|
|
g_free( data );
|
|
}
|
|
|
|
/* user hit "close" in the blocklist-update dialog */
|
|
static void
|
|
onBlocklistUpdateResponse( GtkDialog * dialog, gint response UNUSED, gpointer gdata )
|
|
{
|
|
struct blocklist_data * data = gdata;
|
|
gtk_widget_destroy( GTK_WIDGET( dialog ) );
|
|
gtk_widget_set_sensitive( data->updateBlocklistButton, TRUE );
|
|
data->updateBlocklistDialog = NULL;
|
|
g_signal_handler_disconnect( data->core, data->updateBlocklistTag );
|
|
}
|
|
|
|
/* core says the blocklist was updated */
|
|
static void
|
|
onBlocklistUpdated( TrCore * core, int n, gpointer gdata )
|
|
{
|
|
const char * s = ngettext( "Blocklist now has %'d rule.", "Blocklist now has %'d rules.", n );
|
|
struct blocklist_data * data = gdata;
|
|
GtkMessageDialog * d = GTK_MESSAGE_DIALOG( data->updateBlocklistDialog );
|
|
gtk_widget_set_sensitive( data->updateBlocklistButton, TRUE );
|
|
gtk_message_dialog_set_markup( d, _( "<b>Update succeeded!</b>" ) );
|
|
gtk_message_dialog_format_secondary_text( d, s, n );
|
|
updateBlocklistText( data->check, core );
|
|
}
|
|
|
|
/* user pushed a button to update the blocklist */
|
|
static void
|
|
onBlocklistUpdate( GtkButton * w, gpointer gdata )
|
|
{
|
|
GtkWidget * d;
|
|
struct blocklist_data * data = gdata;
|
|
d = gtk_message_dialog_new( GTK_WINDOW( gtk_widget_get_toplevel( GTK_WIDGET( w ) ) ),
|
|
GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
GTK_MESSAGE_INFO,
|
|
GTK_BUTTONS_CLOSE,
|
|
"%s", _( "Update Blocklist" ) );
|
|
gtk_widget_set_sensitive( data->updateBlocklistButton, FALSE );
|
|
gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG( d ), "%s", _( "Getting new blocklist..." ) );
|
|
data->updateBlocklistDialog = d;
|
|
g_signal_connect( d, "response", G_CALLBACK(onBlocklistUpdateResponse), data );
|
|
gtk_widget_show( d );
|
|
tr_core_blocklist_update( data->core );
|
|
data->updateBlocklistTag = g_signal_connect( data->core, "blocklist-updated", G_CALLBACK( onBlocklistUpdated ), data );
|
|
}
|
|
|
|
static void
|
|
onIntComboChanged( GtkComboBox * w, gpointer core )
|
|
{
|
|
GtkTreeIter iter;
|
|
|
|
if( gtk_combo_box_get_active_iter( w, &iter ) )
|
|
{
|
|
int val = 0;
|
|
const char * key = g_object_get_data( G_OBJECT( w ), PREF_KEY );
|
|
gtk_tree_model_get( gtk_combo_box_get_model( w ), &iter, 0, &val, -1 );
|
|
tr_core_set_pref_int( TR_CORE( core ), key, val );
|
|
}
|
|
}
|
|
|
|
static GtkWidget*
|
|
new_encryption_combo( GObject * core, const char * key )
|
|
{
|
|
int i;
|
|
int selIndex;
|
|
GtkWidget * w;
|
|
GtkCellRenderer * r;
|
|
GtkListStore * store;
|
|
const int currentValue = pref_int_get( key );
|
|
const struct {
|
|
int value;
|
|
const char * text;
|
|
} items[] = {
|
|
{ TR_CLEAR_PREFERRED, N_( "Allow encryption" ) },
|
|
{ TR_ENCRYPTION_PREFERRED, N_( "Prefer encryption" ) },
|
|
{ TR_ENCRYPTION_REQUIRED, N_( "Require encryption" ) }
|
|
};
|
|
|
|
/* build a store for encryption */
|
|
selIndex = -1;
|
|
store = gtk_list_store_new( 2, G_TYPE_INT, G_TYPE_STRING );
|
|
for( i=0; i<(int)G_N_ELEMENTS(items); ++i ) {
|
|
GtkTreeIter iter;
|
|
gtk_list_store_append( store, &iter );
|
|
gtk_list_store_set( store, &iter, 0, items[i].value, 1, _( items[i].text ), -1 );
|
|
if( items[i].value == currentValue )
|
|
selIndex = i;
|
|
}
|
|
|
|
/* build the widget */
|
|
w = gtk_combo_box_new_with_model( GTK_TREE_MODEL( store ) );
|
|
r = gtk_cell_renderer_text_new( );
|
|
gtk_cell_layout_pack_start( GTK_CELL_LAYOUT( w ), r, TRUE );
|
|
gtk_cell_layout_set_attributes( GTK_CELL_LAYOUT( w ), r, "text", 1, NULL );
|
|
g_object_set_data_full( G_OBJECT( w ), PREF_KEY, tr_strdup( key ), g_free );
|
|
if( selIndex >= 0 )
|
|
gtk_combo_box_set_active( GTK_COMBO_BOX( w ), selIndex );
|
|
g_signal_connect( w, "changed", G_CALLBACK( onIntComboChanged ), core );
|
|
|
|
/* cleanup */
|
|
g_object_unref( G_OBJECT( store ) );
|
|
return w;
|
|
}
|
|
|
|
static GtkWidget*
|
|
privacyPage( GObject * core )
|
|
{
|
|
int row = 0;
|
|
const char * s;
|
|
GtkWidget * t;
|
|
GtkWidget * w;
|
|
GtkWidget * b;
|
|
GtkWidget * h;
|
|
struct blocklist_data * data;
|
|
|
|
data = g_new0( struct blocklist_data, 1 );
|
|
data->core = TR_CORE( core );
|
|
|
|
t = hig_workarea_create( );
|
|
hig_workarea_add_section_title( t, &row, _( "Blocklist" ) );
|
|
|
|
w = new_check_button( "", TR_PREFS_KEY_BLOCKLIST_ENABLED, core );
|
|
updateBlocklistText( w, TR_CORE( core ) );
|
|
h = gtk_hbox_new( FALSE, GUI_PAD_BIG );
|
|
gtk_box_pack_start( GTK_BOX( h ), w, TRUE, TRUE, 0 );
|
|
b = data->updateBlocklistButton = gtr_button_new_from_stock( GTK_STOCK_REFRESH, _( "_Update" ) );
|
|
data->check = w;
|
|
g_object_set_data( G_OBJECT( b ), "session",
|
|
tr_core_session( TR_CORE( core ) ) );
|
|
g_signal_connect( b, "clicked", G_CALLBACK( onBlocklistUpdate ), data );
|
|
gtk_box_pack_start( GTK_BOX( h ), b, FALSE, FALSE, 0 );
|
|
g_signal_connect( w, "toggled", G_CALLBACK( target_cb ), b );
|
|
target_cb( w, b );
|
|
hig_workarea_add_wide_control( t, &row, h );
|
|
|
|
s = _( "Enable _automatic updates" );
|
|
w = new_check_button( s, PREF_KEY_BLOCKLIST_UPDATES_ENABLED, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
g_signal_connect( data->check, "toggled", G_CALLBACK( target_cb ), w );
|
|
target_cb( data->check, w );
|
|
|
|
hig_workarea_add_section_divider( t, &row );
|
|
hig_workarea_add_section_title ( t, &row, _( "Privacy" ) );
|
|
|
|
s = _( "_Encryption mode:" );
|
|
w = new_encryption_combo( core, "encryption" );
|
|
hig_workarea_add_row( t, &row, s, w, NULL );
|
|
|
|
s = _( "Use PE_X to find more peers" );
|
|
w = new_check_button( s, TR_PREFS_KEY_PEX_ENABLED, core );
|
|
s = _( "PEX is a tool for exchanging peer lists with the peers you're connected to." );
|
|
gtr_widget_set_tooltip_text( w, s );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
s = _( "Use _DHT to find more peers" );
|
|
w = new_check_button( s, TR_PREFS_KEY_DHT_ENABLED, core );
|
|
s = _( "DHT is a tool for finding peers without a tracker." );
|
|
gtr_widget_set_tooltip_text( w, s );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
hig_workarea_finish( t, &row );
|
|
g_object_weak_ref( G_OBJECT( t ), privacyPageDestroyed, data );
|
|
return t;
|
|
}
|
|
|
|
/****
|
|
***** Web Tab
|
|
****/
|
|
|
|
enum
|
|
{
|
|
COL_ADDRESS,
|
|
N_COLS
|
|
};
|
|
|
|
static GtkTreeModel*
|
|
whitelist_tree_model_new( const char * whitelist )
|
|
{
|
|
int i;
|
|
char ** rules;
|
|
GtkListStore * store = gtk_list_store_new( N_COLS,
|
|
G_TYPE_STRING,
|
|
G_TYPE_STRING );
|
|
|
|
rules = g_strsplit( whitelist, ",", 0 );
|
|
|
|
for( i = 0; rules && rules[i]; ++i )
|
|
{
|
|
GtkTreeIter iter;
|
|
const char * s = rules[i];
|
|
while( isspace( *s ) ) ++s;
|
|
gtk_list_store_append( store, &iter );
|
|
gtk_list_store_set( store, &iter, COL_ADDRESS, s, -1 );
|
|
}
|
|
|
|
g_strfreev( rules );
|
|
return GTK_TREE_MODEL( store );
|
|
}
|
|
|
|
struct remote_page
|
|
{
|
|
TrCore * core;
|
|
GtkTreeView * view;
|
|
GtkListStore * store;
|
|
GtkWidget * remove_button;
|
|
GSList * widgets;
|
|
GSList * auth_widgets;
|
|
GSList * whitelist_widgets;
|
|
GtkToggleButton * rpc_tb;
|
|
GtkToggleButton * auth_tb;
|
|
GtkToggleButton * whitelist_tb;
|
|
};
|
|
|
|
static void
|
|
refreshWhitelist( struct remote_page * page )
|
|
{
|
|
GtkTreeIter iter;
|
|
GtkTreeModel * model = GTK_TREE_MODEL( page->store );
|
|
GString * gstr = g_string_new( NULL );
|
|
|
|
if( gtk_tree_model_get_iter_first( model, &iter ) ) do
|
|
{
|
|
char * address;
|
|
gtk_tree_model_get( model, &iter,
|
|
COL_ADDRESS, &address,
|
|
-1 );
|
|
g_string_append( gstr, address );
|
|
g_string_append( gstr, "," );
|
|
g_free( address );
|
|
}
|
|
while( gtk_tree_model_iter_next( model, &iter ) );
|
|
|
|
g_string_truncate( gstr, gstr->len - 1 ); /* remove the trailing comma */
|
|
|
|
tr_core_set_pref( page->core, TR_PREFS_KEY_RPC_WHITELIST, gstr->str );
|
|
|
|
g_string_free( gstr, TRUE );
|
|
}
|
|
|
|
static void
|
|
onAddressEdited( GtkCellRendererText * r UNUSED,
|
|
gchar * path_string,
|
|
gchar * address,
|
|
gpointer gpage )
|
|
{
|
|
GtkTreeIter iter;
|
|
struct remote_page * page = gpage;
|
|
GtkTreeModel * model = GTK_TREE_MODEL( page->store );
|
|
GtkTreePath * path = gtk_tree_path_new_from_string( path_string );
|
|
|
|
if( gtk_tree_model_get_iter( model, &iter, path ) )
|
|
gtk_list_store_set( page->store, &iter, COL_ADDRESS, address, -1 );
|
|
|
|
gtk_tree_path_free( path );
|
|
refreshWhitelist( page );
|
|
}
|
|
|
|
static void
|
|
onAddWhitelistClicked( GtkButton * b UNUSED,
|
|
gpointer gpage )
|
|
{
|
|
GtkTreeIter iter;
|
|
GtkTreePath * path;
|
|
struct remote_page * page = gpage;
|
|
|
|
gtk_list_store_append( page->store, &iter );
|
|
gtk_list_store_set( page->store, &iter,
|
|
COL_ADDRESS, "0.0.0.0",
|
|
-1 );
|
|
|
|
path = gtk_tree_model_get_path( GTK_TREE_MODEL( page->store ), &iter );
|
|
gtk_tree_view_set_cursor(
|
|
page->view, path,
|
|
gtk_tree_view_get_column( page->view, COL_ADDRESS ),
|
|
TRUE );
|
|
gtk_tree_path_free( path );
|
|
}
|
|
|
|
static void
|
|
onRemoveWhitelistClicked( GtkButton * b UNUSED,
|
|
gpointer gpage )
|
|
{
|
|
struct remote_page * page = gpage;
|
|
GtkTreeSelection * sel = gtk_tree_view_get_selection( page->view );
|
|
GtkTreeIter iter;
|
|
|
|
if( gtk_tree_selection_get_selected( sel, NULL, &iter ) )
|
|
{
|
|
gtk_list_store_remove( page->store, &iter );
|
|
refreshWhitelist( page );
|
|
}
|
|
}
|
|
|
|
static void
|
|
refreshRPCSensitivity( struct remote_page * page )
|
|
{
|
|
GSList * l;
|
|
const int rpc_active = gtk_toggle_button_get_active(
|
|
page->rpc_tb );
|
|
const int auth_active = gtk_toggle_button_get_active(
|
|
page->auth_tb );
|
|
const int whitelist_active = gtk_toggle_button_get_active(
|
|
page->whitelist_tb );
|
|
GtkTreeSelection * sel = gtk_tree_view_get_selection( page->view );
|
|
const int have_addr =
|
|
gtk_tree_selection_get_selected( sel, NULL,
|
|
NULL );
|
|
const int n_rules = gtk_tree_model_iter_n_children(
|
|
GTK_TREE_MODEL( page->store ), NULL );
|
|
|
|
for( l = page->widgets; l != NULL; l = l->next )
|
|
gtk_widget_set_sensitive( GTK_WIDGET( l->data ), rpc_active );
|
|
|
|
for( l = page->auth_widgets; l != NULL; l = l->next )
|
|
gtk_widget_set_sensitive( GTK_WIDGET(
|
|
l->data ), rpc_active && auth_active );
|
|
|
|
for( l = page->whitelist_widgets; l != NULL; l = l->next )
|
|
gtk_widget_set_sensitive( GTK_WIDGET( l->data ),
|
|
rpc_active && whitelist_active );
|
|
|
|
gtk_widget_set_sensitive( page->remove_button,
|
|
rpc_active && have_addr && n_rules > 1 );
|
|
}
|
|
|
|
static void
|
|
onRPCToggled( GtkToggleButton * tb UNUSED,
|
|
gpointer page )
|
|
{
|
|
refreshRPCSensitivity( page );
|
|
}
|
|
|
|
static void
|
|
onWhitelistSelectionChanged( GtkTreeSelection * sel UNUSED,
|
|
gpointer page )
|
|
{
|
|
refreshRPCSensitivity( page );
|
|
}
|
|
|
|
static void
|
|
onLaunchClutchCB( GtkButton * w UNUSED,
|
|
gpointer data UNUSED )
|
|
{
|
|
int port = pref_int_get( TR_PREFS_KEY_RPC_PORT );
|
|
char * url = g_strdup_printf( "http://localhost:%d/transmission/web",
|
|
port );
|
|
|
|
gtr_open_file( url );
|
|
g_free( url );
|
|
}
|
|
|
|
static GtkWidget*
|
|
webPage( GObject * core )
|
|
{
|
|
const char * s;
|
|
int row = 0;
|
|
GtkWidget * t;
|
|
GtkWidget * w;
|
|
GtkWidget * h;
|
|
struct remote_page * page = g_new0( struct remote_page, 1 );
|
|
|
|
page->core = TR_CORE( core );
|
|
|
|
t = hig_workarea_create( );
|
|
g_object_set_data_full( G_OBJECT( t ), "page", page, g_free );
|
|
|
|
hig_workarea_add_section_title( t, &row, _( "Web Client" ) );
|
|
|
|
/* "enabled" checkbutton */
|
|
s = _( "_Enable web client" );
|
|
w = new_check_button( s, TR_PREFS_KEY_RPC_ENABLED, core );
|
|
page->rpc_tb = GTK_TOGGLE_BUTTON( w );
|
|
g_signal_connect( w, "clicked", G_CALLBACK( onRPCToggled ), page );
|
|
h = gtk_hbox_new( FALSE, GUI_PAD_BIG );
|
|
gtk_box_pack_start( GTK_BOX( h ), w, TRUE, TRUE, 0 );
|
|
w = gtr_button_new_from_stock( GTK_STOCK_OPEN, _( "_Open web client" ) );
|
|
page->widgets = g_slist_append( page->widgets, w );
|
|
g_signal_connect( w, "clicked", G_CALLBACK( onLaunchClutchCB ), NULL );
|
|
gtk_box_pack_start( GTK_BOX( h ), w, FALSE, FALSE, 0 );
|
|
hig_workarea_add_wide_control( t, &row, h );
|
|
|
|
/* port */
|
|
w = new_spin_button( TR_PREFS_KEY_RPC_PORT, core, 0, USHRT_MAX, 1 );
|
|
page->widgets = g_slist_append( page->widgets, w );
|
|
w = hig_workarea_add_row( t, &row, _( "Listening _port:" ), w, NULL );
|
|
page->widgets = g_slist_append( page->widgets, w );
|
|
|
|
/* require authentication */
|
|
s = _( "Use _authentication" );
|
|
w = new_check_button( s, TR_PREFS_KEY_RPC_AUTH_REQUIRED, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
page->auth_tb = GTK_TOGGLE_BUTTON( w );
|
|
page->widgets = g_slist_append( page->widgets, w );
|
|
g_signal_connect( w, "clicked", G_CALLBACK( onRPCToggled ), page );
|
|
|
|
/* username */
|
|
s = _( "_Username:" );
|
|
w = new_entry( TR_PREFS_KEY_RPC_USERNAME, core );
|
|
page->auth_widgets = g_slist_append( page->auth_widgets, w );
|
|
w = hig_workarea_add_row( t, &row, s, w, NULL );
|
|
page->auth_widgets = g_slist_append( page->auth_widgets, w );
|
|
|
|
/* password */
|
|
s = _( "Pass_word:" );
|
|
w = new_entry( TR_PREFS_KEY_RPC_PASSWORD, core );
|
|
gtk_entry_set_visibility( GTK_ENTRY( w ), FALSE );
|
|
page->auth_widgets = g_slist_append( page->auth_widgets, w );
|
|
w = hig_workarea_add_row( t, &row, s, w, NULL );
|
|
page->auth_widgets = g_slist_append( page->auth_widgets, w );
|
|
|
|
/* require authentication */
|
|
s = _( "Only allow these IP a_ddresses to connect:" );
|
|
w = new_check_button( s, TR_PREFS_KEY_RPC_WHITELIST_ENABLED, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
page->whitelist_tb = GTK_TOGGLE_BUTTON( w );
|
|
page->widgets = g_slist_append( page->widgets, w );
|
|
g_signal_connect( w, "clicked", G_CALLBACK( onRPCToggled ), page );
|
|
|
|
/* access control list */
|
|
{
|
|
const char * val = pref_string_get( TR_PREFS_KEY_RPC_WHITELIST );
|
|
GtkTreeModel * m = whitelist_tree_model_new( val );
|
|
GtkTreeViewColumn * c;
|
|
GtkCellRenderer * r;
|
|
GtkTreeSelection * sel;
|
|
GtkTreeView * v;
|
|
GtkWidget * w;
|
|
GtkWidget * h;
|
|
|
|
page->store = GTK_LIST_STORE( m );
|
|
w = gtk_tree_view_new_with_model( m );
|
|
g_signal_connect( w, "button-release-event",
|
|
G_CALLBACK( on_tree_view_button_released ), NULL );
|
|
|
|
page->whitelist_widgets = g_slist_append( page->whitelist_widgets, w );
|
|
v = page->view = GTK_TREE_VIEW( w );
|
|
gtr_widget_set_tooltip_text( w, _( "IP addresses may use wildcards, such as 192.168.*.*" ) );
|
|
sel = gtk_tree_view_get_selection( v );
|
|
g_signal_connect( sel, "changed",
|
|
G_CALLBACK( onWhitelistSelectionChanged ), page );
|
|
g_object_unref( G_OBJECT( m ) );
|
|
gtk_tree_view_set_headers_visible( v, TRUE );
|
|
w = gtk_frame_new( NULL );
|
|
gtk_frame_set_shadow_type( GTK_FRAME( w ), GTK_SHADOW_IN );
|
|
gtk_container_add( GTK_CONTAINER( w ), GTK_WIDGET( v ) );
|
|
|
|
/* ip address column */
|
|
r = gtk_cell_renderer_text_new( );
|
|
g_signal_connect( r, "edited",
|
|
G_CALLBACK( onAddressEdited ), page );
|
|
g_object_set( G_OBJECT( r ), "editable", TRUE, NULL );
|
|
c = gtk_tree_view_column_new_with_attributes( NULL, r,
|
|
"text", COL_ADDRESS,
|
|
NULL );
|
|
gtk_tree_view_column_set_expand( c, TRUE );
|
|
gtk_tree_view_append_column( v, c );
|
|
gtk_tree_view_set_headers_visible( v, FALSE );
|
|
|
|
s = _( "Addresses:" );
|
|
w = hig_workarea_add_row( t, &row, s, w, NULL );
|
|
gtk_misc_set_alignment( GTK_MISC( w ), 0.0f, 0.0f );
|
|
gtk_misc_set_padding( GTK_MISC( w ), 0, GUI_PAD );
|
|
page->whitelist_widgets = g_slist_append( page->whitelist_widgets, w );
|
|
|
|
h = gtk_hbox_new( TRUE, GUI_PAD );
|
|
w = gtk_button_new_from_stock( GTK_STOCK_REMOVE );
|
|
g_signal_connect( w, "clicked", G_CALLBACK(
|
|
onRemoveWhitelistClicked ), page );
|
|
page->remove_button = w;
|
|
onWhitelistSelectionChanged( sel, page );
|
|
gtk_box_pack_start( GTK_BOX( h ), w, TRUE, TRUE, 0 );
|
|
w = gtk_button_new_from_stock( GTK_STOCK_ADD );
|
|
page->whitelist_widgets = g_slist_append( page->whitelist_widgets, w );
|
|
g_signal_connect( w, "clicked", G_CALLBACK( onAddWhitelistClicked ), page );
|
|
gtk_box_pack_start( GTK_BOX( h ), w, TRUE, TRUE, 0 );
|
|
w = gtk_hbox_new( FALSE, 0 );
|
|
gtk_box_pack_start( GTK_BOX( w ), gtk_alignment_new( 0, 0, 0, 0 ),
|
|
TRUE, TRUE, 0 );
|
|
gtk_box_pack_start( GTK_BOX( w ), h, FALSE, FALSE, 0 );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
}
|
|
|
|
refreshRPCSensitivity( page );
|
|
hig_workarea_finish( t, &row );
|
|
return t;
|
|
}
|
|
|
|
/****
|
|
***** Proxy Tab
|
|
****/
|
|
|
|
struct ProxyPage
|
|
{
|
|
TrCore * core;
|
|
GSList * proxy_widgets;
|
|
GSList * proxy_auth_widgets;
|
|
};
|
|
|
|
static void
|
|
refreshProxySensitivity( struct ProxyPage * p )
|
|
{
|
|
GSList * l;
|
|
const gboolean proxy_enabled = pref_flag_get( TR_PREFS_KEY_PROXY_ENABLED );
|
|
const gboolean proxy_auth_enabled = pref_flag_get( TR_PREFS_KEY_PROXY_AUTH_ENABLED );
|
|
|
|
for( l = p->proxy_widgets; l != NULL; l = l->next )
|
|
gtk_widget_set_sensitive( GTK_WIDGET( l->data ), proxy_enabled );
|
|
|
|
for( l = p->proxy_auth_widgets; l != NULL; l = l->next )
|
|
gtk_widget_set_sensitive( GTK_WIDGET( l->data ),
|
|
proxy_enabled && proxy_auth_enabled );
|
|
}
|
|
|
|
static void
|
|
onProxyToggled( GtkToggleButton * tb UNUSED,
|
|
gpointer user_data )
|
|
{
|
|
refreshProxySensitivity( user_data );
|
|
}
|
|
|
|
static void
|
|
proxyPageFree( gpointer gpage )
|
|
{
|
|
struct ProxyPage * page = gpage;
|
|
|
|
g_slist_free( page->proxy_widgets );
|
|
g_slist_free( page->proxy_auth_widgets );
|
|
g_free( page );
|
|
}
|
|
|
|
static GtkTreeModel*
|
|
proxyTypeModelNew( void )
|
|
{
|
|
GtkTreeIter iter;
|
|
GtkListStore * store = gtk_list_store_new( 2, G_TYPE_STRING, G_TYPE_INT );
|
|
|
|
gtk_list_store_append( store, &iter );
|
|
gtk_list_store_set( store, &iter, 0, "HTTP", 1, TR_PROXY_HTTP, -1 );
|
|
gtk_list_store_append( store, &iter );
|
|
gtk_list_store_set( store, &iter, 0, "SOCKS4", 1, TR_PROXY_SOCKS4, -1 );
|
|
gtk_list_store_append( store, &iter );
|
|
gtk_list_store_set( store, &iter, 0, "SOCKS5", 1, TR_PROXY_SOCKS5, -1 );
|
|
return GTK_TREE_MODEL( store );
|
|
}
|
|
|
|
static void
|
|
onProxyTypeChanged( GtkComboBox * w,
|
|
gpointer gpage )
|
|
{
|
|
GtkTreeIter iter;
|
|
|
|
if( gtk_combo_box_get_active_iter( w, &iter ) )
|
|
{
|
|
struct ProxyPage * page = gpage;
|
|
int type = TR_PROXY_HTTP;
|
|
gtk_tree_model_get( gtk_combo_box_get_model( w ), &iter, 1, &type, -1 );
|
|
tr_core_set_pref_int( TR_CORE( page->core ), TR_PREFS_KEY_PROXY_TYPE, type );
|
|
}
|
|
}
|
|
|
|
static GtkWidget*
|
|
trackerPage( GObject * core )
|
|
{
|
|
int row = 0;
|
|
const char * s;
|
|
GtkWidget * t;
|
|
GtkWidget * w;
|
|
GtkTreeModel * m;
|
|
GtkCellRenderer * r;
|
|
struct ProxyPage * page = tr_new0( struct ProxyPage, 1 );
|
|
|
|
page->core = TR_CORE( core );
|
|
|
|
t = hig_workarea_create( );
|
|
hig_workarea_add_section_title ( t, &row, _( "Tracker" ) );
|
|
|
|
s = _( "Connect to tracker via a pro_xy" );
|
|
w = new_check_button( s, TR_PREFS_KEY_PROXY_ENABLED, core );
|
|
g_signal_connect( w, "toggled", G_CALLBACK( onProxyToggled ), page );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
s = _( "Proxy _server:" );
|
|
w = new_entry( TR_PREFS_KEY_PROXY, core );
|
|
page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
|
|
w = hig_workarea_add_row( t, &row, s, w, NULL );
|
|
page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
|
|
|
|
w = new_spin_button( TR_PREFS_KEY_PROXY_PORT, core, 0, USHRT_MAX, 1 );
|
|
page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
|
|
w = hig_workarea_add_row( t, &row, _( "Proxy _port:" ), w, NULL );
|
|
page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
|
|
|
|
s = _( "Proxy _type:" );
|
|
m = proxyTypeModelNew( );
|
|
w = gtk_combo_box_new_with_model( m );
|
|
r = gtk_cell_renderer_text_new( );
|
|
gtk_cell_layout_pack_start( GTK_CELL_LAYOUT( w ), r, TRUE );
|
|
gtk_cell_layout_set_attributes( GTK_CELL_LAYOUT( w ), r, "text", 0, NULL );
|
|
gtk_combo_box_set_active( GTK_COMBO_BOX( w ), pref_int_get( TR_PREFS_KEY_PROXY_TYPE ) );
|
|
g_signal_connect( w, "changed", G_CALLBACK( onProxyTypeChanged ), page );
|
|
g_object_unref( G_OBJECT( m ) );
|
|
page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
|
|
w = hig_workarea_add_row( t, &row, s, w, NULL );
|
|
page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
|
|
|
|
s = _( "Use _authentication" );
|
|
w = new_check_button( s, TR_PREFS_KEY_PROXY_AUTH_ENABLED, core );
|
|
g_signal_connect( w, "toggled", G_CALLBACK( onProxyToggled ), page );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
page->proxy_widgets = g_slist_append( page->proxy_widgets, w );
|
|
|
|
s = _( "_Username:" );
|
|
w = new_entry( TR_PREFS_KEY_PROXY_USERNAME, core );
|
|
page->proxy_auth_widgets = g_slist_append( page->proxy_auth_widgets, w );
|
|
w = hig_workarea_add_row( t, &row, s, w, NULL );
|
|
page->proxy_auth_widgets = g_slist_append( page->proxy_auth_widgets, w );
|
|
|
|
s = _( "Pass_word:" );
|
|
w = new_entry( TR_PREFS_KEY_RPC_PASSWORD, core );
|
|
gtk_entry_set_visibility( GTK_ENTRY( w ), FALSE );
|
|
page->proxy_auth_widgets = g_slist_append( page->proxy_auth_widgets, w );
|
|
w = hig_workarea_add_row( t, &row, s, w, NULL );
|
|
page->proxy_auth_widgets = g_slist_append( page->proxy_auth_widgets, w );
|
|
|
|
hig_workarea_finish( t, &row );
|
|
g_object_set_data_full( G_OBJECT( t ), "page", page, proxyPageFree );
|
|
|
|
refreshProxySensitivity( page );
|
|
return t;
|
|
}
|
|
|
|
/****
|
|
***** Bandwidth Tab
|
|
****/
|
|
|
|
struct BandwidthPage
|
|
{
|
|
TrCore * core;
|
|
GSList * sched_widgets;
|
|
};
|
|
|
|
static void
|
|
refreshSchedSensitivity( struct BandwidthPage * p )
|
|
{
|
|
GSList * l;
|
|
const gboolean sched_enabled = pref_flag_get( TR_PREFS_KEY_ALT_SPEED_TIME_ENABLED );
|
|
|
|
for( l = p->sched_widgets; l != NULL; l = l->next )
|
|
gtk_widget_set_sensitive( GTK_WIDGET( l->data ), sched_enabled );
|
|
}
|
|
|
|
static void
|
|
onSchedToggled( GtkToggleButton * tb UNUSED,
|
|
gpointer user_data )
|
|
{
|
|
refreshSchedSensitivity( user_data );
|
|
}
|
|
|
|
static void
|
|
onTimeComboChanged( GtkComboBox * w,
|
|
gpointer core )
|
|
{
|
|
GtkTreeIter iter;
|
|
|
|
if( gtk_combo_box_get_active_iter( w, &iter ) )
|
|
{
|
|
const char * key = g_object_get_data( G_OBJECT( w ), PREF_KEY );
|
|
int val = 0;
|
|
gtk_tree_model_get( gtk_combo_box_get_model(
|
|
w ), &iter, 0, &val, -1 );
|
|
tr_core_set_pref_int( TR_CORE( core ), key, val );
|
|
}
|
|
}
|
|
|
|
static GtkWidget*
|
|
new_time_combo( GObject * core,
|
|
const char * key )
|
|
{
|
|
int val;
|
|
int i;
|
|
GtkWidget * w;
|
|
GtkCellRenderer * r;
|
|
GtkListStore * store;
|
|
|
|
/* build a store at 15 minute intervals */
|
|
store = gtk_list_store_new( 2, G_TYPE_INT, G_TYPE_STRING );
|
|
for( i = 0; i < 60 * 24; i += 15 )
|
|
{
|
|
char buf[128];
|
|
GtkTreeIter iter;
|
|
struct tm tm;
|
|
tm.tm_hour = i / 60;
|
|
tm.tm_min = i % 60;
|
|
tm.tm_sec = 0;
|
|
strftime( buf, sizeof( buf ), "%H:%M", &tm );
|
|
gtk_list_store_append( store, &iter );
|
|
gtk_list_store_set( store, &iter, 0, i, 1, buf, -1 );
|
|
}
|
|
|
|
/* build the widget */
|
|
w = gtk_combo_box_new_with_model( GTK_TREE_MODEL( store ) );
|
|
gtk_combo_box_set_wrap_width( GTK_COMBO_BOX( w ), 4 );
|
|
r = gtk_cell_renderer_text_new( );
|
|
gtk_cell_layout_pack_start( GTK_CELL_LAYOUT( w ), r, TRUE );
|
|
gtk_cell_layout_set_attributes( GTK_CELL_LAYOUT(
|
|
w ), r, "text", 1, NULL );
|
|
g_object_set_data_full( G_OBJECT( w ), PREF_KEY, tr_strdup(
|
|
key ), g_free );
|
|
val = pref_int_get( key );
|
|
gtk_combo_box_set_active( GTK_COMBO_BOX( w ), val / ( 15 ) );
|
|
g_signal_connect( w, "changed", G_CALLBACK( onTimeComboChanged ), core );
|
|
|
|
/* cleanup */
|
|
g_object_unref( G_OBJECT( store ) );
|
|
return w;
|
|
}
|
|
|
|
static GtkWidget*
|
|
new_week_combo( GObject * core, const char * key )
|
|
{
|
|
int i;
|
|
int selIndex;
|
|
GtkWidget * w;
|
|
GtkCellRenderer * r;
|
|
GtkListStore * store;
|
|
const int currentValue = pref_int_get( key );
|
|
const struct {
|
|
int value;
|
|
const char * text;
|
|
} items[] = {
|
|
{ TR_SCHED_ALL, N_( "Every Day" ) },
|
|
{ TR_SCHED_WEEKDAY, N_( "Weekdays" ) },
|
|
{ TR_SCHED_WEEKEND, N_( "Weekends" ) },
|
|
{ TR_SCHED_SUN, N_( "Sunday" ) },
|
|
{ TR_SCHED_MON, N_( "Monday" ) },
|
|
{ TR_SCHED_TUES, N_( "Tuesday" ) },
|
|
{ TR_SCHED_WED, N_( "Wednesday" ) },
|
|
{ TR_SCHED_THURS, N_( "Thursday" ) },
|
|
{ TR_SCHED_FRI, N_( "Friday" ) },
|
|
{ TR_SCHED_SAT, N_( "Saturday" ) }
|
|
};
|
|
|
|
/* build a store for the days of the week */
|
|
selIndex = -1;
|
|
store = gtk_list_store_new( 2, G_TYPE_INT, G_TYPE_STRING );
|
|
for( i=0; i<(int)G_N_ELEMENTS(items); ++i ) {
|
|
GtkTreeIter iter;
|
|
gtk_list_store_append( store, &iter );
|
|
gtk_list_store_set( store, &iter, 0, items[i].value, 1, _( items[i].text ), -1 );
|
|
if( items[i].value == currentValue )
|
|
selIndex = i;
|
|
}
|
|
|
|
/* build the widget */
|
|
w = gtk_combo_box_new_with_model( GTK_TREE_MODEL( store ) );
|
|
r = gtk_cell_renderer_text_new( );
|
|
gtk_cell_layout_pack_start( GTK_CELL_LAYOUT( w ), r, TRUE );
|
|
gtk_cell_layout_set_attributes( GTK_CELL_LAYOUT( w ), r, "text", 1, NULL );
|
|
g_object_set_data_full( G_OBJECT( w ), PREF_KEY, tr_strdup( key ), g_free );
|
|
if( selIndex >= 0 )
|
|
gtk_combo_box_set_active( GTK_COMBO_BOX( w ), selIndex );
|
|
g_signal_connect( w, "changed", G_CALLBACK( onIntComboChanged ), core );
|
|
|
|
/* cleanup */
|
|
g_object_unref( G_OBJECT( store ) );
|
|
return w;
|
|
}
|
|
|
|
static void
|
|
bandwidthPageFree( gpointer gpage )
|
|
{
|
|
struct BandwidthPage * page = gpage;
|
|
|
|
g_slist_free( page->sched_widgets );
|
|
g_free( page );
|
|
}
|
|
|
|
static GtkWidget*
|
|
bandwidthPage( GObject * core )
|
|
{
|
|
int row = 0;
|
|
const char * s;
|
|
GtkWidget * t;
|
|
GtkWidget * l;
|
|
GtkWidget * w, * w2, * h;
|
|
char buf[512];
|
|
struct BandwidthPage * page = tr_new0( struct BandwidthPage, 1 );
|
|
|
|
page->core = TR_CORE( core );
|
|
|
|
t = hig_workarea_create( );
|
|
hig_workarea_add_section_title( t, &row, _( "Speed Limits" ) );
|
|
|
|
s = _( "Limit _download speed (KB/s):" );
|
|
w = new_check_button( s, TR_PREFS_KEY_DSPEED_ENABLED, core );
|
|
w2 = new_spin_button( TR_PREFS_KEY_DSPEED, core, 0, INT_MAX, 5 );
|
|
gtk_widget_set_sensitive( GTK_WIDGET( w2 ), pref_flag_get( TR_PREFS_KEY_DSPEED_ENABLED ) );
|
|
g_signal_connect( w, "toggled", G_CALLBACK( target_cb ), w2 );
|
|
hig_workarea_add_row_w( t, &row, w, w2, NULL );
|
|
|
|
s = _( "Limit _upload speed (KB/s):" );
|
|
w = new_check_button( s, TR_PREFS_KEY_USPEED_ENABLED, core );
|
|
w2 = new_spin_button( TR_PREFS_KEY_USPEED, core, 0, INT_MAX, 5 );
|
|
gtk_widget_set_sensitive( GTK_WIDGET( w2 ), pref_flag_get( TR_PREFS_KEY_USPEED_ENABLED ) );
|
|
g_signal_connect( w, "toggled", G_CALLBACK( target_cb ), w2 );
|
|
hig_workarea_add_row_w( t, &row, w, w2, NULL );
|
|
|
|
hig_workarea_add_section_divider( t, &row );
|
|
h = gtk_hbox_new( FALSE, GUI_PAD );
|
|
w = gtk_image_new_from_stock( "alt-speed-off", -1 );
|
|
gtk_box_pack_start( GTK_BOX( h ), w, FALSE, FALSE, 0 );
|
|
g_snprintf( buf, sizeof( buf ), "<b>%s</b>", _( "Temporary Speed Limits" ) );
|
|
w = gtk_label_new( buf );
|
|
gtk_misc_set_alignment( GTK_MISC( w ), 0.0f, 0.5f );
|
|
gtk_label_set_use_markup( GTK_LABEL( w ), TRUE );
|
|
gtk_box_pack_start( GTK_BOX( h ), w, FALSE, FALSE, 0 );
|
|
hig_workarea_add_section_title_widget( t, &row, h );
|
|
|
|
s = _( "Override normal speed limits manually or at scheduled times" );
|
|
g_snprintf( buf, sizeof( buf ), "<small>%s</small>", s );
|
|
w = gtk_label_new( buf );
|
|
gtk_label_set_use_markup( GTK_LABEL( w ), TRUE );
|
|
gtk_misc_set_alignment( GTK_MISC( w ), 0.5f, 0.5f );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
s = _( "Limit do_wnload speed (KB/s):" );
|
|
w = new_spin_button( TR_PREFS_KEY_ALT_SPEED_DOWN, core, 0, INT_MAX, 5 );
|
|
hig_workarea_add_row( t, &row, s, w, NULL );
|
|
|
|
s = _( "Limit u_pload speed (KB/s):" );
|
|
w = new_spin_button( TR_PREFS_KEY_ALT_SPEED_UP, core, 0, INT_MAX, 5 );
|
|
hig_workarea_add_row( t, &row, s, w, NULL );
|
|
|
|
s = _( "_Scheduled times:" );
|
|
h = gtk_hbox_new( FALSE, 0 );
|
|
w2 = new_time_combo( core, TR_PREFS_KEY_ALT_SPEED_TIME_BEGIN );
|
|
page->sched_widgets = g_slist_append( page->sched_widgets, w2 );
|
|
gtk_box_pack_start( GTK_BOX( h ), w2, TRUE, TRUE, 0 );
|
|
w2 = l = gtk_label_new_with_mnemonic ( _( " _to " ) );
|
|
page->sched_widgets = g_slist_append( page->sched_widgets, w2 );
|
|
gtk_box_pack_start( GTK_BOX( h ), w2, FALSE, FALSE, 0 );
|
|
w2 = new_time_combo( core, TR_PREFS_KEY_ALT_SPEED_TIME_END );
|
|
gtk_label_set_mnemonic_widget( GTK_LABEL( l ), w2 );
|
|
page->sched_widgets = g_slist_append( page->sched_widgets, w2 );
|
|
gtk_box_pack_start( GTK_BOX( h ), w2, TRUE, TRUE, 0 );
|
|
w = new_check_button( s, TR_PREFS_KEY_ALT_SPEED_TIME_ENABLED, core );
|
|
g_signal_connect( w, "toggled", G_CALLBACK( onSchedToggled ), page );
|
|
hig_workarea_add_row_w( t, &row, w, h, NULL );
|
|
|
|
s = _( "_On days:" );
|
|
w = new_week_combo( core, TR_PREFS_KEY_ALT_SPEED_TIME_DAY );
|
|
page->sched_widgets = g_slist_append( page->sched_widgets, w );
|
|
w = hig_workarea_add_row( t, &row, s, w, NULL );
|
|
page->sched_widgets = g_slist_append( page->sched_widgets, w );
|
|
|
|
hig_workarea_finish( t, &row );
|
|
g_object_set_data_full( G_OBJECT( t ), "page", page, bandwidthPageFree );
|
|
|
|
refreshSchedSensitivity( page );
|
|
return t;
|
|
}
|
|
|
|
/****
|
|
***** Network Tab
|
|
****/
|
|
|
|
struct network_page_data
|
|
{
|
|
TrCore * core;
|
|
GtkWidget * portLabel;
|
|
GtkWidget * portButton;
|
|
GtkWidget * portSpin;
|
|
gulong portTag;
|
|
gulong prefsTag;
|
|
};
|
|
|
|
static void
|
|
onCorePrefsChanged( TrCore * core UNUSED, const char * key, gpointer gdata )
|
|
{
|
|
if( !strcmp( key, TR_PREFS_KEY_PEER_PORT ) )
|
|
{
|
|
struct network_page_data * data = gdata;
|
|
gdk_threads_enter();
|
|
gtk_label_set_text( GTK_LABEL( data->portLabel ), _( "Status unknown" ) );
|
|
gtk_widget_set_sensitive( data->portButton, TRUE );
|
|
gtk_widget_set_sensitive( data->portSpin, TRUE );
|
|
gdk_threads_leave();
|
|
}
|
|
}
|
|
|
|
static void
|
|
peerPageDestroyed( gpointer gdata, GObject * dead UNUSED )
|
|
{
|
|
struct network_page_data * data = gdata;
|
|
if( data->prefsTag > 0 )
|
|
g_signal_handler_disconnect( data->core, data->prefsTag );
|
|
if( data->portTag > 0 )
|
|
g_signal_handler_disconnect( data->core, data->portTag );
|
|
g_free( data );
|
|
}
|
|
|
|
static void
|
|
onPortTested( TrCore * core UNUSED, gboolean isOpen, gpointer vdata )
|
|
{
|
|
struct network_page_data * data = vdata;
|
|
const char * markup = isOpen ? _( "Port is <b>open</b>" ) : _( "Port is <b>closed</b>" );
|
|
gdk_threads_enter();
|
|
gtk_label_set_markup( GTK_LABEL( data->portLabel ), markup );
|
|
gtk_widget_set_sensitive( data->portButton, TRUE );
|
|
gtk_widget_set_sensitive( data->portSpin, TRUE );
|
|
gdk_threads_leave();
|
|
}
|
|
|
|
static void
|
|
onPortTest( GtkButton * button UNUSED, gpointer vdata )
|
|
{
|
|
struct network_page_data * data = vdata;
|
|
gtk_widget_set_sensitive( data->portButton, FALSE );
|
|
gtk_widget_set_sensitive( data->portSpin, FALSE );
|
|
gtk_label_set_markup( GTK_LABEL( data->portLabel ), _( "<i>Testing...</i>" ) );
|
|
if( !data->portTag )
|
|
data->portTag = g_signal_connect( data->core, "port-tested", G_CALLBACK(onPortTested), data );
|
|
tr_core_port_test( data->core );
|
|
}
|
|
|
|
static GtkWidget*
|
|
peerPage( GObject * core )
|
|
{
|
|
int row = 0;
|
|
const char * s;
|
|
GtkWidget * t;
|
|
GtkWidget * w;
|
|
GtkWidget * h;
|
|
GtkWidget * l;
|
|
struct network_page_data * data;
|
|
|
|
/* register to stop listening to core prefs changes when the page is destroyed */
|
|
data = g_new0( struct network_page_data, 1 );
|
|
data->core = TR_CORE( core );
|
|
|
|
/* build the page */
|
|
t = hig_workarea_create( );
|
|
hig_workarea_add_section_title( t, &row, _( "Incoming Peers" ) );
|
|
|
|
s = _( "_Port for incoming connections:" );
|
|
w = data->portSpin = new_spin_button( TR_PREFS_KEY_PEER_PORT, core, 1, USHRT_MAX, 1 );
|
|
hig_workarea_add_row( t, &row, s, w, NULL );
|
|
|
|
h = gtk_hbox_new( FALSE, GUI_PAD_BIG );
|
|
l = data->portLabel = gtk_label_new( _( "Status unknown" ) );
|
|
gtk_misc_set_alignment( GTK_MISC( l ), 0.0f, 0.5f );
|
|
gtk_box_pack_start( GTK_BOX( h ), l, TRUE, TRUE, 0 );
|
|
w = data->portButton = gtk_button_new_with_mnemonic( _( "Te_st Port" ) );
|
|
gtk_box_pack_end( GTK_BOX( h ), w, FALSE, FALSE, 0 );
|
|
g_signal_connect( w, "clicked", G_CALLBACK(onPortTest), data );
|
|
hig_workarea_add_row( t, &row, NULL, h, NULL );
|
|
data->prefsTag = g_signal_connect( TR_CORE( core ), "prefs-changed", G_CALLBACK( onCorePrefsChanged ), data );
|
|
g_object_weak_ref( G_OBJECT( t ), peerPageDestroyed, data );
|
|
|
|
s = _( "Use UPnP or NAT-PMP port _forwarding from my router" );
|
|
w = new_check_button( s, TR_PREFS_KEY_PORT_FORWARDING, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
s = _( "Pick a _random port every time Transmission is started" );
|
|
w = new_check_button( s, TR_PREFS_KEY_PEER_PORT_RANDOM_ON_START, core );
|
|
hig_workarea_add_wide_control( t, &row, w );
|
|
|
|
hig_workarea_add_section_divider( t, &row );
|
|
hig_workarea_add_section_title( t, &row, _( "Limits" ) );
|
|
|
|
w = new_spin_button( TR_PREFS_KEY_PEER_LIMIT_TORRENT, core, 1, 300, 5 );
|
|
hig_workarea_add_row( t, &row, _( "Maximum peers per _torrent:" ), w, NULL );
|
|
w = new_spin_button( TR_PREFS_KEY_PEER_LIMIT_GLOBAL, core, 1, 3000, 5 );
|
|
hig_workarea_add_row( t, &row, _( "Maximum peers _overall:" ), w, NULL );
|
|
|
|
hig_workarea_finish( t, &row );
|
|
return t;
|
|
}
|
|
|
|
/****
|
|
*****
|
|
****/
|
|
|
|
GtkWidget *
|
|
tr_prefs_dialog_new( GObject * core,
|
|
GtkWindow * parent )
|
|
{
|
|
GtkWidget * d;
|
|
GtkWidget * n;
|
|
|
|
d = gtk_dialog_new_with_buttons( _(
|
|
"Transmission Preferences" ),
|
|
parent,
|
|
GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
GTK_STOCK_HELP, GTK_RESPONSE_HELP,
|
|
GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
|
|
NULL );
|
|
gtk_window_set_role( GTK_WINDOW( d ), "transmission-preferences-dialog" );
|
|
gtk_dialog_set_has_separator( GTK_DIALOG( d ), FALSE );
|
|
gtk_container_set_border_width( GTK_CONTAINER( d ), GUI_PAD );
|
|
|
|
n = gtk_notebook_new( );
|
|
gtk_container_set_border_width ( GTK_CONTAINER ( n ), GUI_PAD );
|
|
|
|
gtk_notebook_append_page( GTK_NOTEBOOK( n ),
|
|
torrentPage( core ),
|
|
gtk_label_new ( _( "Torrents" ) ) );
|
|
gtk_notebook_append_page( GTK_NOTEBOOK( n ),
|
|
bandwidthPage( core ),
|
|
gtk_label_new ( _( "Speed" ) ) );
|
|
gtk_notebook_append_page( GTK_NOTEBOOK( n ),
|
|
privacyPage( core ),
|
|
gtk_label_new ( _( "Privacy" ) ) );
|
|
gtk_notebook_append_page( GTK_NOTEBOOK( n ),
|
|
peerPage( core ),
|
|
gtk_label_new ( _( "Network" ) ) );
|
|
gtk_notebook_append_page( GTK_NOTEBOOK( n ),
|
|
desktopPage( core ),
|
|
gtk_label_new ( _( "Desktop" ) ) );
|
|
gtk_notebook_append_page( GTK_NOTEBOOK( n ),
|
|
webPage( core ),
|
|
gtk_label_new ( _( "Web" ) ) );
|
|
gtk_notebook_append_page( GTK_NOTEBOOK( n ),
|
|
trackerPage( core ),
|
|
gtk_label_new ( _( "Proxy" ) ) );
|
|
|
|
g_signal_connect( d, "response", G_CALLBACK( response_cb ), core );
|
|
gtk_box_pack_start( GTK_BOX( GTK_DIALOG( d )->vbox ), n, TRUE, TRUE, 0 );
|
|
gtk_widget_show_all( GTK_DIALOG( d )->vbox );
|
|
return d;
|
|
}
|
|
|