# NAME
autobox::Transform - Autobox methods to transform Arrays and Hashes
# SYNOPSIS
# These are equivalent ways to transform arrays and arrayrefs
### map_by
my @genres = map { $_->genre() } @$books;
my @genres = $books->map_by("genre");
my $genres = [ map { $_->genre() } @$books ];
my $genres = $books->map_by("genre");
# With sum from autobox::Core / List::AllUtils
my $book_order_total = sum(
map { $_->price_with_tax($tax_pct) } @{$order->books}
);
my $book_order_total = $order->books
->map_by(price_with_tax => [$tax_pct])->sum;
### grep_by
my $sold_out_books = [ grep { $_->is_sold_out } @$books ];
my $sold_out_books = $books->grep_by("is_sold_out");
my $books_in_library = [ grep { $_->is_in_library($library) } @$books ];
my $books_in_library = $books->grep_by(is_in_library => [$library]);
### group_by
$books->group_by("title"),
# {
# "Leviathan Wakes" => $books->[0],
# "Caliban's War" => $books->[1],
# "The Tree-Body Problem" => $books->[2],
# "The Name of the Wind" => $books->[3],
# },
$authors->group_by(publisher_affiliation => ["with"]),
# {
# 'James A. Corey with Orbit' => $authors->[0],
# 'Cixin Liu with Head of Zeus' => $authors->[1],
# 'Patrick Rothfuss with Gollanz' => $authors->[2],
# },
$books->group_by_count("genre"),
# {
# "Sci-fi" => 3,
# "Fantasy" => 1,
# },
my $genre_books = $books->group_by_array("genre");
# {
# "Sci-fi" => [ $sf_book_1, $sf_book_2, $sf_book_3 ],
# "Fantasy" => [ $fantasy_book_1 ],
# },
#### flat
my $prolific_author_books = [ map { @{$_->books} } @$authors ]
my $prolific_author_books = $authors->map_by("books")->flat
# DESCRIPTION
Note: This module supercedes autobox::Array::Transform which was
unfortunately named.
High level autobox methods you can call on arrays, arrayrefs, hashes
and hashrefs e.g. map\_by(), grep\_by(), group\_by()
## Raison d'etre
[autobox::Core](
https://metacpan.org/pod/autobox::Core) is awesome, for a variety of reasons.
- It cuts down on dereferencing punctuation clutter.
- It makes map and grep transforms read in the same direction it's executed.
- It makes it easier to write those things in a natural order. No need
to move the cursor around a lot just to fix dereferencing, order of
operations etc.
autobox::Transform provides a few higher level methods for mapping,
greping and sorting common cases which are easier to read and write.
Since they are at a slightly higher semantic level, once you know them
they also provide a more specific meaning than just "map" or "grep".
(Compare the difference between seeing a "map" and seeing a "foreach"
loop. Just seeing the word "map" hints at what type of thing is going
on here: transforming a list into another list).
The methods of autobox::Transform are not suitable for all
cases, but when used appropriately they will lead to much more clear,
succinct and direct code, especially in conjunction with
autobox::Core.
## Examples
my $total_order_amount = $order->books
->map_by(price_with_tax => [ $tax_pct ])
->sum;
my $order_authors = $order->books
->map_by("author")
->map_by("name")->uniq->sort->join(", ");
## List and Scalar Context
All of the methods below are context sensitive, i.e. they return a
list in list context and an arrayref in scalar context, just like
autobox::Core.
When in doubt, assume they work like `map` and `grep`, and convert the
return value to references where you might have an unobvious list
context. E.g.
$self->my_method(
# Wrong, this is list context and wouldn't return an arrayref
books => $books->grep_by("is_published"),
),
$self->my_method(
# Correct, convert the list to an arrayref
books => [ $books->grep_by("is_published") ],
),
$self->my_method(
# Correct, ensure scalar context i.e. an array ref
books => scalar $books->grep_by("is_published"),
),
# AUTOBOX ARRAY METHODS
## map\_by($method, @$args?) : @array | @$array
Call the $method on each item in the list. Like:
map { $_->$method() }
Examples:
my @ahthor_names = $authors->map_by("name");
my $author_names = @publishers->map_by("authors")->map_by("name");
Optionally pass in @$args in the method call. Like:
map { $_->$method(@$args) }
Examples:
my @prices_including_tax = $books->map_by("price_with_tax", [ $tax_pct ]);
my $prices_including_tax = $books->map_by(price_with_tax => [ $tax_pct ]);
## grep\_by($method, @$args?) : @array | @$array
Call the $method on each item in the list. Like:
grep { $_->$method() }
Examples:
my @prolific_authors = $authors->grep_by("is_prolific");
Optionally pass in @$args in the method call. Like:
grep { $_->$method(@$args) }
Examples:
my @books_to_charge_for = $books->grep_by("price_with_tax", [ $tax_pct ]);
## group\_by($method, @$args = \[\], $value\_sub = object) : %key\_value | %$key\_value
Call ->$method(@$args) on each object in the array (just like ->map\_by)
and group the return values as keys in a hashref.
The default $value\_sub puts the objects in the list as the hash
values.
Example:
my $title_book = $books->group_by("title");
# {
# "Leviathan Wakes" => $books->[0],
# "Caliban's War" => $books->[1],
# "The Tree-Body Problem" => $books->[2],
# "The Name of the Wind" => $books->[3],
# },
### The $value\_sub
This is a bit tricky to use, so the most common thing would probably
be to use one of the more specific group\_by-methods which do common
things (see below). It should be capable enough to achieve what you
need though, so here's how it works:
The hash key is whatever is returned from $object->$method(@$args).
The hash value is whatever is returned from
my $new_value = $value_sub->($current_value, $object, $key);
where:
- $current value is the current hash value for this key (or undef if the first one).
- $object is the current item in the list. The current $\_ is also set to this.
- $key is the key returned by $object->$method(@$args)
## group\_by\_count($method, @$args = \[\]) : %key\_count | %$key\_count
Just like group\_by, but the hash values are the the number of
instances each $method value occurs in the list.
Example:
$books->group_by_count("genre"),
# {
# "Sci-fi" => 3,
# "Fantasy" => 1,
# },
$book->genre() returns the genre string. There are three books counted
for the "Sci-fi" key.
## group\_by\_array($method, @$args = \[\]) : %key\_objects | %$key\_objects
Just like group\_by, but the hash values are arrayrefs containing the
objects which has each $method value.
Example:
my $genre_books = $books->group_by_array("genre");
# {
# "Sci-fi" => [ $sf_book_1, $sf_book_2, $sf_book_3 ],
# "Fantasy" => [ $fantasy_book_1 ],
# },
$book->genre() returns the genre string. The three Sci-fi book objects
are collected under the Sci-fi key.
## flat() : @array | @$array
Return a flattened array, assuming the array items themselves are
array refs. I.e.
[
[ 1, 2, 3 ],
[ "a", "b" ],
]->flat
returns
[ 1, 2, 3, "a", "b "]
This is useful if e.g. a map\_by("some\_method") returns arrayrefs of
objects which you want to do further method calls on. Example:
# ->books returns an arrayref of Book objects with a ->title
$authors->map_by("books")->flat->map_by("title")
Note: This is different from autobox::Core's ->flatten, which reurns a
list rather than an array and therefore can't be used in this
way.
# DEVELOPMENT
## Author
Johan Lindstrom, `<johanl [AT] cpan.org>`
## Source code
[
https://github.com/jplindstrom/p5-autobox-Transform](
https://github.com/jplindstrom/p5-autobox-Transform)
## Bug reports
Please report any bugs or feature requests on GitHub:
[
https://github.com/jplindstrom/p5-autobox-Transform/issues](
https://github.com/jplindstrom/p5-autobox-Transform/issues).
# COPYRIGHT & LICENSE
Copyright 2016- Johan Lindstrom, All Rights Reserved.
This program is free software; you can redistribute it and/or modify it
under the same terms as Perl itself.