aboutsummaryrefslogtreecommitdiff
path: root/IkiWiki/Plugin/templatebody.pm
blob: 3848b75c73e0382dacd71d3675bed9dac36d3679 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
#!/usr/bin/perl
# Define self-documenting templates as wiki pages without HTML::Template
# markup leaking into IkiWiki's output.
# Copyright © 2013-2014 Simon McVittie. GPL-2+, see debian/copyright
package IkiWiki::Plugin::templatebody;

use warnings;
use strict;
use IkiWiki 3.00;
use Encode;

sub import {
	hook(type => "getsetup", id => "templatebody", call => \&getsetup);
	hook(type => "preprocess", id => "templatebody", call => \&preprocess,
		scan => 1);
	hook(type => "readtemplate", id => "templatebody",
		call => \&readtemplate);
}

sub getsetup () {
	return
		plugin => {
			safe => 1,
			rebuild => undef,
			section => "core",
		},
}

# This doesn't persist between runs: we're going to read and scan the
# template file regardless, so there's no point in saving it to the index.
# Example contents:
# ("templates/note" => "<div class=\"notebox\">\n<TMPL_VAR text>\n</div>")
my %templates;

sub preprocess (@) {
	my %params=@_;

	# [[!templatebody "<div>hello</div>"]] results in
	# preprocess("<div>hello</div>" => undef, page => ...)
	my $content = $_[0];
	if (length $_[1]) {
		error(gettext("first parameter must be the content"));
	}

	$templates{$params{page}} = $content;

	return "";
}

sub readtemplate {
	my %params = @_;
	my $tpage = $params{page};
	my $content = $params{content};
	my $filename = $params{filename};

	# pass-through if it's a .tmpl attachment or otherwise unsuitable
	return $content unless defined $tpage;
	return $content if $tpage =~ /\.tmpl$/;
	my $src = $pagesources{$tpage};
	return $content unless defined $src;
	return $content unless defined pagetype($src);

	# We might be using the template for [[!template]], which has to run
	# during the scan stage so that templates can include scannable
	# directives which are expanded in the resulting page. Calls to
	# IkiWiki::scan are in arbitrary order, so the template might
	# not have been scanned yet. Make sure.
	require IkiWiki::Render;
	IkiWiki::scan($src);

	# Having scanned it, we know whether it had a [[!templatebody]].
	if (exists $templates{$tpage}) {
		return $templates{$tpage};
	}

	# If not, return the whole thing. (Eventually, after implementing
	# a transition, this can become an error.)
	return $content;
}

1