X-Git-Url: http://www.average.org/gitweb/?p=mkgallery.git;a=blobdiff_plain;f=mkgallery.pl;h=5ba15568d09e77017faf41f784ea350c69b46389;hp=b5a82c8d1871ecc602f517eeeb9432df7ea67860;hb=b9326cf9beda29c785ceabd30fc3ae5d219c743f;hpb=76d48c06a42b2107f0f396b806599d184ae4682a
diff --git a/mkgallery.pl b/mkgallery.pl
index b5a82c8..5ba1556 100755
--- a/mkgallery.pl
+++ b/mkgallery.pl
@@ -29,30 +29,36 @@ package FsObj;
use strict;
use Carp;
use POSIX qw/getcwd strftime/;
+use HTTP::Date;
use CGI qw/:html *table *Tr *td *center *div *Link/;
use Image::Info qw/image_info dim/;
use Term::ReadLine;
use Getopt::Long;
use Encode;
-use encoding 'utf-8';
+use UUID;
+#use encoding 'utf-8';
binmode(STDOUT, ":utf8");
my $haveimagick = eval { require Image::Magick; };
{ package Image::Magick; } # to make perl compiler happy
-my $haverssxml = eval { require XML::RSS; };
-{ package XML::RSS; } # to make perl compiler happy
+my $havefeed = eval { require XML::FeedPP; };
+{ package XML::FeedPP; } # to make perl compiler happy
+
+my $havegeoloc = eval { require Image::ExifTool::Location; };
+{ package Image::ExifTool::Location; } # to make perl compiler happy
my @sizes = (160, 640, 1600);
+my $incdir = ".gallery2";
######################################################################
my $incpath;
-my $rssobj;
+my $feedobj;
my $debug = 0;
my $asktitle = 0;
my $noasktitle = 0;
-my $rssfile = "";
+my $feed = "";
charset("utf-8");
@@ -61,33 +67,40 @@ unless (GetOptions(
'incpath'=>\$incpath,
'asktitle'=>\$asktitle,
'noasktitle'=>\$noasktitle,
- 'rssfile=s'=>\$rssfile,
+ 'feed=s'=>\$feed,
'debug'=>\$debug)) {
&help;
}
-if ($rssfile && ! $haverssxml) {
- print STDERR "You need to get XML::RSS from CPAN to use --rssfile\n";
+if ($feed && !$havefeed) {
+ print STDERR "You need to install XML::FeedPP to use --feed\n";
exit 1;
}
my $term = new Term::ReadLine "Edit Title";
FsObj->new(getcwd)->iterate;
-if ($rssobj) { $rssobj->{'rss'}->save($rssobj->{'file'}); }
+if ($feedobj) {
+ $feedobj->{-feed}->pubDate(time);
+ $feedobj->{-feed}->to_file($feedobj->{-savepath});
+}
sub help {
print STDERR <<__END__;
usage: $0 [options]
--help: print help message and exit
- --incpath: do not try to find .gallery2 diretory upstream, use
+ --incpath: do not try to find .gallery2 directory upstream, use
specified path (absolute or relavive). Use with causion.
--debug: print a lot of debugging info to stdout as you run
--asktitle: ask to edit album titles even if there are ".title" files
--noasktitle: don't ask to enter album titles even where ".title"
files are absent. Use partial directory names as titles.
- --rssfile=...: build RSS feed for newly added "albums", give name of rss file
+ --feed=...: build Atom feed for newly added "albums",
+ enter filename, base URL, and optionally PuSH hub url,
+ separated by commas. (Note: PuSH obviously does not work
+ "out of the box" for static tree! You need a separate
+ "watcher" script to do the publishing for you.)
__END__
exit 1;
@@ -101,14 +114,15 @@ sub new {
$class = ref($this);
my $parent = $this;
my $name = shift;
- my $fullpath = $parent->{-fullpath}.'/'.$name;
$self = {
-parent=>$parent,
-root=>$parent->{-root},
+ -toppath=>$parent->{-toppath},
+ -depth=>$parent->{-depth}+1,
-base=>$name,
- -fullpath=>$fullpath,
+ -fullpath=>$parent->{-fullpath}.'/'.$name,
+ -relpath=>$parent->{-relpath}.$name.'/',
-inc=>'../'.$parent->{-inc},
- -rss=>'../'.$parent->{-rss},
};
} else {
$class = $this;
@@ -116,9 +130,9 @@ sub new {
$self = {
-root=>$root,
-fullpath=>$root,
- -inc=>getinc($root),
- -rss=>getrss($root),
};
+ # fill in -inc, -feed, -relpath
+ initpaths($self); # we are not blessed yet, so cheat.
}
bless $self, $class;
if ($debug) {
@@ -130,52 +144,104 @@ sub new {
return $self;
}
-sub getinc {
- my $fullpath=shift; # this is not a method
+sub initpaths {
+ my $self=shift; # this is not a method but we cheat
my $depth=20; # arbitrary max depth
+ my $fullpath=$self->{-fullpath};
+ my $inc;
+ my $relpath;
if ($incpath) {
- return $incpath."/.gallery2";
- }
-
- my $inc=".gallery2";
- while ( ! -d $fullpath."/".$inc ) {
- $inc = "../".$inc;
- last unless ($depth-- > 0);
+ $inc = $incpath;
+ $inc .= '/' unless ($inc =~ m%/$%);
+ } else {
+ $inc="";
+ while ( ! -d $fullpath."/".$inc."/".$incdir ) {
+ $inc = "../".$inc;
+ last unless ($depth-- > 0);
+ }
}
if ($depth > 0) {
- return $inc.'/'; # prefix with trailing slash
+ $self->{-inc} = $inc;
+ my $dp=0;
+ my $pos;
+ for ($pos=index($inc,'/');$pos>=0;
+ $pos=index($inc,'/',$pos+1)) {
+ $dp++;
+ }
+ $self->{-depth} = $dp;
+ for ($pos=length($fullpath);$dp>0 && $pos>0;
+ $pos=rindex($fullpath,'/',$pos-1)) {
+ $dp--;
+ }
+ my $relpath = substr($fullpath,$pos);
+ $relpath =~ s%^/%%;
+ $relpath .= '/' if ($relpath);
+ $self->{-relpath} = $relpath;
+ $self->{-toppath} = substr($fullpath,0,$pos);
+ #print "rel=$relpath, top=$self->{-toppath}, inc=$inc\n";
+ initfeed($self);
} else {
- return 'NO-.INCLUDE-IN-PATH/'; # won't work anyway
+ $self->{-inc} = 'NO-.INCLUDE-IN-PATH/'; # won't work anyway
+ $self->{-feed} = '';
+ $self->{-relpath} = '';
+ $self->{-depth} = 0;
}
}
-sub getrss {
- my $fullpath=shift; # this is not a method
- my $depth=20; # arbitrary max depth
+sub initfeed {
+ my $self=shift; # this is not a method but we cheat
+ my $fullpath=$self->{-fullpath};
+ my $toppath=$self->{-toppath};
+ my $inc=$self->{-inc}.$incdir.'/';
+ my $conffile=$toppath.'/'.$incdir.'/feed.conf';
+ my $CONF;
+
+ if (! $incpath) {
+ if ($feed) {
+ if (open($CONF,">".$conffile)) {
+ print $CONF $feed,"\n";
+ close($CONF);
+ } else {
+ print STDERR "could not open $conffile: $!\n";
+ }
+ } else {
+ if (open($CONF,$conffile)) {
+ $feed=<$CONF>;
+ close($CONF);
+ chop $feed;
+ }
+ }
+ }
- return "" unless $rssfile;
+ return unless ($feed);
- my $rss=$rssfile;
- while ( ! -f $fullpath."/".$rss ) {
- $rss = "../".$rss;
- last unless ($depth-- > 0);
- }
- if ($depth > 0) {
- $rssobj->{'file'} = $rss;
- $rssobj->{'rss'} = new XML::RSS (version=>2);
- $rssobj->{'rss'}->parsefile($rss);
- my $itemstodel = @{$rssobj->{'rss'}->{'items'}} - 15;
- while ($itemstodel-- > 0) {
- pop(@{$rssobj->{'rss'}->{'items'}})
- }
- $rssobj->{'rss'}->save($rssobj->{'file'});
- return $rss;
+ my ($feedfile, $feedbase, $feedhub) = split(',', $feed);
+ $feedbase .= '/' unless ($feedbase =~ /\/$/);
+ print "($feedfile, $feedbase, $feedhub)\n";
+
+ $feedobj->{-savepath} = $self->{-toppath}.'/'.$feedfile;
+ $feedobj->{-file} = $feedfile;
+ $feedobj->{-base} = $feedbase;
+ $feedobj->{-hub} = $feedhub;
+ if ( -f $feedobj->{-file} ) {
+ $feedobj->{-feed} = XML::FeedPP::Atom::Atom10->new(
+ $feedobj->{-file});
+ $feedobj->{-feed}->limit_item(15);
} else {
- print STDERR "There is no $rssfile in this or parent ".
- "directories, you must create one with mkgalrss.pl\n";
- exit 1;
+ $feedobj->{-feed} = XML::FeedPP::Atom::Atom10->new;
+ $feedobj->{-feed}->title("Gallery");
+ $feedobj->{-feed}->description("generated by ".
+ "".
+ "mkgallery");
+ $feedobj->{-feed}->link($feedbase, 'type' => 'text/html', 'rel' => 'alternate');
+ $feedobj->{-feed}->xmlns("xmlns:ostatus" =>
+ "http://ostatus.org/schema/1.0");
+ #$feedobj->{-feed}->copyright("");
+ #$feedobj->{-feed}->language("en");
+ #$feedobj->{-feed}->image($url, $tit, $link, $desc, $w, $h);
}
+ $self->{-feed} = $feedobj->{-feed};
}
sub iterate {
@@ -292,6 +358,16 @@ sub isimg {
my $self = shift;
my $fullpath = $self->{-fullpath};
return 0 unless ( -f $fullpath );
+
+ if ($havegeoloc) {
+ my $exif = new Image::ExifTool;
+ $exif->ExtractInfo($fullpath);
+ my ($la,$lo) = $exif->GetLocation();
+ if ($la && $lo) {
+ $self->{-geoloc} = [$la,$lo];
+ }
+ }
+
my $info = image_info($fullpath);
if (my $error = $info->{error}) {
if (($error !~ "Unrecognized file format") &&
@@ -359,12 +435,13 @@ sub edittitle {
my $title;
my $T;
if (open($T,'<'.$fullpath.'/.title')) {
+ binmode($T, ":utf8");
$title = <$T>;
$title =~ s/[\r\n]*$//;
close($T);
}
if ($asktitle || (!$title && !$noasktitle)) {
- my $prompt = $self->{-base};
+ my $prompt = $self->{-relpath};
$prompt = '/' unless ($prompt);
my $OUT = $term->OUT || \*STDOUT;
print $OUT "Enter title for $fullpath\n";
@@ -376,7 +453,7 @@ sub edittitle {
}
}
unless ($title) {
- $title=substr($fullpath,length($self->{-root}));
+ $title=$self->{-relpath};
}
$self->{-title}=$title;
print "title in $fullpath is $title\n" if ($debug);
@@ -443,7 +520,7 @@ sub makeaux {
my $dn = $self->{-parent}->{-fullpath};
my $pref = $self->{-previmg}->{-base};
my $nref = $self->{-nextimg}->{-base};
- my $inc = $self->{-inc};
+ my $inc = $self->{-inc}.$incdir.'/';
my $title = $self->{-info}->{'Comment'};
$title = $name unless ($title);
@@ -510,6 +587,7 @@ sub makeaux {
end_table,"\n",
center(table({-class=>'picframe'},
Tr(td(img({-src=>$imgsrc,
+ -class=>'standalone',
-alt=>$title}))))),"\n",
end_html,"\n";
close($F);
@@ -524,6 +602,7 @@ sub makeaux {
warn "cannot open \"$fn\": $!";
return;
}
+ binmode($F, ":utf8");
my $imgsrc = sprintf("../.%s/%s",$sizes[0],$name);
print $F start_html(-title=>$title,
-encoding=>"utf-8",
@@ -538,6 +617,7 @@ sub makeaux {
h1($title),"\n",
table({-class=>'ipage'},
Tr(td(img({-src=>$imgsrc,
+ -class=>'thumbnail',
-alt=>$title})),
td($self->infotable))),
a({-href=>'../index.html',-class=>'conceal'},
@@ -561,19 +641,22 @@ sub startindex {
binmode($IND, ":utf8");
$self->{-IND} = $IND;
- my $inc = $self->{-inc};
+ my $inc = $self->{-inc}.$incdir.'/';
my $title = $self->{-title};
- my $rsslink="";
- if ($self->{-rss}) {
- $rsslink=Link({-rel=>'alternate',
- -type=>'application/rss+xml',
- -title=>'RSS',
- -href=>$self->{-rss}});
+ my $feedlink="";
+ if ($feedobj) {
+ $feedlink=Link({-rel=>'alternate',
+ -type=>'application/atom+xml',
+ -title=>'Gallery Feed',
+ -href=>$feedobj->{-base}.$feedobj->{-file}});
}
print $IND start_html(-title => $title,
-encoding=>"utf-8",
- -head=>$rsslink,
- -style=>{-src=>$inc."gallery.css"},
+ -head=>$feedlink,
+ -style=>[
+ {-src=>$inc."gallery.css"},
+ {-src=>$inc."custom.css"},
+ ],
-script=>[
{-src=>$inc."mootools.js"},
{-src=>$inc."overlay.js"},
@@ -587,32 +670,76 @@ sub startindex {
comment("Created by ".$version),"\n",
start_div({-class => 'indexContainer',
-id => 'indexContainer'}),
- a({-href=>"../index.html"},"UP"),"\n",
- start_center,"\n",
- h1($title),"\n",
"\n";
+ my $EVL;
+ if (open($EVL,$self->{-toppath}.'/'.$incdir.'/header.pl')) {
+ my $prm;
+ while (<$EVL>) {
+ $prm .= $_;
+ }
+ close($EVL);
+ %_ = (
+ -version => $version,
+ -depth => $self->{-depth},
+ -title => $title,
+ -path => $self->{-fullpath},
+ -breadcrumbs => "breadcrumbs unimplemented",
+ );
+ print $IND eval $prm,"\n";
+ } else {
+ print STDERR "could not open ",
+ $self->{-toppath}.'/'.$incdir.'/header.pl',
+ " ($!), reverting to default header";
+ print $IND a({-href=>"../index.html"},"UP"),"\n",
+ h1({-class=>'title'},$title),"\n",
+ }
}
sub endindex {
my $self = shift;
my $IND = $self->{-IND};
- print $IND end_center,end_div,
- # "\n",'',"\n",
- end_html,"\n";
+ print $IND end_div;
+ my $EVL;
+ if (open($EVL,$self->{-toppath}.'/'.$incdir.'/footer.pl')) {
+ my $prm;
+ while (<$EVL>) {
+ $prm .= $_;
+ }
+ close($EVL);
+ %_ = (
+ -version => $version,
+ -depth => $self->{-depth},
+ -title => $self->{-title},
+ -breadcrumbs => "breadcrumbs unimplemented",
+ );
+ print $IND eval $prm,"\n";
+ } else {
+ print STDERR "could not open ",
+ $self->{-toppath}.'/'.$incdir.'/footer.pl',
+ " ($!), reverting to default empty footer";
+ }
+ print $IND end_html,"\n";
close($IND) if ($IND);
undef $self->{-IND};
- if ($rssobj) {
- my $rsstitle=sprintf "%s [%d images, %d subalbums]",
+ if ($feedobj) {
+ my $feedtitle=sprintf "%s [%d images, %d subalbums]",
$self->{-title},
$self->{-numofimgs},
$self->{-numofsubs};
- my $rsslink=$rssobj->{'rss'}->channel('link')."index.html";
- $rssobj->{'rss'}->add_item(
+ my $feedlink=$feedobj->{-feed}->link.
+ $self->{-relpath}."index.html";
+ my $uu;
+ my $us;
+ UUID::generate($uu);
+ UUID::unparse($uu, $us);
+ $feedobj->{-feed}->add_item(
title => $self->{-title},
- link => $rsslink,
- description => $rsstitle,
+ link => $feedlink,
+ description => $feedtitle,
+ pubDate => time,
+ guid => $us,
);
}
}
@@ -621,7 +748,7 @@ sub startsublist {
my $self = shift;
my $IND = $self->{-IND};
- print $IND h2("Albums"),"\n",start_table,"\n";
+ print $IND h2({-class=>"atitle"},"Albums"),"\n",start_table,"\n";
}
sub sub_entry {
@@ -648,8 +775,8 @@ sub startimglist {
my $first = $self->{-firstimg}->{-base};
my $slideref = sprintf(".html/%s-slide.html",$first);
- print $IND h2("Images ",
- a({-href=>$slideref,-class=>'showStart',-id=>$first},
+ print $IND h2({-class=>"ititle"},"Images ",
+ a({-href=>$slideref,-class=>'showStart',-rel=>'i'.$first},
'> slideshow')),"\n";
}
@@ -667,29 +794,41 @@ sub img_entry {
$self->{-parent}->{-numofimgs}++;
print $IND a({-name=>$name}),"\n",
- start_table({-class=>'slide'}),start_Tr,start_td,"\n",
- div({-class=>'slidetitle',-id=>$name},
+ start_table({-class=>'slide'}),start_Tr,start_td,"\n";
+ print $IND div({-class=>'slidetitle'},
"\n ",a({-href=>".html/$name-info.html",
-title=>'Image Info: '.$name,
-class=>'infoBox'},
$title),"\n"),"\n",
- div({-class=>'slideimage',-id=>$name},
- "\n ",a({-href=>".html/$name-static.html",
+ start_div({-class=>'slideimage'});
+ if ($self->{-geoloc}) {
+ my ($la,$lo) = @{$self->{-geoloc}};
+ print $IND a({-href=>"http://maps.google.com/".
+ "?q=$la,$lo&ll=$la,$lo",
+ -title=>"$la,$lo",
+ -class=>'geoloc'},
+ div({-class=>'geoloc'},"")),"\n";
+ }
+ print $IND a({-href=>".html/$name-static.html",
-title=>$title,
-class=>'showImage',
- -id=>$name},
- img({-src=>$thumb,-alt=>$title})),"\n"),"\n",
- start_div({-class=>'varimages',-id=>$name,-title=>$title}),"\n";
+ -rel=>'i'.$name},
+ img({-src=>$thumb,
+ -class=>'thumbnail',
+ -alt=>$title})),"\n",end_div,
+ start_div({-class=>'varimages',-id=>'i'.$name,-title=>$title}),"\n";
foreach my $sz(@sizes) {
my $src=$self->{$sz}->{'url'};
my $w=$self->{$sz}->{'dim'}->[0];
my $h=$self->{$sz}->{'dim'}->[1];
print $IND " ",a({-href=>$src,
-class=>"conceal",
+ -rel=>$w."x".$h,
-title=>"Reduced to ".$w."x".$h},
$w."x".$h)," \n";
}
print $IND " ",a({-href=>$name,
+ -rel=>$w."x".$h,
-title=>'Original'},$w."x".$h),
"\n",end_div,"\n",
end_td,end_Tr,end_table,"\n";