X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FMooseX%2FRole%2FParameterized.pm;h=20fa44cf513629c0d69ed355d5657abec2d098de;hb=82b8405cff525a6fb9624914d5eaf9b60aeee755;hp=4486f75b97f04ccad765e9be47ddb399a7f85d1b;hpb=c2cfd77dcb8b610d0e58f5cca04e4c176f35f7ff;p=gitmo%2FMooseX-Role-Parameterized.git diff --git a/lib/MooseX/Role/Parameterized.pm b/lib/MooseX/Role/Parameterized.pm index 4486f75..20fa44c 100644 --- a/lib/MooseX/Role/Parameterized.pm +++ b/lib/MooseX/Role/Parameterized.pm @@ -1,45 +1,44 @@ package MooseX::Role::Parameterized; - -# ABSTRACT: parameterized roles - use Moose ( extends => { -as => 'moose_extends' }, around => { -as => 'moose_around' }, qw/confess blessed/, ); +moose_extends 'Moose::Exporter'; -use Carp 'croak'; use Moose::Role (); -moose_extends 'Moose::Exporter'; use MooseX::Role::Parameterized::Meta::Role::Parameterizable; our $CURRENT_METACLASS; __PACKAGE__->setup_import_methods( - with_caller => ['parameter', 'role', 'method'], - as_is => [ - 'has', 'with', 'extends', 'requires', 'excludes', 'augment', 'inner', - 'before', 'after', 'around', 'super', 'override', 'confess', - 'blessed', - ], + with_caller => ['parameter', 'role', 'method', 'has', 'with', 'extends', + 'requires', 'excludes', 'augment', 'inner', 'before', + 'after', 'around', 'super', 'override'], + as_is => [ 'confess', 'blessed' ], ); sub parameter { my $caller = shift; - my $names = shift; + confess "'parameter' may not be used inside of the role block" + if $CURRENT_METACLASS; + + my $meta = Class::MOP::class_of($caller); + + my $names = shift; $names = [$names] if !ref($names); for my $name (@$names) { - Class::MOP::Class->initialize($caller)->add_parameter($name, @_); + $meta->add_parameter($name, @_); } } -sub role { +sub role (&) { my $caller = shift; my $role_generator = shift; - Class::MOP::Class->initialize($caller)->role_generator($role_generator); + Class::MOP::class_of($caller)->role_generator($role_generator); } sub init_meta { @@ -50,110 +49,100 @@ sub init_meta { ); } -# give role a (&) prototype -moose_around _make_wrapper => sub { - my $orig = shift; - my ($self, $caller, $sub, $fq_name) = @_; - - if ($fq_name =~ /::role$/) { - return sub (&) { $sub->($caller, @_) }; - } - - return $orig->(@_); -}; - sub has { - confess "has must be called within the role { ... } block." - unless $CURRENT_METACLASS; + my $caller = shift; + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); my $names = shift; $names = [$names] if !ref($names); for my $name (@$names) { - $CURRENT_METACLASS->add_attribute($name, @_); + $meta->add_attribute($name, @_); } } sub method { - confess "method must be called within the role { ... } block." - unless $CURRENT_METACLASS; - my $caller = shift; + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); + my $name = shift; my $body = shift; - my $method = $CURRENT_METACLASS->method_metaclass->wrap( + my $method = $meta->method_metaclass->wrap( package_name => $caller, name => $name, body => $body, ); - $CURRENT_METACLASS->add_method($name => $method); + $meta->add_method($name => $method); } sub before { - confess "before must be called within the role { ... } block." - unless $CURRENT_METACLASS; + my $caller = shift; + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); my $code = pop @_; for (@_) { - croak "Roles do not currently support " + Carp::croak "Roles do not currently support " . ref($_) . " references for before method modifiers" if ref $_; - $CURRENT_METACLASS->add_before_method_modifier($_, $code); + $meta->add_before_method_modifier($_, $code); } } sub after { - confess "after must be called within the role { ... } block." - unless $CURRENT_METACLASS; + my $caller = shift; + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); my $code = pop @_; for (@_) { - croak "Roles do not currently support " + Carp::croak "Roles do not currently support " . ref($_) . " references for after method modifiers" if ref $_; - $CURRENT_METACLASS->add_after_method_modifier($_, $code); + $meta->add_after_method_modifier($_, $code); } } sub around { - confess "around must be called within the role { ... } block." - unless $CURRENT_METACLASS; + my $caller = shift; + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); my $code = pop @_; for (@_) { - croak "Roles do not currently support " + Carp::croak "Roles do not currently support " . ref($_) . " references for around method modifiers" if ref $_; - $CURRENT_METACLASS->add_around_method_modifier($_, $code); + $meta->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, @_); + my $caller = shift; + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); + + Moose::Util::apply_all_roles($meta, @_); } 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(@_); + my $caller = shift; + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); + + Carp::croak "Must specify at least one method" unless @_; + $meta->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(@_); + my $caller = shift; + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); + + Carp::croak "Must specify at least one role" unless @_; + $meta->add_excluded_roles(@_); } # see Moose.pm for discussion @@ -163,30 +152,33 @@ sub super { } sub override { - confess "override must be called within the role { ... } block." - unless $CURRENT_METACLASS; + my $caller = shift; + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); my ($name, $code) = @_; - $CURRENT_METACLASS->add_override_method_modifier($name, $code); + $meta->add_override_method_modifier($name, $code); } -sub extends { croak "Roles do not currently support 'extends'" } +sub extends { Carp::croak "Roles do not currently support 'extends'" } -sub inner { croak "Roles cannot support 'inner'" } +sub inner { Carp::croak "Roles cannot support 'inner'" } -sub augment { croak "Roles cannot support 'augment'" } +sub augment { Carp::croak "Roles cannot support 'augment'" } 1; __END__ +=head1 NAME + +MooseX::Role::Parameterized - parameterized roles + =head1 SYNOPSIS package MyRole::Counter; use MooseX::Role::Parameterized; parameter name => ( - is => 'ro', isa => 'Str', required => 1, ); @@ -225,45 +217,78 @@ L. =head1 DESCRIPTION -Your parameterized role consists of two things: parameter declarations and a -C block. +Your parameterized role consists of two new 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 +resembles L. You can use any option that L accepts. The +default value for the C option is C as that's a very common case. 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! +class provided as parameters. You use the parameters to customize your +role however you wish. + +There are many possible implementations for parameterized roles (hopefully with +a consistent enough API); I believe this to be the easiest and most flexible +design. Coincidentally, Pugs originally had an eerily similar design. + +=head2 Why a parameters object? -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 yet convinced that that -is a good thing). +I've been asked several times "Why use a parameter I and not just a +parameter I? That would eliminate the need to explicitly declare your +parameters." + +The benefits of using an object are similar to the benefits of using Moose. You +get an easy way to specify lazy defaults, type constraint, delegation, and so +on. You get to use MooseX modules. + +You also get the usual introspective and intercessory abilities that come +standard with the metaobject protocol. Ambitious users should be able to add +traits to the parameters metaclass to further customize behavior. Please let +me know if you're doing anything viciously complicated with this extension. :) =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! - -You must use all the keywords in the role block. If it turns out to be correct, -we'll compose the parameterizable role (everything outside the role block) with -the parameterized role (everything inside the role block). We throw an error if -you try to use a keyword outside of the role block, so don't worry about it for -now. +C<< method NAME => sub { ... }; >>. This is due to a limitation in Perl. In +return though you can use parameters I! L and L are not yet supported. I'm completely unsure of whether they should be handled by this module. Until we -figure out a plan, both declaring and providing a parameter named C or +figure out a plan, either declaring or providing a parameter named C or C is an error. =head1 AUTHOR Shawn M Moore, C<< >> +=head1 EXAMPLES + +=over 4 + +=item L + +=item L + +=item L + +=item L + +=item L + +=item L + +=item L + +=item L + +=item L + +=back + =cut