summaryrefslogtreecommitdiffstats
path: root/perl-install/interactive.pm
diff options
context:
space:
mode:
Diffstat (limited to 'perl-install/interactive.pm')
-rw-r--r--perl-install/interactive.pm798
1 files changed, 648 insertions, 150 deletions
diff --git a/perl-install/interactive.pm b/perl-install/interactive.pm
index 2cd911741..49f47d05c 100644
--- a/perl-install/interactive.pm
+++ b/perl-install/interactive.pm
@@ -1,4 +1,4 @@
-package interactive; # $Id$
+package interactive;
use diagnostics;
use strict;
@@ -6,77 +6,174 @@ use strict;
#-######################################################################################
#- misc imports
#-######################################################################################
-use common qw(:common :functional);
-
-#- ask_from_entries takes:
-#- val => reference to the value
-#- label => description
-#- icon => icon to put before the description
-#- help => tooltip
-#- advanced => wether it is shown in by default or only in advanced mode
-#- disabled => function returning wether it should be disabled (grayed)
-#- type =>
-#- button => (with clicked) (type defaults to button if clicked is there) (val need not be a reference)
-#- label => (val need not be a reference) (type defaults to label if val is not a reference)
-#- bool (with text)
-#- range (with min, max)
-#- combo (with list, not_edit)
-#- list (with list, icon2f (aka icon), separator (aka tree), format (aka pre_format function),
-#- help can be a hash or a function)
-#- entry (the default) (with hidden)
-#
-#- heritate from this class and you'll get all made interactivity for same steps.
-#- for this you need to provide
-#- - ask_from_listW(o, title, messages, arrayref, default) returns one string of arrayref
-#-
-#- where
-#- - o is the object
-#- - title is a string
-#- - messages is an refarray of strings
-#- - default is an optional string (default is in arrayref)
-#- - arrayref is an arrayref of strings
-#- - arrayref2 contains booleans telling the default state,
-#-
-#- ask_from_list and ask_from_list_ are wrappers around ask_from_biglist and ask_from_smalllist
-#-
-#- ask_from_list_ just translate arrayref before calling ask_from_list and untranslate the result
-#-
-#- ask_from_listW should handle differently small lists and big ones.
-#-
+use common;
+use do_pkgs;
+=head1 NAME
+
+interactive - a GUI layer with multiple backend (text console, Gtk+ GUI, web)
+
+=head1 SYNOPSYS
+
+B<interactive> enables to write GUIes that will work everywhere:
+
+=head1 Functions
+
+=over 4
+
+=item * text console
+
+implemented by L<interactive::stdio> & L<interactive::curses>
+
+=item * web browser
+
+implemented by L<interactive::http>
+
+=item * GUI
+
+implemented by L<interactive::gtk>
+
+=back
+
+Interactive inherits from L<do_pkgs> and thus $in->do_pkgs will return
+an usable C<do_pkgs> object suitable for installing packages.
+
+=head1 Minimal example using interactive
+
+ use lib qw(/usr/lib/libDrakX);
+ use interactive;
+ my $in = interactive->vnew;
+ $in->ask_okcancel('title', 'question');
+ $in->exit;
+
+=head1 Backends
+
+heritate from this class and you'll get all made interactivity for same steps.
+for this you need to provide
+
+C<ask_from_listW(o, title, messages, arrayref, default)> which returns one string of arrayref
+
+where:
+
+=over 4
+
+=item * B<o> is the object
+
+=item * B<title> is a string
+
+=item * B<messages> is an refarray of strings
+
+=item * B<default> is an optional string (default is in arrayref)
+
+=item * B<arrayref> is an arrayref of strings
+
+=item * B<arrayref>2 contains booleans telling the default state,
+
+=back
+
+=head1 Functions
+
+=over
+
+=cut
#-######################################################################################
#- OO Stuff
#-######################################################################################
+our @ISA = qw(do_pkgs);
+
sub new($) {
my ($type) = @_;
- bless {}, ref $type || $type;
+ bless {}, ref($type) || $type;
}
sub vnew {
- my ($type, $su) = @_;
- $su = $su eq "su";
+ my ($_type, $o_su, $o_icon) = @_;
+ my $su = $o_su eq "su";
+ if ($ENV{INTERACTIVE_HTTP}) {
+ require interactive::http;
+ return interactive::http->new;
+ }
require c;
- if ($ENV{DISPLAY} && system('/usr/X11R6/bin/xtest') == 0) {
- if ($su) {
- $ENV{PATH} = "/sbin:/usr/sbin:$ENV{PATH}";
- if ($>) {
- exec("kdesu", "-c", "$0 @ARGV") or die _("kdesu missing");
- }
+ if ($su) {
+ $ENV{PATH} = "/sbin:/usr/sbin:$ENV{PATH}";
+ $su = '' if $::testing || $ENV{TESTING};
+ }
+ require_root_capability() if $su;
+ if (check_for_xserver()) {
+ eval { require interactive::gtk };
+ if (!$@) {
+ my $o = interactive::gtk->new;
+ if ($o_icon && $o_icon ne 'default' && !$::isWizard) { $o->{icon} = $o_icon } else { undef $o->{icon} }
+ return $o;
+ } elsif ($::testing) {
+ die;
}
- eval { require interactive_gtk };
- !$@ and return interactive_gtk->new;
}
- if ($su && $>) {
- die "you must be root to run this program";
+ require interactive::curses;
+ interactive::curses->new;
+}
+
+sub ok { N_("Ok") }
+sub cancel { N_("Cancel") }
+
+sub markup_parse {
+ my ($s) = @_;
+ my @l;
+ my @attrs;
+
+ while ($s) {
+ if ($s =~ s!^<(\w+)(\s+[^>]*?)?>!!s) {
+ push @attrs, [ $1, $2 ];
+ } elsif ($s =~ s!^</(\w+)>!!) {
+ my $previous = pop @attrs;
+ $previous->[0] eq $1 or return;
+ } elsif ($s =~ s!^(&(amp|lt|gt);)!!) {
+ push @l, [ $1, @attrs ];
+ } elsif ($s =~ s!^([^<>&]+)!!s) {
+ push @l, [ $1, @attrs ];
+ } else {
+ return;
+ }
}
- require 'log.pm';
- undef *log::l;
- *log::l = sub {}; # otherwise, it will bother us :(
- require interactive_newt;
- interactive_newt->new;
+ markup_simplify(\@l);
+ \@l;
+}
+
+sub markup_simplify {
+ my ($l) = @_;
+ foreach (@$l) {
+ my ($s, @attrs) = @$_;
+ my %attrs = map {
+ my ($tag, $attrs) = @$_;
+ my $long = { b => { weight => "bold" },
+ i => { style => "italic" },
+ big => { size => 'larger' },
+ }->{$tag};
+ $long ? %$long : map { /^(.*?)=['"]?(.*?)['"]?$/ } split(' ', $attrs);
+ } @attrs;
+
+ $s = +{ '&amp;' => '&', '&lt;' => '<', '&gt;' => '>' }->{$s} || $s;
+
+ @$_ = ($s, if_(%attrs, \%attrs));
+ }
+}
+
+sub markup_remove {
+ my ($s) = @_;
+ if (my $l = markup_parse($s)) {
+ join('', map { $_->[0] } @$l);
+ } else {
+ $s;
+ }
+}
+
+#- drop markup as fallback
+sub adapt_markup {
+ my ($_o, $s) = @_;
+ markup_remove($s);
}
sub enter_console {}
@@ -84,84 +181,146 @@ sub leave_console {}
sub suspend {}
sub resume {}
sub end {}
-sub exit { exit($_[0]) }
+sub exit {
+ if ($::isStandalone) {
+ require standalone;
+ standalone::exit($_[0]);
+ } else {
+ exit($_[0]);
+ }
+}
+
#-######################################################################################
#- Interactive functions
#-######################################################################################
sub ask_warn {
- my ($o, $title, $message) = @_;
- local $::isWizard=0;
- ask_from_listf_no_check($o, $title, $message, undef, [ _("Ok") ]);
+ my ($o, $title, $message, $o_icon) = @_;
+ ask_warn_($o, { title => $title, messages => $message, icon => $o_icon });
}
-
sub ask_yesorno {
- my ($o, $title, $message, $def, $help) = @_;
- ask_from_list_($o, $title, $message, [ __("Yes"), __("No") ], $def ? "Yes" : "No", $help) eq "Yes";
+ my ($o, $title, $message, $b_def) = @_;
+ ask_yesorno_($o, { title => $title, messages => $message }, $b_def);
}
-
sub ask_okcancel {
- my ($o, $title, $message, $def, $help) = @_;
+ my ($o, $title, $message, $b_def, $o_icon) = @_;
+ ask_okcancel_($o, { title => $title, messages => $message, icon => $o_icon }, $b_def);
+}
+
+sub ask_warn_ {
+ my ($o, $common) = @_;
+ local $o->{modal} = $o->{modal} || $::isInstall; # make these popup at install time
+ ask_from_listf_raw_no_check($o, $common, \&translate, [ $o->ok ]);
+}
+
+sub ask_yesorno_ {
+ my ($o, $common, $b_def) = @_;
+ $common->{cancel} = '';
+ ask_from_listf_raw($o, $common, \&translate, [ N_("Yes"), N_("No") ], $b_def ? "Yes" : "No") eq "Yes";
+}
+
+sub ask_okcancel_ {
+ my ($o, $common, $b_def) = @_;
if ($::isWizard) {
- $::no_separator = 1;
- $o->ask_from_entries_refH_powered_no_check({ title => $title, messages => $message, focus_cancel => !$def });
+ $common->{focus_cancel} = !$b_def;
+ ask_from_no_check($o, $common, []);
} else {
- ask_from_list_($o, $title, $message, [ __("Ok"), __("Cancel") ], $def ? "Ok" : "Cancel", $help) eq "Ok";
+ ask_from_listf_raw($o, $common, \&translate, [ $o->ok, $o->cancel ], $b_def ? $o->ok : $o->cancel) eq $o->ok;
}
}
+sub ask_filename {
+ my ($o, $common) = @_;
+ $common->{want_a_dir} = 0;
+ $o->ask_fileW($common);
+}
+
+sub ask_directory {
+ my ($o, $common) = @_;
+ $common->{want_a_dir} = 1;
+ $o->ask_fileW($common);
+}
+
+#- predecated
+sub ask_file {
+ my ($o, $title, $o_dir) = @_;
+ $o->ask_fileW({ title => $title, want_a_dir => 0, directory => $o_dir });
+}
+
+sub ask_fileW {
+ my ($o, $common) = @_;
+ $o->ask_from_entry($common->{title}, $common->{message} || N("Choose a file"));
+}
+
+=item ask_from_list($o, $title, $message, $l, $o_def)
+
+=item ask_from_list_($o, $title, $message, $l, $o_def)
+
+ask_from_list() and ask_from_list_() are wrappers around ask_from_biglist and ask_from_smalllist
+
+ask_from_list_() just translate arrayref before calling ask_from_list and untranslate the result
+
+ask_from_listW() should handle differently small lists and big ones.
+
+=cut
+
sub ask_from_list {
- my ($o, $title, $message, $l, $def, $help) = @_;
- ask_from_listf($o, $title, $message, undef, $l, $def, $help);
+ my ($o, $title, $message, $l, $o_def) = @_;
+ ask_from_listf($o, $title, $message, undef, $l, $o_def);
}
sub ask_from_list_ {
- my ($o, $title, $message, $l, $def, $help) = @_;
- ask_from_listf($o, $title, $message, sub { translate($_[0]) }, $l, $def, $help);
+ my ($o, $title, $message, $l, $o_def) = @_;
+ ask_from_listf($o, $title, $message, \&translate, $l, $o_def);
}
sub ask_from_listf_ {
- my ($o, $title, $message, $f, $l, $def, $help) = @_;
- ask_from_listf($o, $title, $message, sub { translate($f->(@_)) }, $l, $def, $help);
+ my ($o, $title, $message, $f, $l, $o_def) = @_;
+ ask_from_listf($o, $title, $message, sub { translate($f->(@_)) }, $l, $o_def);
}
sub ask_from_listf {
- my ($o, $title, $message, $f, $l, $def, $help) = @_;
- @$l == 0 and die 'ask_from_list: empty list';
+ my ($o, $title, $message, $f, $l, $o_def) = @_;
+ ask_from_listf_raw($o, { title => $title, messages => $message }, $f, $l, $o_def);
+}
+sub ask_from_listf_raw {
+ my ($_o, $_common, $_f, $l, $_o_def) = @_;
+ @$l == 0 and die "ask_from_list: empty list\n" . backtrace();
@$l == 1 and return $l->[0];
- goto &ask_from_listf_no_check;
+ goto &ask_from_listf_raw_no_check;
}
-sub ask_from_listf_no_check {
- my ($o, $title, $message, $f, $l, $def, $help) = @_;
-
- if (@$l <= 2 && !$::isWizard) {
- my $ret = eval {
- ask_from_entries_refH_powered_no_check($o,
- { title => $title, messages => $message, ok => $l->[0] && may_apply($f, $l->[0]),
- if_($l->[1], cancel => may_apply($f, $l->[1]), focus_cancel => $def eq $l->[1]) }, []
- ) ? $l->[0] : $l->[1];
- };
- die if $@ && $@ !~ /^wizcancel/;
- $@ ? undef : $ret;
- } else {
- ask_from_entries_refH($o, $title, $message, [ { val => \$def, type => 'list', list => $l, help => $help, format => $f } ]) && $def;
+sub ask_from_listf_raw_no_check {
+ my ($o, $common, $f, $l, $o_def) = @_;
+
+ if (@$l <= ($::isWizard ? 1 : 2)) {
+ my ($ok, $cancel) = map { $_ && may_apply($f, $_) } @$l;
+ if (length "$ok$cancel" < 70) {
+ my $ret = eval {
+ put_in_hash($common, { ok => $ok,
+ if_($cancel, cancel => $cancel, focus_cancel => $o_def eq $l->[1]) });
+ ask_from_no_check($o, $common, []) ? $l->[0] : $l->[1];
+ };
+ die if $@ && $@ !~ /^wizcancel/;
+ return $@ ? undef : $ret;
+ }
}
+ ask_from_no_check($o, $common, [ { val => \$o_def, type => 'list', list => $l, format => $f } ]) && $o_def;
}
sub ask_from_treelist {
- my ($o, $title, $message, $separator, $l, $def) = @_;
- ask_from_treelistf($o, $title, $message, $separator, undef, $l, $def);
+ my ($o, $title, $message, $separator, $l, $o_def) = @_;
+ ask_from_treelistf($o, $title, $message, $separator, undef, $l, $o_def);
}
sub ask_from_treelist_ {
- my ($o, $title, $message, $separator, $l, $def) = @_;
+ my ($o, $title, $message, $separator, $l, $o_def) = @_;
my $transl = sub { join '|', map { translate($_) } split(quotemeta($separator), $_[0]) };
- ask_from_treelistf($o, $title, $message, $separator, $transl, $l, $def);
+ ask_from_treelistf($o, $title, $message, $separator, $transl, $l, $o_def);
}
sub ask_from_treelistf {
- my ($o, $title, $message, $separator, $f, $l, $def) = @_;
- ask_from_entries_refH($o, $title, $message, [ { val => \$def, separator => $separator, list => $l, format => $f } ]);
- $def;
+ my ($o, $title, $message, $separator, $f, $l, $o_def) = @_;
+ ask_from($o, $title, $message, [ { val => \$o_def, separator => $separator, list => $l, format => $f, sort => 1 } ]) or return;
+ $o_def;
}
sub ask_many_from_list {
@@ -184,13 +343,13 @@ sub ask_many_from_list {
$h->{list} = [ sort { $h->{e}{$a}{text} cmp $h->{e}{$b}{text} } @{$h->{list}} ];
}
}
- $o->ask_from_entries_refH($title, $message, [ map { my $h = $_; map { $h->{e}{$_} } @{$h->{list}} } @l ]) or return;
+ $o->ask_from($title, $message, [ map { my $h = $_; map { $h->{e}{$_} } @{$h->{list}} } @l ]) or return;
@l = map {
my $h = $_;
[ grep { ${$h->{e}{$_}{val}} } @{$h->{list}} ];
} @l;
- wantarray ? @l : $l[0];
+ wantarray() ? @l : $l[0];
}
sub ask_from_entry {
@@ -202,86 +361,425 @@ sub ask_from_entries {
my @l = map { my $i = ''; { label => $_, val => \$i } } @$l;
- $o->ask_from_entries_refH($title, $message, \@l, %callback) ?
- map { ${$_->{val}} } @l :
- undef;
+ $o->ask_from_({ title => $title, messages => $message, %callback,
+ focus_first => 1 }, \@l) or return;
+ map { ${$_->{val}} } @l;
}
-#- can get a hash of callback: focus_out changed and complete
+sub ask_from__add_modify_remove {
+ my ($o, $title, $message, $l, %callback) = @_;
+ die "ask_from__add_modify_remove only handles one item" if @$l != 1;
+
+ $callback{$_} or internal_error("missing callback $_") foreach qw(Add Modify Remove);
+
+ if ($o->can('ask_from__add_modify_removeW')) {
+ $o->ask_from__add_modify_removeW($title, $message, $l, %callback);
+ } else {
+ my $e = $l->[0];
+ my $chosen_element;
+ put_in_hash($e, { allow_empty_list => 1, val => \$chosen_element, type => 'list' });
+
+ while (1) {
+ my $continue;
+ my @l = (@$l,
+ map { my $s = $_; { val => translate($_), clicked_may_quit => sub {
+ my $r = $callback{$s}->($chosen_element);
+ defined $r or return;
+ $continue = 1;
+ } } }
+ N_("Add"), if_(@{$e->{list}} > 0, N_("Modify"), N_("Remove")));
+ $o->ask_from_({ title => $title, messages => $message, %callback }, \@l) or return;
+ return 1 if !$continue;
+ }
+ }
+}
+
+
+#- can get a hash of callback: validate
#- moreove if you pass a hash with a field list -> combo
#- if you pass a hash with a field hidden -> emulate stty -echo
-sub ask_from_entries_refH {
+sub ask_from {
my ($o, $title, $message, $l, %callback) = @_;
- ask_from_entries_refH_powered($o, { title => $title, messages => $message, callbacks => \%callback }, $l);
+ ask_from_($o, { title => $title, messages => $message, %callback }, $l);
}
-sub ask_from_entries_refH_powered_normalize {
- my ($o, $common, $l) = @_;
+sub _normalize_entry {
+ my ($o, $e) = @_;
- foreach my $e (@$l) {
- if (my $l = $e->{list}) {
- if ($e->{sort} || @$l > 10 && !exists $e->{sort}) {
- my @l2 = map { may_apply($e->{format}, $_) } @$l;
- my @places = sort { $l2[$a] cmp $l2[$b] } 0 .. $#l2;
- $e->{list} = $l = [ map { $l->[$_] } @places ];
+ if (my $li = $e->{list}) {
+ ref($e->{val}) =~ /SCALAR|REF/ or internal_error($e->{val} ? "field {val} must be a reference (it is $e->{val})" : "field {val} is mandatory"); #-#
+ if ($e->{sort} || @$li > 10 && !exists $e->{sort}) {
+ my @l2 = map { may_apply($e->{format}, $_) } @$li;
+ my @places = sort { $l2[$a] cmp $l2[$b] } 0 .. $#l2;
+ $e->{list} = $li = [ map { $li->[$_] } @places ];
+ }
+ $e->{type} = 'iconlist' if $e->{icon2f};
+ $e->{type} = 'treelist' if $e->{separator} && $e->{type} ne 'combo';
+ add2hash_($e, { not_edit => 1 });
+ $e->{type} ||= 'combo';
+
+ if (!$e->{not_edit}) {
+ die q(when using "not_edit" you must use strings, not a data structure) if ref(${$e->{val}}) || any { ref $_ } @$li;
+ }
+ if ($e->{type} ne 'combo' || $e->{not_edit}) {
+ ${$e->{val}} = $li->[0] if !member(may_apply($e->{format}, ${$e->{val}}), map { may_apply($e->{format}, $_) } @$li);
+ }
+ } elsif ($e->{type} eq 'range') {
+ $e->{min} <= $e->{max} or die "bad range min $e->{min} > max $e->{max} (called from " . join(':', caller()) . ")";
+ ${$e->{val}} = max($e->{min}, min(${$e->{val}}, $e->{max}));
+ } elsif ($e->{type} eq 'button' || $e->{clicked} || $e->{clicked_may_quit}) {
+ $e->{type} = 'button';
+ $e->{clicked_may_quit} ||= $e->{clicked} ? sub { $e->{clicked}(); 0 } : sub {};
+ $e->{val} = \ (my $_v = $e->{val}) if !ref($e->{val});
+ } elsif (!$e->{type} && !$e->{val}) {
+ $e->{type} = 'only_label';
+ $e->{val} = \ (my $_v = $o->adapt_markup(delete $e->{label}));
+ } elsif ($e->{type} eq 'label' || !ref($e->{val})) {
+ $e->{type} = 'label';
+ $e->{val} = \ (my $_v = $e->{val}) if !ref($e->{val});
+ } elsif ($e->{type} eq 'expander') {
+ _normalize_entries($o, $e->{children});
+ } else {
+ $e->{type} ||= 'entry';
+ }
+ $e->{label} = $o->adapt_markup($e->{label}) if $e->{label};
+}
+
+sub _normalize_entries {
+ my ($o, $l) = @_;
+
+ ref($l) eq 'ARRAY' or internal_error('ask_from_normalize');
+
+ _normalize_entry($o, $_) foreach @$l;
+
+ #- do not display empty lists and one element lists
+ @$l = grep {
+ if ($_->{list} && $_->{not_edit} && !$_->{allow_empty_list}) {
+ if (!@{$_->{list}}) {
+ eval {
+ require 'log.pm'; #- "require log" causes some pb, perl thinking that "log" is the log() function
+ log::l("ask_from_normalize: empty list for $_->{label}\n" . backtrace());
+ };
}
- $e->{type} = 'iconlist' if $e->{icon2f};
- $e->{type} = 'treelist' if $e->{separator};
- $e->{type} ||= 'list' if $e->{format};
- add2hash_($e, { not_edit => 1, type => 'combo' });
- ${$e->{val}} = $l->[0] if ($e->{type} ne 'combo' || $e->{not_edit}) && !member(${$e->{val}}, @$l);
- } elsif ($e->{type} eq 'range') {
- $e->{min} <= $e->{max} or die "bad range min $e->{min} > max $e->{max} (called from " . join(':', caller()) . ")";
- ${$e->{val}} = max($e->{min}, min(${$e->{val}}, $e->{max}));
- } elsif ($e->{type} eq 'button' || $e->{clicked}) {
- $e->{type} = 'button';
- $e->{clicked} ||= sub {};
- $e->{val} = \ (my $v = $e->{val}) if !ref($e->{val});
- } elsif ($e->{type} eq 'label' || !ref($e->{val})) {
- $e->{type} = 'label';
- $e->{val} = \ (my $v = $e->{val}) if !ref($e->{val});
+ @{$_->{list}} > 1;
+ } else {
+ 1;
+ }
+ } @$l;
+}
+
+sub ask_from_normalize {
+ my ($o, $common, $l) = @_;
+
+ if ($common->{focus_first}) {
+ if (my $e = find { $_->{val} } @$l) {
+ $e->{focus} = sub { 1 };
}
- $e->{disabled} ||= sub { 0 };
}
- #- don't display empty lists
- @$l = grep { !($_->{list} && @{$_->{list}} == () && $_->{not_edit}) } @$l;
+ _normalize_entries($o, $l);
- $common->{advanced_label} ||= _("Advanced");
- $common->{$_} = [ deref($common->{$_}) ] foreach qw(messages advanced_messages);
- add2hash_($common, { ok => _("Ok"), cancel => _("Cancel") }) if !exists $common->{ok} && !$::isWizard;
- add2hash_($common->{callbacks} ||= {}, { changed => sub {}, focus_out => sub {}, complete => sub { 0 }, canceled => sub { 0 } });
+ if (!$common->{title} && $::isStandalone) {
+ ($common->{title} = $0) =~ s|.*/||;
+ }
+ $common->{interactive_help} ||= $o->{interactive_help};
+ $common->{interactive_help} ||= $common->{interactive_help_id} && $o->interactive_help_sub_get_id($common->{interactive_help_id});
+ if (!$::isInstall) {
+ delete $common->{$_} foreach qw(interactive_help interactive_help_id);
+ }
+ $common->{advanced_label} ||= N("Advanced");
+ $common->{advanced_label_close} and log::l("advanced_label_close is not used anymore");
+ $common->{$_} = $common->{$_} ? [ map { $o->adapt_markup($_) } deref($common->{$_}) ] : []
+ foreach qw(messages advanced_messages);
+
+ if ($common->{callbacks}) {
+ $common->{callbacks}{changed} and internal_error(q(global "changed" callback is not handled anymore, use a per-entry changed callback));
+ $common->{callbacks}{focus_out} and internal_error(q(global "focus_out" callback is not handled anymore, use a per-entry focus_out callback));
+ add2hash($common, delete $common->{callbacks});
+ }
+ if (my $complete = delete $common->{complete}) {
+ $common->{validate} = sub { !first($complete->()) };
+ }
+ add2hash_($common, { validate => sub { 1 } });
+}
+
+sub migrate_advanced {
+ my ($common, $l) = @_;
+ my ($l1, $l2) = partition { !$_->{advanced} } @$l;
+ my $advanced_message = join("\n", @{$common->{advanced_messages}});
+ [ @$l1, if_(@$l2, { type => 'expander',
+ if_($advanced_message, message => $advanced_message),
+ text => $common->{advanced_label},
+ expanded => $common->{advanced_state},
+ children => $l2,
+ }) ];
}
-sub ask_from_entries_refH_powered {
+
+=item ask_from_($o, $common, $l)
+
+ask_from_() takes global options ($common):
+
+=over 4
+
+=item * B<title>: window title
+
+=item * B<messages>: message displayed in the upper part of the window
+
+=item * B<ok>: force the name of the "Ok"/"Next" button
+
+=item * B<cancel>: force the name of the "Cancel"/"Previous" button
+
+=item * B<focus_cancel>: force focus on the "Cancel" button
+
+=item * I<focus_first>: (deprecated) force focus on the first entry
+
+=item * B<ok_disabled>: function returning whether {ok} should be disabled (grayed)
+
+=item * B<validate>: function called when {ok} is pressed. If it returns false, the first entry is focused, otherwise it quits
+
+=item * I<advanced>: (deprecated) function called when the "advanced" expander is toggled
+
+=item * I<advanced_messages>: (deprecated) message displayed when "Advanced" is pressed
+
+=item * I<advanced_label>: (deprecated) force the name of the "Advanced" button
+
+=item * I<advanced_label_close>: (deprecated) force the name of the "Basic" button
+
+=item * I<advanced_state>: (deprecated) if set to 1, force the "Advanced" part of the dialog to be opened initially
+
+=item * I<advanced_title>: title of the advanced item popup dialog (else reusing main title)
+
+=item * I<callbacks>: (deprecated) functions called when something happen: complete advanced ok_disabled
+
+=back
+
+ask_from_ takes a list of entries with fields:
+
+=over 4
+
+=item * B<val>: reference to the value
+
+=item * B<label>: description
+
+=item * B<title>: a boolean: whether the label should be displayed as a title (see GNOME's HIG)
+
+=item * B<icon>: icon to put before the description
+
+=item * B<help>: tooltip
+
+=item * I<advanced>: (deprecated) whether it is shown in by default or only in advanced mode
+
+=item * B<focus_out>: function called when the entry is focused out
+
+=item * B<changed>: function called when the entry is modified
+
+=item * B<validate>: function called when "Ok" is pressed. If it returns false, this entry is focused, otherwise it quits
+
+=item * B<disabled>: function returning whether it should be disabled (grayed)
+
+=item * B<focus>: function returning whether it should be focused
+
+=item * B<alignment>: preferred alignment
+
+=item * B<do_not_expand>: do not eat all horizontal space
+
+=item * B<install_button>: if possible, use improved graphical style
+
+=item * B<gtk>: gtk preferences
+
+=item * B<type>:
+
+=over 4
+
+=item * B<button>: (with clicked or clicked_may_quit)
+
+I<type> defaults to button if clicked or clicked_may_quit is there.
+I<val> need not be a reference.
+If I<clicked_may_quit> return true, it's as if "Ok" was pressed.
+
+=item * B<label>:
+I<val> need not be a reference.
+I<type> defaults to label if val is not a reference.
+
+=item * B<bool>: (with "text" or "image" (which overrides text) giving an image filename)
+
+=item * B<range>: (with min, max, SpinButton)
+
+=item * B<combo>: (with list, not_edit, format)
+
+=item * B<list>: (with list, icon2f (aka icon), separator (aka tree), format (aka pre_format function),
+
+It has these optional parameters:
+
+=over 4
+
+=item * B<help>: can be a hash or a function,
+
+=item * B<tree_expanded>: boolean telling whether the tree should be wide open by default
+
+=item * B<quit_if_double_click>: boolean
+
+=item * B<allow_empty_list>: disables the special cases for 0 and 1 element lists
+
+=item * B<image2f>: a subroutine which takes a value of the list as parameter, and returns image_file_name
+
+=back
+
+=item * B<entry>: (the default) (with hidden)
+
+=item * B<expander>: (with text, expanded, message, children(a list of sub entries))
+
+=back
+
+=back
+
+=cut
+
+
+sub ask_from_ {
my ($o, $common, $l) = @_;
- ask_from_entries_refH_powered_normalize($o, $common, $l);
+ ask_from_normalize($o, $common, $l);
@$l or return 1;
- $o->ask_from_entries_refW($common, [ grep { !$_->{advanced} } @$l ], [ grep { $_->{advanced} } @$l ]);
+ $common->{cancel} = '' if !defined wantarray();
+ ask_from_real($o, $common, $l);
}
-sub ask_from_entries_refH_powered_no_check {
+sub ask_from_no_check {
my ($o, $common, $l) = @_;
- ask_from_entries_refH_powered_normalize($o, $common, $l);
- $o->ask_from_entries_refW($common, [ grep { !$_->{advanced} } @$l ], [ grep { $_->{advanced} } @$l ]);
+ ask_from_normalize($o, $common, $l);
+ $common->{cancel} = '' if !defined wantarray();
+ my $l_ = migrate_advanced($common, $l);
+ $o->ask_fromW($common, $l_);
}
+sub ask_from_real {
+ my ($o, $common, $l) = @_;
+ my $l_ = migrate_advanced($common, $l);
+ my $v = $o->ask_fromW($common, $l_);
+ foreach my $e (@$l) {
+ if ($e->{type} eq 'range') {
+ ${$e->{val}} = max($e->{min}, min(${$e->{val}}, $e->{max}));
+ }
+ }
+
+ %$common = ();
+ $v;
+}
+
+
+sub ask_browse_tree_info {
+ my ($o, $title, $message, $common) = @_;
+ $common->{interactive_help} ||= $common->{interactive_help_id} && $o->interactive_help_sub_get_id($common->{interactive_help_id});
+ add2hash_($common, { ok => $::isWizard ? ($::Wizard_finished ? N("Finish") : N("Next")) : N("Ok"),
+ cancel => $::isWizard ? N("Previous") : N("Cancel") });
+ add2hash_($common, { title => $title, message => $message });
+ add2hash_($common, { grep_allowed_to_toggle => sub { @_ },
+ grep_unselected => sub { grep { $common->{node_state}($_) eq 'unselected' } @_ },
+ check_interactive_to_toggle => sub { 1 },
+ toggle_nodes => sub {
+ my ($set_state, @nodes) = @_;
+ my $new_state = !$common->{grep_unselected}($nodes[0]) ? 'selected' : 'unselected';
+ $set_state->($_, $new_state) foreach @nodes;
+ },
+ });
+ $o->ask_browse_tree_info_refW($common);
+}
+sub ask_browse_tree_info_refW { #- default definition, do not use with too many items (memory consuming)
+ my ($o, $common) = @_;
+ my ($l, $v, $h) = ([], [], {});
+ $common->{build_tree}(sub {
+ my ($node) = $common->{grep_allowed_to_toggle}(@_);
+ if (my $state = $node && $common->{node_state}($node)) {
+ push @$l, $node;
+ $state eq 'selected' and push @$v, $node;
+ $h->{$node} = $state eq 'selected';
+ }
+ }, 'flat');
+ add2hash_($common, { list => $l, #- TODO interactivity of toggle is missing
+ values => $v,
+ help => sub { $common->{get_info}($_[0]) },
+ });
+ my ($new_v) = $o->ask_many_from_list($common->{title}, $common->{message}, $common) or return;
+ $common->{toggle_nodes}(sub {}, grep { ! delete $h->{$_} } @$new_v);
+ $common->{toggle_nodes}(sub {}, grep { $h->{$_} } keys %$h);
+ 1;
+}
sub wait_message {
- my ($o, $title, $message, $temp) = @_;
+ my ($o, $title, $message, $b_temp) = @_;
+ my $inline_title = $::isInstall ? $title : '';
+ $inline_title ||= N("Please wait");
- my $w = $o->wait_messageW($title, [ _("Please wait"), deref($message) ]);
- push @tempory::objects, $w if $temp;
+ my $w = $o->wait_messageW($title, $inline_title, $message);
+ push @tempory::objects, $w if $b_temp;
my $b = before_leaving { $o->wait_message_endW($w) };
#- enable access through set
- common::add_f4before_leaving(sub { $o->wait_message_nextW([ deref($_[1]) ], $w) }, $b, 'set');
+ MDK::Common::Func::add_f4before_leaving(sub { $o->wait_message_nextW($_[1], $w) }, $b, 'set');
$b;
}
-sub kill {}
-#-######################################################################################
-#- Wonderful perl :(
-#-######################################################################################
+sub wait_message_with_progress_bar {
+ my ($in, $o_title) = @_;
+
+ my $w = $in->wait_message($o_title, '');
+ my $last_msg;
+ $w, sub {
+ my ($msg, $current, $total) = @_;
+ if ($msg) {
+ $w->set($last_msg = $msg);
+ }
+ if ($total) {
+ $w or internal_error('You must first give some text to display');
+ $w->set(join("\n", $last_msg, "$current / $total"));
+ }
+ };
+}
+
+sub kill() {}
+
+
+
+sub helper_separator_tree_to_tree {
+ my ($separator, $list, $formatted_list) = @_;
+ my $sep = quotemeta $separator;
+ my $tree = {};
+
+ each_index {
+ my @l = split $sep;
+ my $leaf = pop @l;
+ my $node = $tree;
+ foreach (@l) {
+ $node = $node->{$_} ||= do {
+ my $r = {};
+ push @{$node->{_order_}}, $_;
+ $r;
+ };
+ }
+ push @{$node->{_leaves_}}, [ $leaf, $list->[$::i] ];
+ ();
+ } @$formatted_list;
+
+ $tree;
+}
+
+
+sub interactive_help_sub_get_id {
+ my ($o, $id) = @_;
+ eval { $o->is_help_file_exist($id) }
+ && sub { $o->interactive_help_get_id($id) };
+}
+
+sub interactive_help_sub_display_id {
+ my ($o, $id) = @_;
+ eval { $o->interactive_help_has_id($id) }
+ && sub { $o->ask_warn(N("Help"), $o->interactive_help_get_id($id)) };
+}
+
+=back
+
+=cut
+
1;