X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?p=p5sagit%2FDevel-REPL.git;a=blobdiff_plain;f=lib%2FDevel%2FREPL%2FPlugin%2FCompletion.pm;h=37821922892fa6c27432230726168f29688ec568;hp=e99ef09c8d54c30c5e12970cbe9fea319e597b31;hb=5ac988901b5ccbf3b31cdf980534d0c9631d9e4a;hpb=f283346085ff7da029cd511bde407fd1519d7ce6 diff --git a/lib/Devel/REPL/Plugin/Completion.pm b/lib/Devel/REPL/Plugin/Completion.pm index e99ef09..3782192 100644 --- a/lib/Devel/REPL/Plugin/Completion.pm +++ b/lib/Devel/REPL/Plugin/Completion.pm @@ -1,113 +1,144 @@ +use strict; +use warnings; package Devel::REPL::Plugin::Completion; +# ABSTRACT: Extensible tab completion + +our $VERSION = '1.003028'; + use Devel::REPL::Plugin; use Scalar::Util 'weaken'; use PPI; -use namespace::clean -except => [ 'meta' ]; +use namespace::autoclean; has current_matches => ( - is => 'rw', - isa => 'ArrayRef', - lazy => 1, - default => sub { [] }, + is => 'rw', + isa => 'ArrayRef', + lazy => 1, + default => sub { [] }, ); has match_index => ( - is => 'rw', - isa => 'Int', - lazy => 1, - default => sub { 0 }, + is => 'rw', + isa => 'Int', + lazy => 1, + default => sub { 0 }, ); has no_term_class_warning => ( - isa => "Bool", - is => "rw", - default => 0, + isa => "Bool", + is => "rw", + default => 0, +); + +has do_readline_filename_completion => ( # so default is no if Completion loaded + isa => "Bool", + is => "rw", + lazy => 1, + default => sub { 0 }, ); before 'read' => sub { - my ($self) = @_; + my ($self) = @_; - if ((!$self->term->isa("Term::ReadLine::Gnu") and !$self->term->isa("Term::ReadLine::Perl")) - and !$self->no_term_class_warning) { - warn "Term::ReadLine::Gnu or Term::ReadLine::Perl is required for the Completion plugin to work"; - $self->no_term_class_warning(1); - } + if ((!$self->term->isa("Term::ReadLine::Gnu") and !$self->term->isa("Term::ReadLine::Perl")) + and !$self->no_term_class_warning) { + warn "Term::ReadLine::Gnu or Term::ReadLine::Perl is required for the Completion plugin to work"; + $self->no_term_class_warning(1); + } - my $weakself = $self; - weaken($weakself); + my $weakself = $self; + weaken($weakself); - $self->term->Attribs->{attempted_completion_function} = sub { - $weakself->_completion(@_); - }; + if ($self->term->isa("Term::ReadLine::Gnu")) { + $self->term->Attribs->{attempted_completion_function} = sub { + $weakself->_completion(@_); + }; + } + + if ($self->term->isa("Term::ReadLine::Perl")) { + $self->term->Attribs->{completion_function} = sub { + $weakself->_completion(@_); + }; + } - $self->term->Attribs->{completion_function} = sub { - $weakself->_completion(@_); - }; }; sub _completion { - my $is_trp = scalar(@_) == 4 ? 1 : 0; - my ($self, $text, $line, $start, $end) = @_; - $end = $start+length($text) if $is_trp; - - # we're discarding everything after the cursor for completion purposes - # we can't just use $text because we want all the code before the cursor to - # matter, not just the current word - substr($line, $end) = ''; - - my $document = PPI::Document->new(\$line); - return unless defined($document); - - $document->prune('PPI::Token::Whitespace'); - - my @matches = $self->complete($text, $document); - - # iterate through the completions - if ($is_trp) { - return @matches; - } else { - return $self->term->completion_matches($text, sub { - my ($text, $state) = @_; - - if (!$state) { - $self->current_matches(\@matches); - $self->match_index(0); - } - else { - $self->match_index($self->match_index + 1); - } - - return $self->current_matches->[$self->match_index]; - }); - } + my $is_trp = scalar(@_) == 4 ? 1 : 0; + my ($self, $text, $line, $start, $end) = @_; + $end = $start+length($text) if $is_trp; + + # we're discarding everything after the cursor for completion purposes + # we can't just use $text because we want all the code before the cursor to + # matter, not just the current word + substr($line, $end) = ''; + + my $document = PPI::Document->new(\$line); + return unless defined($document); + + $document->prune('PPI::Token::Whitespace'); + + my @matches = $self->complete($text, $document); + + # iterate through the completions + if ($is_trp) { + if (scalar(@matches)) { + return @matches; + } else { + return ($self->do_readline_filename_completion) ? readline::rl_filename_list($text) : () ; + } + } else { + $self->term->Attribs->{attempted_completion_over} = 1 unless $self->do_readline_filename_completion; + if (scalar(@matches)) { + return $self->term->completion_matches($text, sub { + my ($text, $state) = @_; + + if (!$state) { + $self->current_matches(\@matches); + $self->match_index(0); + } + else { + $self->match_index($self->match_index + 1); + } + + return $self->current_matches->[$self->match_index]; + }); + } else { + return; + } + } } sub complete { - return (); + return (); } # recursively find the last element sub last_ppi_element { - my ($self, $document, $type) = @_; - my $last = $document; - while ($last->can('last_element') && defined($last->last_element)) { - $last = $last->last_element; - return $last if $type && $last->isa($type); - } - return $last; + my ($self, $document, $type) = @_; + my $last = $document; + while ($last->can('last_element') && defined($last->last_element)) { + $last = $last->last_element; + return $last if $type && $last->isa($type); + } + return $last; } 1; __END__ -=head1 NAME +=pod + +=head1 NOTE -Devel::REPL::Plugin::Completion - Extensible tab completion +By default, the Completion plugin explicitly does I use the Gnu readline +or Term::ReadLine::Perl fallback filename completion. + +Set the attribute C to 1 to enable this feature. =head1 AUTHOR Shawn M Moore, C<< >> =cut -