xref: /plugin/combo/ComboStrap/Prism.php (revision 04fd306c7c155fa133ebb3669986875d65988276)
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