2014-07-28 19:11:15 +02:00
|
|
|
<?php
|
2015-02-04 10:46:36 +01:00
|
|
|
/* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
|
2014-07-28 19:11:15 +02:00
|
|
|
|
|
|
|
namespace Icinga\Module\Doc;
|
|
|
|
|
2014-11-14 11:50:56 +01:00
|
|
|
require_once 'Parsedown/Parsedown.php';
|
2014-07-28 19:11:15 +02:00
|
|
|
|
2014-07-29 11:10:06 +02:00
|
|
|
use DOMDocument;
|
|
|
|
use DOMXPath;
|
2014-07-28 19:11:15 +02:00
|
|
|
use RecursiveIteratorIterator;
|
|
|
|
use Parsedown;
|
|
|
|
use Zend_View_Helper_Url;
|
2014-07-29 11:10:06 +02:00
|
|
|
use Icinga\Module\Doc\Exception\ChapterNotFoundException;
|
|
|
|
use Icinga\Web\Url;
|
2014-07-28 19:11:15 +02:00
|
|
|
use Icinga\Web\View;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* preg_replace_callback helper to replace links
|
|
|
|
*/
|
|
|
|
class Callback
|
|
|
|
{
|
|
|
|
protected $docTree;
|
|
|
|
|
|
|
|
protected $view;
|
|
|
|
|
|
|
|
protected $zendUrlHelper;
|
|
|
|
|
|
|
|
protected $url;
|
|
|
|
|
|
|
|
protected $urlParams;
|
|
|
|
|
|
|
|
public function __construct(
|
|
|
|
DocTree $docTree,
|
|
|
|
View $view,
|
|
|
|
Zend_View_Helper_Url $zendUrlHelper,
|
|
|
|
$url,
|
|
|
|
array $urlParams)
|
|
|
|
{
|
|
|
|
$this->docTree = $docTree;
|
|
|
|
$this->view = $view;
|
|
|
|
$this->zendUrlHelper = $zendUrlHelper;
|
|
|
|
$this->url = $url;
|
|
|
|
$this->urlParams = $urlParams;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function render($match)
|
|
|
|
{
|
|
|
|
$node = $this->docTree->getNode(Renderer::decodeAnchor($match['fragment']));
|
|
|
|
/* @var $node \Icinga\Data\Tree\Node */
|
|
|
|
if ($node === null) {
|
|
|
|
return $match[0];
|
|
|
|
}
|
|
|
|
$section = $node->getValue();
|
|
|
|
/* @var $section \Icinga\Module\Doc\Section */
|
|
|
|
$path = $this->zendUrlHelper->url(
|
|
|
|
array_merge(
|
|
|
|
$this->urlParams,
|
|
|
|
array(
|
2014-08-19 09:45:53 +02:00
|
|
|
'chapterId' => SectionRenderer::encodeUrlParam($section->getChapterId())
|
2014-07-28 19:11:15 +02:00
|
|
|
)
|
|
|
|
),
|
|
|
|
$this->url,
|
|
|
|
false,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
$url = $this->view->url($path);
|
|
|
|
$url->setAnchor(SectionRenderer::encodeAnchor($section->getId()));
|
|
|
|
return sprintf(
|
|
|
|
'<a %s%shref="%s"',
|
|
|
|
strlen($match['attribs']) ? trim($match['attribs']) . ' ' : '',
|
2014-08-19 09:57:22 +02:00
|
|
|
$section->isNoFollow() ? 'rel="nofollow" ' : '',
|
2014-07-28 19:11:15 +02:00
|
|
|
$url->getAbsoluteUrl()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Section renderer
|
|
|
|
*/
|
|
|
|
class SectionRenderer extends Renderer
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* The documentation tree
|
|
|
|
*
|
|
|
|
* @var DocTree
|
|
|
|
*/
|
|
|
|
protected $docTree;
|
|
|
|
|
2014-08-19 11:30:56 +02:00
|
|
|
protected $tocUrl;
|
|
|
|
|
2014-07-28 19:11:15 +02:00
|
|
|
/**
|
|
|
|
* The URL to replace links with
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $url;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Additional URL parameters
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $urlParams;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parsedown instance
|
|
|
|
*
|
|
|
|
* @var Parsedown
|
|
|
|
*/
|
|
|
|
protected $parsedown;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Content
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $content = array();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new section renderer
|
|
|
|
*
|
|
|
|
* @param DocTree $docTree The documentation tree
|
2014-08-19 09:45:53 +02:00
|
|
|
* @param string|null $chapterId If not null, the chapter ID to filter for
|
2014-08-19 11:30:56 +02:00
|
|
|
* @param string $tocUrl
|
2014-07-28 19:11:15 +02:00
|
|
|
* @param string $url The URL to replace links with
|
|
|
|
* @param array $urlParams Additional URL parameters
|
|
|
|
*
|
|
|
|
* @throws ChapterNotFoundException If the chapter to filter for was not found
|
|
|
|
*/
|
2014-08-19 11:30:56 +02:00
|
|
|
public function __construct(DocTree $docTree, $chapterId, $tocUrl, $url, array $urlParams)
|
2014-07-28 19:11:15 +02:00
|
|
|
{
|
2014-08-19 09:45:53 +02:00
|
|
|
if ($chapterId !== null) {
|
|
|
|
$filter = new SectionFilterIterator($docTree, $chapterId);
|
2014-07-28 19:11:15 +02:00
|
|
|
if ($filter->count() === 0) {
|
|
|
|
throw new ChapterNotFoundException(
|
2014-08-19 16:22:22 +02:00
|
|
|
sprintf(mt('doc', 'Chapter \'%s\' not found'), $chapterId)
|
2014-07-28 19:11:15 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
parent::__construct(
|
|
|
|
$filter,
|
|
|
|
RecursiveIteratorIterator::SELF_FIRST
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
parent::__construct($docTree, RecursiveIteratorIterator::SELF_FIRST);
|
|
|
|
}
|
|
|
|
$this->docTree = $docTree;
|
2014-08-19 11:30:56 +02:00
|
|
|
$this->tocUrl = $tocUrl;
|
2014-07-28 19:11:15 +02:00
|
|
|
$this->url = $url;
|
|
|
|
$this->urlParams = array_map(array($this, 'encodeUrlParam'), $urlParams);
|
|
|
|
$this->parsedown = Parsedown::instance();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Syntax highlighting for PHP code
|
|
|
|
*
|
|
|
|
* @param $match
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function highlightPhp($match)
|
|
|
|
{
|
|
|
|
return '<pre>' . highlight_string(htmlspecialchars_decode($match[1]), true) . '</pre>';
|
|
|
|
}
|
|
|
|
|
2014-07-29 11:10:06 +02:00
|
|
|
/**
|
|
|
|
* 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);
|
|
|
|
/* @var $img \DOMElement */
|
|
|
|
$img->setAttribute('src', Url::fromPath($img->getAttribute('src'))->getAbsoluteUrl());
|
|
|
|
return substr_replace($doc->saveXML($img), '', -2, 1); // Replace '/>' with '>'
|
|
|
|
}
|
|
|
|
|
2014-11-20 15:29:46 +01:00
|
|
|
protected function blubb($match)
|
|
|
|
{
|
|
|
|
$doc = new DOMDocument();
|
|
|
|
$doc->loadHTML($match[0]);
|
|
|
|
$xpath = new DOMXPath($doc);
|
|
|
|
$blockquote = $xpath->query('//blockquote[1]')->item(0);
|
|
|
|
/* @var $blockquote \DOMElement */
|
|
|
|
if (strtolower(substr(trim($blockquote->nodeValue), 0, 5)) === 'note:') {
|
|
|
|
$blockquote->setAttribute('class', 'note');
|
|
|
|
}
|
|
|
|
return $doc->saveXML($blockquote);
|
|
|
|
}
|
|
|
|
|
2014-07-28 19:11:15 +02:00
|
|
|
/**
|
|
|
|
* Render the section
|
|
|
|
*
|
|
|
|
* @param View $view
|
|
|
|
* @param Zend_View_Helper_Url $zendUrlHelper
|
2014-08-19 11:30:56 +02:00
|
|
|
* @param bool $renderNavigation
|
|
|
|
*
|
2014-07-28 19:11:15 +02:00
|
|
|
* @return string
|
|
|
|
*/
|
2014-08-19 11:30:56 +02:00
|
|
|
public function render(View $view, Zend_View_Helper_Url $zendUrlHelper, $renderNavigation = true)
|
2014-07-28 19:11:15 +02:00
|
|
|
{
|
|
|
|
$callback = new Callback($this->docTree, $view, $zendUrlHelper, $this->url, $this->urlParams);
|
|
|
|
$content = array();
|
|
|
|
foreach ($this as $node) {
|
|
|
|
$section = $node->getValue();
|
|
|
|
/* @var $section \Icinga\Module\Doc\Section */
|
|
|
|
$content[] = sprintf(
|
2014-08-19 11:30:56 +02:00
|
|
|
'<a name="%1$s"></a><h%2$d>%3$s</h%2$d>',
|
2014-07-28 19:11:15 +02:00
|
|
|
Renderer::encodeAnchor($section->getId()),
|
|
|
|
$section->getLevel(),
|
|
|
|
$view->escape($section->getTitle())
|
|
|
|
);
|
|
|
|
$html = preg_replace_callback(
|
|
|
|
'#<pre><code class="language-php">(.*?)</code></pre>#s',
|
|
|
|
array($this, 'highlightPhp'),
|
|
|
|
$this->parsedown->text(implode('', $section->getContent()))
|
|
|
|
);
|
2014-07-29 11:10:06 +02:00
|
|
|
$html = preg_replace_callback(
|
|
|
|
'/<img[^>]+>/',
|
|
|
|
array($this, 'replaceImg'),
|
|
|
|
$html
|
|
|
|
);
|
2014-11-20 15:29:46 +01:00
|
|
|
$html = preg_replace_callback(
|
|
|
|
'#<blockquote>.+</blockquote>#ms',
|
|
|
|
array($this, 'blubb'),
|
|
|
|
$html
|
|
|
|
);
|
2014-07-28 19:11:15 +02:00
|
|
|
$content[] = preg_replace_callback(
|
2015-01-23 16:00:15 +01:00
|
|
|
'/<a\s+(?P<attribs>[^>]*?\s+)?href="(?:(?!http:\/\/)[^#]*)#(?P<fragment>[^"]+)"/',
|
2014-07-28 19:11:15 +02:00
|
|
|
array($callback, 'render'),
|
|
|
|
$html
|
|
|
|
);
|
|
|
|
}
|
2014-08-19 11:30:56 +02:00
|
|
|
if ($renderNavigation) {
|
|
|
|
foreach ($this->docTree as $chapter) {
|
|
|
|
if ($chapter->getValue()->getId() === $section->getChapterId()) {
|
2014-08-19 13:38:18 +02:00
|
|
|
$navigation = array('<ul class="navigation">');
|
2014-08-19 11:30:56 +02:00
|
|
|
$this->docTree->prev();
|
|
|
|
$prev = $this->docTree->current();
|
|
|
|
if ($prev !== null) {
|
|
|
|
$prev = $prev->getValue();
|
|
|
|
$path = $zendUrlHelper->url(
|
|
|
|
array_merge(
|
|
|
|
$this->urlParams,
|
|
|
|
array(
|
|
|
|
'chapterId' => $this->encodeUrlParam($prev->getChapterId())
|
|
|
|
)
|
|
|
|
),
|
|
|
|
$this->url,
|
|
|
|
false,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
$url = $view->url($path);
|
|
|
|
$url->setAnchor($this->encodeAnchor($prev->getId()));
|
2014-08-19 13:38:18 +02:00
|
|
|
$navigation[] = sprintf(
|
|
|
|
'<li class="prev"><a %shref="%s">%s</a></li>',
|
2014-08-19 11:30:56 +02:00
|
|
|
$prev->isNoFollow() ? 'rel="nofollow" ' : '',
|
|
|
|
$url->getAbsoluteUrl(),
|
|
|
|
$view->escape($prev->getTitle())
|
|
|
|
);
|
|
|
|
$this->docTree->next();
|
|
|
|
$this->docTree->next();
|
|
|
|
} else {
|
|
|
|
$this->docTree->rewind();
|
|
|
|
$this->docTree->next();
|
|
|
|
}
|
|
|
|
$url = $view->url($this->tocUrl);
|
2014-08-19 13:38:18 +02:00
|
|
|
$navigation[] = sprintf(
|
2014-08-19 11:30:56 +02:00
|
|
|
'<li><a href="%s">%s</a></li>',
|
|
|
|
$url->getAbsoluteUrl(),
|
|
|
|
mt('doc', 'Index')
|
|
|
|
);
|
|
|
|
$next = $this->docTree->current();
|
|
|
|
if ($next !== null) {
|
|
|
|
$next = $next->getValue();
|
|
|
|
$path = $zendUrlHelper->url(
|
|
|
|
array_merge(
|
|
|
|
$this->urlParams,
|
|
|
|
array(
|
|
|
|
'chapterId' => $this->encodeUrlParam($next->getChapterId())
|
|
|
|
)
|
|
|
|
),
|
|
|
|
$this->url,
|
|
|
|
false,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
$url = $view->url($path);
|
|
|
|
$url->setAnchor($this->encodeAnchor($next->getId()));
|
2014-08-19 13:38:18 +02:00
|
|
|
$navigation[] = sprintf(
|
|
|
|
'<li class="next"><a %shref="%s">%s</a></li>',
|
2014-08-19 11:30:56 +02:00
|
|
|
$next->isNoFollow() ? 'rel="nofollow" ' : '',
|
|
|
|
$url->getAbsoluteUrl(),
|
|
|
|
$view->escape($next->getTitle())
|
|
|
|
);
|
|
|
|
}
|
2014-08-19 13:38:18 +02:00
|
|
|
$navigation[] = '</ul>';
|
|
|
|
$content = array_merge($navigation, $content, $navigation);
|
2014-08-19 11:30:56 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-07-28 19:11:15 +02:00
|
|
|
return implode("\n", $content);
|
|
|
|
}
|
|
|
|
}
|