elements
*/
class Links extends AbstractHelper
{
/**
* Constants used for specifying which link types to find and render
*
* @var int
*/
const RENDER_ALTERNATE = 0x0001;
const RENDER_STYLESHEET = 0x0002;
const RENDER_START = 0x0004;
const RENDER_NEXT = 0x0008;
const RENDER_PREV = 0x0010;
const RENDER_CONTENTS = 0x0020;
const RENDER_INDEX = 0x0040;
const RENDER_GLOSSARY = 0x0080;
const RENDER_COPYRIGHT = 0x0100;
const RENDER_CHAPTER = 0x0200;
const RENDER_SECTION = 0x0400;
const RENDER_SUBSECTION = 0x0800;
const RENDER_APPENDIX = 0x1000;
const RENDER_HELP = 0x2000;
const RENDER_BOOKMARK = 0x4000;
const RENDER_CUSTOM = 0x8000;
const RENDER_ALL = 0xffff;
/**
* Maps render constants to W3C link types
*
* @var array
*/
protected static $RELATIONS = [
self::RENDER_ALTERNATE => 'alternate',
self::RENDER_STYLESHEET => 'stylesheet',
self::RENDER_START => 'start',
self::RENDER_NEXT => 'next',
self::RENDER_PREV => 'prev',
self::RENDER_CONTENTS => 'contents',
self::RENDER_INDEX => 'index',
self::RENDER_GLOSSARY => 'glossary',
self::RENDER_COPYRIGHT => 'copyright',
self::RENDER_CHAPTER => 'chapter',
self::RENDER_SECTION => 'section',
self::RENDER_SUBSECTION => 'subsection',
self::RENDER_APPENDIX => 'appendix',
self::RENDER_HELP => 'help',
self::RENDER_BOOKMARK => 'bookmark',
];
/**
* The helper's render flag
*
* @see render()
* @see setRenderFlag()
* @var int
*/
protected $renderFlag = self::RENDER_ALL;
/**
* Root container
*
* Used for preventing methods to traverse above the container given to
* the {@link render()} method.
*
* @see _findRoot()
* @var AbstractContainer
*/
protected $root;
/**
* Helper entry point
*
* @param string|AbstractContainer $container container to operate on
* @return Links
*/
public function __invoke($container = null)
{
if (null !== $container) {
$this->setContainer($container);
}
return $this;
}
/**
* Magic overload: Proxy calls to {@link findRelation()} or container
*
* Examples of finder calls:
*
* // METHOD // SAME AS
* $h->findRelNext($page); // $h->findRelation($page, 'rel', 'next')
* $h->findRevSection($page); // $h->findRelation($page, 'rev', 'section');
* $h->findRelFoo($page); // $h->findRelation($page, 'rel', 'foo');
*
*
* @param string $method
* @param array $arguments
* @return mixed
* @throws Exception\ExceptionInterface
*/
public function __call($method, array $arguments = [])
{
ErrorHandler::start(E_WARNING);
$result = preg_match('/find(Rel|Rev)(.+)/', $method, $match);
ErrorHandler::stop();
if ($result) {
return $this->findRelation($arguments[0], strtolower($match[1]), strtolower($match[2]));
}
return parent::__call($method, $arguments);
}
/**
* Renders helper
*
* Implements {@link HelperInterface::render()}.
*
* @param AbstractContainer|string|null $container [optional] container to render.
* Default is to render the
* container registered in the
* helper.
* @return string
*/
public function render($container = null)
{
$this->parseContainer($container);
if (null === $container) {
$container = $this->getContainer();
}
$active = $this->findActive($container);
if ($active) {
$active = $active['page'];
} else {
// no active page
return '';
}
$output = '';
$indent = $this->getIndent();
$this->root = $container;
$result = $this->findAllRelations($active, $this->getRenderFlag());
foreach ($result as $attrib => $types) {
foreach ($types as $relation => $pages) {
foreach ($pages as $page) {
$r = $this->renderLink($page, $attrib, $relation);
if ($r) {
$output .= $indent . $r . PHP_EOL;
}
}
}
}
$this->root = null;
// return output (trim last newline by spec)
return strlen($output) ? rtrim($output, PHP_EOL) : '';
}
/**
* Renders the given $page as a link element, with $attrib = $relation
*
* @param AbstractPage $page the page to render the link for
* @param string $attrib the attribute to use for $type,
* either 'rel' or 'rev'
* @param string $relation relation type, muse be one of;
* alternate, appendix, bookmark,
* chapter, contents, copyright,
* glossary, help, home, index, next,
* prev, section, start, stylesheet,
* subsection
* @return string
* @throws Exception\DomainException
*/
public function renderLink(AbstractPage $page, $attrib, $relation)
{
if (! in_array($attrib, ['rel', 'rev'])) {
throw new Exception\DomainException(sprintf(
'Invalid relation attribute "%s", must be "rel" or "rev"',
$attrib
));
}
if (! $href = $page->getHref()) {
return '';
}
// TODO: add more attribs
// http://www.w3.org/TR/html401/struct/links.html#h-12.2
$attribs = [
$attrib => $relation,
'href' => $href,
'title' => $page->getLabel()
];
return 'htmlAttribs($attribs) .
$this->getClosingBracket();
}
// Finder methods:
/**
* Finds all relations (forward and reverse) for the given $page
*
* The form of the returned array:
*
* // $page denotes an instance of Zend\Navigation\Page\AbstractPage
* $returned = array(
* 'rel' => array(
* 'alternate' => array($page, $page, $page),
* 'start' => array($page),
* 'next' => array($page),
* 'prev' => array($page),
* 'canonical' => array($page)
* ),
* 'rev' => array(
* 'section' => array($page)
* )
* );
*
*
* @param AbstractPage $page page to find links for
* @param null|int
* @return array
*/
public function findAllRelations(AbstractPage $page, $flag = null)
{
if (! is_int($flag)) {
$flag = self::RENDER_ALL;
}
$result = ['rel' => [], 'rev' => []];
$native = array_values(static::$RELATIONS);
foreach (array_keys($result) as $rel) {
$meth = 'getDefined' . ucfirst($rel);
$types = array_merge($native, array_diff($page->$meth(), $native));
foreach ($types as $type) {
if (! $relFlag = array_search($type, static::$RELATIONS)) {
$relFlag = self::RENDER_CUSTOM;
}
if (! ($flag & $relFlag)) {
continue;
}
$found = $this->findRelation($page, $rel, $type);
if ($found) {
if (! is_array($found)) {
$found = [$found];
}
$result[$rel][$type] = $found;
}
}
}
return $result;
}
/**
* Finds relations of the given $rel=$type from $page
*
* This method will first look for relations in the page instance, then
* by searching the root container if nothing was found in the page.
*
* @param AbstractPage $page page to find relations for
* @param string $rel relation, "rel" or "rev"
* @param string $type link type, e.g. 'start', 'next'
* @return AbstractPage|array|null
* @throws Exception\DomainException if $rel is not "rel" or "rev"
*/
public function findRelation(AbstractPage $page, $rel, $type)
{
if (! in_array($rel, ['rel', 'rev'])) {
throw new Exception\DomainException(sprintf(
'Invalid argument: $rel must be "rel" or "rev"; "%s" given',
$rel
));
}
if (! $result = $this->findFromProperty($page, $rel, $type)) {
$result = $this->findFromSearch($page, $rel, $type);
}
return $result;
}
/**
* Finds relations of given $type for $page by checking if the
* relation is specified as a property of $page
*
* @param AbstractPage $page page to find relations for
* @param string $rel relation, 'rel' or 'rev'
* @param string $type link type, e.g. 'start', 'next'
* @return AbstractPage|array|null
*/
protected function findFromProperty(AbstractPage $page, $rel, $type)
{
$method = 'get' . ucfirst($rel);
$result = $page->$method($type);
if ($result) {
$result = $this->convertToPages($result);
if ($result) {
if (! is_array($result)) {
$result = [$result];
}
foreach ($result as $key => $page) {
if (! $this->accept($page)) {
unset($result[$key]);
}
}
return count($result) == 1 ? $result[0] : $result;
}
}
return;
}
/**
* Finds relations of given $rel=$type for $page by using the helper to
* search for the relation in the root container
*
* @param AbstractPage $page page to find relations for
* @param string $rel relation, 'rel' or 'rev'
* @param string $type link type, e.g. 'start', 'next', etc
* @return array|null
*/
protected function findFromSearch(AbstractPage $page, $rel, $type)
{
$found = null;
$method = 'search' . ucfirst($rel) . ucfirst($type);
if (method_exists($this, $method)) {
$found = $this->$method($page);
}
return $found;
}
// Search methods:
/**
* Searches the root container for the forward 'start' relation of the given
* $page
*
* From {@link http://www.w3.org/TR/html4/types.html#type-links}:
* Refers to the first document in a collection of documents. This link type
* tells search engines which document is considered by the author to be the
* starting point of the collection.
*
* @param AbstractPage $page
* @return AbstractPage|null
*/
public function searchRelStart(AbstractPage $page)
{
$found = $this->findRoot($page);
if (! $found instanceof AbstractPage) {
$found->rewind();
$found = $found->current();
}
if ($found === $page || ! $this->accept($found)) {
$found = null;
}
return $found;
}
/**
* Searches the root container for the forward 'next' relation of the given
* $page
*
* From {@link http://www.w3.org/TR/html4/types.html#type-links}:
* Refers to the next document in a linear sequence of documents. User
* agents may choose to preload the "next" document, to reduce the perceived
* load time.
*
* @param AbstractPage $page
* @return AbstractPage|null
*/
public function searchRelNext(AbstractPage $page)
{
$found = null;
$break = false;
$iterator = new RecursiveIteratorIterator($this->findRoot($page), RecursiveIteratorIterator::SELF_FIRST);
foreach ($iterator as $intermediate) {
if ($intermediate === $page) {
// current page; break at next accepted page
$break = true;
continue;
}
if ($break && $this->accept($intermediate)) {
$found = $intermediate;
break;
}
}
return $found;
}
/**
* Searches the root container for the forward 'prev' relation of the given
* $page
*
* From {@link http://www.w3.org/TR/html4/types.html#type-links}:
* Refers to the previous document in an ordered series of documents. Some
* user agents also support the synonym "Previous".
*
* @param AbstractPage $page
* @return AbstractPage|null
*/
public function searchRelPrev(AbstractPage $page)
{
$found = null;
$prev = null;
$iterator = new RecursiveIteratorIterator(
$this->findRoot($page),
RecursiveIteratorIterator::SELF_FIRST
);
foreach ($iterator as $intermediate) {
if (! $this->accept($intermediate)) {
continue;
}
if ($intermediate === $page) {
$found = $prev;
break;
}
$prev = $intermediate;
}
return $found;
}
/**
* Searches the root container for forward 'chapter' relations of the given
* $page
*
* From {@link http://www.w3.org/TR/html4/types.html#type-links}:
* Refers to a document serving as a chapter in a collection of documents.
*
* @param AbstractPage $page
* @return AbstractPage|array|null
*/
public function searchRelChapter(AbstractPage $page)
{
$found = [];
// find first level of pages
$root = $this->findRoot($page);
// find start page(s)
$start = $this->findRelation($page, 'rel', 'start');
if (! is_array($start)) {
$start = [$start];
}
foreach ($root as $chapter) {
// exclude self and start page from chapters
if ($chapter !== $page &&
! in_array($chapter, $start) &&
$this->accept($chapter)) {
$found[] = $chapter;
}
}
switch (count($found)) {
case 0:
return;
case 1:
return $found[0];
default:
return $found;
}
}
/**
* Searches the root container for forward 'section' relations of the given
* $page
*
* From {@link http://www.w3.org/TR/html4/types.html#type-links}:
* Refers to a document serving as a section in a collection of documents.
*
* @param AbstractPage $page
* @return AbstractPage|array|null
*/
public function searchRelSection(AbstractPage $page)
{
$found = [];
// check if given page has pages and is a chapter page
if ($page->hasPages() && $this->findRoot($page)->hasPage($page)) {
foreach ($page as $section) {
if ($this->accept($section)) {
$found[] = $section;
}
}
}
switch (count($found)) {
case 0:
return;
case 1:
return $found[0];
default:
return $found;
}
}
/**
* Searches the root container for forward 'subsection' relations of the
* given $page
*
* From {@link http://www.w3.org/TR/html4/types.html#type-links}:
* Refers to a document serving as a subsection in a collection of
* documents.
*
* @param AbstractPage $page
* @return AbstractPage|array|null
*/
public function searchRelSubsection(AbstractPage $page)
{
$found = [];
if ($page->hasPages()) {
// given page has child pages, loop chapters
foreach ($this->findRoot($page) as $chapter) {
// is page a section?
if ($chapter->hasPage($page)) {
foreach ($page as $subsection) {
if ($this->accept($subsection)) {
$found[] = $subsection;
}
}
}
}
}
switch (count($found)) {
case 0:
return;
case 1:
return $found[0];
default:
return $found;
}
}
/**
* Searches the root container for the reverse 'section' relation of the
* given $page
*
* From {@link http://www.w3.org/TR/html4/types.html#type-links}:
* Refers to a document serving as a section in a collection of documents.
*
* @param AbstractPage $page
* @return AbstractPage|null
*/
public function searchRevSection(AbstractPage $page)
{
$found = null;
$parent = $page->getParent();
if ($parent) {
if ($parent instanceof AbstractPage &&
$this->findRoot($page)->hasPage($parent)) {
$found = $parent;
}
}
return $found;
}
/**
* Searches the root container for the reverse 'section' relation of the
* given $page
*
* From {@link http://www.w3.org/TR/html4/types.html#type-links}:
* Refers to a document serving as a subsection in a collection of
* documents.
*
* @param AbstractPage $page
* @return AbstractPage|null
*/
public function searchRevSubsection(AbstractPage $page)
{
$found = null;
$parent = $page->getParent();
if ($parent) {
if ($parent instanceof AbstractPage) {
$root = $this->findRoot($page);
foreach ($root as $chapter) {
if ($chapter->hasPage($parent)) {
$found = $parent;
break;
}
}
}
}
return $found;
}
// Util methods:
/**
* Returns the root container of the given page
*
* When rendering a container, the render method still store the given
* container as the root container, and unset it when done rendering. This
* makes sure finder methods will not traverse above the container given
* to the render method.
*
* @param AbstractPage $page
* @return AbstractContainer
*/
protected function findRoot(AbstractPage $page)
{
if ($this->root) {
return $this->root;
}
$root = $page;
while ($parent = $page->getParent()) {
$root = $parent;
if ($parent instanceof AbstractPage) {
$page = $parent;
} else {
break;
}
}
return $root;
}
/**
* Converts a $mixed value to an array of pages
*
* @param mixed $mixed mixed value to get page(s) from
* @param bool $recursive whether $value should be looped
* if it is an array or a config
* @return AbstractPage|array|null
*/
protected function convertToPages($mixed, $recursive = true)
{
if ($mixed instanceof AbstractPage) {
// value is a page instance; return directly
return $mixed;
} elseif ($mixed instanceof AbstractContainer) {
// value is a container; return pages in it
$pages = [];
foreach ($mixed as $page) {
$pages[] = $page;
}
return $pages;
} elseif ($mixed instanceof Traversable) {
$mixed = ArrayUtils::iteratorToArray($mixed);
} elseif (is_string($mixed)) {
// value is a string; make a URI page
return AbstractPage::factory([
'type' => 'uri',
'uri' => $mixed
]);
}
if (is_array($mixed) && ! empty($mixed)) {
if ($recursive && is_numeric(key($mixed))) {
// first key is numeric; assume several pages
$pages = [];
foreach ($mixed as $value) {
$value = $this->convertToPages($value, false);
if ($value) {
$pages[] = $value;
}
}
return $pages;
} else {
// pass array to factory directly
try {
$page = AbstractPage::factory($mixed);
return $page;
} catch (\Exception $e) {
}
}
}
// nothing found
return;
}
/**
* Sets the helper's render flag
*
* The helper uses the bitwise '&' operator against the hex values of the
* render constants. This means that the flag can is "bitwised" value of
* the render constants. Examples:
*
* // render all links except glossary
* $flag = Links:RENDER_ALL ^ Links:RENDER_GLOSSARY;
* $helper->setRenderFlag($flag);
*
* // render only chapters and sections
* $flag = Links:RENDER_CHAPTER | Links:RENDER_SECTION;
* $helper->setRenderFlag($flag);
*
* // render only relations that are not native W3C relations
* $helper->setRenderFlag(Links:RENDER_CUSTOM);
*
* // render all relations (default)
* $helper->setRenderFlag(Links:RENDER_ALL);
*
*
* Note that custom relations can also be rendered directly using the
* {@link renderLink()} method.
*
* @param int $renderFlag
* @return Links
*/
public function setRenderFlag($renderFlag)
{
$this->renderFlag = (int) $renderFlag;
return $this;
}
/**
* Returns the helper's render flag
*
* @return int
*/
public function getRenderFlag()
{
return $this->renderFlag;
}
}