separate version option from help
[clawsker.git] / clawsker
index 7ce0da4ea10d222fe7a270c795699651a09baf8e..f2dae0a109f1ceb533ff703b7544485f8f298b9b 100755 (executable)
--- a/clawsker
+++ b/clawsker
@@ -19,20 +19,36 @@ Clawsker is an applet to edit the so called Claws Mail hidden preferences.
 
 Claws Mail is a fast, lightweight and feature-rich MUA with a high number 
 of configurable options. To keep the binary small and fast some of these 
-preferences which not widely used are not provided with a graphical interface 
-for inspection and/or modification.
+preferences which are not widely used are not provided with a graphical
+interface for inspection and/or modification.
 
-Users wanting to edit such preferences had to face raw edition of their 
+Users wanting to edit such preferences had to face raw editing of their 
 configuration files, now you can do it with a convenient GTK2 interface
 using Clawsker.
 
 =head1 OPTIONS
 
-No options are currently available.
+--help
+       Shows a brief help screen.
 
+--verbose
+       Tells more on the standard output.
+
+--alternate-config-dir <dir>
+       Uses <dir> as Claws Mail configuration dir.
+
+--clawsrc <file>
+       Uses <file> as Claws Mail resource configuration file. This sets
+       the full file name overriding any previous setting.
+
+Multiple options are allowed, although only the last one has effect. Weird
+option specifications may produce weird results (but otherwise correct).
+       
 =head1 LIMITATIONS
 
-Alternate configuration directories are not (yet) supported.
+A running Claws Mail cannot be detected if using the --clawsrc option because
+the directory is not assumed to be a Claws Mail configuration dir. If that is 
+the case use the --alternate-config-dir option instead.
 
 =head1 AUTHOR
 
@@ -40,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
@@ -58,133 +74,215 @@ along with this program.  If not, see E<lt>http://www.gnu.org/licenses/E<gt>.
 =cut
 
 use strict;
+use encoding 'utf8';
 use Glib qw(TRUE FALSE);
 use Gtk2 -init;
+use POSIX qw(setlocale);
+use Locale::gettext;
 
+my $NAME = 'clawsker';
 my $PREFIX = '@PREFIX@';
 my $LIBDIR = '@LIBDIR@';
 my $VERSION = '@VERSION@';
