1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68
|
package Devel::REPL::Plugin::Completion;
use Devel::REPL::Plugin;
use Scalar::Util 'weaken';
use PPI;
use namespace::clean -except => [ 'meta' ];
has current_matches => (
is => 'rw',
isa => 'ArrayRef',
lazy => 1,
default => sub { [] },
);
has match_index => (
is => 'rw',
isa => 'Int',
lazy => 1,
default => sub { 0 },
);
sub BEFORE_PLUGIN {
my ($self) = @_;
my $weakself = $self;
weaken($weakself);
$self->term->Attribs->{attempted_completion_function} = sub {
$weakself->_completion(@_);
};
}
sub _completion {
my ($self, $text, $line, $start, $end) = @_;
# 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
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];
});
}
sub complete {
return ();
}
1;
|