X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FMouse%2FMeta%2FRole.pm;h=9072312c680bf26204e0a9903de2a2fa8c46d987;hb=b1b8155380073bc8170b50b17893474865604300;hp=79f3b7fc697070fb99d277a3a06a69d4ce6251f7;hpb=fc0e0bbda277a99354170a4b8de6eb1275920555;p=gitmo%2FMouse.git diff --git a/lib/Mouse/Meta/Role.pm b/lib/Mouse/Meta/Role.pm index 79f3b7f..9072312 100644 --- a/lib/Mouse/Meta/Role.pm +++ b/lib/Mouse/Meta/Role.pm @@ -2,6 +2,7 @@ package Mouse::Meta::Role; use strict; use warnings; +use Carp 'confess'; do { my %METACLASS_CACHE; @@ -27,13 +28,20 @@ sub new { my $class = shift; my %args = @_; - $args{attributes} ||= {}; + $args{attributes} ||= {}; + $args{required_methods} ||= []; bless \%args, $class; } sub name { $_[0]->{name} } +sub add_required_methods { + my $self = shift; + my @methods = @_; + push @{$self->{required_methods}}, @methods; +} + sub add_attribute { my $self = shift; my $name = shift; @@ -45,14 +53,71 @@ sub has_attribute { exists $_[0]->{attributes}->{$_[1]} } sub get_attribute_list { keys %{ $_[0]->{attributes} } } sub get_attribute { $_[0]->{attributes}->{$_[1]} } +# copied from Class::Inspector +sub get_method_list { + my $self = shift; + my $name = $self->name; + + no strict 'refs'; + # Get all the CODE symbol table entries + my @functions = grep !/^meta$/, + grep { /\A[^\W\d]\w*\z/o } + grep { defined &{"${name}::$_"} } + keys %{"${name}::"}; + wantarray ? @functions : \@functions; +} + sub apply { my $self = shift; + my $selfname = $self->name; my $class = shift; + my $classname = $class->name; + + for my $name (@{$self->{required_methods}}) { + unless ($classname->can($name)) { + confess "'$selfname' requires the method '$name' to be implemented by '$classname'"; + } + } + + { + no strict 'refs'; + for my $name ($self->get_method_list) { + next if $name eq 'has' || $name eq 'requires' || $name eq 'meta' || $name eq 'with' || $name eq 'around' || $name eq 'before' || $name eq 'after' || $name eq 'blessed' || $name eq 'extends' || $name eq 'confess' || $name eq 'excludes'; + if ($classname->can($name)) { + # XXX what's Moose's behavior? + next; + } + *{"${classname}::${name}"} = *{"${selfname}::${name}"}; + } + } + + if ($class->isa('Mouse::Meta::Class')) { + # apply role to class + for my $name ($self->get_attribute_list) { + next if $class->has_attribute($name); + my $spec = $self->get_attribute($name); + Mouse::Meta::Attribute->create($class, $name, %$spec); + } + } else { + # apply role to role + # XXX Room for speed improvement + for my $name ($self->get_attribute_list) { + next if $class->has_attribute($name); + my $spec = $self->get_attribute($name); + $class->add_attribute($name, $spec); + } + } + + # XXX Room for speed improvement in role to role + for my $modifier_type (qw/before after around/) { + my $add_method = "add_${modifier_type}_method_modifier"; + my $modified = $self->{"${modifier_type}_method_modifiers"}; - for my $name ($self->get_attribute_list) { - next if $class->has_attribute($name); - my $spec = $self->get_attribute($name); - Mouse::Meta::Attribute->create($class, $name, %$spec); + for my $method_name (keys %$modified) { + for my $code (@{ $modified->{$method_name} }) { + $class->$add_method($method_name => $code); + } + } } }