* Make all templates have a footer div to ease themeing. Required template
[ikiwiki.git] / IkiWiki / Plugin / img.pm
1 #!/usr/bin/perl
2 # Ikiwiki enhanced image handling plugin
3 # Christian Mock cm@tahina.priv.at 20061002
4 package IkiWiki::Plugin::img;
5
6 use warnings;
7 use strict;
8 use IkiWiki 2.00;
9
10 my %imgdefaults;
11
12 sub import { #{{{
13         hook(type => "preprocess", id => "img", call => \&preprocess);
14 } #}}}
15
16 sub preprocess (@) { #{{{
17         my ($image) = $_[0] =~ /$config{wiki_file_regexp}/; # untaint
18         my %params=@_;
19
20         if (! exists $imgdefaults{$params{page}}) {
21                 $imgdefaults{$params{page}} = {};
22         }
23         my $size = $params{size} || $imgdefaults{$params{page}}->{size} || 'full';
24         my $alt = $params{alt} || $imgdefaults{$params{page}}->{alt} || '';
25
26         if ($image eq 'defaults') {
27                 $imgdefaults{$params{page}} = {
28                         size => $size,
29                         alt => $alt,
30                 };
31                 return '';
32         }
33
34         add_depends($params{page}, $image);
35         my $file = bestlink($params{page}, $image)
36                 || return "[[img ".sprintf(gettext("%s not found"), $image)."]]";
37
38         my $dir = IkiWiki::dirname($file);
39         my $base = IkiWiki::basename($file);
40
41         eval q{use Image::Magick};
42         error($@) if $@;
43         my $im = Image::Magick->new;
44         my $imglink;
45         my $r;
46
47         if ($size ne 'full') {
48                 my ($w, $h) = ($size =~ /^(\d+)x(\d+)$/);
49                 return "[[img ".sprintf(gettext('bad size "%s"'), $size)."]]"
50                         unless (defined $w && defined $h);
51
52                 my $outfile = "$config{destdir}/$dir/${w}x${h}-$base";
53                 $imglink = "$dir/${w}x${h}-$base";
54                                 
55                 will_render($params{page}, $imglink);
56
57                 if (-e $outfile && (-M srcfile($file) >= -M $outfile)) {
58                         $r = $im->Read($outfile);
59                         return "[[img ".sprintf(gettext("failed to read %s: %s"), $outfile, $r)."]]" if $r;
60                 }
61                 else {
62                         $r = $im->Read(srcfile($file));
63                         return "[[img ".sprintf(gettext("failed to read %s: %s"), $file, $r)."]]" if $r;
64
65                         $r = $im->Resize(geometry => "${w}x${h}");
66                         return "[[img ".sprinftf(gettext("failed to resize: %s"), $r)."]]" if $r;
67
68                         # don't actually write file in preview mode
69                         if (! $params{preview}) {
70                                 my @blob = $im->ImageToBlob();
71                                 writefile($imglink, $config{destdir}, $blob[0], 1);
72                         }
73                         else {
74                                 $imglink = $file;
75                         }
76                 }
77         }
78         else {
79                 $r = $im->Read(srcfile($file));
80                 return "[[img ".sprintf(gettext("failed to read %s: %s"), $file, $r)."]]" if $r;
81                 $imglink = $file;
82         }
83
84         add_depends($imglink, $params{page});
85
86         my ($fileurl, $imgurl);
87         if (! $params{preview}) {
88                 $fileurl=urlto($file, $params{destpage});
89                 $imgurl=urlto($imglink, $params{destpage});
90         }
91         else {
92                 $fileurl="$config{url}/$file";
93                 $imgurl="$config{url}/$imglink";
94         }
95
96         return '<a href="'.$fileurl.'"><img src="'.$imgurl.
97                 '" alt="'.$alt.'" width="'.$im->Get("width").
98                 '" height="'.$im->Get("height").'" /></a>';
99 } #}}}
100
101 1