added french translation
[clawsker.git] / clawsker
index fe9759a3d347b83c569afaea7b83454aafb76087..64831d60c120e37da1ce0859610bddc33eddee49 100755 (executable)
--- a/clawsker
+++ b/clawsker
@@ -56,7 +56,7 @@ Ricardo Mones E<lt>ricardo@mones.orgE<gt>
 
 =head1 LICENSE
 
-Copyright (c) 2007 by Ricardo Mones
+Copyright (c) 2007-2008 by Ricardo Mones Lastra
 
 This program is free software: you can redistribute it and/or modify
 it under the terms of the GNU General Public License as published by
@@ -74,10 +74,11 @@ along with this program.  If not, see E<lt>http://www.gnu.org/licenses/E<gt>.
 =cut
 
 use strict;
-use POSIX qw(setlocale);
-use Locale::gettext;
+use encoding 'utf8';
 use Glib qw(TRUE FALSE);
 use Gtk2 -init;
+use POSIX qw(setlocale);
+use Locale::gettext;
 
 my $NAME = 'clawsker';
 my $PREFIX = '@PREFIX@';
@@ -87,6 +88,7 @@ my $VERBOSE = FALSE;
 my $main_window = undef;
 
 my $locale = (defined($ENV{LC_MESSAGES}) ? $ENV{LC_MESSAGES} : $ENV{LANG});
+$locale = "C" unless defined($locale);
 setlocale (LC_ALL, $locale);
 bindtextdomain ($NAME, sprintf ('%s/share/locale', $PREFIX));
 textdomain ($NAME);
