#=======================================================================
# ____ ____ _____ _ ____ ___ ____
# | _ \| _ \| ___| _ _ / \ | _ \_ _| |___ \
# | |_) | | | | |_ (_) (_) / _ \ | |_) | | __) |
# | __/| |_| | _| _ _ / ___ \| __/| | / __/
# |_| |____/|_| (_) (_) /_/ \_\_| |___| |_____|
#
# A Perl Module Chain to faciliate the Creation and Modification
# of High-Quality "Portable Document Format (PDF)" Files.
#
# Copyright 1999-2005 Alfred Reibenschuh <areibens@cpan.org>.
#
#=======================================================================
#
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2 of the License, or (at your option) any later version.
#
# This library is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this library; if not, write to the
# Free Software Foundation, Inc., 59 Temple Place - Suite 330,
# Boston, MA 02111-1307, USA.
#
# $Id: GIF.pm,v 2.0 2005/11/16 02:18:23 areibens Exp $
#
#=======================================================================
package PDF::API2::Resource::XObject::Image::GIF;
BEGIN {
use PDF::API2::Util;
use PDF::API2::Basic::PDF::Utils;
use PDF::API2::Resource::XObject::Image;
use POSIX;
use vars qw(@ISA $VERSION);
@ISA = qw( PDF::API2::Resource::XObject::Image );
( $VERSION ) = sprintf '%i.%03i', split(/\./,('$Revision: 2.0 $' =~ /Revision: (\S+)\s/)[0]); # $Date: 2005/11/16 02:18:23 $
}
no warnings qw[ deprecated recursion uninitialized ];
=item $res = PDF::API2::Resource::XObject::Image::GIF->new $pdf, $file [, $name]
Returns a gif-image object.
=cut
# added from PDF::Create:
# PDF::Image::GIFImage - GIF image support
# Author: Michael Gross <mdgrosse@sbox.tugraz.at>
# modified for internal use. (c) 2004 fredo.
sub unInterlace {
my $self = shift;
my $data = $self->{' stream'};
my $row;
my @result;
my $width = $self->width;
my $height = $self->height;
my $idx = 0;
#Pass 1 - every 8th row, starting with row 0
$row = 0;
while ($row < $height) {
$result[$row] = substr($data, $idx*$width, $width);
$row+=8;
$idx++;
}
#Pass 2 - every 8th row, starting with row 4
$row = 4;
while ($row < $height) {
$result[$row] = substr($data, $idx*$width, $width);
$row+=8;
$idx++;
}
#Pass 3 - every 4th row, starting with row 2
$row = 2;
while ($row < $height) {
$result[$row] = substr($data, $idx*$width, $width);
$row+=4;
$idx++;
}
#Pass 4 - every 2th row, starting with row 1
$row = 1;
while ($row < $height) {
$result[$row] = substr($data, $idx*$width, $width);
$row+=2;
$idx++;
}
$self->{' stream'}=join('', @result);
}
sub deGIF {
my ($ibits,$stream)=@_;
my $bits=$ibits;
my $resetcode=1<<($ibits-1);
my $endcode=$resetcode+1;
my $nextcode=$endcode+1;
my $ptr=0;
my $maxptr=8*length($stream);
my $tag;
my $out='';
my $outptr=0;
# print STDERR "reset=$resetcode\nend=$endcode\nmax=$maxptr\n";
my @d=map { chr($_) } (0..$resetcode-1);
while(($ptr+$bits)<=$maxptr) {
$tag=0;
foreach my $off (reverse 0..$bits-1) {
$tag<<=1;
$tag|=vec($stream,$ptr+$off,1);
}
# foreach my $off (0..$bits-1) {
# $tag<<=1;
# $tag|=vec($stream,$ptr+$off,1);
# }
# print STDERR "ptr=$ptr,tag=$tag,bits=$bits,next=$nextcode\n";
# print STDERR "tag to large\n" if($tag>$nextcode);
$ptr+=$bits;
$bits++ if($nextcode == (1<<$bits));
if($tag==$resetcode) {
$bits=$ibits;
$nextcode=$endcode+1;
next;
} elsif($tag==$endcode) {
last;
} elsif($tag<$resetcode) {
$d[$nextcode]=$d[$tag];
$out.=$d[$nextcode];
$nextcode++;
} elsif($tag>$endcode) {
$d[$nextcode]=$d[$tag];
$d[$nextcode].=substr($d[$tag+1],0,1);
$out.=$d[$nextcode];
$nextcode++;
}
}
return($out);
}
sub new {
my ($class,$pdf,$file,$name,%opts) = @_;
my $self;
my $inter=0;
$class = ref $class if ref $class;
$self=$class->SUPER::new($pdf,$name || 'Gx'.pdfkey());
$pdf->new_obj($self) unless($self->is_obj($pdf));
$self->{' apipdf'}=$pdf;
my $fh = IO::File->new;
open($fh,$file);
binmode($fh,':raw');
my $buf;
$fh->read($buf,6); # signature
die "unknown image signature '$buf' -- not a gif." unless($buf=~/^GIF[0-9][0-9][a-b]/);
$fh->read($buf,7); # logical descr.
my($wg,$hg,$flags,$bgColorIndex,$aspect)=unpack('vvCCC',$buf);
if($flags&0x80) {
my $colSize=2**(($flags&0x7)+1);
my $dict=PDFDict();
$pdf->new_obj($dict);
$self->colorspace(PDFArray(PDFName('Indexed'),PDFName('DeviceRGB'),PDFNum($colSize-1),$dict));
$fh->read($dict->{' stream'},3*$colSize); # color-table
}
while(!$fh->eof) {
$fh->read($buf,1); # tag.
my $sep=unpack('C',$buf);
if($sep==0x2C){
$fh->read($buf,9); # image-descr.
my ($left,$top,$w,$h,$flags)=unpack('vvvvC',$buf);
$self->width($w||$wg);
$self->height($h||$hg);
$self->bpc(8);
if($flags&0x80) { # local colormap
my $colSize=2**(($flags&0x7)+1);
my $dict=PDFDict();
$pdf->new_obj($dict);
$self->colorspace(PDFArray(PDFName('Indexed'),PDFName('DeviceRGB'),PDFNum($colSize-1),$dict));
$fh->read($dict->{' stream'},3*$colSize); # color-table
}
if($flags&0x40) { # need de-interlace
$inter=1;
}
$fh->read($buf,1); # image-lzw-start (should be 9).
my ($sep)=unpack('C',$buf);
$fh->read($buf,1); # first chunk.
my ($len)=unpack('C',$buf);
my $stream='';
while($len>0) {
$fh->read($buf,$len);
$stream.=$buf;
$fh->read($buf,1);
$len=unpack('C',$buf);
}
$self->{' stream'}=deGIF($sep+1,$stream);
$self->unInterlace if($inter);
last;
} elsif($sep==0x3b) {
last;
} elsif($sep==0x21) {
# Graphic Control Extension
$fh->read($buf,1); # tag.
my $tag=unpack('C',$buf);
die "unsupported graphic control extension ($tag)" unless($tag==0xF9);
$fh->read($buf,1); # len.
my $len=unpack('C',$buf);
my $stream='';
while($len>0) {
$fh->read($buf,$len);
$stream.=$buf;
$fh->read($buf,1);
$len=unpack('C',$buf);
}
my ($cFlags,$delay,$transIndex)=unpack('CvC',$stream);
if(($cFlags&0x01) && !$opts{-notrans}) {
$self->{Mask}=PDFArray(PDFNum($transIndex),PDFNum($transIndex));
}
} else {
# extension
$fh->read($buf,1); # tag.
my $tag=unpack('C',$buf);
$fh->read($buf,1); # tag.
my $len=unpack('C',$buf);
while($len>0) {
$fh->read($buf,$len);
$fh->read($buf,1);
$len=unpack('C',$buf);
}
}
}
$fh->close;
$self->filters('FlateDecode');
return($self);
}
=item $res = PDF::API2::Resource::XObject::Image::GIF->new_api $api, $file [, $name]
Returns a gif-image object. This method is different from 'new' that
it needs an PDF::API2-object rather than a Text::PDF::File-object.
=cut
sub new_api {
my ($class,$api,@opts)=@_;
my $obj=$class->new($api->{pdf},@opts);
$obj->{' api'}=$api;
return($obj);
}
1;
__END__
=head1 AUTHOR
alfred reibenschuh
=head1 HISTORY
$Log: GIF.pm,v $
Revision 2.0 2005/11/16 02:18:23 areibens
revision workaround for SF cvs import not to screw up CPAN
Revision 1.2 2005/11/16 01:27:50 areibens
genesis2
Revision 1.1 2005/11/16 01:19:27 areibens
genesis
Revision 1.11 2005/06/17 19:44:04 fredo
fixed CPAN modulefile versioning (again)
Revision 1.10 2005/06/17 18:53:34 fredo
fixed CPAN modulefile versioning (dislikes cvs)
Revision 1.9 2005/03/14 22:01:31 fredo
upd 2005
Revision 1.8 2004/12/16 00:30:55 fredo
added no warn for recursion
Revision 1.7 2004/06/15 09:14:54 fredo
removed cr+lf
Revision 1.6 2004/06/07 19:44:44 fredo
cleaned out cr+lf for lf
Revision 1.5 2004/03/20 08:37:43 fredo
added de-interlacing
Revision 1.4 2004/02/12 14:48:41 fredo
removed duplicate definition of $fh
Revision 1.3 2003/12/08 13:06:11 Administrator
corrected to proper licencing statement
Revision 1.2 2003/11/30 17:37:16 Administrator
merged into default
Revision 1.1.1.1.2.2 2003/11/30 16:57:09 Administrator
merged into default
Revision 1.1.1.1.2.1 2003/11/30 16:00:42 Administrator
added CVS id/log
=cut