137748cd8SNickeau<?php 237748cd8SNickeau 337748cd8SNickeaunamespace ComboStrap; 437748cd8SNickeau 537748cd8SNickeau 637748cd8SNickeauuse Doku_Renderer_xhtml; 737748cd8SNickeauuse syntax_plugin_combo_code; 837748cd8SNickeau 9*04fd306cSNickeau/** 10*04fd306cSNickeau * Concurrent: https://highlightjs.org/ used by remark powerpoint 11*04fd306cSNickeau */ 1237748cd8SNickeauclass Prism 1337748cd8SNickeau{ 1437748cd8SNickeau 1537748cd8SNickeau const SNIPPET_NAME = 'prism'; 1637748cd8SNickeau /** 1737748cd8SNickeau * The class used to mark the added prism code 18*04fd306cSNickeau * See: https://cdnjs.com/libraries/prism/ 1937748cd8SNickeau */ 201fa8c418SNickeau const BASE_PRISM_CDN = "https://cdnjs.cloudflare.com/ajax/libs/prism/1.25.0"; 2137748cd8SNickeau /** 2237748cd8SNickeau * The default prompt for bash 2337748cd8SNickeau */ 2437748cd8SNickeau const CONF_BASH_PROMPT = "bashPrompt"; 2537748cd8SNickeau /** 2637748cd8SNickeau * The default prompt for batch (dos) 2737748cd8SNickeau */ 2837748cd8SNickeau const CONF_BATCH_PROMPT = "batchPrompt"; 2937748cd8SNickeau /** 3037748cd8SNickeau * The default prompt for powershell 3137748cd8SNickeau */ 3237748cd8SNickeau const CONF_POWERSHELL_PROMPT = "powershellPrompt"; 3337748cd8SNickeau 3437748cd8SNickeau /** 3537748cd8SNickeau * The default name of prism 3637748cd8SNickeau * It does not follow the naming of the theming 3737748cd8SNickeau */ 3837748cd8SNickeau const PRISM_THEME = "prism"; 3937748cd8SNickeau 4037748cd8SNickeau /** 411fa8c418SNickeau * @var string[] https://cdnjs.cloudflare.com/ajax/libs/prism/1.25.0/themes/prism-{theme}.min.css 4237748cd8SNickeau * 4337748cd8SNickeau * or default 4437748cd8SNickeau * 451fa8c418SNickeau * https://cdnjs.cloudflare.com/ajax/libs/prism/1.25.0/themes/prism.min.css 4637748cd8SNickeau * 4737748cd8SNickeau * or 4837748cd8SNickeau * 4937748cd8SNickeau * https://github.com/PrismJS/prism-themes 5037748cd8SNickeau * 5137748cd8SNickeau * from https://cdnjs.com/libraries/prism 5237748cd8SNickeau */ 5337748cd8SNickeau const THEMES_INTEGRITY = [ 541fa8c418SNickeau Prism::PRISM_THEME => "sha256-ko4j5rn874LF8dHwW29/xabhh8YBleWfvxb8nQce4Fc=", 55cb4149a8Sgerardnico "coy" => "sha256-gkHLZLptZZHaBY+jqrRkAVzOGfMa4HBhSCJteem8wy8=", 56cb4149a8Sgerardnico "dark" => "sha256-l+VX6V333ll/PXrjqG1W6DyZvDEw+50M7aAP6dcD7Qc=", 57cb4149a8Sgerardnico "funky" => "sha256-l9GTgvTMmAvPQ6IlNCd/I2FQwXVlJCLbGId7z6QlOpo=", 58cb4149a8Sgerardnico "okaidia" => "sha256-zzHVEO0xOoVm0I6bT9v5SgpRs1cYNyvEvHXW/1yCgqU=", 59cb4149a8Sgerardnico "solarizedlight" => "sha256-Lr49DyE+/KstnLdBxqZBoDYgNi6ONfZyAZw3LDhxB9I=", 60cb4149a8Sgerardnico "tomorrow" => "sha256-GxX+KXGZigSK67YPJvbu12EiBx257zuZWr0AMiT1Kpg=", 61cb4149a8Sgerardnico "twilight" => "sha256-R7PF7y9XAuz19FB93NgH/WQUVGk30iytl7EwtETrypo=" 6237748cd8SNickeau ]; 6337748cd8SNickeau 6437748cd8SNickeau /** 6537748cd8SNickeau * The theme 6637748cd8SNickeau */ 6737748cd8SNickeau const CONF_PRISM_THEME = "prismTheme"; 6837748cd8SNickeau const PRISM_THEME_DEFAULT = "tomorrow"; 6937748cd8SNickeau const SNIPPET_ID_AUTOLOADER = self::SNIPPET_NAME . "-autoloader"; 7037748cd8SNickeau 7137748cd8SNickeau 7237748cd8SNickeau /** 7337748cd8SNickeau * 7437748cd8SNickeau * @param $theme 7537748cd8SNickeau * 7637748cd8SNickeau * Ter info: The theme of the default wiki is in the print.css file (search for code blocks) 7737748cd8SNickeau */ 7837748cd8SNickeau public static function addSnippet($theme) 7937748cd8SNickeau { 8037748cd8SNickeau $BASE_PRISM_CDN = self::BASE_PRISM_CDN; 8137748cd8SNickeau 8237748cd8SNickeau if ($theme == self::PRISM_THEME) { 8337748cd8SNickeau $themeStyleSheet = "prism.min.css"; 8437748cd8SNickeau } else { 8537748cd8SNickeau $themeStyleSheet = "prism-$theme.min.css"; 8637748cd8SNickeau } 8737748cd8SNickeau $themeIntegrity = self::THEMES_INTEGRITY[$theme]; 8837748cd8SNickeau 8937748cd8SNickeau /** 9037748cd8SNickeau * We miss a bottom margin 9137748cd8SNickeau * as a paragraph 9237748cd8SNickeau */ 934cadd4f8SNickeau $snippetManager = PluginUtility::getSnippetManager(); 94*04fd306cSNickeau $snippetManager->attachCssInternalStyleSheet(self::SNIPPET_NAME); 9537748cd8SNickeau 9637748cd8SNickeau /** 9737748cd8SNickeau * Javascript 9837748cd8SNickeau */ 99*04fd306cSNickeau $snippetManager->attachRemoteJavascriptLibrary( 1004cadd4f8SNickeau self::SNIPPET_NAME, 1014cadd4f8SNickeau "$BASE_PRISM_CDN/components/prism-core.min.js", 1024cadd4f8SNickeau "sha256-vlRYHThwdq55dA+n1BKQRzzLwFtH9VINdSI68+5JhpU="); 103*04fd306cSNickeau $snippetManager->attachRemoteJavascriptLibrary( 1044cadd4f8SNickeau self::SNIPPET_NAME, 1054cadd4f8SNickeau "$BASE_PRISM_CDN/plugins/toolbar/prism-toolbar.min.js", 1064cadd4f8SNickeau "sha256-FyIVdIHL0+ppj4Q4Ft05K3wyCsYikpHIDGI7dcaBalU=" 1071fa8c418SNickeau ); 108*04fd306cSNickeau $snippetManager->attachRemoteCssStyleSheet( 109d4d4add3Sgerardnico self::SNIPPET_NAME, 110d4d4add3Sgerardnico "$BASE_PRISM_CDN/plugins/toolbar/prism-toolbar.css", 111d4d4add3Sgerardnico "sha256-kK4/JIYJUKI4Zdg9ZQ7FYyRIqeWPfYKi5QZHO2n/lJI=" 112d4d4add3Sgerardnico ); 11337748cd8SNickeau // https://prismjs.com/plugins/normalize-whitespace/ 114*04fd306cSNickeau $snippetManager->attachRemoteJavascriptLibrary( 1154cadd4f8SNickeau self::SNIPPET_NAME, 1164cadd4f8SNickeau "$BASE_PRISM_CDN/plugins/normalize-whitespace/prism-normalize-whitespace.min.js", 1174cadd4f8SNickeau "sha256-gBzABGbXfQYYnyr8xmDFjx6KGO9dBYuypG1QBjO76pY="); 118d4d4add3Sgerardnico // https://prismjs.com/plugins/copy-to-clipboard/ 119*04fd306cSNickeau $snippetManager->attachRemoteJavascriptLibrary( 120d4d4add3Sgerardnico self::SNIPPET_NAME, 121d4d4add3Sgerardnico "$BASE_PRISM_CDN/plugins/copy-to-clipboard/prism-copy-to-clipboard.min.js", 122d4d4add3Sgerardnico "sha512-pUNGXbOrc+Y3dm5z2ZN7JYQ/2Tq0jppMDOUsN4sQHVJ9AUQpaeERCUfYYBAnaRB9r8d4gtPKMWICNhm3tRr4Fg=="); 1234cadd4f8SNickeau // https://prismjs.com/plugins/show-language/ 124*04fd306cSNickeau $snippetManager->attachRemoteJavascriptLibrary( 1254cadd4f8SNickeau self::SNIPPET_NAME, 1264cadd4f8SNickeau "$BASE_PRISM_CDN/plugins/show-language/prism-show-language.min.js", 1274cadd4f8SNickeau "sha256-Z3GTw2RIadLG7KyP/OYB+aAxVYzvg2PByKzYrJlA1EM="); 1284cadd4f8SNickeau // https://prismjs.com/plugins/command-line/ 129*04fd306cSNickeau $snippetManager->attachRemoteJavascriptLibrary( 1304cadd4f8SNickeau self::SNIPPET_NAME, 1314cadd4f8SNickeau "$BASE_PRISM_CDN/plugins/command-line/prism-command-line.min.js", 1324cadd4f8SNickeau "sha256-9WlakH0Upf3N8DDteHlbeKCHxSsljby+G9ucUCQNiU0="); 133*04fd306cSNickeau $snippetManager->attachRemoteCssStyleSheet( 134d4d4add3Sgerardnico self::SNIPPET_NAME, 135d4d4add3Sgerardnico "$BASE_PRISM_CDN/plugins/command-line/prism-command-line.css", 136d4d4add3Sgerardnico "sha256-UvoA9bIYCYQkCMTYG5p2LM8ZpJmnC4G8k0oIc89nuQA=" 137d4d4add3Sgerardnico ); 1384cadd4f8SNickeau //https://prismjs.com/plugins/line-numbers/ 139*04fd306cSNickeau $snippetManager->attachRemoteJavascriptLibrary( 1404cadd4f8SNickeau self::SNIPPET_NAME, 1414cadd4f8SNickeau "$BASE_PRISM_CDN/plugins/line-numbers/prism-line-numbers.min.js", 1424cadd4f8SNickeau "sha256-K837BwIyiXo5k/9fCYgqUyA14bN4/Ve9P2SIT0KmZD0="); 143*04fd306cSNickeau $snippetManager->attachRemoteCssStyleSheet( 144d4d4add3Sgerardnico self::SNIPPET_NAME, 145d4d4add3Sgerardnico "$BASE_PRISM_CDN/plugins/line-numbers/prism-line-numbers.css", 146d4d4add3Sgerardnico "sha256-ye8BkHf2lHXUtqZ18U0KI3xjJ1Yv7P8lvdKBt9xmVJM=" 147d4d4add3Sgerardnico ); 1484cadd4f8SNickeau 1494cadd4f8SNickeau // https://prismjs.com/plugins/download-button/--> 150*04fd306cSNickeau $snippetManager->attachRemoteJavascriptLibrary( 1514cadd4f8SNickeau self::SNIPPET_NAME, 1524cadd4f8SNickeau "$BASE_PRISM_CDN/plugins/download-button/prism-download-button.min.js", 1534cadd4f8SNickeau "sha256-CQyVQ5ejeTshlzOS/eCiry40br9f4fQ9jb5e4qPl7ZA="); 1544cadd4f8SNickeau 155d4d4add3Sgerardnico // Loading the theme 156*04fd306cSNickeau $snippetManager->attachRemoteCssStyleSheet( 157d4d4add3Sgerardnico self::SNIPPET_NAME, 158d4d4add3Sgerardnico "$BASE_PRISM_CDN/themes/$themeStyleSheet", 159d4d4add3Sgerardnico $themeIntegrity 160d4d4add3Sgerardnico ); 16137748cd8SNickeau 16237748cd8SNickeau $javascriptCode = <<<EOD 1634cadd4f8SNickeauwindow.addEventListener('load', (event) => { 16437748cd8SNickeau 16537748cd8SNickeau Prism.plugins.NormalizeWhitespace.setDefaults({ 16637748cd8SNickeau 'remove-trailing': true, 16737748cd8SNickeau 'remove-indent': true, 16837748cd8SNickeau 'left-trim': true, 16937748cd8SNickeau 'right-trim': true, 17037748cd8SNickeau }); 17137748cd8SNickeau 17237748cd8SNickeau}); 17337748cd8SNickeauEOD; 174*04fd306cSNickeau $snippetManager->attachJavascriptFromComponentId(self::SNIPPET_NAME, $javascriptCode); 17537748cd8SNickeau 17637748cd8SNickeau } 17737748cd8SNickeau 17837748cd8SNickeau /** 17937748cd8SNickeau * Add the first block of prism 18037748cd8SNickeau * @param \Doku_Renderer_xhtml $renderer 18137748cd8SNickeau * @param TagAttributes $attributes 18237748cd8SNickeau * @param \DokuWiki_Syntax_Plugin $plugin 18337748cd8SNickeau */ 18437748cd8SNickeau public static function htmlEnter(\Doku_Renderer_xhtml $renderer, \DokuWiki_Syntax_Plugin $plugin, $attributes = null) 18537748cd8SNickeau { 18637748cd8SNickeau 18737748cd8SNickeau if ($attributes == null) { 18837748cd8SNickeau $attributes = TagAttributes::createEmpty(); 18937748cd8SNickeau } 19037748cd8SNickeau 19137748cd8SNickeau /** 19237748cd8SNickeau * Display none, no rendering 19337748cd8SNickeau */ 19437748cd8SNickeau $display = $attributes->getValueAndRemove("display"); 19537748cd8SNickeau if ($display != null) { 19637748cd8SNickeau if ($display == "none") { 19737748cd8SNickeau return; 19837748cd8SNickeau } 19937748cd8SNickeau } 20037748cd8SNickeau 20137748cd8SNickeau 20237748cd8SNickeau /** 20337748cd8SNickeau * Add prism theme 20437748cd8SNickeau */ 205*04fd306cSNickeau $theme = $plugin->getConf(Prism::CONF_PRISM_THEME,Prism::PRISM_THEME_DEFAULT); 20637748cd8SNickeau Prism::addSnippet($theme); 20737748cd8SNickeau 20837748cd8SNickeau /** 20937748cd8SNickeau * Logical tag 21037748cd8SNickeau */ 21137748cd8SNickeau $logicalTag = $plugin->getPluginComponent(); 21237748cd8SNickeau if ($attributes->getLogicalTag() != null) { 21337748cd8SNickeau $logicalTag = $attributes->getLogicalTag(); 21437748cd8SNickeau } 21537748cd8SNickeau // for the https://combostrap.com/styling/userstyle 21637748cd8SNickeau $attributes->setLogicalTag($logicalTag . "-container"); 21737748cd8SNickeau 21837748cd8SNickeau /** 21937748cd8SNickeau * The child element (code) of the `pre` element 22037748cd8SNickeau * The container is the passed `attributes` 22137748cd8SNickeau * We can then constrained in height ... 22237748cd8SNickeau * It contains the language 22337748cd8SNickeau */ 22437748cd8SNickeau $codeAttributes = TagAttributes::createEmpty($logicalTag); 22537748cd8SNickeau $codeAttributes->setType($attributes->getType()); 22637748cd8SNickeau $language = $attributes->getValue(TagAttributes::TYPE_KEY); 22737748cd8SNickeau if ($language == null) { 22837748cd8SNickeau // Prism does not have any default language 22937748cd8SNickeau // There is a bug has it tried to download the txt javascript 23037748cd8SNickeau // but without language, there is no styling 23137748cd8SNickeau $language = "txt"; 23237748cd8SNickeau } else { 23337748cd8SNickeau $language = strtolower($language); 23437748cd8SNickeau Prism::addAutoloaderSnippet(); 23537748cd8SNickeau } 23637748cd8SNickeau 237*04fd306cSNickeau if (in_array($language, Tag\WebCodeTag::MARKIS)) { 23837748cd8SNickeau // Marki is not fully markdown 23937748cd8SNickeau // because it accepts space in super set html container and 24037748cd8SNickeau // prism will highlight them as indented code 24137748cd8SNickeau $language = "html"; 24237748cd8SNickeau } 24337748cd8SNickeau /** 244*04fd306cSNickeau * Language name mapping between the syntax name and prism 24537748cd8SNickeau */ 246c3437056SNickeau switch ($language) { 247c3437056SNickeau case "rsplus": 24837748cd8SNickeau $language = "r"; 249c3437056SNickeau break; 250c3437056SNickeau case "dos": 2512f24f7a8Sgerardnico case "bat": 25237748cd8SNickeau $language = "batch"; 253c3437056SNickeau break; 254*04fd306cSNickeau case "grok": 255*04fd306cSNickeau $language = "regex"; 256*04fd306cSNickeau break; 257*04fd306cSNickeau case "jinja": 258*04fd306cSNickeau // https://github.com/PrismJS/prism/issues/759 259*04fd306cSNickeau $language = "twig"; 260*04fd306cSNickeau break; 261c3437056SNickeau case "apache": 26237748cd8SNickeau $language = "apacheconf"; 263c3437056SNickeau break; 264c3437056SNickeau case "babel": 265c3437056SNickeau $language = "jsx"; 266c3437056SNickeau break; 267c3437056SNickeau case "antlr": 268c3437056SNickeau $language = "g4"; 269c3437056SNickeau break; 2702f24f7a8Sgerardnico 27137748cd8SNickeau } 27237748cd8SNickeau 27337748cd8SNickeau StringUtility::addEolCharacterIfNotPresent($renderer->doc); 27437748cd8SNickeau $codeAttributes->addClassName('language-' . $language); 27537748cd8SNickeau /** 27637748cd8SNickeau * Code element 27737748cd8SNickeau * Don't put a fucking EOL after it 27837748cd8SNickeau * Otherwise it fucked up the output as the text below a code tag is printed 27937748cd8SNickeau */ 28037748cd8SNickeau $codeHtml = $codeAttributes->toHtmlEnterTag('code'); 28137748cd8SNickeau $attributes->addHtmlAfterEnterTag($codeHtml); 28237748cd8SNickeau 28337748cd8SNickeau 28437748cd8SNickeau /** 28537748cd8SNickeau * Pre Element 28637748cd8SNickeau * Line numbers 28737748cd8SNickeau */ 28837748cd8SNickeau if ($attributes->hasComponentAttribute("line-numbers")) { 28937748cd8SNickeau $attributes->removeComponentAttribute("line-numbers"); 29037748cd8SNickeau $attributes->addClassName('line-numbers'); 29137748cd8SNickeau } 29237748cd8SNickeau 29337748cd8SNickeau 29437748cd8SNickeau // Command line 29537748cd8SNickeau if ($attributes->hasComponentAttribute("prompt")) { 29637748cd8SNickeau $attributes->addClassName("command-line"); 2974cadd4f8SNickeau $attributes->addOutputAttributeValue("data-prompt", $attributes->getValueAndRemove("prompt")); 29837748cd8SNickeau } else { 29937748cd8SNickeau switch ($language) { 30037748cd8SNickeau case "bash": 30137748cd8SNickeau $attributes->addClassName("command-line"); 3024cadd4f8SNickeau $attributes->addOutputAttributeValue("data-prompt", $plugin->getConf(self::CONF_BASH_PROMPT)); 30337748cd8SNickeau break; 30437748cd8SNickeau case "batch": 30537748cd8SNickeau $attributes->addClassName("command-line"); 30637748cd8SNickeau $batch = trim($plugin->getConf(self::CONF_BATCH_PROMPT)); 30737748cd8SNickeau if (!empty($batch)) { 30837748cd8SNickeau if (!strpos($batch, -1) == ">") { 30937748cd8SNickeau $batch .= ">"; 31037748cd8SNickeau } 31137748cd8SNickeau } 3124cadd4f8SNickeau $attributes->addOutputAttributeValue("data-prompt", $batch); 31337748cd8SNickeau break; 31437748cd8SNickeau case "powershell": 31537748cd8SNickeau $attributes->addClassName("command-line"); 31637748cd8SNickeau $powerShell = trim($plugin->getConf(self::CONF_POWERSHELL_PROMPT)); 31737748cd8SNickeau if (!empty($powerShell)) { 31837748cd8SNickeau if (!strpos($powerShell, -1) == ">") { 31937748cd8SNickeau $powerShell .= ">"; 32037748cd8SNickeau } 32137748cd8SNickeau } 3224cadd4f8SNickeau $attributes->addOutputAttributeValue("data-prompt", $powerShell); 32337748cd8SNickeau break; 32437748cd8SNickeau } 32537748cd8SNickeau } 32637748cd8SNickeau 32737748cd8SNickeau // Download 3284cadd4f8SNickeau $attributes->addOutputAttributeValue('data-download-link', true); 32937748cd8SNickeau if ($attributes->hasComponentAttribute(syntax_plugin_combo_code::FILE_PATH_KEY)) { 33037748cd8SNickeau $fileSrc = $attributes->getValueAndRemove(syntax_plugin_combo_code::FILE_PATH_KEY); 3314cadd4f8SNickeau $attributes->addOutputAttributeValue('data-src', $fileSrc); 3324cadd4f8SNickeau $attributes->addOutputAttributeValue('data-download-link-label', "Download " . $fileSrc); 33337748cd8SNickeau } else { 33437748cd8SNickeau $fileName = "file." . $language; 3354cadd4f8SNickeau $attributes->addOutputAttributeValue('data-src', $fileName); 33637748cd8SNickeau } 33737748cd8SNickeau /** 33837748cd8SNickeau * No end of line after the pre, please, otherwise we get a new line 33937748cd8SNickeau * in the code output 34037748cd8SNickeau */ 34137748cd8SNickeau $htmlCode = $attributes->toHtmlEnterTag("pre"); 34237748cd8SNickeau 34337748cd8SNickeau 34437748cd8SNickeau /** 34537748cd8SNickeau * Return 34637748cd8SNickeau */ 34737748cd8SNickeau $renderer->doc .= $htmlCode; 34837748cd8SNickeau 34937748cd8SNickeau } 35037748cd8SNickeau 35137748cd8SNickeau /** 35237748cd8SNickeau * @param Doku_Renderer_xhtml $renderer 35337748cd8SNickeau * @param TagAttributes $attributes 35437748cd8SNickeau */ 35537748cd8SNickeau public static function htmlExit(\Doku_Renderer_xhtml $renderer, $attributes = null) 35637748cd8SNickeau { 35737748cd8SNickeau 35837748cd8SNickeau if ($attributes != null) { 35937748cd8SNickeau /** 36037748cd8SNickeau * Display none, no rendering 36137748cd8SNickeau */ 36237748cd8SNickeau $display = $attributes->getValueAndRemove("display"); 36337748cd8SNickeau if ($display != null) { 36437748cd8SNickeau if ($display == "none") { 36537748cd8SNickeau return; 36637748cd8SNickeau } 36737748cd8SNickeau } 36837748cd8SNickeau } 36937748cd8SNickeau $renderer->doc .= '</code>' . DOKU_LF . '</pre>' . DOKU_LF; 37037748cd8SNickeau } 37137748cd8SNickeau 37237748cd8SNickeau /** 37337748cd8SNickeau * The autoloader try to download all language 37437748cd8SNickeau * Even the one such as txt that does not exist 37537748cd8SNickeau * This function was created to add it conditionally 37637748cd8SNickeau */ 37737748cd8SNickeau private static function addAutoloaderSnippet() 37837748cd8SNickeau { 3794cadd4f8SNickeau PluginUtility::getSnippetManager() 380*04fd306cSNickeau ->attachRemoteJavascriptLibrary( 3814cadd4f8SNickeau self::SNIPPET_ID_AUTOLOADER, 3824cadd4f8SNickeau self::BASE_PRISM_CDN . "/plugins/autoloader/prism-autoloader.min.js" 3834cadd4f8SNickeau ); 38437748cd8SNickeau } 38537748cd8SNickeau 38637748cd8SNickeau 38737748cd8SNickeau} 388