Merge branch 'master' of ssh://git.ikiwiki.info/srv/git/ikiwiki.info
[ikiwiki.git] / IkiWiki / Plugin / highlight.pm
blob65e372db1336dbdedd6f1c5e256d4ef3b953aa2a
1 #!/usr/bin/perl
2 package IkiWiki::Plugin::highlight;
4 # This has been tested with highlight 2.16 and highlight 3.2+svn19.
5 # In particular version 3.2 won't work. It detects the different
6 # versions by the presence of the the highlight::DataDir class.
8 use warnings;
9 use strict;
10 use IkiWiki 3.00;
11 use Encode;
13 my $data_dir;
15 sub import {
16 hook(type => "getsetup", id => "highlight", call => \&getsetup);
17 hook(type => "checkconfig", id => "highlight", call => \&checkconfig);
18 # this hook is used by the format plugin
19 hook(type => "htmlizeformat", id => "highlight",
20 call => \&htmlizeformat, last => 1);
23 sub getsetup () {
24 return
25 plugin => {
26 safe => 1,
27 rebuild => 1, # format plugin
28 section => "format",
30 tohighlight => {
31 type => "string",
32 example => ".c .h .cpp .pl .py Makefile:make",
33 description => "types of source files to syntax highlight",
34 safe => 1,
35 rebuild => 1,
37 filetypes_conf => {
38 type => "string",
39 example => "/etc/highlight/filetypes.conf",
40 description => "location of highlight's filetypes.conf",
41 safe => 0,
42 rebuild => undef,
44 langdefdir => {
45 type => "string",
46 example => "/usr/share/highlight/langDefs",
47 description => "location of highlight's langDefs directory",
48 safe => 0,
49 rebuild => undef,
53 sub checkconfig () {
55 eval q{use highlight};
56 if ($@) {
57 print STDERR "Failed to load highlight. Configuring anyway.\n";
60 if (highlight::DataDir->can('new')){
61 $data_dir=new highlight::DataDir();
62 $data_dir->searchDataDir("");
63 } else {
64 $data_dir=undef;
67 if (! exists $config{filetypes_conf}) {
68 $config{filetypes_conf}=
69 ($data_dir ? $data_dir->getConfDir() : "/etc/highlight/")
70 . "filetypes.conf";
72 if (! exists $config{langdefdir}) {
73 $config{langdefdir}=
74 ($data_dir ? $data_dir->getLangPath("")
75 : "/usr/share/highlight/langDefs");
78 if (exists $config{tohighlight} && read_filetypes()) {
79 foreach my $file (split ' ', $config{tohighlight}) {
80 my @opts = $file=~s/^\.// ?
81 (keepextension => 1) :
82 (noextension => 1);
83 my $ext = $file=~s/:(.*)// ? $1 : $file;
85 my $langfile=ext2langfile($ext);
86 if (! defined $langfile) {
87 error(sprintf(gettext(
88 "tohighlight contains unknown file type '%s'"),
89 $ext));
92 hook(
93 type => "htmlize",
94 id => $file,
95 call => sub {
96 my %params=@_;
97 highlight($langfile, $params{content});
99 longname => sprintf(gettext("Source code: %s"), $file),
100 @opts,
106 sub htmlizeformat {
107 my $format=lc shift;
108 my $langfile=ext2langfile($format);
110 if (! defined $langfile) {
111 return;
114 return Encode::decode_utf8(highlight($langfile, shift));
117 my %ext2lang;
118 my $filetypes_read=0;
119 my %highlighters;
121 # Parse highlight's config file to get extension => language mappings.
122 sub read_filetypes () {
123 my $f;
124 if (!open($f, $config{filetypes_conf})) {
125 warn($config{filetypes_conf}.": ".$!);
126 return 0;
129 local $/=undef;
130 my $config=<$f>;
131 close $f;
133 # highlight >= 3.2 format (bind-style)
134 while ($config=~m/Lang\s*=\s*\"([^"]+)\"[,\s]+Extensions\s*=\s*{([^}]+)}/sg) {
135 my $lang=$1;
136 foreach my $bit (split ',', $2) {
137 $bit=~s/.*"(.*)".*/$1/s;
138 $ext2lang{$bit}=$lang;
142 # highlight < 3.2 format
143 if (! keys %ext2lang) {
144 foreach (split("\n", $config)) {
145 if (/^\$ext\((.*)\)=(.*)$/) {
146 $ext2lang{$_}=$1 foreach $1, split ' ', $2;
151 return $filetypes_read=1;
155 # Given a filename extension, determines the language definition to
156 # use to highlight it.
157 sub ext2langfile ($) {
158 my $ext=shift;
160 my $langfile="$config{langdefdir}/$ext.lang";
161 return $langfile if exists $highlighters{$langfile};
163 read_filetypes() unless $filetypes_read;
164 if (exists $ext2lang{$ext}) {
165 return "$config{langdefdir}/$ext2lang{$ext}.lang";
167 # If a language only has one common extension, it will not
168 # be listed in filetypes, so check the langfile.
169 elsif (-e $langfile) {
170 return $langfile;
172 else {
173 return undef;
177 # Interface to the highlight C library.
178 sub highlight ($$) {
179 my $langfile=shift;
180 my $input=shift;
182 eval q{use highlight};
183 if ($@) {
184 print STDERR gettext("warning: highlight perl module not available; falling back to pass through");
185 return $input;
188 my $gen;
189 if (! exists $highlighters{$langfile}) {
190 $gen = highlight::CodeGenerator::getInstance($highlight::XHTML);
191 $gen->setFragmentCode(1); # generate html fragment
192 $gen->setHTMLEnclosePreTag(1); # include stylish <pre>
193 if ($data_dir){
194 # new style, requires a real theme, but has no effect
195 $gen->initTheme($data_dir->getThemePath("seashell.theme"));
196 } else {
197 # old style, anything works.
198 $gen->initTheme("/dev/null");
200 $gen->loadLanguage($langfile); # must come after initTheme
201 $gen->setEncoding("utf-8");
202 $highlighters{$langfile}=$gen;
204 else {
205 $gen=$highlighters{$langfile};
208 return $gen->generateString($input);