improvements from MX::MS and MX::Declare
[p5sagit/Devel-Declare.git] / lib / Devel / Declare / MethodInstaller / Simple.pm
1 package Devel::Declare::MethodInstaller::Simple;
2
3 use base 'Devel::Declare::Context::Simple';
4
5 use Devel::Declare ();
6 use Sub::Name;
7 use strict;
8 use warnings;
9
10 sub install_methodhandler {
11   my $class = shift;
12   my %args  = @_;
13   {
14     no strict 'refs';
15     *{$args{into}.'::'.$args{name}}   = sub (&) {};
16   }
17
18   my $ctx = $class->new(%args);
19   Devel::Declare->setup_for(
20     $args{into},
21     { $args{name} => { const => sub { $ctx->parser(@_) } } }
22   );
23 }
24
25 sub strip_attrs {
26   my $self = shift;
27   $self->skipspace;
28
29   my $Offset  = $self->offset;
30   my $linestr = Devel::Declare::get_linestr;
31   my $attrs   = '';
32
33   if (substr($linestr, $Offset, 1) eq ':') {
34     while (substr($linestr, $Offset, 1) ne '{') {
35       if (substr($linestr, $Offset, 1) eq ':') {
36         substr($linestr, $Offset, 1) = '';
37         Devel::Declare::set_linestr($linestr);
38
39         $attrs .= ':';
40       }
41
42       $self->skipspace;
43       $Offset  = $self->offset;
44       $linestr = Devel::Declare::get_linestr();
45
46       if (my $len = Devel::Declare::toke_scan_word($Offset, 0)) {
47         my $name = substr($linestr, $Offset, $len);
48         substr($linestr, $Offset, $len) = '';
49         Devel::Declare::set_linestr($linestr);
50
51         $attrs .= " ${name}";
52
53         if (substr($linestr, $Offset, 1) eq '(') {
54           my $length = Devel::Declare::toke_scan_str($Offset);
55           my $arg    = Devel::Declare::get_lex_stuff();
56           Devel::Declare::clear_lex_stuff();
57           $linestr = Devel::Declare::get_linestr();
58           substr($linestr, $Offset, $length) = '';
59           Devel::Declare::set_linestr($linestr);
60
61           $attrs .= "(${arg})";
62         }
63       }
64     }
65
66     $linestr = Devel::Declare::get_linestr();
67   }
68
69   return $attrs;
70 }
71
72 sub parser {
73   my $self = shift;
74   $self->init(@_);
75
76   $self->skip_declarator;
77   my $name   = $self->strip_name;
78   my $proto  = $self->strip_proto;
79   my $attrs  = $self->strip_attrs;
80   my @decl   = $self->parse_proto($proto);
81   my $inject = $self->inject_parsed_proto(@decl);
82   if (defined $name) {
83     $inject = $self->scope_injector_call() . $inject;
84   }
85   $self->inject_if_block($inject, $attrs ? "sub ${attrs} " : '');
86   if (defined $name) {
87     my $pkg = $self->get_curstash_name;
88     $name = join( '::', $pkg, $name )
89       unless( $name =~ /::/ );
90     $self->shadow( sub (&) {
91       my $code = shift;
92       # So caller() gets the subroutine name
93       no strict 'refs';
94       *{$name} = subname $name => $code;
95     });
96   } else {
97     $self->shadow(sub (&) { shift });
98   }
99 }
100
101 sub parse_proto { }
102
103 sub inject_parsed_proto {
104   return $_[1];
105 }
106
107 1;
108