+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);
+
+my $SHOWHINTS = FALSE;
+$SHOWHINTS = TRUE if ($Gtk2::VERSION >= 1.040 and Gtk2->CHECK_VERSION (2, 12, 0));
+
+sub _ {
+    my $str = shift;
+    my %par = @_;
+    my $xla = gettext ($str);
+    if (scalar(keys(%par)) > 0) {
+        foreach my $key (keys %par) {
+            $xla =~ s/\{$key\}/$par{$key}/g;
+        }
+    }
+    return $xla;
+}
 
 # default messages
 %xl::s = (
-    win_title => 'Claws Mail Hidden Preferences',
-    about => 'About...',
-    about_title => 'Clawsker ~ A Claws Mail Tweaker',
-    about_license => 'License:',
-
-    tab_colours => 'Colours',
-    tab_behaviour => 'Behaviour',
-    tab_gui => 'GUI',
-    tab_other => 'Other',
-
-    ab_frame => 'Addressbook',
-    mem_frame => 'Memory',
-    msgview_frame => 'Message view',
-    log_frame => 'Log window',
-
-    l_oth_use_dlg => 'Use address book dialog',
-    h_oth_use_dlg => 'If true use a separate dialogue to edit a person\'s details. Otherwise will use a form embedded in the addressbook\'s main window.',
-    l_oth_max_use => 'Maximum memory for message cache (Kb)',
-    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.',
+    win_title => _('Claws Mail Hidden Preferences'),
+    about => _('About...'),
+    about_title => _('Clawsker :: A Claws Mail Tweaker'),
+    about_license => _('License:'),
+    about_version => _('Version:'),
+
+    tab_colours => _('Colours'),
+    tab_behaviour => _('Behaviour'),
+    tab_gui => _('GUI'),
+    tab_other => _('Other'),
+
+    ab_frame => _('Addressbook'),
+    mem_frame => _('Memory'),
+    msgview_frame => _('Message view'),
+    log_frame => _('Log window'),
+    dnd_frame => _('Drag \'n\' drop'),
+    ssl_frame => _('Secure Sockets Layer'),
+    msgs_frame => _('Messages'),
+    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.'),
+    l_oth_max_use => _('Maximum memory for message cache (kB)'),
+    h_oth_max_use => _('The maximum amount of memory to use to cache messages, in kilobytes.'),
+    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.',
-    l_gui_no_markup => 'Don\'t use markup in compose window',
-    h_gui_no_markup => 'Don\'t use bold and italic text in Compose dialogue\'s account selector.',
-    l_gui_dot_lines => 'Use dotted lines in tree views',
-    h_gui_dot_lines => 'Use the old dotted line look in the main window tree views (folder and message lists) instead of the modern lineless look.',
-    l_gui_h_scroll => 'Enable horizontal scrollbar',
-    h_gui_h_scroll => 'Enable the horizontal scrollbar in the Message List.',
-    l_gui_swp_from => 'Swap From field in Sent folder',
-    h_gui_swp_from => 'Display the sender\'s email address in the To column of the Sent folder instead of the recipient\'s.',
-    l_gui_v_scroll => 'Folder list scrollbar behaviour',
-    h_gui_v_scroll => 'Specify the policy of vertical scrollbar of Folder List: show always, automatic or hide always.',
-    l_gui_v_scroll_show => 'Show always',
-    l_gui_v_scroll_auto => 'Automatic',
-    l_gui_v_scroll_hide => 'Hide always',
-    l_gui_strip_off => 'Striped lines contrast',
-    h_gui_strip_off => 'Specify the value to use when creating alternately coloured lines in tree view components. The smaller the value, the less visible the difference in the alternating colours of the lines.',
-    l_gui_cursor_v => 'Cursor visible in message view',
-    h_gui_cursor_v => 'Display the cursor in the message view.',
-    l_gui_toolbar_d => 'Detachable toolbars',
-    h_gui_toolbar_d => 'Show handles in the toolbars.',
-    l_gui_strip_all => 'Use stripes in tree view components',
-    h_gui_strip_all => 'Enable alternately coloured lines in all tree view components.',
-    l_gui_strip_sum => 'Use stripes in summary view',
-    h_gui_strip_sum => 'Enable alternately coloured lines in summary view and folder list',
-
-    l_beh_hover_t => 'Drag and 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.',
-    l_beh_dangerous => 'Don\'t confirm deletions',
-    h_beh_dangerous => 'Don\'t ask for confirmation before definitive deletion of emails.',
-    l_beh_flowed => 'Respect flowed format in messages',
-    h_beh_flowed => 'Respect format=flowed on text/plain message parts. This will cause some mails to have long lines, but will fix some URLs that would otherwise be wrapped.',
-    l_beh_parts_rw => 'Allow writable temporary files',
-    h_beh_parts_rw => 'Saves temporary files when opening attachment with write bit set.',
-    l_beh_skip_ssl => 'Don\'t check SSL certificates',
-    h_beh_skip_ssl => 'Disables the verification of SSL certificates.',
-    l_beh_up_step => 'Progress update step (items)',
-    h_beh_up_step => 'Update stepping in progress bars.',
-    l_beh_thread_a => 'Maximum age in threading by subject (days)', 
-    h_beh_thread_a => 'Number of days to include a message in a thread when using "Thread using subject in addition to standard headers".',
-    l_beh_unsafe_ssl => 'Allow unsafe SSL certificates',
-    h_beh_unsafe_ssl => 'Allows Claws Mail to remember multiple SSL certificates for a given server/port.',
-    l_beh_use_utf8 => 'Use UTF-8 in broken mails',
-    h_beh_use_utf8 => 'Use UTF-8 encoding for broken mails instead of current locale.',
-    l_beh_warn_dnd => 'Warning on drag and drop',
-    h_beh_warn_dnd => 'Display a confirmation dialogue on drag \'n\' drop of folders.',
-
-    l_col_emphasis => 'Emphasised X-Mailer colour',
-    h_col_emphasis => 'The colour used for the X-Mailer line when its value is Claws Mail.',
-    l_col_log_err => 'Errors colour',
-    h_col_log_err => 'Colour for error messages in log window.',
-    l_col_log_in => 'Input colour',
-    h_col_log_in => 'Colour for input messages in log window.',
-    l_col_log_msg => 'Messages colour',
-    h_col_log_msg => 'Colour for messages in log window.',
-    l_col_log_out => 'Output colour',
-    h_col_log_out => 'Colour for output messages in log window.',
-    l_col_log_warn => 'Warnings colour',
-    h_col_log_warn => 'Colour for warning messages in log window.',
-
-    e_error => 'Error: ',
-    e_noclawsrc => 'no $HOME/.claws-mail/clawsrc file found.',
-    e_running => 'seems Claws Mail is currently running, close it first.',
+    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.'),
+    l_gui_no_markup => _('Don\'t use markup in compose window'),
+    h_gui_no_markup => _('Don\'t use bold and italic text in Compose dialogue\'s account selector.'),
+    l_gui_dot_lines => _('Use dotted lines in tree view components'),
+    h_gui_dot_lines => _('Use the old dotted line look in the main window tree views (Folder, Message and other lists) instead of the modern lineless look.'),
+    l_gui_h_scroll => _('Enable horizontal scrollbar'),
+    h_gui_h_scroll => _('Enable the horizontal scrollbar in the Message List.'),
+    l_gui_swp_from => _('Dislay To column instead From column in Sent folder'),
+    h_gui_swp_from => _('Display the recipient\'s email address in a To column of the Sent folder instead of the originator\'s one in a From column.'),
+    l_gui_v_scroll => _('Folder List scrollbar behaviour'),
+    h_gui_v_scroll => _('Specify the policy of vertical scrollbar of Folder List: show always, automatic or hide always.'),
+    l_gui_v_scroll_show => _('Show always'),
+    l_gui_v_scroll_auto => _('Automatic'),
+    l_gui_v_scroll_hide => _('Hide always'),
+    l_gui_strip_off => _('Coloured lines contrast'),
+    h_gui_strip_off => _('Specify the value to use when creating alternately coloured lines in tree view components. The smaller the value, the less visible the difference in the alternating colours of the lines.'),
+    l_gui_cursor_v => _('Show Cursor in message view'),
+    h_gui_cursor_v => _('Display the cursor in the message view.'),
+    l_gui_toolbar_d => _('Detachable toolbars'),
+    h_gui_toolbar_d => _('Show handles in the toolbars.'),
+    l_gui_strip_all => _('Use stripes in all tree view components'),
+    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.'),
+    l_beh_dangerous => _('Don\'t confirm deletions (dangerous!)'),
+    h_beh_dangerous => _('Don\'t ask for confirmation before definitive deletion of emails.'),
+    l_beh_flowed => _('Respect format=flowed in messages'),
+    h_beh_flowed => _('Respect format=flowed on text/plain message parts. This will cause some mails to have long lines, but will fix some URLs that would otherwise be wrapped.'),
+    l_beh_parts_rw => _('Allow writable temporary files'),
+    h_beh_parts_rw => _('Saves temporary files when opening attachment with write bit set.'),
+    l_beh_skip_ssl => _('Don\'t check SSL certificates'),
+    h_beh_skip_ssl => _('Disables the verification of SSL certificates.'),
+    l_beh_up_step => _('Progress bar update step (items)'),
+    h_beh_up_step => _('Update stepping in progress bars.'),
+    l_beh_thread_a => _('Maximum age when threading by subject (days)'),
+    h_beh_thread_a => _('Number of days to include a message in a thread when using "Thread using subject in addition to standard headers".'),
+    l_beh_unsafe_ssl => _('Allow unsafe SSL certificates'),
+    h_beh_unsafe_ssl => _('Allows Claws Mail to remember multiple SSL certificates for a given server/port.'),
+    l_beh_use_utf8 => _('Force UTF-8 for broken mails'),
+    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.'),
+    l_col_log_err => _('Error messages'),
+    h_col_log_err => _('Colour for error messages in log window.'),
+    l_col_log_in => _('Server messages'),
+    h_col_log_in => _('Colour for messages received from servers in log window.'),
+    l_col_log_msg => _('Standard messages'),
+    h_col_log_msg => _('Colour for messages in log window.'),
+    l_col_log_out => _('Client messages'),
+    h_col_log_out => _('Colour for messages sent to servers in log window.'),
+    l_col_log_warn => _('Warnings'),
+    h_col_log_warn => _('Colour for warning messages in log window.'),
+
+    e_error => _('Error: '),
+    e_noclawsrc => _('resource file for Claws Mail was not found.'),
+    e_running => _('seems Claws Mail is currently running, close it first.'),
+    e_requireddir => _('option requires a directory name.'),
+    e_requiredfile => _('option requires a file name.'),
+    e_notadir => _('specified name is not a directory or does not exist.'),
+    e_notafile => _('specified name is not a file or does not exist.'),
 );
 
