123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250 |
- use strict;
- use v5.10;
- AddModuleDescription('markdown-rule.pl', 'Markdown Rule Extension');
- our ($q, $bol, %RuleOrder, @MyRules, $UrlProtocols, $FullUrlPattern, @HtmlStack, $Fragment);
- push(@MyRules, \&MarkdownRule);
- $RuleOrder{\&MarkdownRule} = 200;
- sub MarkdownRule {
- my $alignment;
-
- if (m/\G\\([-#>*`=])/cg) {
- return $1;
- }
-
- elsif ($bol and m~\G(\s*\n)*(
- my $header_depth = length($2);
- return CloseHtmlEnvironments()
- . AddHtmlEnvironment("h" . $header_depth);
- }
-
- elsif ((InElement('h1') or InElement('h2') or InElement('h3') or
- InElement('h4') or InElement('h5') or InElement('h6'))
- and m/\G\n/cg) {
- return CloseHtmlEnvironments()
- . AddHtmlEnvironment("p");
- }
-
-
- elsif ($bol and m/\G((?:>.*\n?)+)/cg) {
- Clean(CloseHtmlEnvironments());
- Dirty($1);
- my $text = $1;
- my ($oldpos, $old_) = ((pos), $_);
- print '<blockquote>';
- $text =~ s/^> ?//gm;
- ApplyRules($text, 1, 1, undef, 'p');
- print '</blockquote>';
- Clean(AddHtmlEnvironment('p'));
- ($_, pos) = ($old_, $oldpos);
- }
-
- elsif ($bol and m/\G("""[ \t]*\n(.*?)\n"""[ \t]*(?:\n|$))/cgs) {
- Clean(CloseHtmlEnvironments());
- Dirty($1);
- my ($oldpos, $old_) = ((pos), $_);
- print '<blockquote>';
- ApplyRules($2, 1, 1, undef, 'p');
- print '</blockquote>';
- Clean(AddHtmlEnvironment('p'));
- ($_, pos) = ($old_, $oldpos);
- }
-
- elsif ($bol and m/\G```[ \t]*\n(.*?)\n```[ \t]*(\n|$)/cgs) {
- return CloseHtmlEnvironments() . $q->pre($1)
- . AddHtmlEnvironment("p");
- }
-
- elsif (m/\G`([^\n`][^`]*)`/cg) {
- return $q->code($1);
- }
-
- elsif (not InElement('strong') and not InElement('em') and m/\G\*\*\*/cg) {
- return AddHtmlEnvironment('em') . AddHtmlEnvironment('strong');
- }
- elsif (InElement('strong') and InElement('em') and m/\G\*\*\*/cg) {
- return CloseHtmlEnvironment('strong') . CloseHtmlEnvironment('em');
- }
- # **bold**
- elsif (m/\G\*\*/cg) {
- return AddOrCloseHtmlEnvironment('strong');
- }
- # *italic* (closing before adding environment!)
- elsif (InElement('em') and m/\G\*/cg) {
- return CloseHtmlEnvironment('em');
- }
- elsif ($bol and m/\G\*/cg or m/\G(?<=\P{Word})\*/cg) {
- return AddHtmlEnvironment('em');
- }
- # ~~strikethrough~~ (deleted)
- elsif (m/\G~~/cg) {
- return AddOrCloseHtmlEnvironment('del');
- }
- # indented lists = nested lists
- elsif ($bol and m/\G(\s*\n)*()([*-]|\d+\.)[ \t]+/cg
- or InElement('li') && m/\G(\s*\n)+( *)([*-]|\d+\.)[ \t]+/cg) {
- my $nesting_goal = int(length($2)/4) + 1;
- my $tag = ($3 eq '*' or $3 eq '-') ? 'ul' : 'ol';
- my $nesting_current = 0;
- my @nesting = grep(/^[uo]l$/, @HtmlStack);
- my $html = CloseHtmlEnvironmentUntil('li');
-
-
-
- while (@nesting > $nesting_goal) {
- $html .= CloseHtmlEnvironment(pop(@nesting));
-
- }
-
- if (@nesting == $nesting_goal
- and $nesting[$#nesting] ne $tag) {
- $html .= CloseHtmlEnvironment(pop(@nesting));
-
- }
-
- if (@nesting < $nesting_goal) {
- $html .= AddHtmlEnvironment($tag);
-
- }
-
- if (InElement('li')) {
- $html .= CloseHtmlEnvironmentUntil($nesting[$#nesting]);
-
- }
- $html .= AddHtmlEnvironment('li');
-
- return $html;
- }
-
- elsif ($bol and !InElement('table') and m/\G\|/cg) {
- # warn pos . " beginning of a table";
- $alignment = 'style="text-align: right"' if m/\G([ \t]+)/cg;
- $alignment = 'style="text-align: center"' if $alignment and m/\G(?=[^|]+[ \t]+\|)/cg;
- $Fragment =~ s/[ \t]+$//;
- return OpenHtmlEnvironment('table',1)
- . AddHtmlEnvironment('tr')
- . AddHtmlEnvironment('th', $alignment);
- }
-
- elsif (InElement('table') and m/\G\|?\n\|/cg) {
- # warn pos . " end of a row and beginning of a new row";
- $alignment = 'style="text-align: right"' if m/\G([ \t]+)/cg;
- $alignment = 'style="text-align: center"' if $alignment and m/\G(?=[^|]+[ \t]+\|)/cg;
- $Fragment =~ s/[ \t]+$//;
- return CloseHtmlEnvironment('tr')
- . AddHtmlEnvironment('tr')
- . AddHtmlEnvironment('td', $alignment);
- }
-
- elsif (InElement('table') and m/\G\|?(\n|$)/cg) {
-
- $Fragment =~ s/[ \t]+$//;
- return CloseHtmlEnvironment('table')
- . AddHtmlEnvironment('p');
- }
-
- elsif (InElement('th') and m/\G\|/cg) {
- # warn pos . " continuation of the first row";
- $alignment = 'style="text-align: right"' if m/\G([ \t]+)/cg;
- $alignment = 'style="text-align: center"' if $alignment and m/\G(?=[^|]+[ \t]+\|)/cg;
- $Fragment =~ s/[ \t]+$//;
- return CloseHtmlEnvironment('th')
- . AddHtmlEnvironment('th', $alignment);
- }
-
- elsif (InElement('td') and m/\G\|/cg) {
- # warn pos . " continuation of other rows";
- $alignment = 'style="text-align: right"' if m/\G([ \t]+)/cg;
- $alignment = 'style="text-align: center"' if $alignment and m/\G(?=[^|]+[ \t]+\|)/cg;
- $Fragment =~ s/[ \t]+$//;
- return CloseHtmlEnvironment('td')
- . AddHtmlEnvironment('td', $alignment);
- }
-
- elsif ($bol and m/\G(\s*\n)*( .+)\n?/cg) {
- my $str = substr($2, 4);
- while (m/\G( .*)\n?/cg) {
- $str .= "\n" . substr($1, 4);
- }
- return OpenHtmlEnvironment('pre',1) . $str; # always level 1
- }
- # link: [an example](http://example.com/ "Title")
- elsif (m/\G\[((?:[^]\n]+\n?)+)\]\((\S+)(\s+"(.+?)")?\)/cg) {
- my ($text, $url, $title) = ($1, $2, $4);
- my %params;
- $params{-href} = $url;
- $params{-class} = "url";
- $params{-title} = $title if $title;
- return $q->a(\%params, $text);
- }
-
- elsif (m/\G\[((?:[^]\n]+\n?)+)\]\((#\S)+(\s+"(.+?)")?\)/cg) {
- my ($text, $url, $title) = ($1, $2, $4);
- my %params;
- $params{-href} = $url;
- $params{-class} = "named-anchor";
- $params{-title} = $title if $title;
- return $q->a(\%params, $text);
- }
-
- elsif ($bol and m/\G((\s*\n)*(.+?)[ \t]*\n(-+|=+)[ \t]*\n)/cg) {
- return CloseHtmlEnvironments()
- . (substr($4,0,1) eq '=' ? $q->h2($3) : $q->h3($3))
- . AddHtmlEnvironment('p');
- }
- return;
- }
- push(@MyRules, \&MarkdownExtraRule);
- sub MarkdownExtraRule {
-
- if (m/\G__/cg) {
- return AddOrCloseHtmlEnvironment('em', 'style="font-style: normal; text-decoration: underline"');
- }
-
- elsif (InElement('em', 'style="font-style: normal; text-decoration: underline"') and m/\G_/cg) {
- return CloseHtmlEnvironment('em');
- }
- elsif ($bol and m/\G_/cg or m/\G(?<=\P{Word})_(?=\S)/cg) {
- return AddHtmlEnvironment('em', 'style="font-style: normal; text-decoration: underline"');
- }
-
- elsif (m/\G\/\//cg) {
- return AddOrCloseHtmlEnvironment('em');
- }
-
- elsif (InElement('em') and m/\G\//cg) {
- return CloseHtmlEnvironment('em');
- }
- elsif ($bol and m/\G\//cg or m/\G(?<=[|[:space:]])\/(?=\S)/cg) {
- return AddHtmlEnvironment('em');
- }
- return;
- }
|