improvements from MX::MS and MX::Declare
[p5sagit/Devel-Declare.git] / lib / Devel / Declare / MethodInstaller / Simple.pm
CommitLineData
e7be1784 1package Devel::Declare::MethodInstaller::Simple;
2
3use base 'Devel::Declare::Context::Simple';
4
5use Devel::Declare ();
6use Sub::Name;
7use strict;
8use warnings;
9
10sub install_methodhandler {
5b27c9b2 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 );
e7be1784 23}
24
b0a89632 25sub 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
e7be1784 72sub parser {
5b27c9b2 73 my $self = shift;
74 $self->init(@_);
75
76 $self->skip_declarator;
77 my $name = $self->strip_name;
78 my $proto = $self->strip_proto;
b0a89632 79 my $attrs = $self->strip_attrs;
5b27c9b2 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 }
b0a89632 85 $self->inject_if_block($inject, $attrs ? "sub ${attrs} " : '');
5b27c9b2 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 }
e7be1784 99}
5b27c9b2 100
e7be1784 101sub parse_proto { }
5b27c9b2 102
e7be1784 103sub inject_parsed_proto {
5b27c9b2 104 return $_[1];
e7be1784 105}
106
e7be1784 1071;
108