X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FMooseX%2FRole%2FParameterized.pm;h=0c1e5d00ca0af5f0e195bd70f9fe936dce0458db;hb=cbee2cbcba9ab385b9b8a4ceefc891bfcbb4ffaf;hp=52f7d054955e24a83eb57eb6c8853757e23a2373;hpb=d5487cd9641688cda3ef47bca221e55b0cd69a07;p=gitmo%2FMooseX-Role-Parameterized.git diff --git a/lib/MooseX/Role/Parameterized.pm b/lib/MooseX/Role/Parameterized.pm index 52f7d05..0c1e5d0 100644 --- a/lib/MooseX/Role/Parameterized.pm +++ b/lib/MooseX/Role/Parameterized.pm @@ -25,7 +25,7 @@ sub parameter { confess "'parameter' may not be used inside of the role block" if $CURRENT_METACLASS; - my $meta = Class::MOP::Class->initialize($caller); + my $meta = Class::MOP::class_of($caller); my $names = shift; $names = [$names] if !ref($names); @@ -35,10 +35,10 @@ sub parameter { } } -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 { @@ -49,21 +49,9 @@ 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 { my $caller = shift; - my $meta = $CURRENT_METACLASS || Class::MOP::Class->initialize($caller); + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); my $names = shift; $names = [$names] if !ref($names); @@ -75,7 +63,7 @@ sub has { sub method { my $caller = shift; - my $meta = $CURRENT_METACLASS || Class::MOP::Class->initialize($caller); + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); my $name = shift; my $body = shift; @@ -89,61 +77,46 @@ sub method { $meta->add_method($name => $method); } -sub before { +sub _add_method_modifier { + my $type = shift; my $caller = shift; - my $meta = $CURRENT_METACLASS || Class::MOP::Class->initialize($caller); + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); my $code = pop @_; for (@_) { Carp::croak "Roles do not currently support " . ref($_) - . " references for before method modifiers" + . " references for $type method modifiers" if ref $_; - $meta->add_before_method_modifier($_, $code); + + my $add_method = "add_${type}_method_modifier"; + $meta->$add_method($_, $code); } } -sub after { - my $caller = shift; - my $meta = $CURRENT_METACLASS || Class::MOP::Class->initialize($caller); - - my $code = pop @_; +sub before { + _add_method_modifier('before', @_); +} - for (@_) { - Carp::croak "Roles do not currently support " - . ref($_) - . " references for after method modifiers" - if ref $_; - $meta->add_after_method_modifier($_, $code); - } +sub after { + _add_method_modifier('after', @_); } sub around { - my $caller = shift; - my $meta = $CURRENT_METACLASS || Class::MOP::Class->initialize($caller); - - my $code = pop @_; - - for (@_) { - Carp::croak "Roles do not currently support " - . ref($_) - . " references for around method modifiers" - if ref $_; - $meta->add_around_method_modifier($_, $code); - } + _add_method_modifier('around', @_); } sub with { my $caller = shift; - my $meta = $CURRENT_METACLASS || Class::MOP::Class->initialize($caller); + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); Moose::Util::apply_all_roles($meta, @_); } sub requires { my $caller = shift; - my $meta = $CURRENT_METACLASS || Class::MOP::Class->initialize($caller); + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); Carp::croak "Must specify at least one method" unless @_; $meta->add_required_methods(@_); @@ -151,7 +124,7 @@ sub requires { sub excludes { my $caller = shift; - my $meta = $CURRENT_METACLASS || Class::MOP::Class->initialize($caller); + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); Carp::croak "Must specify at least one role" unless @_; $meta->add_excluded_roles(@_); @@ -165,7 +138,7 @@ sub super { sub override { my $caller = shift; - my $meta = $CURRENT_METACLASS || Class::MOP::Class->initialize($caller); + my $meta = $CURRENT_METACLASS || Class::MOP::class_of($caller); my ($name, $code) = @_; $meta->add_override_method_modifier($name, $code); @@ -229,25 +202,39 @@ 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. The -default value for the "is" option is "ro" as that's a very common case. These +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 @@ -257,7 +244,7 @@ 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 @@ -270,10 +257,10 @@ Shawn M Moore, C<< >> =item L -=item L - =item L +=item L + =item L =item L