getIterator(), $chapter); if ($filter->isEmpty()) { throw new ChapterNotFoundException( mt('doc', 'Chapter %s not found'), $chapter ); } parent::__construct( $filter, RecursiveIteratorIterator::SELF_FIRST ); } else { parent::__construct($tree->getIterator(), RecursiveIteratorIterator::SELF_FIRST); } $this->tree = $tree; $this->parsedown = Parsedown::instance(); } /** * Set the search criteria to highlight * * @param string $highlightSearch * * @return $this */ public function setHighlightSearch($highlightSearch) { $this->highlightSearch = $highlightSearch; return $this; } /** * Get the search criteria to highlight * * @return string */ public function getHighlightSearch() { return $this->highlightSearch; } /** * Syntax highlighting for PHP code * * @param array $match * * @return string */ protected function highlightPhp($match) { return '
' . highlight_string(htmlspecialchars_decode($match[1]), true) . '
'; } /** * Highlight search criteria * * @param string $html * @param DocSearch $search Search criteria * * @return string */ protected function highlightSearch($html, DocSearch $search) { $doc = new DOMDocument(); $fragment = $doc->createDocumentFragment(); $fragment->appendXML($html); $doc->appendChild($fragment); $iter = new RecursiveIteratorIterator(new DomNodeIterator($doc), RecursiveIteratorIterator::SELF_FIRST); foreach ($iter as $node) { if ($node->nodeType !== XML_TEXT_NODE || ($node->parentNode->nodeType === XML_ELEMENT_NODE && $node->parentNode->tagName === 'code') ) { continue; } $text = $node->nodeValue; if (($match = $search->search($text)) === null) { continue; } $matches = $match->getMatches(); ksort($matches); $offset = 0; $fragment = $doc->createDocumentFragment(); foreach ($matches as $position => $match) { $fragment->appendChild($doc->createTextNode(substr($text, $offset, $position - $offset))); $fragment->appendChild($doc->createElement('span', $match)) ->setAttribute('class', DocSearchMatch::HIGHLIGHT_CSS_CLASS); $offset = $position + strlen($match); } $fragment->appendChild($doc->createTextNode(substr($text, $offset))); $node->parentNode->replaceChild($fragment, $node); } return $doc->saveHTML(); } /** * Markup notes * * @param array $match * * @return string */ protected function markupNotes($match) { $doc = new DOMDocument(); $doc->loadHTML($match[0]); $xpath = new DOMXPath($doc); $blockquote = $xpath->query('//blockquote[1]')->item(0); /** @type \DOMElement $blockquote */ if (strtolower(substr(trim($blockquote->nodeValue), 0, 5)) === 'note:') { $blockquote->setAttribute('class', 'note'); } return $doc->saveXML($blockquote); } /** * Replace img src tags * * @param $match * * @return string */ protected function replaceImg($match) { $doc = new DOMDocument(); $doc->loadHTML($match[0]); $xpath = new DOMXPath($doc); $img = $xpath->query('//img[1]')->item(0); /** @type \DOMElement $img */ $img->setAttribute('src', Url::fromPath($img->getAttribute('src'))->getAbsoluteUrl()); return substr_replace($doc->saveXML($img), '', -2, 1); // Replace '/>' with '>' } /** * Replace link * * @param array $match * * @return string */ protected function replaceLink($match) { if (($section = $this->tree->getNode($this->decodeAnchor($match['fragment']))) === null) { return $match[0]; } /** @type \Icinga\Module\Doc\DocSection $section */ $path = $this->getView()->getHelper('Url')->url( array_merge( $this->urlParams, array( 'chapter' => $this->encodeUrlParam($section->getChapter()->getId()) ) ), $this->url, false, false ); $url = $this->getView()->url($path); /** @type \Icinga\Web\Url $url */ $url->setAnchor($this->encodeAnchor($section->getId())); return sprintf( 'getNoFollow() ? 'rel="nofollow" ' : '', $url->getAbsoluteUrl() ); } /** * {@inheritdoc} */ public function render() { $search = null; if (($highlightSearch = $this->getHighlightSearch()) !== null) { $search = new DocSearch($highlightSearch); } foreach ($this as $section) { $title = $section->getTitle(); if ($search !== null && ($match = $search->search($title)) !== null) { $title = $match->highlight(); } else { $title = $this->getView()->escape($title); } $this->content[] = sprintf( '%3$s', static::encodeAnchor($section->getId()), $section->getLevel(), $title ); $content = $this->parsedown->text(implode('', $section->getContent())); if (empty($content)) { continue; } $html = preg_replace_callback( '#
(.*?)
#s', array($this, 'highlightPhp'), $content ); $html = preg_replace_callback( '/]+>/', array($this, 'replaceImg'), $html ); $html = preg_replace_callback( '#
.+
#ms', array($this, 'markupNotes'), $html ); $html = preg_replace_callback( '/[^>]*?\s+)?href="(?:(?!http:\/\/)[^#]*)#(?P[^"]+)"/', array($this, 'replaceLink'), $html ); if ($search !== null) { $html = $this->highlightSearch($html, $search); } $this->content[] = $html; } return implode("\n", $this->content); } }