-# load i18n messages if available
-my $pname = "clawsker";
-my @spp = split('/',$0);
-$spp[$#spp] = '';
-my $sps = join('/',@spp);
-my $lang = (defined($ENV{'LANG'}))? $ENV{'LANG'}: 'en';
-my $langf = "";
-for ($lang) {
-    /es.*/ && do { $langf = "es"; };
-}
-if ($langf ne "") { 
-    my $lf = $sps . $pname . "." . $langf . ".pl";
-    -x $lf or print "Warning: cant load $langf translation file\n";
-    do $lf unless ! -x $lf ;
-}
-
-# check if claws is running
-my $socket = "/tmp/claws-mail-$<";
--S $socket and die "$xl::s{e_error}$xl::s{e_running}\n";
-
 # all preferences read by load_preferences
 my %PREFS = ();
 # values of all preferences handled by clawsker
 my %HPVALUE = ();
+# default config dir and file name
+my $ALTCONFIGDIR = FALSE;
+my $CONFIGDIR = $ENV{HOME} . '/.claws-mail/';
+my $CONFIGRC = 'clawsrc';
 
 # index constants for preference arrays
 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;
+use constant FRAME_SPC => 2;
+use constant PAGE_SPC => 5;
+
+# version functions
+
+sub version_greater() {
+  my ($version, $refvers) = @_;
+  my @version = split (/\./, $version);
+  my @refvers = split (/\./, $refvers);
+  while ($#version < $#refvers) {
+    push (@version, '0');
+  }
+  my $idx = 0;
+  while (($idx <= $#refvers) 
+         and (int ($version[$idx]) == int ($refvers[$idx]))) {
+    ++$idx;
+  }
+  return TRUE if (($idx > $#refvers) 
+               or (int ($version[$idx]) > int ($refvers[$idx])));
+  return FALSE;
+}
+
+sub get_claws_version() {
+    my @cmbin = (
+        'claws-mail',
+    );
+    my $res = "";
+    foreach my $bin (@cmbin) {
+        $_ = qx/which $bin/;
+        chomp;
+        last if ($_ ne "");
+    }
+    return "" unless ($_); # not installed
+    $_ = qx/$_ -v/;
+    chomp;
+    my @fver = split (/ /);
+    die "Invalid version string" unless ($fver[2] eq "version");
+    my @ver = split (/\./, $fver[3]);
+    $res .= "$ver[0].";
+    $res .= "$ver[1].";
+    if ($ver[2] =~ /(\d+)cvs(\d+)/) {
+        $res .= "$1.$2";
+    }
+    else {
+        $res .= "$ver[2].0";
+    }
+    return $res;
+}
 
 # data handlers and auxiliar functions
 
@@ -245,6 +343,70 @@ sub handle_selection_value {
     $$dataref = $widget->get_active;
 }
 
+sub get_rc_filename {
+    return $CONFIGDIR . $CONFIGRC;
+}
+
+sub set_rc_filename {
+    my ($fullname) = @_;
+    my @parts = split ('/', $fullname);
+    $CONFIGRC = $parts[$#parts];
+    $parts[$#parts] = '';
+    $CONFIGDIR = join ('/', @parts);
+}
+
+sub log_message {
+    my ($mesg, $fatal) = @_;
+    if (defined($fatal) && $fatal eq 'die') {
+        die "$NAME: $mesg\n";
+    }
+    if ($VERBOSE) {
+        print "$NAME: $mesg\n";
+    }
+}
+
+sub error_dialog {
+    my ($emsg) = @_;
+    my $markup = "<span weight=\"bold\" size=\"large\">" . $emsg . "</span>";
+    my $errordlg = Gtk2::MessageDialog->new_with_markup ($main_window, 'modal', 'error', 'cancel', $markup);
+    $errordlg->set_title (_('Error message'));
+    $errordlg->run;
+    $errordlg->destroy;
+}
+
+sub check_claws_not_running() {
+    my $socket = (not $ALTCONFIGDIR)? "/tmp/": $CONFIGDIR;
+    $socket .= "claws-mail-$<";
+    -S $socket and do {
+        my $emsg = "$xl::s{e_error}$xl::s{e_running}";
+        log_message ($emsg);
+        error_dialog ($emsg);
+       return FALSE;
+     };
+
+     return TRUE;
+}
+
+sub check_rc_file() {
+    my ($rcfile) = @_;
+    (defined($rcfile) && -f $rcfile) or do {
+       my $emsg = "$xl::s{e_error}$xl::s{e_noclawsrc}\n";
+        log_message ($emsg);
+        error_dialog ($emsg);
+       return FALSE;
+    };
+
+    return TRUE;
+}
+
+sub set_widget_hint() {
+    if ($SHOWHINTS) {
+        my ($wdgt, $hint) = @_;    
+        $wdgt->set_tooltip_text ($hint);
+        $wdgt->set_has_tooltip (TRUE);
+    }
+}
+
 # graphic element creation 
 
 sub new_check_button_for {
@@ -252,33 +414,45 @@ sub new_check_button_for {
     my $name = $$hash{$key}[NAME];
     my $label = $$hash{$key}[LABEL];
     #
+    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});
         });
+    &set_widget_hint ($cb, $$hash{$key}[DESC]);
+    $hbox->pack_start ($cb, FALSE, FALSE, HBOX_SPC);
     #
-    return $cb;
+    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;
+    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});
         });
