clean
[ikiwiki.git] / IkiWiki / Plugin / tag.pm
blob13dabce0c16cef34b85eab82fcc8c6c9aab2af54
1 #!/usr/bin/perl
2 # Ikiwiki tag plugin.
3 package IkiWiki::Plugin::tag;
5 use warnings;
6 use strict;
7 use IkiWiki 3.00;
9 sub import {
10 hook(type => "checkconfig", id => "tag", call => \&checkconfig);
11 hook(type => "getopt", id => "tag", call => \&getopt);
12 hook(type => "getsetup", id => "tag", call => \&getsetup);
13 hook(type => "preprocess", id => "tag", call => \&preprocess_tag, scan => 1);
14 hook(type => "preprocess", id => "taglink", call => \&preprocess_taglink, scan => 1);
15 hook(type => "pagetemplate", id => "tag", call => \&pagetemplate);
17 IkiWiki::loadplugin("transient");
20 sub getopt () {
21 eval q{use Getopt::Long};
22 error($@) if $@;
23 Getopt::Long::Configure('pass_through');
24 GetOptions("tagbase=s" => \$config{tagbase});
27 sub getsetup () {
28 return
29 plugin => {
30 safe => 1,
31 rebuild => undef,
33 tagbase => {
34 type => "string",
35 example => "tag",
36 description => "parent page tags are located under",
37 safe => 1,
38 rebuild => 1,
40 tag_autocreate => {
41 type => "boolean",
42 example => 1,
43 description => "autocreate new tag pages?",
44 safe => 1,
45 rebuild => undef,
47 tag_autocreate_commit => {
48 type => "boolean",
49 example => 1,
50 default => 1,
51 description => "commit autocreated tag pages",
52 safe => 1,
53 rebuild => 0,
57 sub checkconfig () {
58 if (! defined $config{tag_autocreate_commit}) {
59 $config{tag_autocreate_commit} = 1;
63 sub taglink ($) {
64 my $tag=shift;
66 if ($tag !~ m{^/} &&
67 defined $config{tagbase}) {
68 $tag="/".$config{tagbase}."/".$tag;
69 $tag=~y#/#/#s; # squash dups
72 return $tag;
75 # Returns a tag name from a tag link
76 sub tagname ($) {
77 my $tag=shift;
78 if (defined $config{tagbase}) {
79 $tag =~ s!^/\Q$config{tagbase}\E/!!;
80 } else {
81 $tag =~ s!^\.?/!!;
83 return pagetitle($tag, 1);
86 sub htmllink_tag ($$$;@) {
87 my $page=shift;
88 my $destpage=shift;
89 my $tag=shift;
90 my %opts=@_;
92 return htmllink($page, $destpage, taglink($tag), %opts);
95 sub gentag ($) {
96 my $tag=shift;
98 if ($config{tag_autocreate} ||
99 ($config{tagbase} && ! defined $config{tag_autocreate})) {
100 my $tagpage=taglink($tag);
101 if ($tagpage=~/^\.\/(.*)/) {
102 $tagpage=$1;
104 else {
105 $tagpage=~s/^\///;
108 my $tagfile = newpagefile($tagpage, $config{default_pageext});
110 add_autofile($tagfile, "tag", sub {
111 my $message=sprintf(gettext("creating tag page %s"), $tagpage);
112 debug($message);
114 my $template=template("autotag.tmpl");
115 $template->param(tagname => tagname($tag));
116 $template->param(tag => $tag);
118 my $dir = $config{srcdir};
119 if (! $config{tag_autocreate_commit}) {
120 $dir = $IkiWiki::Plugin::transient::transientdir;
123 writefile($tagfile, $dir, $template->output);
124 if ($config{rcs} && $config{tag_autocreate_commit}) {
125 IkiWiki::disable_commit_hook();
126 IkiWiki::rcs_add($tagfile);
127 IkiWiki::rcs_commit_staged(message => $message);
128 IkiWiki::enable_commit_hook();
134 sub preprocess_tag (@) {
135 if (! @_) {
136 return "";
138 my %params=@_;
139 my $page = $params{page};
140 delete $params{page};
141 delete $params{destpage};
142 delete $params{preview};
144 foreach my $tag (keys %params) {
145 $tag=linkpage($tag);
147 # hidden WikiLink
148 add_link($page, taglink($tag), 'tag');
150 gentag($tag);
153 return "";
156 sub preprocess_taglink (@) {
157 if (! @_) {
158 return "";
160 my %params=@_;
161 return join(" ", map {
162 if (/(.*)\|(.*)/) {
163 my $tag=linkpage($2);
164 add_link($params{page}, taglink($tag), 'tag');
165 gentag($tag);
166 return htmllink_tag($params{page}, $params{destpage}, $tag,
167 linktext => pagetitle($1));
169 else {
170 my $tag=linkpage($_);
171 add_link($params{page}, taglink($tag), 'tag');
172 gentag($tag);
173 return htmllink_tag($params{page}, $params{destpage}, $tag);
176 grep {
177 $_ ne 'page' && $_ ne 'destpage' && $_ ne 'preview'
178 } keys %params);
181 sub pagetemplate (@) {
182 my %params=@_;
183 my $page=$params{page};
184 my $destpage=$params{destpage};
185 my $template=$params{template};
187 my $tags = $typedlinks{$page}{tag};
189 $template->param(tags => [
190 map {
191 link => htmllink_tag($page, $destpage, $_,
192 rel => "tag", linktext => tagname($_))
193 }, sort keys %$tags
194 ]) if defined $tags && %$tags && $template->query(name => "tags");
196 if ($template->query(name => "categories")) {
197 # It's an rss/atom template. Add any categories.
198 if (defined $tags && %$tags) {
199 $template->param(categories => [map { category => tagname($_) },
200 sort keys %$tags]);
205 package IkiWiki::PageSpec;
207 sub match_tagged ($$;@) {
208 my $page=shift;
209 my $glob=IkiWiki::Plugin::tag::taglink(shift);
210 return match_link($page, $glob, linktype => 'tag', @_);