typo fixes
[gitmo/Moose.git] / lib / Moose / Meta / Attribute / Native / Trait.pm
CommitLineData
e3c07b19 1
c466e58f 2package Moose::Meta::Attribute::Native::Trait;
e3c07b19 3use Moose::Role;
3cf2f9ec 4
efa806d8 5use List::MoreUtils qw( any uniq );
e3c07b19 6use Moose::Util::TypeConstraints;
34d6d196 7use Moose::Deprecated;
e3c07b19 8
efa728b4 9our $VERSION = '1.15';
e3c07b19 10$VERSION = eval $VERSION;
11our $AUTHORITY = 'cpan:STEVAN';
12
2e069f5a 13requires '_helper_type';
e3c07b19 14
efa806d8 15has _used_default_is => (
16 is => 'rw',
17 isa => 'Bool',
18 default => 0,
19);
20
2edb73d9 21before '_process_options' => sub {
22 my ( $self, $name, $options ) = @_;
e3c07b19 23
2edb73d9 24 $self->_check_helper_type( $options, $name );
e3c07b19 25
3cf2f9ec 26 if ( !( any { exists $options->{$_} } qw( is reader writer accessor ) )
27 && $self->can('_default_is') ) {
28
34d6d196 29 $options->{is} = $self->_default_is;
30
efa806d8 31 $options->{_used_default_is} = 1;
34d6d196 32 }
33
3cf2f9ec 34 if (
35 !(
36 $options->{required}
b558f8a6 37 || any { exists $options->{$_} } qw( default builder lazy_build )
3cf2f9ec 38 )
39 && $self->can('_default_default')
40 ) {
41
34d6d196 42 $options->{default} = $self->_default_default;
43
44 Moose::Deprecated::deprecated(
45 feature => 'default default for Native Trait',
46 message =>
47 'Allowing a native trait to automatically supply a default is deprecated'
48 );
49 }
2edb73d9 50};
e3c07b19 51
efa806d8 52after 'install_accessors' => sub {
53 my $self = shift;
54
55 return unless $self->_used_default_is;
56
57 my @methods
58 = $self->_default_is eq 'rw'
59 ? qw( reader writer accessor )
60 : 'reader';
61
62 my $name = $self->name;
63 my $class = $self->associated_class->name;
64
65 for my $meth ( uniq grep {defined} map { $self->$_ } @methods ) {
66
67 my $message
68 = "The $meth method in the $class class was automatically created"
69 . " by the native delegation trait for the $name attribute."
70 . q{ This "default is" feature is deprecated.}
71 . q{ Explicitly set "is" or define accessor names to avoid this};
72
73 $self->associated_class->add_before_method_modifier(
74 $meth => sub {
75 Moose::Deprecated::deprecated(
76 feature => 'default is for Native Trait',
77 message =>$message,
78 );
79 }
80 );
81 }
82 };
83
2edb73d9 84sub _check_helper_type {
85 my ( $self, $options, $name ) = @_;
e3c07b19 86
2e069f5a 87 my $type = $self->_helper_type;
2edb73d9 88
2e069f5a 89 $options->{isa} = $type
2edb73d9 90 unless exists $options->{isa};
91
92 my $isa = Moose::Util::TypeConstraints::find_or_create_type_constraint(
93 $options->{isa} );
94
95 ( $isa->is_a_type_of($type) )
96 || confess
97 "The type constraint for $name must be a subtype of $type but it's a $isa";
98}
e3c07b19 99
2edb73d9 100before 'install_accessors' => sub { (shift)->_check_handles_values };
5404f169 101
2edb73d9 102sub _check_handles_values {
e3c07b19 103 my $self = shift;
104
5404f169 105 my %handles = $self->_canonicalize_handles;
e3c07b19 106
046c8b5e 107 for my $original_method ( values %handles ) {
5404f169 108 my $name = $original_method->[0];
f7fd22b6 109
ffc2e25f 110 my $accessor_class = $self->_native_accessor_class_for($name);
f7fd22b6 111
78aee58f 112 ( $accessor_class && $accessor_class->can('new') )
113 || confess
114 "$name is an unsupported method type - $accessor_class";
e3c07b19 115 }
5404f169 116}
e3c07b19 117
d4dc38ed 118around '_canonicalize_handles' => sub {
119 shift;
120 my $self = shift;
121 my $handles = $self->handles;
122
123 return unless $handles;
124
125 unless ( 'HASH' eq ref $handles ) {
126 $self->throw_error(
127 "The 'handles' option must be a HASH reference, not $handles");
128 }
129
130 return map {
131 my $to = $handles->{$_};
132 $to = [$to] unless ref $to;
133 $_ => $to
134 } keys %$handles;
135};
136
18281451 137around '_make_delegation_method' => sub {
138 my $next = shift;
046c8b5e 139 my ( $self, $handle_name, $method_to_call ) = @_;
18281451 140
3c573ca4 141 my ( $name, @curried_args ) = @$method_to_call;
18281451 142
ffc2e25f 143 my $accessor_class = $self->_native_accessor_class_for($name);
18281451 144
78aee58f 145 die "Cannot find an accessor class for $name"
146 unless $accessor_class && $accessor_class->can('new');
147
148 return $accessor_class->new(
8b9641b8 149 name => $handle_name,
150 package_name => $self->associated_class->name,
151 delegate_to_method => $name,
152 attribute => $self,
153 is_inline => 1,
154 curried_arguments => \@curried_args,
155 root_types => [ $self->_root_types ],
78aee58f 156 );
18281451 157};
158
a6ae7438 159sub _root_types {
160 return $_[0]->_helper_type;
161}
162
ffc2e25f 163sub _native_accessor_class_for {
164 my ( $self, $suffix ) = @_;
165
8b9641b8 166 my $role
167 = 'Moose::Meta::Method::Accessor::Native::'
168 . $self->_native_type . '::'
169 . $suffix;
170
171 return Moose::Meta::Class->create_anon_class(
172 superclasses =>
173 [ $self->accessor_metaclass, $self->delegation_metaclass ],
174 roles => [$role],
175 cache => 1,
176 )->name;
f7fd22b6 177}
178
15715245 179sub _build_native_type {
180 my $self = shift;
181
182 for my $role_name ( map { $_->name } $self->meta->calculate_all_roles ) {
183 return $1 if $role_name =~ /::Native::Trait::(\w+)$/;
184 }
185
186 die "Cannot calculate native type for " . ref $self;
187}
188
189has '_native_type' => (
190 is => 'ro',
191 isa => 'Str',
192 lazy => 1,
193 builder => '_build_native_type',
194);
195
e3c07b19 196no Moose::Role;
197no Moose::Util::TypeConstraints;
198
1991;
200
201__END__
202
203=head1 NAME
204
e132fd56 205Moose::Meta::Attribute::Native::Trait - Shared role for native delegation traits
e3c07b19 206
e3c07b19 207=head1 BUGS
208
d4048ef3 209See L<Moose/BUGS> for details on reporting bugs.
e3c07b19 210
1af5d116 211=head1 SEE ALSO
212
e132fd56 213Documentation for Moose native traits can be found in
214L<Moose::Meta::Attribute::Native>.
1af5d116 215
e3c07b19 216=head1 AUTHORS
217
218Yuval Kogman
219
220Shawn M Moore
221
222Jesse Luehrs
223
224=head1 COPYRIGHT AND LICENSE
225
226Copyright 2007-2009 by Infinity Interactive, Inc.
227
228L<http://www.iinteractive.com>
229
230This library is free software; you can redistribute it and/or modify
231it under the same terms as Perl itself.
232
233=cut