Catalyst-Action-Serialize-SimpleXLSX
view release on metacpan or search on metacpan
lib/Catalyst/Action/Serialize/SimpleXLSX.pm view on Meta::CPAN
$c,
entity => $entity
);
}
In your jQuery webpage, to initiate a file download:
<script>
$(document).ready(function () {
function export_to_excel() {
$('<iframe ' + 'src="/item?content-type=application/vnd.openxmlformats-officedocument.spreadsheetml.sheet">').hide().appendTo('body');
}
$("#books").on("click", export_to_excel);
});
</script>
Note, the content-type query param is required if you're just linking to the
action. It tells L<Catalyst::Controller::REST> what you're serializing the data
as.
=head1 DESCRIPTION
Your entity should be either:
=over 4
=item * an array of arrays
=item * an array of arrays of arrays
=item * a hash with the keys as described below and in the L</SYNOPSIS>
=back
If entity is a hashref, keys should be:
=head2 sheets
An array of worksheets. Either sheets or a worksheet specification at the top
level is required.
=head2 filename
Optional. The name of the file before .xlsx. Defaults to "data".
Each sheet should be an array of arrays, or a hashref with the following fields:
=head2 name
Optional. The name of the worksheet.
=head2 rows
Required. The array of arrays of rows.
=head2 header
Optional, an array for the first line of the sheet, which will be in bold.
=head2 column_widths
Optional, the widths in characters of the columns. Otherwise the widths are
calculated automatically from the data and header.
If you only have one sheet, you can put it in the top level hash.
=cut
has 'content_type' => (
is => 'ro',
required => 1,
default => sub { return 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' },
);
sub execute {
my $self = shift;
my ( $controller, $c ) = @_;
my $stash_key = (
$controller->config->{'serialize'}
? $controller->config->{'serialize'}->{'stash_key'}
: $controller->config->{'stash_key'}
)
|| 'rest';
my $data = $c->stash->{$stash_key};
open my $fh, '>', \my $buf;
my $workbook = Excel::Writer::XLSX->new($fh);
my ( $filename, $sheets ) = $self->_parse_entity($data);
for my $sheet (@$sheets) {
$self->_add_sheet( $workbook, $sheet );
}
$workbook->close;
$self->_write_file( $c, $filename, $buf );
return 1;
}
sub _write_file {
my ( $self, $c, $filename, $data ) = @_;
$c->res->content_type( $self->content_type );
$c->res->header(
'Content-Disposition' => "attachment; filename=${filename}.xlsx" );
$c->res->output($data);
}
sub _parse_entity {
my ( $self, $data ) = @_;
my @sheets;
my $filename = 'data';
if ( ref $data eq 'ARRAY' ) {
if ( not ref $data->[0][0] ) {
$sheets[0] = { rows => $data };
}
else {
@sheets =
map ref $_ eq 'HASH' ? $_
: ref $_ eq 'ARRAY' ? { rows => $_ }
: Catalyst::Exception->throw(
'Unsupported sheet reference type: ' . ref($_) ), @{$data};
}
}
elsif ( ref $data eq 'HASH' ) {
$filename = $data->{filename} if $data->{filename};
my $sheets = $data->{sheets};
my $rows = $data->{rows};
if ( $sheets && $rows ) {
Catalyst::Exception->throw('Use either sheets or rows, not both.');
}
if ($sheets) {
@sheets =
map ref $_ eq 'HASH' ? $_
: ref $_ eq 'ARRAY' ? { rows => $_ }
: Catalyst::Exception->throw(
'Unsupported sheet reference type: ' . ref($_) ), @{$sheets};
}
elsif ($rows) {
$sheets[0] = $data;
}
else {
Catalyst::Exception->throw('Must supply either sheets or rows.');
}
}
else {
Catalyst::Exception->throw(
'Unsupported workbook reference type: ' . ref($data) );
}
return ( $filename, \@sheets );
}
sub _add_sheet {
my ( $self, $workbook, $sheet ) = @_;
my $worksheet = $workbook->add_worksheet( $sheet->{name} ? $sheet->{name} : () );
$worksheet->keep_leading_zeros(1);
my ( $row, $col ) = ( 0, 0 );
my @auto_widths;
# Write Header
if ( exists $sheet->{header} ) {
my $header_format = $workbook->add_format;
$header_format->set_bold;
for my $header ( @{ $sheet->{header} } ) {
if (defined $auto_widths[$col] && $auto_widths[$col] < length $header) {
$auto_widths[$col] = length $header;
}
$worksheet->write( $row, $col++, $header, $header_format );
}
$row++;
$col = 0;
}
# Write data
for my $the_row ( @{ $sheet->{rows} } ) {
for my $the_col (@$the_row) {
if (defined $auto_widths[$col] && $auto_widths[$col] < length $the_col) {
$auto_widths[$col] = length $the_col;
}
$worksheet->write( $row, $col++, $the_col );
}
$row++;
$col = 0;
}
# Set column widths
$sheet->{column_widths} = \@auto_widths unless exists $sheet->{column_widths};
for my $width ( @{ $sheet->{column_widths} } ) {
$worksheet->set_column( $col, $col++, $width );
}
$worksheet->set_column( 0, 0, $sheet->{column_widths}[0] );
return $worksheet;
}
=head1 AUTHOR
Mike Baas <mbaas at cpan.org>
=head1 ORIGINAL AUTHOR
Rafael Kitover <rkitover at cpan.org>
=head1 ACKNOWLEDGEMENTS
This module is really nothing more than a tweak to L<Catalyst::Action::Serialize::SimpleExcel> that drops in L<Excel::Writer::XLSX> for compatibility with Excel 2007 and later. I just needed more rows!
=head1 SEE ALSO
L<Catalyst>, L<Catalyst::Controller::REST>, L<Catalyst::Action::REST>, L<Catalyst::Action::Serialize::SimpleExcel>, L<Excel::Writer::XLSX>
=head1 REPOSITORY
L<https://github.com/initself/Catalyst-Action-Serialize-SimpleXLSX>
=head1 COPYRIGHT & LICENSE
This is free software; you can redistribute it and/or modify it under the same terms as the Perl 5 programming language system itself.
=cut
1; # End of Catalyst::Action::Serialize::SimpleXLSX
( run in 1.897 second using v1.01-cache-2.11-cpan-39bf76dae61 )