sub add_attribute {
my $self = shift;
- my $name = shift;
- if (scalar @_ == 1 && ref($_[0]) eq 'HASH') {
- # NOTE:
- # if it is a HASH ref, we de-ref it.
- # this will usually mean that it is
- # coming from a role
- $self->SUPER::add_attribute($self->_process_attribute($name => %{$_[0]}));
- }
- else {
- # otherwise we just pass the args
- $self->SUPER::add_attribute($self->_process_attribute($name => @_));
- }
+ $self->SUPER::add_attribute($self->_process_attribute(@_));
}
sub add_override_method_modifier {
}
# NOTE:
-# this was crap anyway, see
-# Moose::Util::apply_all_roles
+# this was crap anyway, see
+# Moose::Util::apply_all_roles
# instead
sub _apply_all_roles { die "DEPRECATED" }
+my %ANON_CLASSES;
+
sub _process_attribute {
- my ($self, $name, %options) = @_;
+ my $self = shift;
+ my $name = shift;
+ my %options = ((scalar @_ == 1 && ref($_[0]) eq 'HASH') ? %{$_[0]} : @_);
+
if ($name =~ /^\+(.*)/) {
return $self->_process_inherited_attribute($1, %options);
}
else {
+ my $attr_metaclass_name;
if ($options{metaclass}) {
my $metaclass_name = $options{metaclass};
eval {
if ($@) {
Class::MOP::load_class($metaclass_name);
}
- return $metaclass_name->new($name, %options);
+ $attr_metaclass_name = $metaclass_name;
}
else {
- return $self->attribute_metaclass->new($name, %options);
+ $attr_metaclass_name = $self->attribute_metaclass;
}
+
+ if ($options{traits}) {
+
+ my $anon_role_key = join "|" => @{$options{traits}};
+
+ my $class;
+ if (exists $ANON_CLASSES{$anon_role_key} && defined $ANON_CLASSES{$anon_role_key}) {
+ $class = $ANON_CLASSES{$anon_role_key};
+ }
+ else {
+ $class = Moose::Meta::Class->create_anon_class(
+ superclasses => [ $attr_metaclass_name ]
+ );
+ $ANON_CLASSES{$anon_role_key} = $class;
+ Moose::Util::apply_all_roles($class, @{$options{traits}});
+ }
+
+ $attr_metaclass_name = $class->name;
+ }
+
+ return $attr_metaclass_name->new($name, %options);
}
}
--- /dev/null
+#!/usr/bin/perl
+
+use strict;
+use warnings;
+
+use Test::More tests => 5;
+use Test::Exception;
+
+BEGIN {
+ use_ok('Moose');
+}
+
+{
+ package My::Attribute::Trait;
+ use Moose::Role;
+
+ has 'alias_to' => (is => 'ro', isa => 'Str');
+
+ after 'install_accessors' => sub {
+ my $self = shift;
+ $self->associated_class->add_method(
+ $self->alias_to,
+ $self->get_read_method_ref
+ );
+ };
+}
+
+{
+ package My::Class;
+ use Moose;
+
+ has 'bar' => (
+ traits => [qw/My::Attribute::Trait/],
+ is => 'ro',
+ isa => 'Int',
+ alias_to => 'baz',
+ );
+}
+
+my $c = My::Class->new(bar => 100);
+isa_ok($c, 'My::Class');
+
+is($c->bar, 100, '... got the right value for bar');
+
+can_ok($c, 'baz');
+is($c->baz, 100, '... got the right value for baz');
--- /dev/null
+#!/usr/bin/perl
+
+use strict;
+use warnings;
+
+use Test::More no_plan => 1;
+use Test::Exception;
+
+BEGIN {
+ use_ok('Moose');
+}
+
+{
+ package My::Role;
+ use Moose::Role;
+
+ around 'baz' => sub {
+ my $next = shift;
+ 'My::Role::baz(' . $next->(@_) . ')';
+ };
+}
+
+{
+ package Foo;
+ use Moose;
+
+ sub baz { 'Foo::baz' }
+
+ __PACKAGE__->meta->make_immutable(debug => 0);
+}
+
+my $foo = Foo->new;
+isa_ok($foo, 'Foo');
+
+is($foo->baz, 'Foo::baz', '... got the right value');
+
+lives_ok {
+ My::Role->meta->apply($foo)
+} '... successfully applied the role to immutable instance';
+
+is($foo->baz, 'My::Role::baz(Foo::baz)', '... got the right value');
+
+