X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FMooseX%2FRole%2FParameterized.pm;h=8d3ee66ca40de3d745d0c64321676c0763e117b8;hb=d9e02904b2452dfcbbd103fd47287e266e0ca173;hp=24152fe895465f6a883e4d5fe63fc39f54708f8a;hpb=fc4a95b67fb93b9b80dfbf0c9e6663cf3dccb602;p=gitmo%2FMooseX-Role-Parameterized.git diff --git a/lib/MooseX/Role/Parameterized.pm b/lib/MooseX/Role/Parameterized.pm index 24152fe..8d3ee66 100644 --- a/lib/MooseX/Role/Parameterized.pm +++ b/lib/MooseX/Role/Parameterized.pm @@ -1,8 +1,11 @@ -#!/usr/bin/env perl package MooseX::Role::Parameterized; + +# ABSTRACT: parameterized roles + use Moose ( extends => { -as => 'moose_extends' }, - qw/around confess/, + around => { -as => 'moose_around' }, + qw/confess blessed/, ); use Carp 'croak'; @@ -15,7 +18,11 @@ our $CURRENT_METACLASS; __PACKAGE__->setup_import_methods( with_caller => ['parameter', 'role', 'method'], - as_is => ['has', 'extends', 'augment', 'inner'], + as_is => [ + 'has', 'with', 'extends', 'requires', 'excludes', 'augment', 'inner', + 'before', 'after', 'around', 'super', 'override', 'confess', + 'blessed', + ], ); sub parameter { @@ -44,7 +51,7 @@ sub init_meta { } # give role a (&) prototype -around _make_wrapper => sub { +moose_around _make_wrapper => sub { my $orig = shift; my ($self, $caller, $sub, $fq_name) = @_; @@ -84,6 +91,85 @@ sub method { $CURRENT_METACLASS->add_method($name => $method); } +sub before { + confess "before must be called within the role { ... } block." + unless $CURRENT_METACLASS; + + my $code = pop @_; + + for (@_) { + croak "Roles do not currently support " + . ref($_) + . " references for before method modifiers" + if ref $_; + $CURRENT_METACLASS->add_before_method_modifier($_, $code); + } +} + +sub after { + confess "after must be called within the role { ... } block." + unless $CURRENT_METACLASS; + + my $code = pop @_; + + for (@_) { + croak "Roles do not currently support " + . ref($_) + . " references for after method modifiers" + if ref $_; + $CURRENT_METACLASS->add_after_method_modifier($_, $code); + } +} + +sub around { + confess "around must be called within the role { ... } block." + unless $CURRENT_METACLASS; + + my $code = pop @_; + + for (@_) { + croak "Roles do not currently support " + . ref($_) + . " references for around method modifiers" + if ref $_; + $CURRENT_METACLASS->add_around_method_modifier($_, $code); + } +} + +sub with { + confess "with must be called within the role { ... } block." + unless $CURRENT_METACLASS; + Moose::Util::apply_all_roles($CURRENT_METACLASS, @_); +} + +sub requires { + confess "requires must be called within the role { ... } block." + unless $CURRENT_METACLASS; + croak "Must specify at least one method" unless @_; + $CURRENT_METACLASS->add_required_methods(@_); +} + +sub excludes { + confess "excludes must be called within the role { ... } block." + unless $CURRENT_METACLASS; + croak "Must specify at least one role" unless @_; + $CURRENT_METACLASS->add_excluded_roles(@_); +} + +# see Moose.pm for discussion +sub super { + return unless $Moose::SUPER_BODY; + $Moose::SUPER_BODY->(@Moose::SUPER_ARGS); +} + +sub override { + confess "override must be called within the role { ... } block." + unless $CURRENT_METACLASS; + + my ($name, $code) = @_; + $CURRENT_METACLASS->add_override_method_modifier($name, $code); +} + sub extends { croak "Roles do not currently support 'extends'" } sub inner { croak "Roles cannot support 'inner'" } @@ -92,3 +178,85 @@ sub augment { croak "Roles cannot support 'augment'" } 1; +__END__ + +=head1 SYNOPSIS + + package MyRole::Counter; + use MooseX::Role::Parameterized; + + parameter name => ( + is => 'ro', + isa => 'Str', + required => 1, + ); + + role { + my $p = shift; + + my $name = $p->name; + + has $name => ( + is => 'rw', + isa => 'Int', + default => 0, + ); + + method "increment_$name" => sub { + my $self = shift; + $self->$name($self->$name + 1); + }; + + method "decrement_$name" => sub { + my $self = shift; + $self->$name($self->$name - 1); + }; + }; + + package MyGame::Tile; + use Moose; + + with 'MyRole::Counter' => { name => 'stepped_on' }; + +=head1 L + +B If you're new here, please read +L. + +=head1 DESCRIPTION + +Your parameterized role consists of two things: parameter declarations and a +C block. + +Parameters are declared using the L keyword which very much +resembles L. You can use any option that L accepts. +These parameters will get their values when the consuming class (or role) uses +L. A parameter object will be constructed with these values, and +passed to the C block. + +The C block then uses the usual L keywords to build up a +role. You can shift off the parameter object to inspect what the consuming +class provided as parameters. You can use the parameters to make your role +customizable! + +There are many paths to parameterized roles (hopefully with a consistent enough +API); I believe this to be the easiest and most flexible implementation. +Coincidentally, Pugs has a very similar design (I'm not convinced that that is +a good thing yet). + +=head1 CAVEATS + +You must use this syntax to declare methods in the role block: +C sub { ... };>. This is due to a limitation in Perl. In return +though you can use parameters I! + +L and L are not yet supported. Because +I'm totally unsure of whether they should be handled by this module, both +declaring and providing a parameter named C or C is an error. + +=head1 AUTHOR + +Shawn M Moore, C<< >> + +=cut +