use Mouse::Meta::TypeConstraint;
-#use Mouse::Meta::Method::Accessor;
-use Mouse::Meta::Method::Delegation;
+sub new {
+ my $class = shift;
+ my $name = shift;
-sub _process_options{
- my($class, $name, $args) = @_;
+ my %args = (@_ == 1) ? %{ $_[0] } : @_;
# XXX: for backward compatibility (with method modifiers)
if($class->can('canonicalize_args') != \&canonicalize_args){
- %{$args} = $class->canonicalize_args($name, %{$args});
- }
-
- # taken from Class::MOP::Attribute::new
-
- defined($name)
- or $class->throw_error('You must provide a name for the attribute');
-
- if(!exists $args->{init_arg}){
- $args->{init_arg} = $name;
- }
-
- # 'required' requires eigher 'init_arg', 'builder', or 'default'
- my $can_be_required = defined( $args->{init_arg} );
-
- if(exists $args->{builder}){
- # XXX:
- # Moose refuses a CODE ref builder, but Mouse doesn't for backward compatibility
- # This feature will be changed in a future. (gfx)
- $class->throw_error('builder must be a defined scalar value which is a method name')
- #if ref $args->{builder} || !defined $args->{builder};
- if !defined $args->{builder};
-
- $can_be_required++;
- }
- elsif(exists $args->{default}){
- if(ref $args->{default} && ref($args->{default}) ne 'CODE'){
- $class->throw_error("References are not allowed as default values, you must "
- . "wrap the default of '$name' in a CODE reference (ex: sub { [] } and not [])");
- }
- $can_be_required++;
- }
-
- if( $args->{required} && !$can_be_required ) {
- $class->throw_error("You cannot have a required attribute ($name) without a default, builder, or an init_arg");
+ %args = $class->canonicalize_args($name, %args);
}
- # taken from Mouse::Meta::Attribute->new and _process_args->
-
- if(exists $args->{is}){
- my $is = $args->{is};
-
- if($is eq 'ro'){
- $args->{reader} ||= $name;
- }
- elsif($is eq 'rw'){
- if(exists $args->{writer}){
- $args->{reader} ||= $name;
- }
- else{
- $args->{accessor} ||= $name;
- }
- }
- elsif($is eq 'bare'){
- # do nothing, but don't complain (later) about missing methods
- }
- else{
- $is = 'undef' if !defined $is;
- $class->throw_error("I do not understand this option (is => $is) on attribute ($name)");
- }
- }
-
- my $tc;
- if(exists $args->{isa}){
- $args->{type_constraint} = Mouse::Util::TypeConstraints::find_or_create_isa_type_constraint($args->{isa});
- }
- elsif(exists $args->{does}){
- $args->{type_constraint} = Mouse::Util::TypeConstraints::find_or_create_does_type_constraint($args->{does});
- }
- $tc = $args->{type_constraint};
-
- if($args->{coerce}){
- defined($tc)
- || $class->throw_error("You cannot have coercion without specifying a type constraint on attribute ($name)");
-
- $args->{weak_ref}
- && $class->throw_error("You cannot have a weak reference to a coerced value on attribute ($name)");
- }
-
- if ($args->{lazy_build}) {
- exists($args->{default})
- && $class->throw_error("You can not use lazy_build and default for the same attribute ($name)");
-
- $args->{lazy} = 1;
- $args->{builder} ||= "_build_${name}";
- if ($name =~ /^_/) {
- $args->{clearer} ||= "_clear${name}";
- $args->{predicate} ||= "_has${name}";
- }
- else {
- $args->{clearer} ||= "clear_${name}";
- $args->{predicate} ||= "has_${name}";
- }
- }
-
- if ($args->{auto_deref}) {
- defined($tc)
- || $class->throw_error("You cannot auto-dereference without specifying a type constraint on attribute ($name)");
-
- ( $tc->is_a_type_of('ArrayRef') || $tc->is_a_type_of('HashRef') )
- || $class->throw_error("You cannot auto-dereference anything other than a ArrayRef or HashRef on attribute ($name)");
- }
-
- if (exists $args->{trigger}) {
- ('CODE' eq ref $args->{trigger})
- || $class->throw_error("Trigger must be a CODE ref on attribute ($name)");
- }
-
- if ($args->{lazy}) {
- (exists $args->{default} || defined $args->{builder})
- || $class->throw_error("You cannot have lazy attribute ($name) without specifying a default value for it");
- }
-
- return;
-}
-
-sub new {
- my $class = shift;
- my $name = shift;
-
- my %args = (@_ == 1) ? %{ $_[0] } : @_;
-
$class->_process_options($name, \%args);
$args{name} = $name;
my ($self, $name, %args) = @_;
Carp::cluck("$self->canonicalize_args has been deprecated."
- . "Use \$self->_process_options instead.")
- if _MOUSE_VERBOSE;
+ . "Use \$self->_process_options instead.");
return %args;
}
my ($self, $class, $name, %args) = @_;
Carp::cluck("$self->create has been deprecated."
- . "Use \$meta->add_attribute and \$attr->install_accessors instead.")
- if _MOUSE_VERBOSE;
+ . "Use \$meta->add_attribute and \$attr->install_accessors instead.");
# noop
return $self;
return 1 if !$type_constraint;
return 1 if $type_constraint->check($value);
- $self->verify_type_constraint_error($self->name, $value, $type_constraint);
-}
-
-sub verify_type_constraint_error {
- my($self, $name, $value, $type) = @_;
- $self->throw_error("Attribute ($name) does not pass the type constraint because: "
- . $type->get_message($value));
+ $self->_throw_type_constraint_error($value, $type_constraint);
}
-sub coerce_constraint { # DEPRECATED
- my $type = $_[0]->{type_constraint}
- or return $_[1];
-
- Carp::cluck("coerce_constraint() has been deprecated, which was an internal utility anyway");
+sub _throw_type_constraint_error {
+ my($self, $value, $type) = @_;
- return Mouse::Util::TypeConstraints->typecast_constraints($_[0]->associated_class->name, $type, $_[1]);
+ $self->throw_error(
+ sprintf q{Attribute (%s) does not pass the type constraint because: %s},
+ $self->name,
+ $type->get_message($value),
+ );
}
sub clone_and_inherit_options{
my %args = ($self->get_parent_args($class, $name), @_);
Carp::cluck("$self->clone_parent has been deprecated."
- . "Use \$meta->add_attribute and \$attr->install_accessors instead.")
- if _MOUSE_VERBOSE;
+ . "Use \$meta->add_attribute and \$attr->install_accessors instead.");
$self->clone_and_inherited_args($class, $name, %args);
}
sub get_read_method {
- $_[0]->reader || $_[0]->accessor
+ return $_[0]->reader || $_[0]->accessor
}
sub get_write_method {
- $_[0]->writer || $_[0]->accessor
+ return $_[0]->writer || $_[0]->accessor
}
-sub get_read_method_ref{
- my($self) = @_;
+sub _get_accessor_method_ref {
+ my($self, $type, $generator) = @_;
- $self->{_read_method_ref} ||= do{
- my $metaclass = $self->associated_class
- or $self->throw_error('No asocciated class for ' . $self->name);
+ my $metaclass = $self->associated_class
+ || $self->throw_error('No asocciated class for ' . $self->name);
- my $reader = $self->{reader} || $self->{accessor};
- if($reader){
- $metaclass->name->can($reader);
- }
- else{
- $self->accessor_metaclass->_generate_reader($self, $metaclass);
- }
- };
+ my $accessor = $self->$type();
+ if($accessor){
+ return $metaclass->get_method_body($accessor);
+ }
+ else{
+ return $self->accessor_metaclass->$generator($self, $metaclass);
+ }
+}
+
+sub get_read_method_ref{
+ my($self) = @_;
+ return $self->{_read_method_ref} ||= $self->_get_accessor_method_ref('get_read_method', '_generate_reader');
}
sub get_write_method_ref{
my($self) = @_;
+ return $self->{_write_method_ref} ||= $self->_get_accessor_method_ref('get_write_method', '_generate_writer');
+}
- $self->{_write_method_ref} ||= do{
- my $metaclass = $self->associated_class
- or $self->throw_error('No asocciated class for ' . $self->name);
+sub set_value {
+ my($self, $object, $value) = @_;
+ return $self->get_write_method_ref()->($object, $value);
+}
- my $reader = $self->{writer} || $self->{accessor};
- if($reader){
- $metaclass->name->can($reader);
- }
- else{
- $self->accessor_metaclass->_generate_writer($self, $metaclass);
- }
- };
+sub get_value {
+ my($self, $object) = @_;
+ return $self->get_read_method_ref()->($object);
}
-sub _canonicalize_handles {
- my($self, $handles) = @_;
+sub has_value {
+ my($self, $object) = @_;
+ my $accessor_ref = $self->{_predicate_ref}
+ ||= $self->_get_accessor_method_ref('predicate', '_generate_predicate');
- if (ref($handles) eq 'HASH') {
- return %$handles;
- }
- elsif (ref($handles) eq 'ARRAY') {
- return map { $_ => $_ } @$handles;
- }
- elsif (ref($handles) eq 'Regexp') {
- my $class_or_role = ($self->{isa} || $self->{does})
- || $self->throw_error("Cannot delegate methods based on a Regexp without a type constraint (isa)");
+ return $accessor_ref->($object);
+}
- my $meta = Mouse::Meta::Class->initialize("$class_or_role"); # "" for stringify
- return map { $_ => $_ }
- grep { $_ ne 'meta' && !Mouse::Object->can($_) && $_ =~ $handles }
- Mouse::Util::is_a_metarole($meta)
- ? $meta->get_method_list
- : $meta->get_all_method_names;
- }
- else {
- $self->throw_error("Unable to canonicalize the 'handles' option with $handles");
- }
+sub clear_value {
+ my($self, $object) = @_;
+ my $accessor_ref = $self->{_crealer_ref}
+ ||= $self->_get_accessor_method_ref('clearer', '_generate_clearer');
+
+ return $accessor_ref->($object);
}
+
sub associate_method{
- my ($attribute, $method) = @_;
+ my ($attribute, $method_name) = @_;
$attribute->{associated_methods}++;
return;
}
-
-sub delegation_metaclass() { 'Mouse::Meta::Method::Delegation' }
-
sub install_accessors{
my($attribute) = @_;
my $generator = '_generate_' . $type;
my $code = $accessor_class->$generator($attribute, $metaclass);
$metaclass->add_method($attribute->{$type} => $code);
- $attribute->associate_method($code);
+ $attribute->associate_method($attribute->{$type});
}
}
# install delegation
if(exists $attribute->{handles}){
- my $delegation_class = $attribute->delegation_metaclass;
my %handles = $attribute->_canonicalize_handles($attribute->{handles});
- my $reader = $attribute->get_read_method_ref;
- while(my($handle_name, $method_to_call) = each %handles){
- my $code = $delegation_class->_generate_delegation($attribute, $metaclass,
- $reader, $handle_name, $method_to_call);
+ while(my($handle, $method_to_call) = each %handles){
+ $metaclass->add_method($handle =>
+ $attribute->_make_delegation_method(
+ $handle, $method_to_call));
- $metaclass->add_method($handle_name => $code);
- $attribute->associate_method($code);
+ $attribute->associate_method($handle);
}
}
-
if($attribute->can('create') != \&create){
# backword compatibility
$attribute->create($metaclass, $attribute->name, %{$attribute});
return;
}
+sub delegation_metaclass() { 'Mouse::Meta::Method::Delegation' }
+
+sub _canonicalize_handles {
+ my($self, $handles) = @_;
+
+ if (ref($handles) eq 'HASH') {
+ return %$handles;
+ }
+ elsif (ref($handles) eq 'ARRAY') {
+ return map { $_ => $_ } @$handles;
+ }
+ elsif ( ref($handles) eq 'CODE' ) {
+ my $class_or_role = ( $self->{isa} || $self->{does} )
+ || $self->throw_error( "Cannot find delegate metaclass for attribute " . $self->name );
+ return $handles->( $self, Mouse::Meta::Class->initialize("$class_or_role"));
+ }
+ elsif (ref($handles) eq 'Regexp') {
+ my $class_or_role = ($self->{isa} || $self->{does})
+ || $self->throw_error("Cannot delegate methods based on a Regexp without a type constraint (isa)");
+
+ my $meta = Mouse::Meta::Class->initialize("$class_or_role"); # "" for stringify
+ return map { $_ => $_ }
+ grep { !Mouse::Object->can($_) && $_ =~ $handles }
+ Mouse::Util::is_a_metarole($meta)
+ ? $meta->get_method_list
+ : $meta->get_all_method_names;
+ }
+ else {
+ $self->throw_error("Unable to canonicalize the 'handles' option with $handles");
+ }
+}
+
+sub _make_delegation_method {
+ my($self, $handle, $method_to_call) = @_;
+ my $delegator = $self->delegation_metaclass;
+ Mouse::Util::load_class($delegator);
+
+ return $delegator->_generate_delegation($self, $handle, $method_to_call);
+}
+
sub throw_error{
my $self = shift;
}
1;
-
__END__
=head1 NAME
=head1 VERSION
-This document describes Mouse version 0.40_03
+This document describes Mouse version 0.47
=head1 METHODS
=back
-=head2 C<< associate_method(Method) >>
+=head2 C<< associate_method(MethodName) >>
Associates a method with the attribute. Typically, this is called internally
when an attribute generates its accessors.
-Currently the argument I<Method> is ignored in Mouse.
+Currently the argument I<MethodName> is ignored in Mouse.
=head2 C<< verify_against_type_constraint(Item) -> TRUE | ERROR >>