X-Git-Url: https://git.immae.eu/?a=blobdiff_plain;f=plugins%2Fmarkdown%2Fmarkdown.php;h=0cf6e6e2d283e32de9ec978389cad40ad5c309c0;hb=514185e14b09d1e37c41eb5d2720e3e45b12eea9;hp=544ed22ecb6c71f0b8b769df580b661d5e3aa5be;hpb=11609d9fd8ba53f049e6c913d8e3affab6cfc9ce;p=github%2Fshaarli%2FShaarli.git diff --git a/plugins/markdown/markdown.php b/plugins/markdown/markdown.php index 544ed22e..0cf6e6e2 100644 --- a/plugins/markdown/markdown.php +++ b/plugins/markdown/markdown.php @@ -6,13 +6,10 @@ * Shaare's descriptions are parsed with Markdown. */ -require_once 'Parsedown.php'; - /* * If this tag is used on a shaare, the description won't be processed by Parsedown. - * Using a private tag so it won't appear for visitors. */ -define('NO_MD_TAG', '.nomarkdown'); +define('NO_MD_TAG', 'nomarkdown'); /** * Parse linklist descriptions. @@ -25,6 +22,26 @@ function hook_markdown_render_linklist($data) { foreach ($data['links'] as &$value) { if (!empty($value['tags']) && noMarkdownTag($value['tags'])) { + $value = stripNoMarkdownTag($value); + continue; + } + $value['description'] = process_markdown($value['description']); + } + return $data; +} + +/** + * Parse feed linklist descriptions. + * + * @param array $data linklist data. + * + * @return mixed linklist data parsed in markdown (and converted to HTML). + */ +function hook_markdown_render_feed($data) +{ + foreach ($data['links'] as &$value) { + if (!empty($value['tags']) && noMarkdownTag($value['tags'])) { + $value = stripNoMarkdownTag($value); continue; } $value['description'] = process_markdown($value['description']); @@ -46,6 +63,7 @@ function hook_markdown_render_daily($data) foreach ($data['cols'] as &$value) { foreach ($value as &$value2) { if (!empty($value2['tags']) && noMarkdownTag($value2['tags'])) { + $value2 = stripNoMarkdownTag($value2); continue; } $value2['formatedDescription'] = process_markdown($value2['formatedDescription']); @@ -64,7 +82,30 @@ function hook_markdown_render_daily($data) */ function noMarkdownTag($tags) { - return strpos($tags, NO_MD_TAG) !== false; + return preg_match('/(^|\s)'. NO_MD_TAG .'(\s|$)/', $tags); +} + +/** + * Remove the no-markdown meta tag so it won't be displayed. + * + * @param array $link Link data. + * + * @return array Updated link without no markdown tag. + */ +function stripNoMarkdownTag($link) +{ + if (! empty($link['taglist'])) { + $offset = array_search(NO_MD_TAG, $link['taglist']); + if ($offset !== false) { + unset($link['taglist'][$offset]); + } + } + + if (!empty($link['tags'])) { + str_replace(NO_MD_TAG, '', $link['tags']); + } + + return $link; } /** @@ -119,7 +160,45 @@ function hook_markdown_render_editlink($data) */ function reverse_text2clickable($description) { - return preg_replace('![^ ]+!m', '$1', $description); + $descriptionLines = explode(PHP_EOL, $description); + $descriptionOut = ''; + $codeBlockOn = false; + $lineCount = 0; + + foreach ($descriptionLines as $descriptionLine) { + // Detect line of code: starting with 4 spaces, + // except lists which can start with +/*/- or `2.` after spaces. + $codeLineOn = preg_match('/^ +(?=[^\+\*\-])(?=(?!\d\.).)/', $descriptionLine) > 0; + // Detect and toggle block of code + if (!$codeBlockOn) { + $codeBlockOn = preg_match('/^```/', $descriptionLine) > 0; + } + elseif (preg_match('/^```/', $descriptionLine) > 0) { + $codeBlockOn = false; + } + + $hashtagTitle = ' title="Hashtag [^"]+"'; + // Reverse `inline code` hashtags. + $descriptionLine = preg_replace( + '!(`[^`\n]*)([^<]+)([^`\n]*`)!m', + '$1$2$3', + $descriptionLine + ); + + // Reverse all links in code blocks, only non hashtag elsewhere. + $hashtagFilter = (!$codeBlockOn && !$codeLineOn) ? '(?!'. $hashtagTitle .')': '(?:'. $hashtagTitle .')?'; + $descriptionLine = preg_replace( + '#([^<]+)#m', + '$1', + $descriptionLine + ); + + $descriptionOut .= $descriptionLine; + if ($lineCount++ < count($descriptionLines) - 1) { + $descriptionOut .= PHP_EOL; + } + } + return $descriptionOut; } /** @@ -194,9 +273,9 @@ function process_markdown($description) $parsedown = new Parsedown(); $processedDescription = $description; - $processedDescription = reverse_text2clickable($processedDescription); $processedDescription = reverse_nl2br($processedDescription); $processedDescription = reverse_space2nbsp($processedDescription); + $processedDescription = reverse_text2clickable($processedDescription); $processedDescription = unescape($processedDescription); $processedDescription = $parsedown ->setMarkupEscaped(false)