@@ -129,6 +131,7 @@ sub _ {
     stripes_frame => _('Coloured stripes'),
     sbar_frame => _('Scroll bars'),
     mlist_frame => _('Message list'),
+    netm_frame => _('NetworkManager'),
 
     l_oth_use_dlg => _('Use detached address book edit dialogue'),
     h_oth_use_dlg => _('If true use a separate dialogue to edit a person\'s details. Otherwise will use a form embedded in the address book\'s main window.'),
@@ -136,6 +139,8 @@ sub _ {
     h_oth_max_use => _('The maximum amount of memory to use to cache messages, in kB.'),
     l_oth_min_time => _('Minimun time for cache elements (minutes)'),
     h_oth_min_time => _('The minimum time in minutes to keep a cache in memory. Caches more recent than this time will not be freed, even if the memory usage is too high.'),
+    l_oth_use_netm => _('Use NetworkManager'),
+    h_oth_use_netm => _('Use NetworkManager to switch offline automatically.'),
     
     l_gui_b_unread => _('Show unread messages with bold font'),
     h_gui_b_unread => _('Show unread messages in the Message List using a bold font.'),
@@ -162,6 +167,8 @@ sub _ {
     h_gui_strip_all => _('Enable alternately coloured lines in all tree view components.'),
     l_gui_strip_sum => _('Use stripes in Folder List and Message List'),
     h_gui_strip_sum => _('Enable alternately coloured lines in message list and folder list'),
+    l_gui_two_line_v => _('2 lines per Message List item in 3-column layout'),
+    h_gui_two_line_v => _('Spread Message List information over two lines when using the three column mode'),
 
     l_beh_hover_t => _('Drag \'n\' drop hover timeout (ms)'),
     h_beh_hover_t => _('Time in milliseconds that will cause a folder tree to expand when the mouse cursor is held over it during drag and drop.'),
@@ -183,6 +190,8 @@ sub _ {
     h_beh_use_utf8 => _('Use UTF-8 encoding for broken mails instead of current locale.'),
     l_beh_warn_dnd => _('Warn on drag \'n\' drop'),
     h_beh_warn_dnd => _('Display a confirmation dialogue on drag \'n\' drop of folders.'),
+    l_beh_out_ascii => _('Ougoing messages fallback to ASCII'),
+    h_beh_out_ascii => _('If content allows, ASCII will be used to encode outgoing messages, otherwise the user-defined encoding is enforced always.'),
 
     l_col_emphasis => _('X-Mailer header'),
     h_col_emphasis => _('The colour used for the X-Mailer line when its value is Claws Mail.'),
@@ -220,7 +229,9 @@ use constant NAME  => 0; # the name on the rc file
 use constant LABEL => 1; # the label on the GUI
 use constant DESC  => 2; # the description for the hint/help
 use constant TYPE  => 3; # data type: bool, int, float, string, color
-use constant GUI   => 4; # GUI element
+use constant CMVER => 4; # lowest Claws Mail version the feature exists
+use constant CMDEF => 5; # default value for the preference in Claws Mail
+use constant GUI   => 6; # GUI element
 
 # constants for GUI spacing
 use constant HBOX_SPC => 5;
@@ -300,10 +311,10 @@ sub set_rc_filename {
 
 sub log_message {
     my ($mesg, $fatal) = @_;
+    if (defined($fatal) && $fatal eq 'die') {
+        die "$NAME: $mesg\n";
+    }
     if ($VERBOSE) {
-        if (defined($fatal) && $fatal eq 'die') {
-          die "$NAME: $mesg\n";
-        }
         print "$NAME: $mesg\n";
     }
 }
@@ -360,7 +371,9 @@ sub new_check_button_for {
     my $hbox = Gtk2::HBox->new (FALSE, 5);
     my $cb = Gtk2::CheckButton->new ($label);
     $$hash{$key}[GUI] = $cb;
-    $cb->set_active ($HPVALUE{$name} eq '1');
+    if (defined ($HPVALUE{$name})) {
+        $cb->set_active ($HPVALUE{$name} eq '1');
+    }
     $cb->signal_connect (clicked => sub {
             my ($w, $e) = @_;
            &handle_bool_value($w, $e, \$HPVALUE{$name});
@@ -371,20 +384,25 @@ sub new_check_button_for {
     return $hbox;
 }
 
-sub new_text_box_for {
+sub new_text_box_for_int {
     my ($hash, $key) = @_;
     my $name = $$hash{$key}[NAME];
     my $label = $$hash{$key}[LABEL];
+    my @type = split (/,/, $$hash{$key}[TYPE]);
+    push (@type, 0), push (@type, 10000) unless ($#type > 0); 
     #
     my $hbox = Gtk2::HBox->new (FALSE, 5);
     my $glabel = Gtk2::Label->new ($label);
-    my $gentry = Gtk2::Entry->new;
-    $gentry->set_width_chars (8);
+    my $pagei = int (($type[2] - $type[1]) / 10);
+    my $adjust = Gtk2::Adjustment->new (
+            $HPVALUE{$name}, $type[1], $type[2], 1, $pagei, 10
+       );
+    my $gentry = Gtk2::SpinButton->new ($adjust, 1, 0);
+    $gentry->set_numeric (TRUE);
     $$hash{$key}[GUI] = $gentry;
-    $gentry->set_text($HPVALUE{$name});
-    $gentry->signal_connect(changed => sub {
+    $gentry->signal_connect('value-changed' => sub {
             my ($w, $e) = @_;
-           &handle_int_value($w, $e, \$HPVALUE{$name}); # FIXME int only
+           &handle_int_value($w, $e, \$HPVALUE{$name});
         });
     &set_widget_hint ($gentry, $$hash{$key}[DESC]);
     $hbox->pack_start ($glabel, FALSE, FALSE, HBOX_SPC);
@@ -450,20 +468,35 @@ sub new_selection_box_for {
        $xl::s{l_oth_use_dlg},
         $xl::s{h_oth_use_dlg},
        'bool',
+       '2.7.0',
+       '0',
        undef,
     ],
     max_use => [
        'cache_max_mem_usage',
        $xl::s{l_oth_max_use},
        $xl::s{h_oth_max_use},
-       'int',
+       'int,0,262144', # 0 Kb - 256 Mb
+       '0.0.0',
+       '4096',
        undef,
     ],
     min_time => [
         'cache_min_keep_time',
        $xl::s{l_oth_min_time},
        $xl::s{h_oth_min_time},
-       'int',
+       'int,0,120', # 0 minutes - 2 hours
+       '0.0.0',
+       '15',
+       undef,
+    ],
+    use_netm => [
+        'use_networkmanager',
+       $xl::s{l_oth_use_netm},
+       $xl::s{h_oth_use_netm},
+       'bool',
+       '3.3.1',
+       '1',
        undef,
     ],
 );
@@ -480,16 +513,24 @@ sub new_other_page() {
     $ab_frame->add ($vb1);
 
     my $mem_frame = Gtk2::Frame->new ($xl::s{mem_frame}); 
-    my $tb_max_use = &new_text_box_for(\%pr::oth, 'max_use');
-    my $tb_min_time = &new_text_box_for(\%pr::oth, 'min_time');
+    my $tb_max_use = &new_text_box_for_int(\%pr::oth, 'max_use');
+    my $tb_min_time = &new_text_box_for_int(\%pr::oth, 'min_time');
     my $vb2 = Gtk2::VBox->new (FALSE, 5);
     $vb2->set_border_width (PAGE_SPC); 
     $vb2->pack_start ($tb_max_use, FALSE, FALSE, 0);
     $vb2->pack_start ($tb_min_time, FALSE, FALSE, 0);
     $mem_frame->add ($vb2);
 
+    my $netm_frame = Gtk2::Frame->new ($xl::s{netm_frame});
+    my $cb_use_netm = &new_check_button_for(\%pr::oth, 'use_netm');
+    my $vb3 = Gtk2::VBox->new (FALSE, 5);
+    $vb3->set_border_width (PAGE_SPC);
+    $vb3->pack_start ($cb_use_netm, FALSE, FALSE, 0);
+    $netm_frame->add ($vb3);
+
     $of->pack_start ($ab_frame, FALSE, FALSE, FRAME_SPC);
     $of->pack_start ($mem_frame, FALSE, FALSE, FRAME_SPC);
+    $of->pack_start ($netm_frame, FALSE, FALSE, FRAME_SPC);
 
     return $of;
 }
@@ -500,13 +541,17 @@ sub new_other_page() {
         $xl::s{l_gui_b_unread},
         $xl::s{h_gui_b_unread},
        'bool',
+       '0.0.0',
+       '1',
        undef,
     ],
     no_markup => [
         'compose_no_markup',
         $xl::s{l_gui_no_markup},
         $xl::s{h_gui_no_markup},
-       'bool',
+        'bool',
+        '0.0.0',
+        '0',
        undef,
     ],
     dot_lines => [
@@ -514,6 +559,8 @@ sub new_other_page() {
         $xl::s{l_gui_dot_lines},
         $xl::s{h_gui_dot_lines},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     h_scroll => [
@@ -521,6 +568,8 @@ sub new_other_page() {
         $xl::s{l_gui_h_scroll},
         $xl::s{h_gui_h_scroll},
        'bool',
+       '0.0.0',
+       '1',
        undef,
     ],
     swp_from => [
@@ -528,6 +577,8 @@ sub new_other_page() {
         $xl::s{l_gui_swp_from},
         $xl::s{h_gui_swp_from},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     v_scroll => [
@@ -535,13 +586,17 @@ sub new_other_page() {
         $xl::s{l_gui_v_scroll},
         $xl::s{h_gui_v_scroll},
        '0=l_gui_v_scroll_show;1=l_gui_v_scroll_auto;2=l_gui_v_scroll_hide',
+       '0.0.0',
+       '0',
        undef,
     ],
     strip_off => [
         'stripes_color_offset',
        $xl::s{l_gui_strip_off},
        $xl::s{h_gui_strip_off},
-       'int',
+       'int,0,10000', # no idea what this number means
+       '0.0.0',
+       '4000',
        undef,
     ],
     cursor_v => [
@@ -549,6 +604,8 @@ sub new_other_page() {
         $xl::s{l_gui_cursor_v},
         $xl::s{h_gui_cursor_v},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     toolbar_d => [
@@ -556,6 +613,8 @@ sub new_other_page() {
         $xl::s{l_gui_toolbar_d},
         $xl::s{h_gui_toolbar_d},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     strip_all => [
@@ -563,6 +622,8 @@ sub new_other_page() {
         $xl::s{l_gui_strip_all},
         $xl::s{h_gui_strip_all},
        'bool',
+       '0.0.0',
+       '1',
        undef,
     ],
     strip_sum => [
@@ -570,6 +631,17 @@ sub new_other_page() {
         $xl::s{l_gui_strip_sum},
         $xl::s{h_gui_strip_sum},
        'bool',
+       '0.0.0',
+       '1',
+       undef,
+    ],
+    two_linev => [
+        'two_line_vertical',
+       $xl::s{l_gui_two_line_v},
+       $xl::s{h_gui_two_line_v},
+       'bool',
+       '3.4.0.7',
+       '0',
        undef,
     ],
 );
@@ -581,7 +653,7 @@ sub new_gui_page() {
     my $stripes_frame = Gtk2::Frame->new ($xl::s{stripes_frame});
     my $cb_strip_all = &new_check_button_for (\%pr::gui, 'strip_all');
     my $cb_strip_sum = &new_check_button_for (\%pr::gui, 'strip_sum');
-    my $tb_strip_off = &new_text_box_for (\%pr::gui, 'strip_off');
+    my $tb_strip_off = &new_text_box_for_int (\%pr::gui, 'strip_off');
     my $vb1 = Gtk2::VBox->new (FALSE, 5);
     $vb1->set_border_width (PAGE_SPC);
     $vb1->pack_start ($cb_strip_all, FALSE, FALSE, 0);
@@ -611,6 +683,7 @@ sub new_gui_page() {
     my $cb_dot_lines = &new_check_button_for (\%pr::gui, 'dot_lines'); 
     my $cb_cursor_v = &new_check_button_for (\%pr::gui, 'cursor_v');
     my $cb_toolbar_d = &new_check_button_for (\%pr::gui, 'toolbar_d');
+    my $cb_two_linev = &new_check_button_for (\%pr::gui, 'two_linev');
 
     $gf->pack_start ($stripes_frame, FALSE, FALSE, FRAME_SPC);
     $gf->pack_start ($mlist_frame, FALSE, FALSE, FRAME_SPC);
@@ -618,6 +691,7 @@ sub new_gui_page() {
     $gf->pack_start ($cb_dot_lines, FALSE, FALSE, 0);
     $gf->pack_start ($cb_cursor_v, FALSE, FALSE, 0);
     $gf->pack_start ($cb_toolbar_d, FALSE, FALSE, 0);
+    $gf->pack_start ($cb_two_linev, FALSE, FALSE, 0);
     $gf->pack_start ($sbar_frame, FALSE, FALSE, FRAME_SPC);
 
     return $gf;
@@ -628,7 +702,9 @@ sub new_gui_page() {
         'hover_timeout',
         $xl::s{l_beh_hover_t},
         $xl::s{h_beh_hover_t},
-       'int',
+       'int,100,3000', # 0.1 seconds - 3 seconds
+       '0.0.0',
+       '500',
        undef,
     ],
     dangerous => [
@@ -636,6 +712,8 @@ sub new_gui_page() {
         $xl::s{l_beh_dangerous},
         $xl::s{h_beh_dangerous},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     flowed => [
@@ -643,6 +721,8 @@ sub new_gui_page() {
         $xl::s{l_beh_flowed},
         $xl::s{h_beh_flowed},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     parts_rw => [
@@ -650,6 +730,8 @@ sub new_gui_page() {
         $xl::s{l_beh_parts_rw},
         $xl::s{h_beh_parts_rw},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     skip_ssl => [
@@ -657,20 +739,26 @@ sub new_gui_page() {
         $xl::s{l_beh_skip_ssl},
         $xl::s{h_beh_skip_ssl},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     up_step => [
         'statusbar_update_step',
         $xl::s{l_beh_up_step},
         $xl::s{h_beh_up_step},
-       'int',
+       'int,1,200', # 1 item - 200 items
+       '0.0.0',
+       '10',
        undef,
     ],
     thread_a => [
         'thread_by_subject_max_age',
         $xl::s{l_beh_thread_a},
         $xl::s{h_beh_thread_a},
-       'int',
+       'int,1,30', # 1 day - 30 days
+       '0.0.0',
+       '10',
        undef,
     ],
     unsafe_ssl => [
@@ -678,6 +766,8 @@ sub new_gui_page() {
         $xl::s{l_beh_unsafe_ssl},
         $xl::s{h_beh_unsafe_ssl},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     use_utf8 => [
@@ -685,6 +775,8 @@ sub new_gui_page() {
         $xl::s{l_beh_use_utf8},
         $xl::s{h_beh_use_utf8},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     warn_dnd => [
@@ -692,6 +784,17 @@ sub new_gui_page() {
         $xl::s{l_beh_warn_dnd},
         $xl::s{h_beh_warn_dnd},
        'bool',
+       '0.0.0',
+       '1',
+       undef,
+    ],
+    out_ascii => [
+        'outgoing_fallback_to_ascii',
+       $xl::s{l_beh_out_ascii},
+       $xl::s{h_beh_out_ascii},
+       'bool',
+       '3.4.0.37',
+       '1',
        undef,
     ],
 );
@@ -701,7 +804,7 @@ sub new_behaviour_page() {
     $bf->set_border_width (PAGE_SPC);
 
     my $dnd_frame = Gtk2::Frame->new ($xl::s{dnd_frame});
-    my $tb_hoover_t = &new_text_box_for (\%pr::beh, 'hover_t');
+    my $tb_hoover_t = &new_text_box_for_int (\%pr::beh, 'hover_t');
     my $cb_warn_dnd = &new_check_button_for (\%pr::beh, 'warn_dnd');
     my $vb1 = Gtk2::VBox->new (FALSE, 5);
     $vb1->set_border_width (PAGE_SPC);
@@ -718,20 +821,22 @@ sub new_behaviour_page() {
     $hb1->pack_start ($cb_unsafe_ssl, FALSE, FALSE, 0);
     $ssl_frame->add ($hb1);
 
-    my $tb_up_step = &new_text_box_for (\%pr::beh, 'up_step');
-    my $tb_thread_a = &new_text_box_for (\%pr::beh, 'thread_a');
+    my $tb_up_step = &new_text_box_for_int (\%pr::beh, 'up_step');
+    my $tb_thread_a = &new_text_box_for_int (\%pr::beh, 'thread_a');
 
     my $msgs_frame = Gtk2::Frame->new ($xl::s{msgs_frame});
     my $cb_flowed = &new_check_button_for (\%pr::beh, 'flowed');
     my $cb_parts_rw = &new_check_button_for (\%pr::beh, 'parts_rw');
     my $cb_use_utf8 = &new_check_button_for (\%pr::beh, 'use_utf8');
     my $cb_dangerous = &new_check_button_for (\%pr::beh, 'dangerous');
+    my $cb_out_ascii = &new_check_button_for (\%pr::beh, 'out_ascii');
     my $vb2 = Gtk2::VBox->new (FALSE, 5);    
     $vb2->set_border_width (PAGE_SPC);
     $vb2->pack_start ($cb_flowed, FALSE, FALSE, 0);
     $vb2->pack_start ($cb_parts_rw, FALSE, FALSE, 0);
     $vb2->pack_start ($cb_use_utf8, FALSE, FALSE, 0);
     $vb2->pack_start ($cb_dangerous, FALSE, FALSE, 0);
+    $vb2->pack_start ($cb_out_ascii, FALSE, FALSE, 0);
     $msgs_frame->add ($vb2);
 
     $bf->pack_start ($dnd_frame, FALSE, FALSE, FRAME_SPC);
@@ -749,6 +854,8 @@ sub new_behaviour_page() {
         $xl::s{l_col_emphasis},
         $xl::s{h_col_emphasis},
        'color',
+       '0.0.0',
+       '#0000cf',
        undef,
     ],
     log_err => [
@@ -756,6 +863,8 @@ sub new_behaviour_page() {
         $xl::s{l_col_log_err},
         $xl::s{h_col_log_err},
         'color',
+       '0.0.0',
+       '#af0000',
         undef,
     ],
     log_in => [
@@ -763,6 +872,8 @@ sub new_behaviour_page() {
         $xl::s{l_col_log_in},
         $xl::s{h_col_log_in},
         'color',
+       '0.0.0',
+       '#000000',
         undef,
     ],
     log_msg => [
@@ -770,6 +881,8 @@ sub new_behaviour_page() {
         $xl::s{l_col_log_msg},
         $xl::s{h_col_log_msg},
         'color',
+       '0.0.0',
+       '#00af00',
         undef,
     ],
     log_out => [
@@ -777,6 +890,8 @@ sub new_behaviour_page() {
         $xl::s{l_col_log_out},
         $xl::s{h_col_log_out},
         'color',
+       '0.0.0',
+       '#0000ef',
         undef,
     ],
     log_warn => [
@@ -784,6 +899,8 @@ sub new_behaviour_page() {
         $xl::s{l_col_log_warn},
         $xl::s{h_col_log_warn},
         'color',
+       '0.0.0',
+       '#af0000',
         undef,
     ],
 );
@@ -851,6 +968,14 @@ sub print_help() {
     print _("    --clawsrc <file>               Uses <file> as full resource name.\n");
 }
 
+# handle errors which don't allow to run
+sub command_line_fatal() {
+    my $reason = shift;
+    my $emsg = $xl::s{e_error} . $reason;
+    error_dialog ($emsg);
+    log_message ("$emsg", 'die');
+}
+
 # parse the command line
 sub parse_command_line() {
     my $arg = 0;
@@ -866,25 +991,27 @@ sub parse_command_line() {
            };
            /--alternate-config-dir/ && do {
                ++$arg;
-               die "$xl::s{e_error}$xl::s{e_requireddir}\n" 
+               &command_line_fatal ($xl::s{e_requireddir})
                    unless defined($ARGV[$arg]);
-               die "$xl::s{e_error}$xl::s{e_notadir}\n" 
+               &command_line_fatal ($xl::s{e_notadir})
                    unless -d $ARGV[$arg];
                $CONFIGDIR = $ARGV[$arg];
+               $CONFIGDIR .= "/" 
+                   unless ($CONFIGDIR =~ /.*\/$/);
                $ALTCONFIGDIR = TRUE;
                last;
            };
            /--clawsrc/ && do {
                ++$arg;
-               die "$xl::s{e_error}$xl::s{e_requiredfile}\n" 
+               &command_line_fatal($xl::s{e_requiredfile}) 
                    unless defined($ARGV[$arg]);
-               die "$xl::s{e_error}$xl::s{e_notafile}\n" 
+               &command_line_fatal($xl::s{e_notafile}) 
                    unless -f $ARGV[$arg];
                &set_rc_filename ($ARGV[$arg]);
                last;
            };
-           /.*/ && die $xl::s{e_error} 
-                       . _("unknown option '{opt}'.\n", opt => $ARGV[$arg]);
+           /.*/ && &command_line_fatal (
+                       _("unknown option '{opt}'.\n", opt => $ARGV[$arg]));
        }
        ++$arg;
     }