Commit | Line | Data |
2b4ce4bd |
1 | #!/usr/bin/env perl |
8eec3c69 |
2 | package MooseX::Singleton::Role::Meta::Method::Constructor; |
3 | use Moose::Role; |
2b4ce4bd |
4 | |
8eec3c69 |
5 | override _initialize_body => sub { |
2b4ce4bd |
6 | my $self = shift; |
7 | # TODO: |
8 | # the %options should also include a both |
9 | # a call 'initializer' and call 'SUPER::' |
10 | # options, which should cover approx 90% |
11 | # of the possible use cases (even if it |
12 | # requires some adaption on the part of |
13 | # the author, after all, nothing is free) |
0272982a |
14 | my $source = 'sub {'; |
2b4ce4bd |
15 | $source .= "\n" . 'my $class = shift;'; |
8eec3c69 |
16 | |
32bf84e9 |
17 | $source .= "\n" . 'my $existing = do { no strict "refs"; no warnings "once"; \${"$class\::singleton"}; };'; |
2b4ce4bd |
18 | $source .= "\n" . 'return ${$existing} if ${$existing};'; |
19 | |
20 | $source .= "\n" . 'return $class->Moose::Object::new(@_)'; |
21 | $source .= "\n" . ' if $class ne \'' . $self->associated_metaclass->name . '\';'; |
22 | |
0cd38a85 |
23 | $source .= $self->_generate_params('$params', '$class'); |
24 | $source .= $self->_generate_instance('$instance', '$class'); |
25 | $source .= $self->_generate_slot_initializers; |
2b4ce4bd |
26 | |
a06ef25a |
27 | $source .= ";\n" . $self->_generate_triggers(); |
2b4ce4bd |
28 | $source .= ";\n" . $self->_generate_BUILDALL(); |
29 | |
30 | $source .= ";\n" . 'return ${$existing} = $instance'; |
31 | $source .= ";\n" . '}'; |
32 | warn $source if $self->options->{debug}; |
33 | |
0cd38a85 |
34 | my $attrs = $self->_attributes; |
35 | |
36 | my @type_constraints = map { |
37 | $_->can('type_constraint') ? $_->type_constraint : undef |
38 | } @$attrs; |
0272982a |
39 | |
0cd38a85 |
40 | my @type_constraint_bodies = map { |
41 | defined $_ ? $_->_compiled_type_constraint : undef; |
42 | } @type_constraints; |
2b4ce4bd |
43 | |
ade9ece0 |
44 | my ( $code, $e ) = $self->_compile_code( |
0cd38a85 |
45 | code => $source, |
46 | environment => { |
47 | '$meta' => \$self, |
48 | '$attrs' => \$attrs, |
49 | '@type_constraints' => \@type_constraints, |
50 | '@type_constraint_bodies' => \@type_constraint_bodies, |
51 | }, |
ade9ece0 |
52 | ); |
53 | |
54 | $self->throw_error("Could not eval the constructor :\n\n$source\n\nbecause :\n\n$e", error => $e, data => $source ) |
55 | if $e; |
2b4ce4bd |
56 | |
ede8dce0 |
57 | $self->{'body'} = $code; |
8eec3c69 |
58 | }; |
59 | |
60 | # Ideally we'd be setting this in the constructor, but the new() methods in |
61 | # what the parent classes are not well-factored. |
62 | # |
63 | # This is all a nasty hack, though. We need to fix Class::MOP::Inlined to |
64 | # allow constructor class roles to say "if the parent class has role X, |
65 | # inline". |
66 | override _expected_method_class => sub { |
67 | my $self = shift; |
dbeedf9e |
68 | |
8eec3c69 |
69 | my $super_value = super(); |
70 | if ( $super_value eq 'Moose::Object' ) { |
71 | for my $parent ( map { Class::MOP::class_of($_) } |
72 | $self->associated_metaclass->superclasses ) { |
73 | return $parent->name |
74 | if $parent->is_anon_class |
75 | && grep { $_->name eq 'Moose::Object' } |
76 | map { Class::MOP::class_of($_) } $parent->superclasses; |
77 | } |
78 | } |
79 | |
80 | return $super_value; |
81 | }; |
c87dffa8 |
82 | |
2b4ce4bd |
83 | no Moose; |
84 | |
85 | 1; |