Add LexEnv completion plugin. I still owe mst a refactor of the completion subsystem :)
Sartak [Wed, 26 Sep 2007 01:39:33 +0000 (01:39 +0000)]
git-svn-id: http://dev.catalyst.perl.org/repos/bast/trunk/Devel-REPL@3783 bd8105ee-0ff8-0310-8827-fb3f25b6796d

lib/Devel/REPL/Plugin/Completion.pm
lib/Devel/REPL/Plugin/CompletionDriver/LexEnv.pm [new file with mode: 0644]

index ae4a8ec..e9ac902 100644 (file)
@@ -5,17 +5,17 @@ use PPI;
 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 {
@@ -33,11 +33,15 @@ 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
diff --git a/lib/Devel/REPL/Plugin/CompletionDriver/LexEnv.pm b/lib/Devel/REPL/Plugin/CompletionDriver/LexEnv.pm
new file mode 100644 (file)
index 0000000..2011896
--- /dev/null
@@ -0,0 +1,38 @@
+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;
+