-    $hbox->pack_start ($glabel, FALSE, FALSE, 0);
-    $hbox->pack_start ($gentry, FALSE, FALSE, 0);
+    &set_widget_hint ($gentry, $$hash{$key}[DESC]);
+    $hbox->pack_start ($glabel, FALSE, FALSE, HBOX_SPC);
+    $hbox->pack_start ($gentry, FALSE, FALSE, HBOX_SPC);
     #
     return $hbox;
 }
@@ -294,12 +468,14 @@ sub new_color_button_for {
     my $button = Gtk2::ColorButton->new_with_color ($col);
     $$hash{$key}[GUI] = $button;
     $button->set_title ($label);
+    $button->set_relief ('none');
     $button->signal_connect ('color-set' => sub {
             my ($w, $e) = @_;
            &handle_color_value($w, $e, \$HPVALUE{$name}); 
         });
-    $hbox->pack_start ($button, FALSE, FALSE, 0);
-    $hbox->pack_start ($glabel, FALSE, FALSE, 0);
+    &set_widget_hint ($button, $$hash{$key}[DESC]);
+    $hbox->pack_start ($button, FALSE, FALSE, HBOX_SPC);
+    $hbox->pack_start ($glabel, FALSE, FALSE, HBOX_SPC);
     #
     return $hbox;
 }
