summaryrefslogtreecommitdiff
path: root/IkiWiki/Plugin/highlight.pm
blob: 872537c726c2f62500e836babc9bccd50a2e95fd (plain)
  1. #!/usr/bin/perl
  2. package IkiWiki::Plugin::highlight;
  3. use warnings;
  4. use strict;
  5. use IkiWiki 3.00;
  6. use Encode;
  7. sub import {
  8. hook(type => "getsetup", id => "highlight", call => \&getsetup);
  9. hook(type => "checkconfig", id => "highlight", call => \&checkconfig);
  10. # this hook is used by the format plugin
  11. hook(type => "htmlizeformat", id => "highlight",
  12. call => \&htmlizeformat, last => 1);
  13. }
  14. sub getsetup () {
  15. return
  16. plugin => {
  17. safe => 1,
  18. rebuild => 1, # format plugin
  19. section => "format",
  20. },
  21. tohighlight => {
  22. type => "string",
  23. example => ".c .h .cpp .pl .py Makefile:make",
  24. description => "types of source files to syntax highlight",
  25. safe => 1,
  26. rebuild => 1,
  27. },
  28. filetypes_conf => {
  29. type => "string",
  30. example => "/etc/highlight/filetypes.conf",
  31. description => "location of highlight's filetypes.conf",
  32. safe => 0,
  33. rebuild => undef,
  34. },
  35. langdefdir => {
  36. type => "string",
  37. example => "/usr/share/highlight/langDefs",
  38. description => "location of highlight's langDefs directory",
  39. safe => 0,
  40. rebuild => undef,
  41. },
  42. }
  43. sub checkconfig () {
  44. if (! exists $config{filetypes_conf}) {
  45. $config{filetypes_conf}="/etc/highlight/filetypes.conf";
  46. }
  47. if (! exists $config{langdefdir}) {
  48. $config{langdefdir}="/usr/share/highlight/langDefs";
  49. }
  50. if (exists $config{tohighlight}) {
  51. foreach my $file (split ' ', $config{tohighlight}) {
  52. my @opts = $file=~s/^\.// ?
  53. (keepextension => 1) :
  54. (noextension => 1);
  55. my $ext = $file=~s/:(.*)// ? $1 : $file;
  56. my $langfile=ext2langfile($ext);
  57. if (! defined $langfile) {
  58. error(sprintf(gettext(
  59. "tohighlight contains unknown file type '%s'"),
  60. $ext));
  61. }
  62. hook(
  63. type => "htmlize",
  64. id => $file,
  65. call => sub {
  66. my %params=@_;
  67. highlight($langfile, $params{content});
  68. },
  69. longname => sprintf(gettext("Source code: %s"), $file),
  70. @opts,
  71. last => 1,
  72. );
  73. }
  74. }
  75. }
  76. sub htmlizeformat {
  77. my $format=lc shift;
  78. my $langfile=ext2langfile($format);
  79. if (! defined $langfile) {
  80. return;
  81. }
  82. return Encode::decode_utf8(highlight($langfile, shift));
  83. }
  84. my %ext2lang;
  85. my $filetypes_read=0;
  86. my %highlighters;
  87. # Parse highlight's config file to get extension => language mappings.
  88. sub read_filetypes () {
  89. open (IN, $config{filetypes_conf}) || error("$config{filetypes_conf}: $!");
  90. while (<IN>) {
  91. chomp;
  92. if (/^\$ext\((.*)\)=(.*)$/) {
  93. $ext2lang{$_}=$1 foreach $1, split ' ', $2;
  94. }
  95. }
  96. close IN;
  97. $filetypes_read=1;
  98. }
  99. # Given a filename extension, determines the language definition to
  100. # use to highlight it.
  101. sub ext2langfile ($) {
  102. my $ext=shift;
  103. my $langfile="$config{langdefdir}/$ext.lang";
  104. return $langfile if exists $highlighters{$langfile};
  105. read_filetypes() unless $filetypes_read;
  106. if (exists $ext2lang{$ext}) {
  107. return "$config{langdefdir}/$ext2lang{$ext}.lang";
  108. }
  109. # If a language only has one common extension, it will not
  110. # be listed in filetypes, so check the langfile.
  111. elsif (-e $langfile) {
  112. return $langfile;
  113. }
  114. else {
  115. return undef;
  116. }
  117. }
  118. # Interface to the highlight C library.
  119. sub highlight ($$) {
  120. my $langfile=shift;
  121. my $input=shift;
  122. eval q{use highlight};
  123. if ($@) {
  124. print STDERR gettext("warning: highlight perl module not available; falling back to pass through");
  125. return $input;
  126. }
  127. my $gen;
  128. if (! exists $highlighters{$langfile}) {
  129. $gen = highlightc::CodeGenerator_getInstance($highlightc::XHTML);
  130. $gen->setFragmentCode(1); # generate html fragment
  131. $gen->setHTMLEnclosePreTag(1); # include stylish <pre>
  132. $gen->initTheme("/dev/null"); # theme is not needed because CSS is not emitted
  133. $gen->initLanguage($langfile); # must come after initTheme
  134. $gen->setEncoding("utf-8");
  135. $highlighters{$langfile}=$gen;
  136. }
  137. else {
  138. $gen=$highlighters{$langfile};
  139. }
  140. return $gen->generateString($input);
  141. }
  142. 1