5 # Recursively create image gallery index and slideshow wrappings.
6 # Makes use of (slightly modified) "lightbox" Javascript/CSS as published
7 # at http://www.huddletogether.com/projects/lightbox/
9 # Copyright (c) 2006 Eugene G. Crosser
11 # This software is provided 'as-is', without any express or implied
12 # warranty. In no event will the authors be held liable for any damages
13 # arising from the use of this software.
15 # Permission is granted to anyone to use this software for any purpose,
16 # including commercial applications, and to alter it and redistribute it
17 # freely, subject to the following restrictions:
19 # 1. The origin of this software must not be misrepresented; you must not
20 # claim that you wrote the original software. If you use this software
21 # in a product, an acknowledgment in the product documentation would be
22 # appreciated but is not required.
23 # 2. Altered source versions must be plainly marked as such, and must not be
24 # misrepresented as being the original software.
25 # 3. This notice may not be removed or altered from any source distribution.
31 use POSIX qw/getcwd strftime/;
32 use CGI qw/:html *table *Tr *center *div/;
33 use Image::Info qw/image_info dim/;
38 binmode(STDOUT, ":utf8");
40 my $haveimagick = eval { require Image::Magick; };
41 { package Image::Magick; } # to make perl compiler happy
43 my @sizes = (160, 640);
45 ######################################################################
54 GetOptions( 'incpath'=>\$incpath,
55 'asktitle'=>\$asktitle,
56 'noasktitle'=>\$noasktitle,
59 my $term = new Term::ReadLine "Edit Title";
61 FsObj->new(getcwd)->iterate;
71 my $fullpath = $parent->{-fullpath}.'/'.$name;
74 -root=>$parent->{-root},
77 -inc=>'../'.$parent->{-inc},
90 print "new $class:\n";
91 foreach my $k(keys %$self) {
92 print "\t$k\t=\t$self->{$k}\n";
99 my $fullpath=shift; # this is not a method
100 my $depth=20; # arbitrary max depth
103 return $incpath."/.include";
107 while ( ! -d $fullpath."/".$inc ) {
109 last unless ($depth-- > 0);
112 return $inc.'/'; # prefix with trailing slash
114 return 'NO-.INCLUDE-IN-PATH/'; # won't work anyway
120 my $fullpath .= $self->{-fullpath};
121 print "iterate in dir $fullpath\n" if ($debug);
127 unless (opendir($D,$fullpath)) {
128 warn "cannot opendir $fullpath: $!";
131 while (my $de = readdir($D)) {
132 next if ($de =~ /^\./);
133 my $child = $self->new($de);
134 my @stat = stat($child->{-fullpath});
135 $youngest = $stat[9] if ($youngest < $stat[9]);
137 push(@rdirlist,$child);
138 } elsif ($child->isimg) {
139 push(@rimglist,$child);
143 my @dirlist = sort {$a->{-base} cmp $b->{-base}} @rdirlist;
144 undef @rdirlist; # inplace sorting would be handy here
145 my @imglist = sort {$a->{-base} cmp $b->{-base}} @rimglist;
146 undef @rimglist; # optimize away unsorted versions
147 $self->{-firstimg} = $imglist[0];
149 print "Dir: $self->{-fullpath}\n" if ($debug);
151 # 1. first of all, fill title for this directory and create hidden subdirs
155 # 2. recurse into subdirectories to get their titles filled
156 # before we start writing out subalbum list
158 foreach my $dir(@dirlist) {
162 # 3. iterate through images to build cross-links,
165 foreach my $img(@imglist) {
166 # list-linking must be done before generating
167 # aux html because aux pages rely on prev/next refs
169 $previmg->{-nextimg} = $img;
170 $img->{-previmg} = $previmg;
175 # 4. create scaled versions and aux html pages
177 foreach my $img(@imglist) {
178 # scaled versions must be generated before aux html
179 # and main image index because they both rely on
180 # refs to scaled images and they may be just original
181 # images, this is not known before we try scaling.
183 # finally, make aux html pages
187 # no need to go beyond this point if the directory timestamp did not
188 # change since we built index.html file last time.
190 my @istat = stat($self->{-fullpath}.'/index.html');
191 return unless ($youngest > $istat[9]);
193 # 5. start building index.html for the directory
197 # 6. iterate through subdirectories to build subalbums list
201 foreach my $dir(@dirlist) {
207 # 7. iterate through images to build thumb list
211 foreach my $img(@imglist) {
212 print "Img: $img->{-fullpath}\n" if ($debug);
218 # 8. comlplete building index.html for the directory
225 return ( -d $self->{-fullpath} );
230 my $fullpath = $self->{-fullpath};
231 return 0 unless ( -f $fullpath );
232 my $info = image_info($fullpath);
233 if (my $error = $info->{error}) {
234 if (($error !~ "Unrecognized file format") &&
235 ($error !~ "Can't read head")) {
236 warn "File \"$fullpath\": $error\n";
241 tryapp12($info) unless ($info->{'ExifVersion'});
244 $self->{-info} = $info;
249 my $info = shift; # this is not a method
251 # dirty hack to take care of Image::Info parser strangeness
252 foreach my $k(keys %$info) {
253 $app12=substr($k,6).$info->{$k} if ($k =~ /^App12-/);
255 return unless ($app12); # bad luck
257 foreach my $ln(split /[\r\n]+/,$app12) {
258 $ln =~ s/[[:^print:]\000]/ /g;
259 unless ($seenfirstline) {
264 my ($k,$v)=split /=/,$ln,2;
265 if ($k eq 'TimeDate') {
266 $info->{'DateTime'} =
267 strftime("%Y:%m:%d %H:%M:%S", localtime($v))
269 } elsif ($k eq 'Shutter') {
270 $info->{'ExposureTime'} = '1/'.int(1000000/$v+.5);
271 } elsif ($k eq 'Flash') {
272 $info->{'Flash'} = $v?'Flash fired':'Flash did not fire';
273 } elsif ($k eq 'Type') {
274 $info->{'Model'} = $v;
275 } elsif ($k eq 'Version') {
276 $info->{'Software'} = $v;
277 } elsif ($k eq 'Fnumber') {
278 $info->{'FNumber'} = $v;
285 my $fullpath = $self->{-fullpath};
286 for my $subdir(@sizes, 'html') {
287 my $tdir=sprintf "%s/.%s",$self->{-fullpath},$subdir;
288 mkdir($tdir,0755) unless ( -d $tdir );
295 my $fullpath = $self->{-fullpath};
298 if (open($T,'<'.$fullpath.'/.title')) {
300 $title =~ s/[\r\n]*$//;
303 if ($asktitle || (!$title && !$noasktitle)) {
304 my $prompt = $self->{-base};
305 $prompt = '/' unless ($prompt);
306 my $OUT = $term->OUT || \*STDOUT;
307 print $OUT "Enter title for $fullpath\n";
308 $title = $term->readline($prompt.' >',$title);
309 $term->addhistory($title) if ($title);
310 if (open($T,'>'.$fullpath.'/.title')) {
311 print $T $title,"\n";
316 $title=substr($fullpath,length($self->{-root}));
318 $self->{-title}=$title;
319 print "title in $fullpath is $title\n" if ($debug);
324 my $fn = $self->{-fullpath};
325 my $name = $self->{-base};
326 my $dn = $self->{-parent}->{-fullpath};
327 my ($w, $h) = dim($self->{-info});
328 my $max = ($w > $h)?$w:$h;
330 foreach my $size(@sizes) {
331 my $nref = '.'.$size.'/'.$name;
332 my $nfn = $dn.'/'.$nref;
333 my $factor=$size/$max;
335 $self->{$size} = $name; # unscaled version will do
337 $self->{$size} = $nref;
338 if (isnewer($fn,$nfn)) {
339 doscaling($fn,$nfn,$factor,$w,$h);
346 my ($fn1,$fn2) = @_; # this is not a method
347 my @stat1=stat($fn1);
348 my @stat2=stat($fn2);
349 return (!@stat2 || ($stat1[9] > $stat2[9]));
350 # true if $fn2 is absent or is older than $fn1
354 my ($src,$dest,$factor,$w,$h) = @_; # this is not a method
358 my $im = new Image::Magick;
359 print "doscaling $src -> $dest by $factor\n" if ($debug);
360 if ($err = $im->Read($src)) {
361 warn "ImageMagick: read \"$src\": $err";
363 $im->Scale(width=>$w*$factor,height=>$h*$factor);
364 $err=$im->Write($dest);
365 warn "ImageMagick: write \"$dest\": $err" if ($err);
369 if ($err) { # fallback to command-line tools
370 system("djpeg \"$src\" | pnmscale \"$factor\" | cjpeg >\"$dest\"");
376 my $name = $self->{-base};
377 my $dn = $self->{-parent}->{-fullpath};
378 my $pref = $self->{-previmg}->{-base};
379 my $nref = $self->{-nextimg}->{-base};
380 my $inc = $self->{-inc};
381 my $title = $self->{-info}->{'Comment'};
382 $title = $name unless ($title);
384 print "slide: \"$title\": \"$pref\"->\"$name\"->\"$nref\"\n" if ($debug);
387 for my $refresh('static', 'slide') {
388 my $fn = sprintf("%s/.html/%s-%s.html",$dn,$name,$refresh);
389 if (isnewer($self->{-fullpath},$fn)) {
390 my $imgsrc = '../'.$self->{$sizes[1]};
394 $fwdref = sprintf("%s-%s.html",$nref,$refresh);
396 $fwdref = '../index.html';
399 $bakref = sprintf("%s-%s.html",$pref,$refresh);
401 $bakref = '../index.html';
405 if ($refresh eq 'slide') {
406 $toggleref=sprintf("%s-static.html",$name);
407 $toggletext = 'Stop!';
409 $toggleref=sprintf("%s-slide.html",$name);
410 $toggletext = 'Play->';
413 unless (open($F,'>'.$fn)) {
414 warn "cannot open \"$fn\": $!";
417 binmode($F, ":utf8");
418 if ($refresh eq 'slide') {
423 -head=>meta({-http_equiv=>'Refresh',
424 -content=>"3; url=$fwdref"}),
425 -style=>{-src=>$inc."gallery.css"},
429 print $F start_html(-title=>$title,
432 -style=>{-src=>$inc."gallery.css"},
435 print $F start_center,"\n",
437 start_table({-class=>'navi'}),start_Tr,"\n",
438 td(a({-href=>"../index.html"},"Index")),"\n",
439 td(a({-href=>$bakref},"<<Prev")),"\n",
440 td(a({-href=>$toggleref},$toggletext)),"\n",
441 td(a({-href=>$fwdref},"Next>>")),"\n",
444 table({-class=>'picframe'},
445 Tr(td(img({-src=>$imgsrc})))),"\n",
453 my $fn = sprintf("%s/.html/%s-info.html",$dn,$name);
454 if (isnewer($self->{-fullpath},$fn)) {
456 unless (open($F,'>'.$fn)) {
457 warn "cannot open \"$fn\": $!";
460 my $imgsrc = sprintf("../.%s/%s",$sizes[0],$name);
461 print $F start_html(-title=>$title,
463 -style=>{-src=>$inc."gallery.css"},),"\n",
466 table({-class=>'ipage'},
467 Tr(td(img({-src=>$imgsrc})),
468 td($self->infotable))),
469 a({-href=>'../index.html'},'Index'),"\n",
478 my $fn = $self->{-fullpath}.'/index.html';
479 my $block = $self->{-fullpath}.'/.noindex';
480 $fn = '/dev/null' if ( -f $block );
482 unless (open($IND,'>'.$fn)) {
483 warn "cannot open $fn: $!";
486 binmode($IND, ":utf8");
487 $self->{-IND} = $IND;
489 my $inc = $self->{-inc};
490 my $title = $self->{-title};
491 print $IND start_html(-title => $title,
493 -style=>{-src=>[$inc."gallery.css",
494 $inc."lightbox.css"]},
495 -script=>[{-code=>"var incPrefix='$inc';"},
496 {-src=>$inc."gallery.js"},
497 {-src=>$inc."lightbox.js"}]),
498 a({-href=>"../index.html"},"UP"),"\n",
506 my $IND = $self->{-IND};
508 print $IND end_center,end_html,"\n";
510 close($IND) if ($IND);
516 my $IND = $self->{-IND};
518 print $IND h2("Albums"),"\n",start_table,"\n";
523 my $IND = $self->{-parent}->{-IND};
524 my $name = $self->{-base};
525 my $title = $self->{-title};
527 print $IND Tr(td(a({-href=>$name.'/index.html'},$name)),
528 td(a({-href=>$name.'/index.html'},$title))),"\n";
533 my $IND = $self->{-IND};
535 print $IND end_table,"\n",br({-clear=>'all'}),hr,"\n\n";
540 my $IND = $self->{-IND};
541 my $first = $self->{-firstimg}->{-base};
542 my $slideref = sprintf(".html/%s-slide.html",$first);
544 print $IND h2("Images"),"\n",
545 a({-href=>$slideref},'Slideshow'),
551 my $IND = $self->{-parent}->{-IND};
552 my $name = $self->{-base};
553 my $title = $self->{-info}->{'Comment'};
554 $title = $name unless ($title);
555 my $thumb = $self->{$sizes[0]};
556 my $medium = $self->{$sizes[1]};
557 my $info = $self->{-info};
558 my ($w, $h) = dim($info);
560 print $IND start_div({-class=>'ibox',-id=>$name,
561 -OnClick=>"HideIbox('$name');"}),"\n",
562 start_div({-class=>'iboxtitle'}),
563 span({-style=>'float: left;'},b("Info for $name")),
564 span({-style=>'float: right;'},
565 a({-href=>"#",-OnClick=>"HideIbox('$name');"},"Close")),
566 br({-clear=>'all'}),"\n",
571 print $IND table({-class=>'slide'},Tr(td(
572 a({-href=>".html/$name-info.html",-title=>'Image Info',
573 -onClick=>"return showIbox('$name');"},$title),
575 a({-href=>$medium,-rel=>"lightbox",-title=>$title},
576 img({-src=>$thumb})),
578 a({-href=>$name,-title=>'Original Image'},"($w x $h)"),
584 my $IND = $self->{-IND};
586 print $IND br({-clear=>'all'}),hr,"\n\n";
591 my $info = $self->{-info};
608 $msg.=start_table({-class=>'infotable'})."\n";
609 foreach my $k(@infokeys) {
610 $msg.=Tr(td($k.":"),td($info->{$k}))."\n" if ($info->{$k});
612 $msg.=end_table."\n";