@@ -323,8 +499,9 @@ sub new_selection_box_for {
            &handle_selection_value($w, $e, \$HPVALUE{$name});
         });
     $combo->set_active ($HPVALUE{$name});
-    $hbox->pack_start ($glabel, FALSE, FALSE, 0);
-    $hbox->pack_start ($combo, FALSE, FALSE, 0);
+    &set_widget_hint ($combo, $$hash{$key}[DESC]);
+    $hbox->pack_start ($glabel, FALSE, FALSE, HBOX_SPC);
+    $hbox->pack_start ($combo, FALSE, FALSE, HBOX_SPC);
     #
     return $hbox;
 }
@@ -337,43 +514,69 @@ 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,
     ],
 );
 
 sub new_other_page() {
     my $of = Gtk2::VBox->new (FALSE, 5);
+    $of->set_border_width (PAGE_SPC);
 
     my $ab_frame = Gtk2::Frame->new ($xl::s{ab_frame}); 
     my $cb_use_dlg = &new_check_button_for(\%pr::oth, 'use_dlg');
     my $vb1 = Gtk2::VBox->new (FALSE, 5);
+    $vb1->set_border_width (PAGE_SPC); 
     $vb1->pack_start ($cb_use_dlg, FALSE, FALSE, 0);
     $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);
 
-    $of->pack_start ($ab_frame, FALSE, FALSE, 0);
-    $of->pack_start ($mem_frame, FALSE, FALSE, 0);
+    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;
 }
