NAME
Devel::Declare - Adding keywords to perl, in perl
SYNOPSIS
use Devel::Declare ();
{
package MethodHandlers;
use strict;
use warnings;
use B::Hooks::EndOfScope;
our ($Declarator, $Offset);
sub skip_declarator {
$Offset += Devel::Declare::toke_move_past_token($Offset);
}
sub skipspace {
$Offset += Devel::Declare::toke_skipspace($Offset);
}
sub strip_name {
skipspace;
if (my $len = Devel::Declare::toke_scan_word($Offset, 1)) {
my $linestr = Devel::Declare::get_linestr();
my $name = substr($linestr, $Offset, $len);
substr($linestr, $Offset, $len) = '';
Devel::Declare::set_linestr($linestr);
return $name;
}
return;
}
sub strip_proto {
skipspace;
my $linestr = Devel::Declare::get_linestr();
if (substr($linestr, $Offset, 1) eq '(') {
my $length = Devel::Declare::toke_scan_str($Offset);
my $proto = Devel::Declare::get_lex_stuff();
Devel::Declare::clear_lex_stuff();
$linestr = Devel::Declare::get_linestr();
substr($linestr, $Offset, $length) = '';
Devel::Declare::set_linestr($linestr);
return $proto;
}
return;
}
sub shadow {
my $pack = Devel::Declare::get_curstash_name;
Devel::Declare::shadow_sub("${pack}::${Declarator}", $_[0]);
}
# undef -> my ($self) = shift;
# '' -> my ($self) = @_;
# '$foo' -> my ($self, $foo) = @_;
sub make_proto_unwrap {
my ($proto) = @_;
my $inject = 'my ($self';
if (defined $proto) {
$inject .= ", $proto" if length($proto);
$inject .= ') = @_; ';
} else {
$inject .= ') = shift;';
}
return $inject;
}
sub inject_if_block {
my $inject = shift;
skipspace;
my $linestr = Devel::Declare::get_linestr;
if (substr($linestr, $Offset, 1) eq '{') {
substr($linestr, $Offset+1, 0) = $inject;
Devel::Declare::set_linestr($linestr);
}
}
sub scope_injector_call {
return ' BEGIN { MethodHandlers::inject_scope }; ';
}
sub parser {
local ($Declarator, $Offset) = @_;
skip_declarator;
my $name = strip_name;
my $proto = strip_proto;
my $inject = make_proto_unwrap($proto);
if (defined $name) {
$inject = scope_injector_call().$inject;
}
inject_if_block($inject);
if (defined $name) {
$name = join('::', Devel::Declare::get_curstash_name(), $name)
unless ($name =~ /::/);
shadow(sub (&) { no strict 'refs'; *{$name} = shift; });
} else {
shadow(sub (&) { shift });
}
}
sub inject_scope {
on_scope_end {
my $linestr = Devel::Declare::get_linestr;
my $offset = Devel::Declare::get_linestr_offset;
substr($linestr, $offset, 0) = ';';
Devel::Declare::set_linestr($linestr);
};
}
}
my ($test_method1, $test_method2, @test_list);
{
package DeclareTest;
sub method (&);
BEGIN {
Devel::Declare->setup_for(
__PACKAGE__,
{ method => { const => \&MethodHandlers::parser } }
);
}
method new {
my $class = ref $self || $self;
return bless({ @_ }, $class);
}
method foo ($foo) {
return (ref $self).': Foo: '.$foo;
}
method upgrade(){ # no spaces to make case pathological
bless($self, 'DeclareTest2');
}
method DeclareTest2::bar () {
return 'DeclareTest2: bar';
}
$test_method1 = method {
return join(', ', $self->{attr}, $_[1]);
};
$test_method2 = method ($what) {
return join(', ', ref $self, $what);
};
method main () { return "main"; }
@test_list = (method { 1 }, sub { 2 }, method () { 3 }, sub { 4 });
}
use Test::More 'no_plan';
my $o = DeclareTest->new(attr => "value");
isa_ok($o, 'DeclareTest');
is($o->{attr}, 'value', '@_ args ok');
is($o->foo('yay'), 'DeclareTest: Foo: yay', 'method with argument ok');
is($o->main, 'main', 'declaration of package named method ok');
$o->upgrade;
isa_ok($o, 'DeclareTest2');
is($o->bar, 'DeclareTest2: bar', 'absolute method declaration ok');
is($o->$test_method1('no', 'yes'), 'value, yes', 'anon method with @_ ok');
is($o->$test_method2('this'), 'DeclareTest2, this', 'anon method with proto ok');
is_deeply([ map { $_->() } @test_list ], [ 1, 2, 3, 4], 'binding ok');
(this is t/method-no-semi.t in this distribution)
DESCRIPTION
setup_for
Devel::Declare->setup_for(
$package,
{
$name => { $op_type => $sub }
}
);
Currently valid op types: 'check', 'rv2cv'
AUTHORS
Matt S Trout - <mst@shadowcat.co.uk>
Company: http://www.shadowcat.co.uk/ Blog: http://chainsawblues.vox.com/
Florian Ragwitz <rafl@debian.org>
LICENSE
This library is free software under the same terms as perl itself