use namespace::clean -except => [ 'meta' ];
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 },
);
sub BEFORE_PLUGIN {
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
--- /dev/null
+package Devel::REPL::Plugin::CompletionDriver::LexEnv;
+use Devel::REPL::Plugin;
+use namespace::clean -except => [ 'meta' ];
+
+sub AFTER_PLUGIN {
+ my ($_REPL) = @_;
+
+ if (!$_REPL->can('lexical_environment')) {
+ warn "Devel::REPL::Plugin::CompletionDriver::LexEnv requires Devel::REPL::Plugin::LexEnv.";
+ }
+}
+
+around complete => sub {
+ my $orig = shift;
+ my ($self, $text, $document) = @_;
+
+ # recursively find the last element
+ my $last = $document;
+ while ($last->can('last_element') && defined($last->last_element)) {
+ $last = $last->last_element;
+ }
+
+ return $orig->(@_)
+ unless $last->isa('PPI::Token::Symbol');
+
+ my $sigil = substr($last, 0, 1, '');
+ my $re = qr/^\Q$last/;
+
+ return $orig->(@_),
+ # ReadLine is weirdly inconsistent
+ map { $sigil eq '%' ? '%' . $_ : $_ }
+ grep { /$re/ }
+ map { substr($_, 1) } # drop lexical's sigil
+ keys %{$self->lexical_environment->get_context('_')};
+};
+
+1;
+