@@ -384,13 +587,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 => [
@@ -398,6 +605,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 => [
@@ -405,6 +614,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 => [
@@ -412,6 +623,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 => [
@@ -419,13 +632,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 => [
@@ -433,6 +650,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 => [
@@ -440,6 +659,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 => [
@@ -447,6 +668,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 => [
@@ -454,36 +677,69 @@ 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,
     ],
 );
 
 sub new_gui_page() {
     my $gf = Gtk2::VBox->new (FALSE, 5);
+    $gf->set_border_width (PAGE_SPC);
+
+    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_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);
+    $vb1->pack_start ($cb_strip_sum, FALSE, FALSE, 0);
+    $vb1->pack_start ($tb_strip_off, FALSE, FALSE, 0);
+    $stripes_frame->add ($vb1);
 
+    my $mlist_frame = Gtk2::Frame->new ($xl::s{mlist_frame});
     my $cb_b_unread = &new_check_button_for (\%pr::gui, 'b_unread');
+    my $cb_swp_from = &new_check_button_for (\%pr::gui, 'swp_from');
+    my $vb3 = Gtk2::VBox->new (FALSE, 5);
+    $vb3->set_border_width (PAGE_SPC);
+    $vb3->pack_start ($cb_b_unread, FALSE, FALSE, 0);
+    $vb3->pack_start ($cb_swp_from, FALSE, FALSE, 0);
+    $mlist_frame->add ($vb3);
+
+    my $sbar_frame = Gtk2::Frame->new ($xl::s{sbar_frame});
+    my $cb_h_scroll = &new_check_button_for (\%pr::gui, 'h_scroll');
+    my $sb_v_scroll = &new_selection_box_for (\%pr::gui, 'v_scroll');
+    my $vb2 = Gtk2::VBox->new (FALSE, 5);
+    $vb2->set_border_width (PAGE_SPC);
+    $vb2->pack_start ($cb_h_scroll, FALSE, FALSE, 0);
+    $vb2->pack_start ($sb_v_scroll, FALSE, FALSE, 0);
+    $sbar_frame->add ($vb2);
+
     my $cb_no_markup = &new_check_button_for (\%pr::gui, 'no_markup'); 
     my $cb_dot_lines = &new_check_button_for (\%pr::gui, 'dot_lines'); 
-    my $cb_h_scroll = &new_check_button_for (\%pr::gui, 'h_scroll');
-    my $cb_swp_from = &new_check_button_for (\%pr::gui, 'swp_from');
     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_strip_all = &new_check_button_for (\%pr::gui, 'strip_all');
-    my $cb_strip_sum = &new_check_button_for (\%pr::gui, 'strip_sum');
-    my $sb_v_scroll = &new_selection_box_for (\%pr::gui, 'v_scroll');
-    my $tb_strip_off = &new_text_box_for (\%pr::gui, 'strip_off');
+    my $cb_two_linev = &new_check_button_for (\%pr::gui, 'two_linev');
 
-    $gf->pack_start ($cb_b_unread, FALSE, FALSE, 0);
+    $gf->pack_start ($stripes_frame, FALSE, FALSE, FRAME_SPC);
+    $gf->pack_start ($mlist_frame, FALSE, FALSE, FRAME_SPC);
     $gf->pack_start ($cb_no_markup, FALSE, FALSE, 0);
     $gf->pack_start ($cb_dot_lines, FALSE, FALSE, 0);
-    $gf->pack_start ($cb_h_scroll, FALSE, FALSE, 0);
-    $gf->pack_start ($cb_swp_from, 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_strip_all, FALSE, FALSE, 0);
-    $gf->pack_start ($cb_strip_sum, FALSE, FALSE, 0);
-    $gf->pack_start ($sb_v_scroll, FALSE, FALSE, 0);
-    $gf->pack_start ($tb_strip_off, FALSE, FALSE, 0);
+    $gf->pack_start ($cb_two_linev, FALSE, FALSE, 0);
+    $gf->pack_start ($sbar_frame, FALSE, FALSE, FRAME_SPC);
+
     return $gf;
 }
 
@@ -492,7 +748,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 => [
@@ -500,6 +758,8 @@ sub new_gui_page() {
         $xl::s{l_beh_dangerous},
         $xl::s{h_beh_dangerous},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     flowed => [
@@ -507,6 +767,8 @@ sub new_gui_page() {
         $xl::s{l_beh_flowed},
         $xl::s{h_beh_flowed},
        'bool',
+       '0.0.0',
+       '0',
        undef,
     ],
     parts_rw => [
@@ -514,6 +776,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 => [
@@ -521,20 +785,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 => [
@@ -542,6 +812,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 => [
@@ -549,6 +821,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 => [
@@ -556,34 +830,67 @@ 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,
     ],
 );
 
 sub new_behaviour_page() {
     my $bf = Gtk2::VBox->new (FALSE, 5);
-  
-    my $tb_hoover_t = &new_text_box_for (\%pr::beh, 'hover_t');
-    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 $cb_dangerous = &new_check_button_for (\%pr::beh, 'dangerous');
-    my $cb_flowed = &new_check_button_for (\%pr::beh, 'flowed');
-    my $cb_parts_rw = &new_check_button_for (\%pr::beh, 'parts_rw');
+    $bf->set_border_width (PAGE_SPC);
+
+    my $dnd_frame = Gtk2::Frame->new ($xl::s{dnd_frame});
+    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);
+    $vb1->pack_start ($cb_warn_dnd, FALSE, FALSE, 0);
+    $vb1->pack_start ($tb_hoover_t, FALSE, FALSE, 0);
+    $dnd_frame->add ($vb1);
+
+    my $ssl_frame = Gtk2::Frame->new ($xl::s{ssl_frame});
     my $cb_skip_ssl = &new_check_button_for (\%pr::beh, 'skip_ssl');
     my $cb_unsafe_ssl = &new_check_button_for (\%pr::beh, 'unsafe_ssl');
-    my $cb_use_utf8 = &new_check_button_for (\%pr::beh, 'use_utf8');
-    my $cb_warn_dnd = &new_check_button_for (\%pr::beh, 'warn_dnd');
+    my $hb1 = Gtk2::HBox->new (FALSE, 5);
+    $hb1->set_border_width (PAGE_SPC);
+    $hb1->pack_start ($cb_skip_ssl, FALSE, FALSE, 0);
+    $hb1->pack_start ($cb_unsafe_ssl, FALSE, FALSE, 0);
+    $ssl_frame->add ($hb1);
+
+    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');
 
-    $bf->pack_start ($tb_hoover_t, FALSE, FALSE, 0);
+    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);
+    $bf->pack_start ($ssl_frame, FALSE, FALSE, FRAME_SPC);
     $bf->pack_start ($tb_up_step, FALSE, FALSE, 0);
     $bf->pack_start ($tb_thread_a, FALSE, FALSE, 0);
-    $bf->pack_start ($cb_dangerous, FALSE, FALSE, 0);
-    $bf->pack_start ($cb_flowed, FALSE, FALSE, 0);
-    $bf->pack_start ($cb_parts_rw, FALSE, FALSE, 0);
-    $bf->pack_start ($cb_skip_ssl, FALSE, FALSE, 0);
-    $bf->pack_start ($cb_unsafe_ssl, FALSE, FALSE, 0);
-    $bf->pack_start ($cb_use_utf8, FALSE, FALSE, 0);
-    $bf->pack_start ($cb_warn_dnd, FALSE, FALSE, 0);
+    $bf->pack_start ($msgs_frame, FALSE, FALSE, FRAME_SPC);
+
     return $bf;
 }
 
@@ -593,6 +900,8 @@ sub new_behaviour_page() {
         $xl::s{l_col_emphasis},
         $xl::s{h_col_emphasis},
        'color',
+       '0.0.0',
+       '#0000cf',
        undef,
     ],
     log_err => [
@@ -600,6 +909,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 => [
@@ -607,6 +918,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 => [
@@ -614,6 +927,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 => [
@@ -621,6 +936,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 => [
@@ -628,16 +945,20 @@ sub new_behaviour_page() {
         $xl::s{l_col_log_warn},
         $xl::s{h_col_log_warn},
         'color',
+       '0.0.0',
+       '#af0000',
         undef,
     ],
 );
 
 sub new_colours_page() {
     my $cf = Gtk2::VBox->new (FALSE, 5);
+    $cf->set_border_width (PAGE_SPC);
 
     my $msgview_frame = Gtk2::Frame->new ($xl::s{msgview_frame}); 
     my $cb_emphasis = &new_color_button_for (\%pr::col, 'emphasis');
     my $vb1 = Gtk2::VBox->new (FALSE, 5);
+    $vb1->set_border_width (PAGE_SPC);
     $vb1->pack_start ($cb_emphasis, FALSE, FALSE, 0);
     $msgview_frame->add ($vb1);
     
@@ -648,6 +969,7 @@ sub new_colours_page() {
     my $cb_log_out = &new_color_button_for (\%pr::col, 'log_out');
     my $cb_log_warn = &new_color_button_for (\%pr::col, 'log_warn');
     my $vb2 = Gtk2::VBox->new (FALSE, 5);
+    $vb2->set_border_width (PAGE_SPC);
     $vb2->pack_start ($cb_log_err, FALSE, FALSE, 0);
     $vb2->pack_start ($cb_log_in, FALSE, FALSE, 0);
     $vb2->pack_start ($cb_log_msg, FALSE, FALSE, 0);
@@ -657,9 +979,102 @@ sub new_colours_page() {
 
     $cf->pack_start ($msgview_frame, FALSE, FALSE, 0);
     $cf->pack_start ($log_frame, FALSE, FALSE, 0);
+
     return $cf;
 }
 
+# version info
+sub print_version() {
+    print $xl::s{about_title} . "\n";
+    print $xl::s{about_version} . " $VERSION\n";
+    print "Perl-GLib " . $Glib::VERSION;
+    # version info stuff appeared in 1.040
+    if ($Glib::VERSION >= 1.040) {
+        print _(", built for ") . join(".", Glib->GET_VERSION_INFO) 
+              . _(", running with ") . join(".", &Glib::major_version, 
+                 &Glib::minor_version, &Glib::micro_version);
+    }
+    print "\n";
+    print "Perl-GTK2 " . $Gtk2::VERSION;
+    if ($Gtk2::VERSION >= 1.040) {
+        print _(", built for ") . join(".", Gtk2->GET_VERSION_INFO)
+             . _(", running with ") . join(".", &Gtk2::major_version, 
+                 &Gtk2::minor_version, &Gtk2::micro_version);
+    }
+    print "\n";
+}
+
+# the command line help
+sub print_help() {
+    my $line = '-' x length ($xl::s{about_title}) . "\n";
+    print $line;
+    print $xl::s{about_title} . "\n";
+    print $line;
+    print _("Syntax:\n");
+    print _("    clawsker [options]\n");
+    print _("Options:\n");
+    print _("    --help                         Prints this help screen.\n");
+    print _("    --version                      Prints version infos.\n");
+    print _("    --verbose                      More messages on standard output.\n");
+    print _("    --alternate-config-dir <dir>   Uses <dir> as Claws Mail config dir.\n");
+    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;
+    while (defined($ARGV[$arg])) {
+        for ($ARGV[$arg]) {
+            /--help/ && do { 
+               &print_help; 
+                return FALSE;
+           };
+            /--version/ && do { 
+               &print_version; 
+                return FALSE;
+           };
+           /--verbose/ && do {
+               $VERBOSE = TRUE;
+               last;
+           };
+           /--alternate-config-dir/ && do {
+               ++$arg;
+               &command_line_fatal ($xl::s{e_requireddir})
+                   unless defined($ARGV[$arg]);
+               &command_line_fatal ($xl::s{e_notadir})
+                   unless -d $ARGV[$arg];
+               $CONFIGDIR = $ARGV[$arg];
+               $CONFIGDIR .= "/" 
+                   unless ($CONFIGDIR =~ /.*\/$/);
+               $ALTCONFIGDIR = TRUE;
+               last;
+           };
+           /--clawsrc/ && do {
+               ++$arg;
+               &command_line_fatal($xl::s{e_requiredfile}) 
+                   unless defined($ARGV[$arg]);
+               &command_line_fatal($xl::s{e_notafile}) 
+                   unless -f $ARGV[$arg];
+               &set_rc_filename ($ARGV[$arg]);
+               last;
+           };
+           /.*/ && &command_line_fatal (
+                       _("unknown option '{opt}'.\n", opt => $ARGV[$arg]));
+       }
+       ++$arg;
+    }
+    # eveything continues...
+    return TRUE;
+}
+
 # update the hidden preferences status from loaded values
 sub init_hidden_preferences() {
     foreach my $hash (\%pr::beh, \%pr::col, \%pr::gui, \%pr::oth) {
@@ -667,12 +1082,15 @@ sub init_hidden_preferences() {
            $HPVALUE{${$hash}{$key}[NAME]} = $PREFS{${$hash}{$key}[NAME]};
         }
     }
+    return TRUE;
 }
 
 # load current status from disc
 sub load_preferences() {
-    my $rc = $ENV{HOME} . '/.claws-mail/clawsrc';
-    -f $rc or die "$xl::s{e_error}$xl::s{e_noclawsrc}\n";
+    my $rc = &get_rc_filename;
+    &log_message ("Loading preferences from $rc\n");
+    return FALSE unless &check_rc_file ($rc);
+    return FALSE unless &check_claws_not_running;
     open (RCF, "<$rc");
     while (<RCF>) {
        chomp;
@@ -681,11 +1099,15 @@ sub load_preferences() {
         }
     }
     close (RCF);
+    return TRUE;
 }
 
 # save current preferences to disc
 sub save_preferences() {
-    my $rc = $ENV{HOME} . '/.claws-mail/clawsrc';
+    my $rc = &get_rc_filename;
+    &log_message ("Saving preferences to $rc\n");
+    return FALSE unless &check_rc_file ($rc);
+    return FALSE unless &check_claws_not_running;
     my $rcbak = "$rc.backup";
     rename ($rc, $rcbak);
     open (RCF, ">$rc");
@@ -706,6 +1128,7 @@ sub save_preferences() {
     }
     close (RCB);
     close (RCF);
+    return TRUE;
 }
 
 # create notebook
@@ -725,6 +1148,7 @@ sub new_about_dialog() {
     my ($parent) = @_;
     my $title = $xl::s{about_title};
     my $lic = $xl::s{about_license};
+    my $vers = $xl::s{about_version} . " $VERSION";
     my $license = 
 "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
@@ -738,11 +1162,17 @@ GNU General Public License for more details.
 
 You should have received a copy of the GNU General Public License
 along with this program.  If not, see &lt;http://www.gnu.org/licenses/&gt;.";
+    my $year = "2007";
+    my $holder = "Ricardo Mones &lt;ricardo\@mones.org&gt;";
 
     my $dialog = Gtk2::MessageDialog->new_with_markup ($parent, 
                     [qw/modal destroy-with-parent/], 
                    'info', 'close', 
-                   "<span size=\"x-large\" weight=\"bold\">$title</span>\n<span>Copyright 2007 by Ricardo Mones &lt;ricardo\@mones.org&gt;</span>\n\n<span size=\"large\">$lic</span>\n\n<span size=\"small\">$license</span>");
+                   "<span size=\"x-large\" weight=\"bold\">$title</span>\n"
+                   . "<span size=\"large\">$vers</span>\n"
+                   . "<span>Copyright $year by $holder</span>\n\n"
+                   . "<span size=\"large\">$lic</span>\n\n"
+                   . "<span size=\"small\">$license</span>");
     $dialog->set_title ($xl::s{about});
     #
     return $dialog;
@@ -750,7 +1180,7 @@ along with this program.  If not, see &lt;http://www.gnu.org/licenses/&gt;.";
 
 # create buttons box
 sub new_button_box() {
-    my ($adlg) = @_;
+    my ($parent, $adlg) = @_;
     my $b_about = Gtk2::Button->new_from_stock ('gtk-about');
     my $b_exit = Gtk2::Button->new_from_stock ('gtk-quit');
     my $b_apply = Gtk2::Button->new_from_stock ('gtk-apply');
@@ -759,30 +1189,32 @@ sub new_button_box() {
     my $hbox = Gtk2::HBox->new (FALSE, 5);
     # signal handlers 
     $b_exit->signal_connect (clicked => sub { Gtk2->main_quit });
-    $b_apply->signal_connect (clicked => sub { &save_preferences });
+    $b_apply->signal_connect (clicked => sub { &save_preferences ($parent) });
     # $b_undo->signal_connect (clicked => sub { &undo_current_changes });
     $b_about->signal_connect (clicked => sub { $adlg->run; $adlg->hide });
     # package them
-    $hbox->pack_end ($b_exit, FALSE, FALSE, 0);
     $hbox->pack_end ($b_apply, FALSE, FALSE, 0);
+    $hbox->pack_end ($b_exit, FALSE, FALSE, 0);
     # $hbox->pack_end ($b_undo, FALSE, FALSE, 0);
     $hbox->pack_start ($b_about, FALSE, FALSE, 0);
     #
     return $hbox;
 }
 
-# initialise values
-&load_preferences;
-&init_hidden_preferences;
+# initialise
+$main_window = Gtk2::Window->new ('toplevel');
+exit unless &parse_command_line;
+exit unless &load_preferences;
+exit unless &init_hidden_preferences;
 # create main GUI
-my $window = Gtk2::Window->new ('toplevel');
 my $box = Gtk2::VBox->new (FALSE, 5);
+$box->set_border_width(3);
 my $about = &new_about_dialog;
 $box->pack_start (&new_notebook, FALSE, FALSE, 0);
-$box->pack_end (&new_button_box ($about), FALSE, FALSE, 0);
-
-$window->set_title ($xl::s{win_title});
-$window->add ($box);
-$window->show_all;
+$box->pack_end (&new_button_box ($main_window, $about), FALSE, FALSE, 0);
+$main_window->signal_connect (delete_event => sub { Gtk2->main_quit });
+$main_window->set_title ($xl::s{win_title});
+$main_window->add ($box);
+$main_window->show_all;
 Gtk2->main;