<?php /** * Zend Framework * * LICENSE * * This source file is subject to the new BSD license that is bundled * with this package in the file LICENSE.txt. * It is also available through the world-wide-web at this URL: * http://framework.zend.com/license/new-bsd * If you did not receive a copy of the license and are unable to * obtain it through the world-wide-web, please send an email * to license@zend.com so we can send you a copy immediately. * * @category Zend * @package Zend_View * @subpackage Helper * @copyright Copyright (c) 2005-2011 Zend Technologies USA Inc. (http://www.zend.com) * @license http://framework.zend.com/license/new-bsd New BSD License * @version $Id$ */ /** * @see Zend_View_Helper_Navigation_Helper */ require_once 'Zend/View/Helper/Navigation/Helper.php'; /** * @see Zend_View_Helper_HtmlElement */ require_once 'Zend/View/Helper/HtmlElement.php'; /** * Base class for navigational helpers * * @category Zend * @package Zend_View * @subpackage Helper * @copyright Copyright (c) 2005-2011 Zend Technologies USA Inc. (http://www.zend.com) * @license http://framework.zend.com/license/new-bsd New BSD License */ abstract class Zend_View_Helper_Navigation_HelperAbstract extends Zend_View_Helper_HtmlElement implements Zend_View_Helper_Navigation_Helper { /** * Container to operate on by default * * @var Zend_Navigation_Container */ protected $_container; /** * The minimum depth a page must have to be included when rendering * * @var int */ protected $_minDepth; /** * The maximum depth a page can have to be included when rendering * * @var int */ protected $_maxDepth; /** * Indentation string * * @var string */ protected $_indent = ''; /** * Translator * * @var Zend_Translate_Adapter */ protected $_translator; /** * ACL to use when iterating pages * * @var Zend_Acl */ protected $_acl; /** * Wheter invisible items should be rendered by this helper * * @var bool */ protected $_renderInvisible = false; /** * ACL role to use when iterating pages * * @var string|Zend_Acl_Role_Interface */ protected $_role; /** * Whether translator should be used for page labels and titles * * @var bool */ protected $_useTranslator = true; /** * Whether ACL should be used for filtering out pages * * @var bool */ protected $_useAcl = true; /** * Default ACL to use when iterating pages if not explicitly set in the * instance by calling {@link setAcl()} * * @var Zend_Acl */ protected static $_defaultAcl; /** * Default ACL role to use when iterating pages if not explicitly set in the * instance by calling {@link setRole()} * * @var string|Zend_Acl_Role_Interface */ protected static $_defaultRole; // Accessors: /** * Sets navigation container the helper operates on by default * * Implements {@link Zend_View_Helper_Navigation_Interface::setContainer()}. * * @param Zend_Navigation_Container $container [optional] container * to operate on. * Default is null, * meaning container * will be reset. * @return Zend_View_Helper_Navigation_HelperAbstract fluent interface, * returns self */ public function setContainer(Zend_Navigation_Container $container = null) { $this->_container = $container; return $this; } /** * Returns the navigation container helper operates on by default * * Implements {@link Zend_View_Helper_Navigation_Interface::getContainer()}. * * If a helper is not explicitly set in this helper instance by calling * {@link setContainer()} or by passing it through the helper entry point, * this method will look in {@link Zend_Registry} for a container by using * the key 'Zend_Navigation'. * * If no container is set, and nothing is found in Zend_Registry, a new * container will be instantiated and stored in the helper. * * @return Zend_Navigation_Container navigation container */ public function getContainer() { if (null === $this->_container) { // try to fetch from registry first require_once 'Zend/Registry.php'; if (Zend_Registry::isRegistered('Zend_Navigation')) { $nav = Zend_Registry::get('Zend_Navigation'); if ($nav instanceof Zend_Navigation_Container) { return $this->_container = $nav; } } // nothing found in registry, create new container require_once 'Zend/Navigation.php'; $this->_container = new Zend_Navigation(); } return $this->_container; } /** * Sets the minimum depth a page must have to be included when rendering * * @param int $minDepth [optional] minimum * depth. Default is * null, which sets * no minimum depth. * @return Zend_View_Helper_Navigation_HelperAbstract fluent interface, * returns self */ public function setMinDepth($minDepth = null) { if (null === $minDepth || is_int($minDepth)) { $this->_minDepth = $minDepth; } else { $this->_minDepth = (int) $minDepth; } return $this; } /** * Returns minimum depth a page must have to be included when rendering * * @return int|null minimum depth or null */ public function getMinDepth() { if (!is_int($this->_minDepth) || $this->_minDepth < 0) { return 0; } return $this->_minDepth; } /** * Sets the maximum depth a page can have to be included when rendering * * @param int $maxDepth [optional] maximum * depth. Default is * null, which sets no * maximum depth. * @return Zend_View_Helper_Navigation_HelperAbstract fluent interface, * returns self */ public function setMaxDepth($maxDepth = null) { if (null === $maxDepth || is_int($maxDepth)) { $this->_maxDepth = $maxDepth; } else { $this->_maxDepth = (int) $maxDepth; } return $this; } /** * Returns maximum depth a page can have to be included when rendering * * @return int|null maximum depth or null */ public function getMaxDepth() { return $this->_maxDepth; } /** * Set the indentation string for using in {@link render()}, optionally a * number of spaces to indent with * * @param string|int $indent indentation string or * number of spaces * @return Zend_View_Helper_Navigation_HelperAbstract fluent interface, * returns self */ public function setIndent($indent) { $this->_indent = $this->_getWhitespace($indent); return $this; } /** * Returns indentation * * @return string */ public function getIndent() { return $this->_indent; } /** * Sets translator to use in helper * * Implements {@link Zend_View_Helper_Navigation_Helper::setTranslator()}. * * @param mixed $translator [optional] translator. * Expects an object of * type * {@link Zend_Translate_Adapter} * or {@link Zend_Translate}, * or null. Default is * null, which sets no * translator. * @return Zend_View_Helper_Navigation_HelperAbstract fluent interface, * returns self */ public function setTranslator($translator = null) { if (null == $translator || $translator instanceof Zend_Translate_Adapter) { $this->_translator = $translator; } elseif ($translator instanceof Zend_Translate) { $this->_translator = $translator->getAdapter(); } return $this; } /** * Returns translator used in helper * * Implements {@link Zend_View_Helper_Navigation_Helper::getTranslator()}. * * @return Zend_Translate_Adapter|null translator or null */ public function getTranslator() { if (null === $this->_translator) { require_once 'Zend/Registry.php'; if (Zend_Registry::isRegistered('Zend_Translate')) { $this->setTranslator(Zend_Registry::get('Zend_Translate')); } } return $this->_translator; } /** * Sets ACL to use when iterating pages * * Implements {@link Zend_View_Helper_Navigation_Helper::setAcl()}. * * @param Zend_Acl $acl [optional] ACL object. * Default is null. * @return Zend_View_Helper_Navigation_HelperAbstract fluent interface, * returns self */ public function setAcl(Zend_Acl $acl = null) { $this->_acl = $acl; return $this; } /** * Returns ACL or null if it isn't set using {@link setAcl()} or * {@link setDefaultAcl()} * * Implements {@link Zend_View_Helper_Navigation_Helper::getAcl()}. * * @return Zend_Acl|null ACL object or null */ public function getAcl() { if ($this->_acl === null && self::$_defaultAcl !== null) { return self::$_defaultAcl; } return $this->_acl; } /** * Sets ACL role(s) to use when iterating pages * * Implements {@link Zend_View_Helper_Navigation_Helper::setRole()}. * * @param mixed $role [optional] role to * set. Expects a string, * an instance of type * {@link Zend_Acl_Role_Interface}, * or null. Default is * null, which will set * no role. * @throws Zend_View_Exception if $role is invalid * @return Zend_View_Helper_Navigation_HelperAbstract fluent interface, * returns self */ public function setRole($role = null) { if (null === $role || is_string($role) || $role instanceof Zend_Acl_Role_Interface) { $this->_role = $role; } else { require_once 'Zend/View/Exception.php'; $e = new Zend_View_Exception(sprintf( '$role must be a string, null, or an instance of ' . 'Zend_Acl_Role_Interface; %s given', gettype($role) )); $e->setView($this->view); throw $e; } return $this; } /** * Returns ACL role to use when iterating pages, or null if it isn't set * using {@link setRole()} or {@link setDefaultRole()} * * Implements {@link Zend_View_Helper_Navigation_Helper::getRole()}. * * @return string|Zend_Acl_Role_Interface|null role or null */ public function getRole() { if ($this->_role === null && self::$_defaultRole !== null) { return self::$_defaultRole; } return $this->_role; } /** * Sets whether ACL should be used * * Implements {@link Zend_View_Helper_Navigation_Helper::setUseAcl()}. * * @param bool $useAcl [optional] whether ACL * should be used. * Default is true. * @return Zend_View_Helper_Navigation_HelperAbstract fluent interface, * returns self */ public function setUseAcl($useAcl = true) { $this->_useAcl = (bool) $useAcl; return $this; } /** * Returns whether ACL should be used * * Implements {@link Zend_View_Helper_Navigation_Helper::getUseAcl()}. * * @return bool whether ACL should be used */ public function getUseAcl() { return $this->_useAcl; } /** * Return renderInvisible flag * * @return bool */ public function getRenderInvisible() { return $this->_renderInvisible; } /** * Render invisible items? * * @param bool $renderInvisible [optional] boolean flag * @return Zend_View_Helper_Navigation_HelperAbstract fluent interface * returns self */ public function setRenderInvisible($renderInvisible = true) { $this->_renderInvisible = (bool) $renderInvisible; return $this; } /** * Sets whether translator should be used * * Implements {@link Zend_View_Helper_Navigation_Helper::setUseTranslator()}. * * @param bool $useTranslator [optional] whether * translator should be * used. Default is true. * @return Zend_View_Helper_Navigation_HelperAbstract fluent interface, * returns self */ public function setUseTranslator($useTranslator = true) { $this->_useTranslator = (bool) $useTranslator; return $this; } /** * Returns whether translator should be used * * Implements {@link Zend_View_Helper_Navigation_Helper::getUseTranslator()}. * * @return bool whether translator should be used */ public function getUseTranslator() { return $this->_useTranslator; } // Magic overloads: /** * Magic overload: Proxy calls to the navigation container * * @param string $method method name in container * @param array $arguments [optional] arguments to pass * @return mixed returns what the container returns * @throws Zend_Navigation_Exception if method does not exist in container */ public function __call($method, array $arguments = array()) { return call_user_func_array( array($this->getContainer(), $method), $arguments); } /** * Magic overload: Proxy to {@link render()}. * * This method will trigger an E_USER_ERROR if rendering the helper causes * an exception to be thrown. * * Implements {@link Zend_View_Helper_Navigation_Helper::__toString()}. * * @return string */ public function __toString() { try { return $this->render(); } catch (Exception $e) { $msg = get_class($e) . ': ' . $e->getMessage(); trigger_error($msg, E_USER_ERROR); return ''; } } // Public methods: /** * Finds the deepest active page in the given container * * @param Zend_Navigation_Container $container container to search * @param int|null $minDepth [optional] minimum depth * required for page to be * valid. Default is to use * {@link getMinDepth()}. A * null value means no minimum * depth required. * @param int|null $minDepth [optional] maximum depth * a page can have to be * valid. Default is to use * {@link getMaxDepth()}. A * null value means no maximum * depth required. * @return array an associative array with * the values 'depth' and * 'page', or an empty array * if not found */ public function findActive(Zend_Navigation_Container $container, $minDepth = null, $maxDepth = -1) { if (!is_int($minDepth)) { $minDepth = $this->getMinDepth(); } if ((!is_int($maxDepth) || $maxDepth < 0) && null !== $maxDepth) { $maxDepth = $this->getMaxDepth(); } $found = null; $foundDepth = -1; $iterator = new RecursiveIteratorIterator($container, RecursiveIteratorIterator::CHILD_FIRST); foreach ($iterator as $page) { $currDepth = $iterator->getDepth(); if ($currDepth < $minDepth || !$this->accept($page)) { // page is not accepted continue; } if ($page->isActive(false) && $currDepth > $foundDepth) { // found an active page at a deeper level than before $found = $page; $foundDepth = $currDepth; } } if (is_int($maxDepth) && $foundDepth > $maxDepth) { while ($foundDepth > $maxDepth) { if (--$foundDepth < $minDepth) { $found = null; break; } $found = $found->getParent(); if (!$found instanceof Zend_Navigation_Page) { $found = null; break; } } } if ($found) { return array('page' => $found, 'depth' => $foundDepth); } else { return array(); } } /** * Checks if the helper has a container * * Implements {@link Zend_View_Helper_Navigation_Helper::hasContainer()}. * * @return bool whether the helper has a container or not */ public function hasContainer() { return null !== $this->_container; } /** * Checks if the helper has an ACL instance * * Implements {@link Zend_View_Helper_Navigation_Helper::hasAcl()}. * * @return bool whether the helper has a an ACL instance or not */ public function hasAcl() { return null !== $this->_acl; } /** * Checks if the helper has an ACL role * * Implements {@link Zend_View_Helper_Navigation_Helper::hasRole()}. * * @return bool whether the helper has a an ACL role or not */ public function hasRole() { return null !== $this->_role; } /** * Checks if the helper has a translator * * Implements {@link Zend_View_Helper_Navigation_Helper::hasTranslator()}. * * @return bool whether the helper has a translator or not */ public function hasTranslator() { return null !== $this->_translator; } /** * Returns an HTML string containing an 'a' element for the given page * * @param Zend_Navigation_Page $page page to generate HTML for * @return string HTML string for the given page */ public function htmlify(Zend_Navigation_Page $page) { // get label and title for translating $label = $page->getLabel(); $title = $page->getTitle(); if ($this->getUseTranslator() && $t = $this->getTranslator()) { if (is_string($label) && !empty($label)) { $label = $t->translate($label); } if (is_string($title) && !empty($title)) { $title = $t->translate($title); } } // get attribs for anchor element $attribs = array( 'id' => $page->getId(), 'title' => $title, 'class' => $page->getClass(), 'href' => $page->getHref(), 'target' => $page->getTarget() ); return '<a' . $this->_htmlAttribs($attribs) . '>' . $this->view->escape($label) . '</a>'; } // Iterator filter methods: /** * Determines whether a page should be accepted when iterating * * Rules: * - If a page is not visible it is not accepted, unless RenderInvisible has * been set to true. * - If helper has no ACL, page is accepted * - If helper has ACL, but no role, page is not accepted * - If helper has ACL and role: * - Page is accepted if it has no resource or privilege * - Page is accepted if ACL allows page's resource or privilege * - If page is accepted by the rules above and $recursive is true, the page * will not be accepted if it is the descendant of a non-accepted page. * * @param Zend_Navigation_Page $page page to check * @param bool $recursive [optional] if true, page will not * be accepted if it is the * descendant of a page that is not * accepted. Default is true. * @return bool whether page should be accepted */ public function accept(Zend_Navigation_Page $page, $recursive = true) { // accept by default $accept = true; if (!$page->isVisible(false) && !$this->getRenderInvisible()) { // don't accept invisible pages $accept = false; } elseif ($this->getUseAcl() && !$this->_acceptAcl($page)) { // acl is not amused $accept = false; } if ($accept && $recursive) { $parent = $page->getParent(); if ($parent instanceof Zend_Navigation_Page) { $accept = $this->accept($parent, true); } } return $accept; } /** * Determines whether a page should be accepted by ACL when iterating * * Rules: * - If helper has no ACL, page is accepted * - If page has a resource or privilege defined, page is accepted * if the ACL allows access to it using the helper's role * - If page has no resource or privilege, page is accepted * * @param Zend_Navigation_Page $page page to check * @return bool whether page is accepted by ACL */ protected function _acceptAcl(Zend_Navigation_Page $page) { if (!$acl = $this->getAcl()) { // no acl registered means don't use acl return true; } $role = $this->getRole(); $resource = $page->getResource(); $privilege = $page->getPrivilege(); if ($resource || $privilege) { // determine using helper role and page resource/privilege return $acl->isAllowed($role, $resource, $privilege); } return true; } // Util methods: /** * Retrieve whitespace representation of $indent * * @param int|string $indent * @return string */ protected function _getWhitespace($indent) { if (is_int($indent)) { $indent = str_repeat(' ', $indent); } return (string) $indent; } /** * Converts an associative array to a string of tag attributes. * * Overloads {@link Zend_View_Helper_HtmlElement::_htmlAttribs()}. * * @param array $attribs an array where each key-value pair is converted * to an attribute name and value * @return string an attribute string */ protected function _htmlAttribs($attribs) { // filter out null values and empty string values foreach ($attribs as $key => $value) { if ($value === null || (is_string($value) && !strlen($value))) { unset($attribs[$key]); } } return parent::_htmlAttribs($attribs); } /** * Normalize an ID * * Overrides {@link Zend_View_Helper_HtmlElement::_normalizeId()}. * * @param string $value * @return string */ protected function _normalizeId($value) { $prefix = get_class($this); $prefix = strtolower(trim(substr($prefix, strrpos($prefix, '_')), '_')); return $prefix . '-' . $value; } // Static methods: /** * Sets default ACL to use if another ACL is not explicitly set * * @param Zend_Acl $acl [optional] ACL object. Default is null, which * sets no ACL object. * @return void */ public static function setDefaultAcl(Zend_Acl $acl = null) { self::$_defaultAcl = $acl; } /** * Sets default ACL role(s) to use when iterating pages if not explicitly * set later with {@link setRole()} * * @param midex $role [optional] role to set. Expects null, * string, or an instance of * {@link Zend_Acl_Role_Interface}. * Default is null, which sets no default * role. * @throws Zend_View_Exception if role is invalid * @return void */ public static function setDefaultRole($role = null) { if (null === $role || is_string($role) || $role instanceof Zend_Acl_Role_Interface) { self::$_defaultRole = $role; } else { require_once 'Zend/View/Exception.php'; throw new Zend_View_Exception( '$role must be null|string|Zend_Acl_Role_Interface' ); } } }