Lexer->addEntryPattern($pattern, $mode, PluginUtility::getModeFromTag($this->getPluginComponent())); } function postConnect() { $this->Lexer->addExitPattern('' . self::TAG . '>', PluginUtility::getModeFromTag($this->getPluginComponent())); } /** * * The handle function goal is to parse the matched syntax through the pattern function * and to return the result for use in the renderer * This result is always cached until the page is modified. * @param string $match * @param int $state * @param int $pos - byte position in the original source file * @param Doku_Handler $handler * @return array|bool * @see DokuWiki_Syntax_Plugin::handle() * */ function handle($match, $state, $pos, Doku_Handler $handler) { switch ($state) { case DOKU_LEXER_ENTER : $tagAttributes = TagAttributes::createFromTagMatch($match); return array( PluginUtility::STATE => $state, PluginUtility::ATTRIBUTES => $tagAttributes->toCallStackArray() ); case DOKU_LEXER_UNMATCHED : return PluginUtility::handleAndReturnUnmatchedData(self::TAG, $match, $handler); case DOKU_LEXER_EXIT : $callStack = CallStack::createFromHandler($handler); $callStack->moveToPreviousCorrespondingOpeningCall(); $bnfCode = ""; $bnfCodeFound = false; while ($actual = $callStack->next()) { if (in_array($actual->getTagName(), WebCodeTag::CODE_TAGS)) { switch ($actual->getState()) { case DOKU_LEXER_ENTER: $actualCodeType = strtolower($actual->getType()); if ($actualCodeType === 'bnf') { $bnfCodeFound = true; }; break; case DOKU_LEXER_UNMATCHED: if ($bnfCodeFound) { $bnfCode = $actual->getCapturedContent(); break 2; } break; } } } return array( PluginUtility::STATE => $state, PluginUtility::PAYLOAD => $bnfCode ); } return array(); } /** * Render the output * @param string $format * @param Doku_Renderer $renderer * @param array $data - what the function handle() return'ed * @return boolean - rendered correctly? (however, returned value is not used at the moment) * @see DokuWiki_Syntax_Plugin::render() * * */ function render($format, Doku_Renderer $renderer, $data) { if ($format == 'xhtml') { /** @var Doku_Renderer_xhtml $renderer */ $state = $data [PluginUtility::STATE]; switch ($state) { case DOKU_LEXER_ENTER : break; case DOKU_LEXER_UNMATCHED : $renderer->doc .= PluginUtility::renderUnmatched($data); break; case DOKU_LEXER_EXIT : $bnfCode = $data[PluginUtility::PAYLOAD]; if (!empty($bnfCode)) { $snippetManager = PluginUtility::getSnippetManager(); $snippetId = self::TAG; $libraryId = "rrdiagram"; $snippetManager->attachCssInternalStyleSheet($snippetId); $snippetManager->attachJavascriptFromComponentId($snippetId); /** * * Calculation * ` * openssl dgst -sha256 -binary rrdiagram.js | openssl base64 -A * ` * $sha256integrity = ; */ $snippetManager->attachJavascriptComboResourceForSlot( $snippetId, "library:$libraryId:0.9.4.1:$libraryId.js", "sha256-" . "noP8Tag5vKjRfh3+8GXy5QSZqKnRt7WQe6I9rGVl+go=" ); /** * This code is replaced at runtime by the diagram */ $class = self::CLASS_NAME; $renderer->doc .= "
" . hsc($bnfCode) . ""; } else { LogUtility::msg("No code component with bnf grammar was found", LogUtility::LVL_MSG_WARNING, self::CANONICAL); } break; } return true; } // unsupported $mode return false; } }