xref: /plugin/combo/ComboStrap/Prism.php (revision d4d4add3f8894284a98648d77172277cb5cae9fa)
137748cd8SNickeau<?php
237748cd8SNickeau
337748cd8SNickeaunamespace ComboStrap;
437748cd8SNickeau
537748cd8SNickeau
637748cd8SNickeauuse Doku_Renderer_xhtml;
737748cd8SNickeauuse syntax_plugin_combo_code;
837748cd8SNickeau
937748cd8SNickeauclass Prism
1037748cd8SNickeau{
1137748cd8SNickeau
1237748cd8SNickeau    const SNIPPET_NAME = 'prism';
1337748cd8SNickeau    /**
1437748cd8SNickeau     * The class used to mark the added prism code
1537748cd8SNickeau     */
161fa8c418SNickeau    const BASE_PRISM_CDN = "https://cdnjs.cloudflare.com/ajax/libs/prism/1.25.0";
1737748cd8SNickeau    /**
1837748cd8SNickeau     * The default prompt for bash
1937748cd8SNickeau     */
2037748cd8SNickeau    const CONF_BASH_PROMPT = "bashPrompt";
2137748cd8SNickeau    /**
2237748cd8SNickeau     * The default prompt for batch (dos)
2337748cd8SNickeau     */
2437748cd8SNickeau    const CONF_BATCH_PROMPT = "batchPrompt";
2537748cd8SNickeau    /**
2637748cd8SNickeau     * The default prompt for powershell
2737748cd8SNickeau     */
2837748cd8SNickeau    const CONF_POWERSHELL_PROMPT = "powershellPrompt";
2937748cd8SNickeau
3037748cd8SNickeau    /**
3137748cd8SNickeau     * The default name of prism
3237748cd8SNickeau     * It does not follow the naming of the theming
3337748cd8SNickeau     */
3437748cd8SNickeau    const PRISM_THEME = "prism";
3537748cd8SNickeau
3637748cd8SNickeau    /**
371fa8c418SNickeau     * @var string[] https://cdnjs.cloudflare.com/ajax/libs/prism/1.25.0/themes/prism-{theme}.min.css
3837748cd8SNickeau     *
3937748cd8SNickeau     * or default
4037748cd8SNickeau     *
411fa8c418SNickeau     * https://cdnjs.cloudflare.com/ajax/libs/prism/1.25.0/themes/prism.min.css
4237748cd8SNickeau     *
4337748cd8SNickeau     * or
4437748cd8SNickeau     *
4537748cd8SNickeau     * https://github.com/PrismJS/prism-themes
4637748cd8SNickeau     *
4737748cd8SNickeau     * from https://cdnjs.com/libraries/prism
4837748cd8SNickeau     */
4937748cd8SNickeau    const THEMES_INTEGRITY = [
501fa8c418SNickeau        Prism::PRISM_THEME => "sha256-ko4j5rn874LF8dHwW29/xabhh8YBleWfvxb8nQce4Fc=",
51cb4149a8Sgerardnico        "coy" => "sha256-gkHLZLptZZHaBY+jqrRkAVzOGfMa4HBhSCJteem8wy8=",
52cb4149a8Sgerardnico        "dark" => "sha256-l+VX6V333ll/PXrjqG1W6DyZvDEw+50M7aAP6dcD7Qc=",
53cb4149a8Sgerardnico        "funky" => "sha256-l9GTgvTMmAvPQ6IlNCd/I2FQwXVlJCLbGId7z6QlOpo=",
54cb4149a8Sgerardnico        "okaidia" => "sha256-zzHVEO0xOoVm0I6bT9v5SgpRs1cYNyvEvHXW/1yCgqU=",
55cb4149a8Sgerardnico        "solarizedlight" => "sha256-Lr49DyE+/KstnLdBxqZBoDYgNi6ONfZyAZw3LDhxB9I=",
56cb4149a8Sgerardnico        "tomorrow" => "sha256-GxX+KXGZigSK67YPJvbu12EiBx257zuZWr0AMiT1Kpg=",
57cb4149a8Sgerardnico        "twilight" => "sha256-R7PF7y9XAuz19FB93NgH/WQUVGk30iytl7EwtETrypo="
5837748cd8SNickeau    ];
5937748cd8SNickeau
6037748cd8SNickeau    /**
6137748cd8SNickeau     * The theme
6237748cd8SNickeau     */
6337748cd8SNickeau    const CONF_PRISM_THEME = "prismTheme";
6437748cd8SNickeau    const PRISM_THEME_DEFAULT = "tomorrow";
6537748cd8SNickeau    const SNIPPET_ID_AUTOLOADER = self::SNIPPET_NAME . "-autoloader";
6637748cd8SNickeau
6737748cd8SNickeau
6837748cd8SNickeau    /**
6937748cd8SNickeau     *
7037748cd8SNickeau     * @param $theme
7137748cd8SNickeau     *
7237748cd8SNickeau     * Ter info: The theme of the default wiki is in the print.css file (search for code blocks)
7337748cd8SNickeau     */
7437748cd8SNickeau    public static function addSnippet($theme)
7537748cd8SNickeau    {
7637748cd8SNickeau        $BASE_PRISM_CDN = self::BASE_PRISM_CDN;
7737748cd8SNickeau
7837748cd8SNickeau        if ($theme == self::PRISM_THEME) {
7937748cd8SNickeau            $themeStyleSheet = "prism.min.css";
8037748cd8SNickeau        } else {
8137748cd8SNickeau            $themeStyleSheet = "prism-$theme.min.css";
8237748cd8SNickeau        }
8337748cd8SNickeau        $themeIntegrity = self::THEMES_INTEGRITY[$theme];
8437748cd8SNickeau
8537748cd8SNickeau        /**
8637748cd8SNickeau         * We miss a bottom margin
8737748cd8SNickeau         * as a paragraph
8837748cd8SNickeau         */
894cadd4f8SNickeau        $snippetManager = PluginUtility::getSnippetManager();
904cadd4f8SNickeau        $snippetManager->attachCssInternalStyleSheetForSlot(self::SNIPPET_NAME);
9137748cd8SNickeau
9237748cd8SNickeau        /**
9337748cd8SNickeau         * Javascript
9437748cd8SNickeau         */
954cadd4f8SNickeau        $snippetManager->attachJavascriptLibraryForSlot(
964cadd4f8SNickeau            self::SNIPPET_NAME,
974cadd4f8SNickeau            "$BASE_PRISM_CDN/components/prism-core.min.js",
984cadd4f8SNickeau            "sha256-vlRYHThwdq55dA+n1BKQRzzLwFtH9VINdSI68+5JhpU=");
994cadd4f8SNickeau        $snippetManager->attachJavascriptLibraryForSlot(
1004cadd4f8SNickeau            self::SNIPPET_NAME,
1014cadd4f8SNickeau            "$BASE_PRISM_CDN/plugins/toolbar/prism-toolbar.min.js",
1024cadd4f8SNickeau            "sha256-FyIVdIHL0+ppj4Q4Ft05K3wyCsYikpHIDGI7dcaBalU="
1031fa8c418SNickeau        );
104*d4d4add3Sgerardnico        $snippetManager->attachCssExternalStyleSheetForSlot(
105*d4d4add3Sgerardnico            self::SNIPPET_NAME,
106*d4d4add3Sgerardnico            "$BASE_PRISM_CDN/plugins/toolbar/prism-toolbar.css",
107*d4d4add3Sgerardnico            "sha256-kK4/JIYJUKI4Zdg9ZQ7FYyRIqeWPfYKi5QZHO2n/lJI="
108*d4d4add3Sgerardnico        );
10937748cd8SNickeau        // https://prismjs.com/plugins/normalize-whitespace/
1104cadd4f8SNickeau        $snippetManager->attachJavascriptLibraryForSlot(
1114cadd4f8SNickeau            self::SNIPPET_NAME,
1124cadd4f8SNickeau            "$BASE_PRISM_CDN/plugins/normalize-whitespace/prism-normalize-whitespace.min.js",
1134cadd4f8SNickeau            "sha256-gBzABGbXfQYYnyr8xmDFjx6KGO9dBYuypG1QBjO76pY=");
114*d4d4add3Sgerardnico        // https://prismjs.com/plugins/copy-to-clipboard/
115*d4d4add3Sgerardnico        $snippetManager->attachJavascriptLibraryForSlot(
116*d4d4add3Sgerardnico            self::SNIPPET_NAME,
117*d4d4add3Sgerardnico            "$BASE_PRISM_CDN/plugins/copy-to-clipboard/prism-copy-to-clipboard.min.js",
118*d4d4add3Sgerardnico            "sha512-pUNGXbOrc+Y3dm5z2ZN7JYQ/2Tq0jppMDOUsN4sQHVJ9AUQpaeERCUfYYBAnaRB9r8d4gtPKMWICNhm3tRr4Fg==");
1194cadd4f8SNickeau        // https://prismjs.com/plugins/show-language/
1204cadd4f8SNickeau        $snippetManager->attachJavascriptLibraryForSlot(
1214cadd4f8SNickeau            self::SNIPPET_NAME,
1224cadd4f8SNickeau            "$BASE_PRISM_CDN/plugins/show-language/prism-show-language.min.js",
1234cadd4f8SNickeau            "sha256-Z3GTw2RIadLG7KyP/OYB+aAxVYzvg2PByKzYrJlA1EM=");
1244cadd4f8SNickeau        // https://prismjs.com/plugins/command-line/
1254cadd4f8SNickeau        $snippetManager->attachJavascriptLibraryForSlot(
1264cadd4f8SNickeau            self::SNIPPET_NAME,
1274cadd4f8SNickeau            "$BASE_PRISM_CDN/plugins/command-line/prism-command-line.min.js",
1284cadd4f8SNickeau            "sha256-9WlakH0Upf3N8DDteHlbeKCHxSsljby+G9ucUCQNiU0=");
129*d4d4add3Sgerardnico        $snippetManager->attachCssExternalStyleSheetForSlot(
130*d4d4add3Sgerardnico            self::SNIPPET_NAME,
131*d4d4add3Sgerardnico            "$BASE_PRISM_CDN/plugins/command-line/prism-command-line.css",
132*d4d4add3Sgerardnico            "sha256-UvoA9bIYCYQkCMTYG5p2LM8ZpJmnC4G8k0oIc89nuQA="
133*d4d4add3Sgerardnico        );
1344cadd4f8SNickeau        //https://prismjs.com/plugins/line-numbers/
1354cadd4f8SNickeau        $snippetManager->attachJavascriptLibraryForSlot(
1364cadd4f8SNickeau            self::SNIPPET_NAME,
1374cadd4f8SNickeau            "$BASE_PRISM_CDN/plugins/line-numbers/prism-line-numbers.min.js",
1384cadd4f8SNickeau            "sha256-K837BwIyiXo5k/9fCYgqUyA14bN4/Ve9P2SIT0KmZD0=");
139*d4d4add3Sgerardnico        $snippetManager->attachCssExternalStyleSheetForSlot(
140*d4d4add3Sgerardnico            self::SNIPPET_NAME,
141*d4d4add3Sgerardnico            "$BASE_PRISM_CDN/plugins/line-numbers/prism-line-numbers.css",
142*d4d4add3Sgerardnico            "sha256-ye8BkHf2lHXUtqZ18U0KI3xjJ1Yv7P8lvdKBt9xmVJM="
143*d4d4add3Sgerardnico        );
1444cadd4f8SNickeau
1454cadd4f8SNickeau        // https://prismjs.com/plugins/download-button/-->
1464cadd4f8SNickeau        $snippetManager->attachJavascriptLibraryForSlot(
1474cadd4f8SNickeau            self::SNIPPET_NAME,
1484cadd4f8SNickeau            "$BASE_PRISM_CDN/plugins/download-button/prism-download-button.min.js",
1494cadd4f8SNickeau            "sha256-CQyVQ5ejeTshlzOS/eCiry40br9f4fQ9jb5e4qPl7ZA=");
1504cadd4f8SNickeau
151*d4d4add3Sgerardnico        // Loading the theme
152*d4d4add3Sgerardnico        $snippetManager->attachCssExternalStyleSheetForSlot(
153*d4d4add3Sgerardnico            self::SNIPPET_NAME,
154*d4d4add3Sgerardnico            "$BASE_PRISM_CDN/themes/$themeStyleSheet",
155*d4d4add3Sgerardnico            $themeIntegrity
156*d4d4add3Sgerardnico        );
15737748cd8SNickeau
15837748cd8SNickeau        $javascriptCode = <<<EOD
1594cadd4f8SNickeauwindow.addEventListener('load', (event) => {
16037748cd8SNickeau
16137748cd8SNickeau    Prism.plugins.NormalizeWhitespace.setDefaults({
16237748cd8SNickeau        'remove-trailing': true,
16337748cd8SNickeau        'remove-indent': true,
16437748cd8SNickeau        'left-trim': true,
16537748cd8SNickeau        'right-trim': true,
16637748cd8SNickeau    });
16737748cd8SNickeau
16837748cd8SNickeau});
16937748cd8SNickeauEOD;
1704cadd4f8SNickeau        $snippetManager->attachInternalJavascriptForSlot(self::SNIPPET_NAME, $javascriptCode);
17137748cd8SNickeau
17237748cd8SNickeau    }
17337748cd8SNickeau
17437748cd8SNickeau    /**
17537748cd8SNickeau     * Add the first block of prism
17637748cd8SNickeau     * @param \Doku_Renderer_xhtml $renderer
17737748cd8SNickeau     * @param TagAttributes $attributes
17837748cd8SNickeau     * @param \DokuWiki_Syntax_Plugin $plugin
17937748cd8SNickeau     */
18037748cd8SNickeau    public static function htmlEnter(\Doku_Renderer_xhtml $renderer, \DokuWiki_Syntax_Plugin $plugin, $attributes = null)
18137748cd8SNickeau    {
18237748cd8SNickeau
18337748cd8SNickeau        if ($attributes == null) {
18437748cd8SNickeau            $attributes = TagAttributes::createEmpty();
18537748cd8SNickeau        }
18637748cd8SNickeau
18737748cd8SNickeau        /**
18837748cd8SNickeau         * Display none, no rendering
18937748cd8SNickeau         */
19037748cd8SNickeau        $display = $attributes->getValueAndRemove("display");
19137748cd8SNickeau        if ($display != null) {
19237748cd8SNickeau            if ($display == "none") {
19337748cd8SNickeau                return;
19437748cd8SNickeau            }
19537748cd8SNickeau        }
19637748cd8SNickeau
19737748cd8SNickeau
19837748cd8SNickeau        /**
19937748cd8SNickeau         * Add prism theme
20037748cd8SNickeau         */
20137748cd8SNickeau        $theme = $plugin->getConf(Prism::CONF_PRISM_THEME);
20237748cd8SNickeau        Prism::addSnippet($theme);
20337748cd8SNickeau
20437748cd8SNickeau        /**
20537748cd8SNickeau         * Logical tag
20637748cd8SNickeau         */
20737748cd8SNickeau        $logicalTag = $plugin->getPluginComponent();
20837748cd8SNickeau        if ($attributes->getLogicalTag() != null) {
20937748cd8SNickeau            $logicalTag = $attributes->getLogicalTag();
21037748cd8SNickeau        }
21137748cd8SNickeau        // for the https://combostrap.com/styling/userstyle
21237748cd8SNickeau        $attributes->setLogicalTag($logicalTag . "-container");
21337748cd8SNickeau
21437748cd8SNickeau        /**
21537748cd8SNickeau         * The child element (code) of the `pre` element
21637748cd8SNickeau         * The container is the passed `attributes`
21737748cd8SNickeau         * We can then constrained in height ...
21837748cd8SNickeau         * It contains the language
21937748cd8SNickeau         */
22037748cd8SNickeau        $codeAttributes = TagAttributes::createEmpty($logicalTag);
22137748cd8SNickeau        $codeAttributes->setType($attributes->getType());
22237748cd8SNickeau        $language = $attributes->getValue(TagAttributes::TYPE_KEY);
22337748cd8SNickeau        if ($language == null) {
22437748cd8SNickeau            // Prism does not have any default language
22537748cd8SNickeau            // There is a bug has it tried to download the txt javascript
22637748cd8SNickeau            // but without language, there is no styling
22737748cd8SNickeau            $language = "txt";
22837748cd8SNickeau        } else {
22937748cd8SNickeau            $language = strtolower($language);
23037748cd8SNickeau            Prism::addAutoloaderSnippet();
23137748cd8SNickeau        }
23237748cd8SNickeau
23337748cd8SNickeau        if (in_array($language, \syntax_plugin_combo_webcode::MARKIS)) {
23437748cd8SNickeau            // Marki is not fully markdown
23537748cd8SNickeau            // because it accepts space in super set html container and
23637748cd8SNickeau            // prism will highlight them as indented code
23737748cd8SNickeau            $language = "html";
23837748cd8SNickeau        }
23937748cd8SNickeau        /**
24037748cd8SNickeau         * Language name mapping between the dokuwiki default
24137748cd8SNickeau         * and prism
24237748cd8SNickeau         */
243c3437056SNickeau        switch ($language) {
244c3437056SNickeau            case "rsplus":
24537748cd8SNickeau                $language = "r";
246c3437056SNickeau                break;
247c3437056SNickeau            case "dos":
2482f24f7a8Sgerardnico            case "bat":
24937748cd8SNickeau                $language = "batch";
250c3437056SNickeau                break;
251c3437056SNickeau            case "apache":
25237748cd8SNickeau                $language = "apacheconf";
253c3437056SNickeau                break;
254c3437056SNickeau            case "babel":
255c3437056SNickeau                $language = "jsx";
256c3437056SNickeau                break;
257c3437056SNickeau            case "antlr":
258c3437056SNickeau                $language = "g4";
259c3437056SNickeau                break;
2602f24f7a8Sgerardnico
26137748cd8SNickeau        }
26237748cd8SNickeau
26337748cd8SNickeau        StringUtility::addEolCharacterIfNotPresent($renderer->doc);
26437748cd8SNickeau        $codeAttributes->addClassName('language-' . $language);
26537748cd8SNickeau        /**
26637748cd8SNickeau         * Code element
26737748cd8SNickeau         * Don't put a fucking EOL after it
26837748cd8SNickeau         * Otherwise it fucked up the output as the text below a code tag is printed
26937748cd8SNickeau         */
27037748cd8SNickeau        $codeHtml = $codeAttributes->toHtmlEnterTag('code');
27137748cd8SNickeau        $attributes->addHtmlAfterEnterTag($codeHtml);
27237748cd8SNickeau
27337748cd8SNickeau
27437748cd8SNickeau        /**
27537748cd8SNickeau         * Pre Element
27637748cd8SNickeau         * Line numbers
27737748cd8SNickeau         */
27837748cd8SNickeau        if ($attributes->hasComponentAttribute("line-numbers")) {
27937748cd8SNickeau            $attributes->removeComponentAttribute("line-numbers");
28037748cd8SNickeau            $attributes->addClassName('line-numbers');
28137748cd8SNickeau        }
28237748cd8SNickeau
28337748cd8SNickeau
28437748cd8SNickeau        // Command line
28537748cd8SNickeau        if ($attributes->hasComponentAttribute("prompt")) {
28637748cd8SNickeau            $attributes->addClassName("command-line");
2874cadd4f8SNickeau            $attributes->addOutputAttributeValue("data-prompt", $attributes->getValueAndRemove("prompt"));
28837748cd8SNickeau        } else {
28937748cd8SNickeau            switch ($language) {
29037748cd8SNickeau                case "bash":
29137748cd8SNickeau                    $attributes->addClassName("command-line");
2924cadd4f8SNickeau                    $attributes->addOutputAttributeValue("data-prompt", $plugin->getConf(self::CONF_BASH_PROMPT));
29337748cd8SNickeau                    break;
29437748cd8SNickeau                case "batch":
29537748cd8SNickeau                    $attributes->addClassName("command-line");
29637748cd8SNickeau                    $batch = trim($plugin->getConf(self::CONF_BATCH_PROMPT));
29737748cd8SNickeau                    if (!empty($batch)) {
29837748cd8SNickeau                        if (!strpos($batch, -1) == ">") {
29937748cd8SNickeau                            $batch .= ">";
30037748cd8SNickeau                        }
30137748cd8SNickeau                    }
3024cadd4f8SNickeau                    $attributes->addOutputAttributeValue("data-prompt", $batch);
30337748cd8SNickeau                    break;
30437748cd8SNickeau                case "powershell":
30537748cd8SNickeau                    $attributes->addClassName("command-line");
30637748cd8SNickeau                    $powerShell = trim($plugin->getConf(self::CONF_POWERSHELL_PROMPT));
30737748cd8SNickeau                    if (!empty($powerShell)) {
30837748cd8SNickeau                        if (!strpos($powerShell, -1) == ">") {
30937748cd8SNickeau                            $powerShell .= ">";
31037748cd8SNickeau                        }
31137748cd8SNickeau                    }
3124cadd4f8SNickeau                    $attributes->addOutputAttributeValue("data-prompt", $powerShell);
31337748cd8SNickeau                    break;
31437748cd8SNickeau            }
31537748cd8SNickeau        }
31637748cd8SNickeau
31737748cd8SNickeau        // Download
3184cadd4f8SNickeau        $attributes->addOutputAttributeValue('data-download-link', true);
31937748cd8SNickeau        if ($attributes->hasComponentAttribute(syntax_plugin_combo_code::FILE_PATH_KEY)) {
32037748cd8SNickeau            $fileSrc = $attributes->getValueAndRemove(syntax_plugin_combo_code::FILE_PATH_KEY);
3214cadd4f8SNickeau            $attributes->addOutputAttributeValue('data-src', $fileSrc);
3224cadd4f8SNickeau            $attributes->addOutputAttributeValue('data-download-link-label', "Download " . $fileSrc);
32337748cd8SNickeau        } else {
32437748cd8SNickeau            $fileName = "file." . $language;
3254cadd4f8SNickeau            $attributes->addOutputAttributeValue('data-src', $fileName);
32637748cd8SNickeau        }
32737748cd8SNickeau        /**
32837748cd8SNickeau         * No end of line after the pre, please, otherwise we get a new line
32937748cd8SNickeau         * in the code output
33037748cd8SNickeau         */
33137748cd8SNickeau        $htmlCode = $attributes->toHtmlEnterTag("pre");
33237748cd8SNickeau
33337748cd8SNickeau
33437748cd8SNickeau        /**
33537748cd8SNickeau         * Return
33637748cd8SNickeau         */
33737748cd8SNickeau        $renderer->doc .= $htmlCode;
33837748cd8SNickeau
33937748cd8SNickeau    }
34037748cd8SNickeau
34137748cd8SNickeau    /**
34237748cd8SNickeau     * @param Doku_Renderer_xhtml $renderer
34337748cd8SNickeau     * @param TagAttributes $attributes
34437748cd8SNickeau     */
34537748cd8SNickeau    public static function htmlExit(\Doku_Renderer_xhtml $renderer, $attributes = null)
34637748cd8SNickeau    {
34737748cd8SNickeau
34837748cd8SNickeau        if ($attributes != null) {
34937748cd8SNickeau            /**
35037748cd8SNickeau             * Display none, no rendering
35137748cd8SNickeau             */
35237748cd8SNickeau            $display = $attributes->getValueAndRemove("display");
35337748cd8SNickeau            if ($display != null) {
35437748cd8SNickeau                if ($display == "none") {
35537748cd8SNickeau                    return;
35637748cd8SNickeau                }
35737748cd8SNickeau            }
35837748cd8SNickeau        }
35937748cd8SNickeau        $renderer->doc .= '</code>' . DOKU_LF . '</pre>' . DOKU_LF;
36037748cd8SNickeau    }
36137748cd8SNickeau
36237748cd8SNickeau    /**
36337748cd8SNickeau     * The autoloader try to download all language
36437748cd8SNickeau     * Even the one such as txt that does not exist
36537748cd8SNickeau     * This function was created to add it conditionally
36637748cd8SNickeau     */
36737748cd8SNickeau    private static function addAutoloaderSnippet()
36837748cd8SNickeau    {
3694cadd4f8SNickeau        PluginUtility::getSnippetManager()
3704cadd4f8SNickeau            ->attachJavascriptLibraryForSlot(
3714cadd4f8SNickeau                self::SNIPPET_ID_AUTOLOADER,
3724cadd4f8SNickeau                self::BASE_PRISM_CDN . "/plugins/autoloader/prism-autoloader.min.js"
3734cadd4f8SNickeau            );
37437748cd8SNickeau    }
37537748cd8SNickeau
37637748cd8SNickeau
37737748cd8SNickeau}
378