Add Array->each_n.
[gitmo/Moose-Autobox.git] / lib / Moose / Autobox / Array.pm
CommitLineData
5f654d8e 1package Moose::Autobox::Array;
2use Moose::Role 'with';
7fc99864 3use Perl6::Junction;
7dad2765 4use Moose::Autobox;
5f654d8e 5
e3598a18 6our $VERSION = '0.10';
5f654d8e 7
e6bb88b0 8with 'Moose::Autobox::Ref',
31d40d73 9 'Moose::Autobox::List',
10 'Moose::Autobox::Indexed';
6cf5bcf2 11
12## Array Interface
13
14sub pop {
15 my ($array) = @_;
16 CORE::pop @$array;
17}
18
19sub push {
20 my ($array, @rest) = @_;
21 CORE::push @$array, @rest;
22 $array;
23}
24
25sub unshift {
26 my ($array, @rest) = @_;
27 CORE::unshift @$array, @rest;
28 $array;
29}
5f654d8e 30
6cf5bcf2 31sub delete {
32 my ($array, $index) = @_;
33 CORE::delete $array->[$index];
34}
35
36sub shift {
37 my ($array) = @_;
38 CORE::shift @$array;
c11e6a74 39}
40
41sub slice {
42 my ($array, $indicies) = @_;
43 [ @{$array}[ @{$indicies} ] ];
44}
6cf5bcf2 45
46# NOTE:
47# sprintf args need to be reversed,
48# because the invocant is the array
49sub sprintf { CORE::sprintf $_[1], @{$_[0]} }
50
51## ::List interface implementation
52
53sub head { $_[0]->[0] }
54sub tail { [ @{$_[0]}[ 1 .. $#{$_[0]} ] ] }
e6bb88b0 55
5f654d8e 56sub length {
57 my ($array) = @_;
58 CORE::scalar @$array;
59}
60
61sub grep {
62 my ($array, $sub) = @_;
63 [ CORE::grep { $sub->($_) } @$array ];
64}
65
66sub map {
67 my ($array, $sub) = @_;
68 [ CORE::map { $sub->($_) } @$array ];
69}
70
71sub join {
feffe00c 72 my ($array, $sep) = @_;
73 $sep ||= '';
5f654d8e 74 CORE::join $sep, @$array;
75}
76
77sub reverse {
78 my ($array) = @_;
e6bb88b0 79 [ CORE::reverse @$array ];
5f654d8e 80}
81
82sub sort {
83 my ($array, $sub) = @_;
84 $sub ||= sub { $a cmp $b };
85 [ CORE::sort { $sub->($a, $b) } @$array ];
5dc78481 86}
87
260cc81f 88## ::Indexed implementation
5dc78481 89
260cc81f 90sub at {
91 my ($array, $index) = @_;
92 $array->[$index];
93}
5dc78481 94
260cc81f 95sub put {
96 my ($array, $index, $value) = @_;
97 $array->[$index] = $value;
98}
5dc78481 99
6cf5bcf2 100sub exists {
101 my ($array, $index) = @_;
102 CORE::exists $array->[$index];
103}
5dc78481 104
105sub keys {
106 my ($array) = @_;
107 [ 0 .. $#{$array} ];
108}
109
110sub values {
111 my ($array) = @_;
112 [ @$array ];
113}
114
115sub kv {
116 my ($array) = @_;
feffe00c 117 $array->keys->map(sub { [ $_, $array->[$_] ] });
5dc78481 118}
e6bb88b0 119
2dcdd7d7 120sub each {
121 my ($array, $sub) = @_;
122 for my $i (0 .. $#$array) {
123 $sub->($i, $array->[ $i ]);
124 }
125}
126
127sub each_key {
128 my ($array, $sub) = @_;
2224d5b4 129 $sub->($_) for (0 .. $#$array);
2dcdd7d7 130}
131
132sub each_value {
133 my ($array, $sub) = @_;
134 $sub->($_) for @$array;
135}
136
0e480911 137sub each_n {
138 my ($array, $n, $sub) = @_;
139 my $it = List::MoreUtils::natatime($n, @$array);
140
141 while (my @vals = $it->()) {
142 $sub->(@vals);
143 }
144
145 return;
146}
147
2dcdd7d7 148# end indexed
149
2197a7c0 150sub flatten {
151 @{$_[0]}
152}
153
e477b088 154sub _flatten_deep {
155 my @array = @_;
156 my $depth = CORE::pop @array;
157 --$depth if (defined($depth));
158
159 CORE::map {
160 (ref eq 'ARRAY')
161 ? (defined($depth) && $depth == -1) ? $_ : _flatten_deep( @$_, $depth )
162 : $_
163 } @array;
164
165}
166
167sub flatten_deep {
168 my ($array, $depth) = @_;
169 [ _flatten_deep(@$array, $depth) ];
170}
171
7fc99864 172## Junctions
173
174sub all {
175 my ($array) = @_;
aaf111de 176 return Perl6::Junction::all(@$array);
7fc99864 177}
178
179sub any {
180 my ($array) = @_;
aaf111de 181 return Perl6::Junction::any(@$array);
7fc99864 182}
183
184sub none {
185 my ($array) = @_;
aaf111de 186 return Perl6::Junction::none(@$array);
7fc99864 187}
188
189sub one {
190 my ($array) = @_;
aaf111de 191 return Perl6::Junction::one(@$array);
7fc99864 192}
193
3f4dd8b7 194## Print
195
196sub print { CORE::print @{$_[0]} }
197sub say { CORE::print @{$_[0]}, "\n" }
198
5f654d8e 1991;
31d40d73 200
201__END__
202
203=pod
204
205=head1 NAME
206
207Moose::Autobox::Array - the Array role
208
209=head1 SYNOPOSIS
210
211 use Moose::Autobox;
31d40d73 212
5272f13f 213 [ 1..5 ]->isa('ARRAY'); # true
214 [ a..z ]->does('Moose::Autobox::Array'); # true
215 [ 0..2 ]->does('Moose::Autobox::List'); # true
216
31d40d73 217 print "Squares: " . [ 1 .. 10 ]->map(sub { $_ * $_ })->join(', ');
5272f13f 218
219 print [ 1, 'number' ]->sprintf('%d is the loneliest %s');
f6e003cc 220
221 print ([ 1 .. 5 ]->any == 3) ? 'true' : 'false'; # prints 'true'
31d40d73 222
223=head1 DESCRIPTION
224
8937074a 225This is a role to describe operations on the Array type.
226
260cc81f 227=head1 METHODS
228
229=over 4
230
260cc81f 231=item B<pop>
232
5272f13f 233=item B<push ($value)>
260cc81f 234
235=item B<shift>
236
5272f13f 237=item B<unshift ($value)>
260cc81f 238
5272f13f 239=item B<delete ($index)>
260cc81f 240
5272f13f 241=item B<sprintf ($format_string)>
260cc81f 242
c11e6a74 243=item B<slice (@indices)>
244
2197a7c0 245=item B<flatten>
246
e477b088 247=item B<flatten_deep ($depth)>
248
0e480911 249=item B<each_n ($n, $callback)>
250
260cc81f 251=back
252
5272f13f 253=head2 Indexed implementation
260cc81f 254
255=over 4
256
5272f13f 257=item B<at ($index)>
260cc81f 258
5272f13f 259=item B<put ($index, $value)>
260cc81f 260
5272f13f 261=item B<exists ($index)>
260cc81f 262
263=item B<keys>
264
260cc81f 265=item B<values>
266
5272f13f 267=item B<kv>
268
c0ab09e3 269=item B<each>
270
271=item B<each_key>
272
273=item B<each_value>
274
260cc81f 275=back
276
5272f13f 277=head2 List implementation
260cc81f 278
279=over 4
280
281=item B<head>
282
283=item B<tail>
284
5272f13f 285=item B<join (?$seperator)>
260cc81f 286
287=item B<length>
288
5272f13f 289=item B<map (\&block)>
260cc81f 290
5272f13f 291=item B<grep (\&block)>
260cc81f 292
09a0196c 293Note that, in both the above, $_ is in scope within the code block, as well as
294being passed as $_[0]. As per CORE::map and CORE::grep, $_ is an alias to
295the list value, so can be used to to modify the list, viz:
296
297 use Moose::Autobox;
298
299 my $foo = [1, 2, 3];
300 $foo->map( sub {$_++} );
301 print $foo->dump;
302
303yields
304
305 $VAR1 = [
306 2,
307 3,
308 4
309 ];
310
260cc81f 311=item B<reverse>
312
5272f13f 313=item B<sort (?\&block)>
314
315=back
316
7fc99864 317=head2 Junctions
318
319=over 4
320
321=item B<all>
322
323=item B<any>
324
325=item B<none>
326
327=item B<one>
328
329=back
330
5272f13f 331=over 4
332
333=item B<meta>
260cc81f 334
3f4dd8b7 335=item B<print>
336
337=item B<say>
338
260cc81f 339=back
340
31d40d73 341=head1 BUGS
342
343All complex software has bugs lurking in it, and this module is no
344exception. If you find a bug please either email me, or add the bug
345to cpan-RT.
346
347=head1 AUTHOR
348
349Stevan Little E<lt>stevan@iinteractive.comE<gt>
350
351=head1 COPYRIGHT AND LICENSE
352
ea4e64bf 353Copyright 2006-2008 by Infinity Interactive, Inc.
31d40d73 354
355L<http://www.iinteractive.com>
356
357This library is free software; you can redistribute it and/or modify
358it under the same terms as Perl itself.
359
f6e003cc 360=cut