bump copyright year to 2010
[gitmo/Moose.git] / lib / Moose / Cookbook / Basics / Recipe6.pod
CommitLineData
a909a4df 1
2=pod
3
4=head1 NAME
5
021b8139 6Moose::Cookbook::Basics::Recipe6 - The augment/inner example
a909a4df 7
8=head1 SYNOPSIS
f1c9feb4 9
a909a4df 10 package Document::Page;
11 use Moose;
f1c9feb4 12
13 has 'body' => ( is => 'rw', isa => 'Str', default => sub {''} );
14
a909a4df 15 sub create {
16 my $self = shift;
17 $self->open_page;
18 inner();
19 $self->close_page;
20 }
f1c9feb4 21
22 sub append_body {
23 my ( $self, $appendage ) = @_;
24 $self->body( $self->body . $appendage );
a909a4df 25 }
f1c9feb4 26
d876e049 27 sub open_page { (shift)->append_body('<page>') }
f1c9feb4 28 sub close_page { (shift)->append_body('</page>') }
29
d876e049 30 package Document::PageWithHeadersAndFooters;
a909a4df 31 use Moose;
f1c9feb4 32
a909a4df 33 extends 'Document::Page';
f1c9feb4 34
d876e049 35 augment 'create' => sub {
a909a4df 36 my $self = shift;
37 $self->create_header;
38 inner();
39 $self->create_footer;
d876e049 40 };
f1c9feb4 41
a909a4df 42 sub create_header { (shift)->append_body('<header/>') }
f1c9feb4 43 sub create_footer { (shift)->append_body('<footer/>') }
44
d876e049 45 package TPSReport;
a909a4df 46 use Moose;
f1c9feb4 47
d876e049 48 extends 'Document::PageWithHeadersAndFooters';
f1c9feb4 49
d876e049 50 augment 'create' => sub {
a909a4df 51 my $self = shift;
52 $self->create_tps_report;
71181776 53 inner();
a909a4df 54 };
f1c9feb4 55
a909a4df 56 sub create_tps_report {
f1c9feb4 57 (shift)->append_body('<report type="tps"/>');
a909a4df 58 }
f1c9feb4 59
60 # <page><header/><report type="tps"/><footer/></page>
c79239a2 61 my $report_xml = TPSReport->new->create;
a909a4df 62
63=head1 DESCRIPTION
64
71181776 65This recipe shows how the C<augment> method modifier works. This
66modifier reverses the normal subclass to parent method resolution
67order. With an C<augment> modifier the I<least> specific method is
68called first. Each successive call to C<inner> descends the
69inheritance tree, ending at the most specific subclass.
7125b244 70
71181776 71The C<augment> modifier lets you design a parent class that can be
72extended in a specific way. The parent provides generic wrapper
73functionality, and the subclasses fill in the details.
a909a4df 74
71181776 75In the example above, we've created a set of document classes, with
76the most specific being the C<TPSReport> class.
a909a4df 77
71181776 78We start with the least specific class, C<Document::Page>. Its create
79method contains a call to C<inner()>:
a909a4df 80
71181776 81 sub create {
82 my $self = shift;
83 $self->open_page;
84 inner();
85 $self->close_page;
86 }
87
88The C<inner> function is exported by C<Moose>, and is like C<super>
89for augmented methods. When C<inner> is called, Moose finds the next
90method in the chain, which is the C<augment> modifier in
91C<Document::PageWithHeadersAndFooters>. You'll note that we can call
92C<inner> in our modifier:
93
94 augment 'create' => sub {
95 my $self = shift;
96 $self->create_header;
97 inner();
98 $self->create_footer;
99 };
100
101This finds the next most specific modifier, in the C<TPSReport> class.
102
103Finally, in the C<TPSReport> class, the chain comes to an end:
104
105 augment 'create' => sub {
106 my $self = shift;
107 $self->create_tps_report;
108 inner();
109 };
110
111We do call the C<inner> function one more time, but since there is no
112more specific subclass, this is a no-op. Making this call means we can
113easily subclass C<TPSReport> in the future.
114
115=head1 CONCLUSION
116
117The C<augment> modifier is a powerful tool for creating a set of
118nested wrappers. It's not something you will need often, but when you
0367f4bf 119do, it is very handy.
a909a4df 120
121=head1 AUTHOR
122
123Stevan Little E<lt>stevan@iinteractive.comE<gt>
124
71181776 125Dave Rolsky E<lt>autarch@urth.orgE<gt>
126
a909a4df 127=head1 COPYRIGHT AND LICENSE
128
8e5dd3fb 129Copyright 2007-2010 by Infinity Interactive, Inc.
a909a4df 130
131L<http://www.iinteractive.com>
132
133This library is free software; you can redistribute it and/or modify
134it under the same terms as Perl itself.
135
c79239a2 136=begin testing
137
138my $tps_report = TPSReport->new;
139isa_ok( $tps_report, 'TPSReport' );
140
141is(
142 $tps_report->create,
143 q{<page><header/><report type="tps"/><footer/></page>},
144 '... got the right TPS report'
145);
146
147=end testing
148
71181776 149=cut