Mouse::Role supports 'with'
[gitmo/Mouse.git] / lib / Mouse / Meta / Role.pm
CommitLineData
a2227e71 1#!/usr/bin/env perl
2package Mouse::Meta::Role;
3use strict;
4use warnings;
59089ec3 5use Carp 'confess';
a2227e71 6
acf0f643 7do {
8 my %METACLASS_CACHE;
9
10 # because Mouse doesn't introspect existing classes, we're forced to
11 # only pay attention to other Mouse classes
12 sub _metaclass_cache {
13 my $class = shift;
14 my $name = shift;
15 return $METACLASS_CACHE{$name};
16 }
17
18 sub initialize {
19 my $class = shift;
20 my $name = shift;
21 $METACLASS_CACHE{$name} = $class->new(name => $name)
22 if !exists($METACLASS_CACHE{$name});
23 return $METACLASS_CACHE{$name};
24 }
25};
26
27sub new {
28 my $class = shift;
29 my %args = @_;
30
59089ec3 31 $args{attributes} ||= {};
32 $args{required_methods} ||= [];
274b6cce 33
acf0f643 34 bless \%args, $class;
35}
a2227e71 36
513854c7 37sub name { $_[0]->{name} }
38
59089ec3 39sub add_required_methods {
40 my $self = shift;
41 my @methods = @_;
42 push @{$self->{required_methods}}, @methods;
43}
44
274b6cce 45sub add_attribute {
46 my $self = shift;
47 my $name = shift;
69ac1dcf 48 my $spec = shift;
49 $self->{attributes}->{$name} = $spec;
da0c885d 50}
51
274b6cce 52sub has_attribute { exists $_[0]->{attributes}->{$_[1]} }
53sub get_attribute_list { keys %{ $_[0]->{attributes} } }
69ac1dcf 54sub get_attribute { $_[0]->{attributes}->{$_[1]} }
274b6cce 55
2e92bb89 56# copied from Class::Inspector
57sub get_method_list {
58 my $self = shift;
59 my $name = $self->name;
60
61 no strict 'refs';
62 # Get all the CODE symbol table entries
63 my @functions = grep !/^meta$/,
64 grep { /\A[^\W\d]\w*\z/o }
65 grep { defined &{"${name}::$_"} }
66 keys %{"${name}::"};
67 wantarray ? @functions : \@functions;
68}
69
da0c885d 70sub apply {
71 my $self = shift;
2e92bb89 72 my $selfname = $self->name;
da0c885d 73 my $class = shift;
2e92bb89 74 my $classname = $class->name;
da0c885d 75
59089ec3 76 for my $name (@{$self->{required_methods}}) {
2e92bb89 77 unless ($classname->can($name)) {
78 confess "'$selfname' requires the method '$name' to be implemented by '$classname'";
79 }
80 }
81
82 {
83 no strict 'refs';
84 for my $name ($self->get_method_list) {
85 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';
86 if ($classname->can($name)) {
87 # XXX what's Moose's behavior?
88 next;
89 }
90 *{"${classname}::${name}"} = *{"${selfname}::${name}"};
59089ec3 91 }
92 }
93
b1b81553 94 if ($class->isa('Mouse::Meta::Class')) {
95 # apply role to class
96 for my $name ($self->get_attribute_list) {
97 next if $class->has_attribute($name);
98 my $spec = $self->get_attribute($name);
99 Mouse::Meta::Attribute->create($class, $name, %$spec);
100 }
101 } else {
102 # apply role to role
103 # XXX Room for speed improvement
104 for my $name ($self->get_attribute_list) {
105 next if $class->has_attribute($name);
106 my $spec = $self->get_attribute($name);
107 $class->add_attribute($name, $spec);
108 }
da0c885d 109 }
d99db7b6 110
b1b81553 111 # XXX Room for speed improvement in role to role
d99db7b6 112 for my $modifier_type (qw/before after around/) {
113 my $add_method = "add_${modifier_type}_method_modifier";
114 my $modified = $self->{"${modifier_type}_method_modifiers"};
115
116 for my $method_name (keys %$modified) {
117 for my $code (@{ $modified->{$method_name} }) {
118 $class->$add_method($method_name => $code);
119 }
120 }
121 }
da0c885d 122}
0fc8adbc 123
c2f128e7 124for my $modifier_type (qw/before after around/) {
125 no strict 'refs';
fc0e0bbd 126 *{ __PACKAGE__ . '::' . "add_${modifier_type}_method_modifier" } = sub {
127 my ($self, $method_name, $method) = @_;
128
129 push @{ $self->{"${modifier_type}_method_modifiers"}->{$method_name} },
130 $method;
131 };
132
c2f128e7 133 *{ __PACKAGE__ . '::' . "get_${modifier_type}_method_modifiers" } = sub {
134 my ($self, $method_name, $method) = @_;
135 @{ $self->{"${modifier_type}_method_modifiers"}->{$method_name} || [] }
136 };
137}
138
a2227e71 1391;
140