Fix potential value for clear
[gitmo/Moose.git] / lib / Moose / Meta / Attribute / Native / Trait.pm
CommitLineData
e3c07b19 1
c466e58f 2package Moose::Meta::Attribute::Native::Trait;
e3c07b19 3use Moose::Role;
4use Moose::Util::TypeConstraints;
5
b6cca0d5 6our $VERSION = '1.14';
e3c07b19 7$VERSION = eval $VERSION;
8our $AUTHORITY = 'cpan:STEVAN';
9
2e069f5a 10requires '_helper_type';
e3c07b19 11
2edb73d9 12before '_process_options' => sub {
13 my ( $self, $name, $options ) = @_;
e3c07b19 14
2edb73d9 15 $self->_check_helper_type( $options, $name );
e3c07b19 16
2edb73d9 17 $options->{is} = $self->_default_is
d4dc38ed 18 if !exists $options->{is} && $self->can('_default_is');
e3c07b19 19
2edb73d9 20 $options->{default} = $self->_default_default
d4dc38ed 21 if !exists $options->{default} && $self->can('_default_default');
2edb73d9 22};
e3c07b19 23
2edb73d9 24sub _check_helper_type {
25 my ( $self, $options, $name ) = @_;
e3c07b19 26
2e069f5a 27 my $type = $self->_helper_type;
2edb73d9 28
2e069f5a 29 $options->{isa} = $type
2edb73d9 30 unless exists $options->{isa};
31
32 my $isa = Moose::Util::TypeConstraints::find_or_create_type_constraint(
33 $options->{isa} );
34
35 ( $isa->is_a_type_of($type) )
36 || confess
37 "The type constraint for $name must be a subtype of $type but it's a $isa";
38}
e3c07b19 39
2edb73d9 40before 'install_accessors' => sub { (shift)->_check_handles_values };
5404f169 41
2edb73d9 42sub _check_handles_values {
e3c07b19 43 my $self = shift;
44
45 my $method_constructors = $self->method_constructors;
46
5404f169 47 my %handles = $self->_canonicalize_handles;
e3c07b19 48
046c8b5e 49 for my $original_method ( values %handles ) {
5404f169 50 my $name = $original_method->[0];
f7fd22b6 51
ffc2e25f 52 my $accessor_class = $self->_native_accessor_class_for($name);
f7fd22b6 53
ffc2e25f 54 # XXX - bridge code
55 ( ( $accessor_class && $accessor_class->can('new') )
56 || exists $method_constructors->{$name} )
04e05413 57 || confess "$name is an unsupported method type - $accessor_class";
e3c07b19 58 }
5404f169 59}
e3c07b19 60
d4dc38ed 61around '_canonicalize_handles' => sub {
62 shift;
63 my $self = shift;
64 my $handles = $self->handles;
65
66 return unless $handles;
67
68 unless ( 'HASH' eq ref $handles ) {
69 $self->throw_error(
70 "The 'handles' option must be a HASH reference, not $handles");
71 }
72
73 return map {
74 my $to = $handles->{$_};
75 $to = [$to] unless ref $to;
76 $_ => $to
77 } keys %$handles;
78};
79
18281451 80around '_make_delegation_method' => sub {
81 my $next = shift;
046c8b5e 82 my ( $self, $handle_name, $method_to_call ) = @_;
18281451 83
3c573ca4 84 my ( $name, @curried_args ) = @$method_to_call;
18281451 85
ffc2e25f 86 my $accessor_class = $self->_native_accessor_class_for($name);
18281451 87
ffc2e25f 88 # XXX - bridge code
89 if ( $accessor_class && $accessor_class->can('new') ) {
f7fd22b6 90 return $accessor_class->new(
91 name => $handle_name,
92 package_name => $self->associated_class->name,
93 attribute => $self,
94 curried_arguments => \@curried_args,
a6ae7438 95 root_types => [ $self->_root_types ],
f7fd22b6 96 );
97 }
ffc2e25f 98 # XXX - bridge code
f7fd22b6 99 else {
100 my $method_constructors = $self->method_constructors;
101
102 my $code = $method_constructors->{$name}->(
103 $self,
104 $self->get_read_method_ref,
105 $self->get_write_method_ref,
106 );
107
108 return $next->(
109 $self,
110 $handle_name,
111 sub {
112 my $instance = shift;
113 return $code->( $instance, @curried_args, @_ );
114 }
115 );
116 }
18281451 117};
118
a6ae7438 119sub _root_types {
120 return $_[0]->_helper_type;
121}
122
ffc2e25f 123sub _native_accessor_class_for {
124 my ( $self, $suffix ) = @_;
125
ffc2e25f 126 return 'Moose::Meta::Method::Accessor::Native::' . $self->_native_type . '::' . $suffix;
f7fd22b6 127}
128
15715245 129sub _build_native_type {
130 my $self = shift;
131
132 for my $role_name ( map { $_->name } $self->meta->calculate_all_roles ) {
133 return $1 if $role_name =~ /::Native::Trait::(\w+)$/;
134 }
135
136 die "Cannot calculate native type for " . ref $self;
137}
138
139has '_native_type' => (
140 is => 'ro',
141 isa => 'Str',
142 lazy => 1,
143 builder => '_build_native_type',
144);
145
f7fd22b6 146has 'method_constructors' => (
147 is => 'ro',
148 isa => 'HashRef',
149 lazy => 1,
150 default => sub {
151 my $self = shift;
a7821be5 152 return +{}
153 unless $self->can('has_method_provider')
154 && $self->has_method_provider;
f7fd22b6 155
156 # or grab them from the role/class
157 my $method_provider = $self->method_provider->meta;
158 return +{ map { $_->name => $_ }
159 $method_provider->_get_local_methods };
160 },
161);
162
e3c07b19 163no Moose::Role;
164no Moose::Util::TypeConstraints;
165
1661;
167
168__END__
169
170=head1 NAME
171
2420461c 172Moose::Meta::Attribute::Native::Trait - Base role for helpers
e3c07b19 173
e3c07b19 174=head1 BUGS
175
d4048ef3 176See L<Moose/BUGS> for details on reporting bugs.
e3c07b19 177
1af5d116 178=head1 SEE ALSO
179
180Documentation for Moose native traits starts at L<Moose::Meta::Attribute Native>
181
e3c07b19 182=head1 AUTHORS
183
184Yuval Kogman
185
186Shawn M Moore
187
188Jesse Luehrs
189
190=head1 COPYRIGHT AND LICENSE
191
192Copyright 2007-2009 by Infinity Interactive, Inc.
193
194L<http://www.iinteractive.com>
195
196This library is free software; you can redistribute it and/or modify
197it under the same terms as Perl itself.
198
199=cut