diff options
Diffstat (limited to 'perl-install/interactive.pm')
| -rw-r--r-- | perl-install/interactive.pm | 302 | 
1 files changed, 214 insertions, 88 deletions
| diff --git a/perl-install/interactive.pm b/perl-install/interactive.pm index f4b95ae9d..34ae7a4de 100644 --- a/perl-install/interactive.pm +++ b/perl-install/interactive.pm @@ -1,4 +1,4 @@ -package interactive; # $Id$ +package interactive; # $Id: interactive.pm 247292 2008-10-01 15:23:37Z tv $  use diagnostics;  use strict; @@ -6,7 +6,6 @@ use strict;  #-######################################################################################  #- misc imports  #-###################################################################################### -use MDK::Common::Func;  use common;  use do_pkgs; @@ -21,23 +20,35 @@ use do_pkgs;  #- ask_from_ takes global options ($common):  #-  title                => window title  #-  messages             => message displayed in the upper part of the window -#-  advanced_messages    => message displayed when "Advanced" is pressed  #-  ok                   => force the name of the "Ok"/"Next" button  #-  cancel               => force the name of the "Cancel"/"Previous" button -#-  advanced_label       => force the name of the "Advanced" button -#-  advanced_label_close => force the name of the "Basic" button -#-  advanced_state       => if set to 1, force the "Advanced" part of the dialog to be opened initially  #-  focus_cancel         => force focus on the "Cancel" button -#-  focus_first          => force focus on the first entry -#-  callbacks            => functions called when something happen: complete canceled advanced changed focus_out ok_disabled +#-  focus_first          => (deprecated) force focus on the first entry +#-  ok_disabled          => function returning wether {ok} should be disabled (grayed) +#-  validate             => function called when {ok} is pressed. If it returns false, the first entry is focused, otherwise it quits +#-  advanced             => (deprecated) function called when the "advanced" expander is toggled +#-  advanced_messages    => (deprecated) message displayed when "Advanced" is pressed +#-  advanced_label       => (deprecated) force the name of the "Advanced" button +#-  advanced_label_close => (deprecated) force the name of the "Basic" button +#-  advanced_state       => (deprecated) if set to 1, force the "Advanced" part of the dialog to be opened initially +#-  advanced_title       => (deprecated) title of the advanced item popup dialog (else reusing main title) +#-  callbacks            => (deprecated) functions called when something happen: complete advanced ok_disabled  #- ask_from_ takes a list of entries with fields:  #-  val      => reference to the value  #-  label    => description +#-  title    => a boolean: whether the label should be displayed as a title (see GNOME's HIG)  #-  icon     => icon to put before the description  #-  help     => tooltip -#-  advanced => wether it is shown in by default or only in advanced mode +#-  advanced => (deprecated) wether it is shown in by default or only in advanced mode +#-  focus_out => function called when the entry is focused out +#-  changed  => function called when the entry is modified +#-  validate => function called when "Ok" is pressed. If it returns false, this entry is focused, otherwise it quits  #-  disabled => function returning wether it should be disabled (grayed) +#-  focus    => function returning wether it should be focused +#-  alignment => preferred alignment +#-  do_not_expand => do not eat all horizontal space +#-  install_button => if possible, use improved graphical style  #-  gtk      => gtk preferences  #-  type     =>   #-     button => (with clicked or clicked_may_quit) @@ -45,15 +56,16 @@ use do_pkgs;  #-               (val need not be a reference) (if clicked_may_quit return true, it's as if "Ok" was pressed)  #-     label => (val need not be a reference) (type defaults to label if val is not a reference)   #-     bool (with "text" or "image" (which overrides text) giving an image filename) -#-     range (with min, max) +#-     range (with min, max, SpinButton)  #-     combo (with list, not_edit, format)  #-     list (with list, icon2f (aka icon), separator (aka tree), format (aka pre_format function),  #-           help can be a hash or a function,  #-           tree_expanded boolean telling wether the tree should be wide open by default  #-           quit_if_double_click boolean  #-           allow_empty_list disables the special cases for 0 and 1 element lists -#-           image2f is a subroutine which takes a value of the list as parameter, and returns an array (text, image_file_name)) +#-           image2f is a subroutine which takes a value of the list as parameter, and returns image_file_name  #-     entry (the default) (with hidden) +#-     expander (with text, expanded, message, children(a list of sub entries))  #  #- heritate from this class and you'll get all made interactivity for same steps.  #- for this you need to provide @@ -110,11 +122,68 @@ sub vnew {  	}      } -    require 'log.pm'; #- "require log" causes some pb, perl thinking that "log" is the log() function -    undef *log::l; -    *log::l = sub {}; # otherwise, it will bother us :( -    require interactive::newt; -    interactive::newt->new; +    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; +	} +    } +    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 = +{ '&' => '&', '<' => '<', '>' => '>' }->{$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 {} @@ -131,42 +200,43 @@ sub exit {      }  } +  #-######################################################################################  #- Interactive functions  #-######################################################################################  sub ask_warn { -    my ($o, $title, $message) = @_; -    ask_warn_($o, { title => $title, messages => $message }); +    my ($o, $title, $message, $o_icon) = @_; +    ask_warn_($o, { title => $title, messages => $message, icon => $o_icon });  }  sub ask_yesorno {      my ($o, $title, $message, $b_def) = @_;      ask_yesorno_($o, { title => $title, messages => $message }, $b_def);  }  sub ask_okcancel { -    my ($o, $title, $message, $b_def) = @_; -    ask_okcancel_($o, { title => $title, messages => $message }, $b_def); +    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) = @_; -    ask_from_listf_raw_no_check($o, $common, undef, [ $o->ok ]); +    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, sub { translate($_[0]) }, [ N_("Yes"), N_("No") ], $b_def ? "Yes" : "No") eq "Yes"; +    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;  	$common->{focus_cancel} = !$b_def;      	ask_from_no_check($o, $common, []);      } else { -	ask_from_listf_raw($o, $common, sub { translate($_[0]) }, [ $o->ok, $o->cancel ], $b_def ? $o->ok : "Cancel") eq $o->ok; +	ask_from_listf_raw($o, $common, \&translate, [ $o->ok, $o->cancel ], $b_def ? $o->ok : $o->cancel) eq $o->ok;      }  } @@ -200,7 +270,7 @@ sub ask_from_list {  sub ask_from_list_ {      my ($o, $title, $message, $l, $o_def) = @_; -    ask_from_listf($o, $title, $message, sub { translate($_[0]) }, $l, $o_def); +    ask_from_listf($o, $title, $message, \&translate, $l, $o_def);  }  sub ask_from_listf_ { @@ -289,7 +359,7 @@ sub ask_from_entries {      my @l = map { my $i = ''; { label => $_, val => \$i } } @$l; -    $o->ask_from_({ title => $title, messages => $message, callbacks => \%callback,  +    $o->ask_from_({ title => $title, messages => $message, %callback,   		    focus_first => 1 }, \@l) or return;      map { ${$_->{val}} } @l;  } @@ -316,60 +386,70 @@ sub ask_from__add_modify_remove {  					     $continue = 1;  					 } } }  		     N_("Add"), if_(@{$e->{list}} > 0, N_("Modify"), N_("Remove"))); -	    $o->ask_from_({ title => $title, messages => $message, callbacks => \%callback }, \@l) or return; +	    $o->ask_from_({ title => $title, messages => $message, %callback }, \@l) or return;  	    return 1 if !$continue;  	}      }  } -#- can get a hash of callback: focus_out changed and complete +#- 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 {      my ($o, $title, $message, $l, %callback) = @_; -    ask_from_($o, { title => $title, messages => $message, callbacks => \%callback }, $l); +    ask_from_($o, { title => $title, messages => $message, %callback }, $l);  } -sub ask_from_normalize { -    my ($o, $common, $l) = @_; +sub _normalize_entry { +    my ($o, $e) = @_; -    ref($l) eq 'ARRAY' or internal_error('ask_from_normalize'); -    foreach my $e (@$l) { -	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}; -	    add2hash_($e, { not_edit => 1 }); -	    $e->{type} ||= 'combo'; +    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} eq 'label' || !ref($e->{val})) { -	    $e->{type} = 'label'; -	    $e->{val} = \ (my $_v = $e->{val}) if !ref($e->{val}); -	} else { -	    $e->{type} ||= 'entry'; +	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);  	} -	$e->{disabled} ||= sub { 0 }; +    } 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 {  @@ -385,16 +465,53 @@ sub ask_from_normalize {  	    1;  	}      } @$l; +} + +sub ask_from_normalize { +    my ($o, $common, $l) = @_; + +    if ($common->{focus_first}) { +	if (my $e = find { $_->{val} } @$l) { +	    $e->{focus} = sub { 1 }; +	} +    } + +    _normalize_entries($o, $l);      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} ||= N("Basic"); -    $common->{$_} = $common->{$_} ? [ deref($common->{$_}) ] : [] foreach qw(messages advanced_messages); -    add2hash_($common->{callbacks} ||= {}, { changed => sub {}, focus_out => sub {}, complete => sub { 0 }, canceled => sub { 0 }, advanced => sub {} }); +    $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_ { @@ -408,15 +525,15 @@ sub ask_from_no_check {      my ($o, $common, $l) = @_;      ask_from_normalize($o, $common, $l);      $common->{cancel} = '' if !defined wantarray(); -    my ($l1, $l2) = partition { !$_->{advanced} } @$l; -    $o->ask_fromW($common, $l1, $l2); +    my $l_ = migrate_advanced($common, $l); +    $o->ask_fromW($common, $l_);  }  sub ask_from_real {      my ($o, $common, $l) = @_; -    my ($l1, $l2) = partition { !$_->{advanced} } @$l; -    my $v = $o->ask_fromW($common, $l1, $l2); +    my $l_ = migrate_advanced($common, $l); +    my $v = $o->ask_fromW($common, $l_); -    foreach my $e (@$l1, @$l2) { +    foreach my $e (@$l) {  	if ($e->{type} eq 'range') {  	    ${$e->{val}} = max($e->{min}, min(${$e->{val}}, $e->{max}));  	} @@ -467,16 +584,36 @@ sub ask_browse_tree_info_refW { #- default definition, do not use with too many  sub wait_message {      my ($o, $title, $message, $b_temp) = @_; +    my $inline_title = $::isInstall ? $title : ''; +    $inline_title ||= N("Please wait"); -    my $w = $o->wait_messageW($title, [ N("Please wait"), deref($message) ]); +    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 -    MDK::Common::Func::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 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() {} @@ -505,27 +642,16 @@ sub helper_separator_tree_to_tree {  } -sub interactive_help_has_id { -    my ($_o, $id) = @_; -    exists $help::{$id}; -} - -sub interactive_help_get_id { -    my ($_o, @l) = @_; -    @l = map {  -	join("\n\n", map { s/\n/ /mg; $_ } split("\n\n", translate($help::{$_}->()))); -    } grep { exists $help::{$_} } @l; -    join("\n\n\n", @l); -} -  sub interactive_help_sub_get_id {      my ($o, $id) = @_; -    $o->interactive_help_has_id($id) && sub { $o->interactive_help_get_id($id) }; +    eval { $o->is_help_file_exist($id) } +      && sub { $o->interactive_help_get_id($id) };  }  sub interactive_help_sub_display_id {      my ($o, $id) = @_; -    $o->interactive_help_has_id($id) && sub { $o->ask_warn(N("Help"), $o->interactive_help_get_id($id)) }; +    eval { $o->interactive_help_has_id($id) } +      && sub { $o->ask_warn(N("Help"), $o->interactive_help_get_id($id)) };  }  1; | 
