CommunityID/libs/Zend/Paginator.php

1088 lines
28 KiB
PHP

<?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_Paginator
* @copyright Copyright (c) 2005-2008 Zend Technologies USA Inc. (http://www.zend.com)
* @license http://framework.zend.com/license/new-bsd New BSD License
* @version $Id: Paginator.php 16142 2009-06-18 23:57:26Z norm2782 $
*/
/**
* @see Zend_Loader_PluginLoader
*/
require_once 'Zend/Loader/PluginLoader.php';
/**
* @see Zend_Json
*/
require_once 'Zend/Json.php';
/**
* @category Zend
* @package Zend_Paginator
* @copyright Copyright (c) 2005-2008 Zend Technologies USA Inc. (http://www.zend.com)
* @license http://framework.zend.com/license/new-bsd New BSD License
*/
class Zend_Paginator implements Countable, IteratorAggregate
{
/**
* Specifies that the factory should try to detect the proper adapter type first
*
* @var string
*/
const INTERNAL_ADAPTER = 'Zend_Paginator_Adapter_Internal';
/**
* The cache tag prefix used to namespace Paginator results in the cache
*
*/
const CACHE_TAG_PREFIX = 'Zend_Paginator_';
/**
* Adapter plugin loader
*
* @var Zend_Loader_PluginLoader
*/
protected static $_adapterLoader = null;
/**
* Configuration file
*
* @var Zend_Config
*/
protected static $_config = null;
/**
* Default scrolling style
*
* @var string
*/
protected static $_defaultScrollingStyle = 'Sliding';
/**
* Default item count per page
*
* @var int
*/
protected static $_defaultItemCountPerPage = 10;
/**
* Scrolling style plugin loader
*
* @var Zend_Loader_PluginLoader
*/
protected static $_scrollingStyleLoader = null;
/**
* Cache object
*
* @var Zend_Cache_Core
*/
protected static $_cache;
/**
* Enable or desable the cache by Zend_Paginator instance
*
* @var bool
*/
protected $_cacheEnabled = true;
/**
* Adapter
*
* @var Zend_Paginator_Adapter_Interface
*/
protected $_adapter = null;
/**
* Number of items in the current page
*
* @var integer
*/
protected $_currentItemCount = null;
/**
* Current page items
*
* @var Traversable
*/
protected $_currentItems = null;
/**
* Current page number (starting from 1)
*
* @var integer
*/
protected $_currentPageNumber = 1;
/**
* Result filter
*
* @var Zend_Filter_Interface
*/
protected $_filter = null;
/**
* Number of items per page
*
* @var integer
*/
protected $_itemCountPerPage = null;
/**
* Number of pages
*
* @var integer
*/
protected $_pageCount = null;
/**
* Number of local pages (i.e., the number of discrete page numbers
* that will be displayed, including the current page number)
*
* @var integer
*/
protected $_pageRange = 10;
/**
* Pages
*
* @var array
*/
protected $_pages = null;
/**
* View instance used for self rendering
*
* @var Zend_View_Interface
*/
protected $_view = null;
/**
* Adds an adapter prefix path to the plugin loader.
*
* @param string $prefix
* @param string $path
*/
public static function addAdapterPrefixPath($prefix, $path)
{
self::getAdapterLoader()->addPrefixPath($prefix, $path);
}
/**
* Adds an array of adapter prefix paths to the plugin
* loader.
*
* <code>
* $prefixPaths = array(
* 'My_Paginator_Adapter' => 'My/Paginator/Adapter/',
* 'Your_Paginator_Adapter' => 'Your/Paginator/Adapter/'
* );
* </code>
*
* @param array $prefixPaths
*/
public static function addAdapterPrefixPaths(array $prefixPaths)
{
if (isset($prefixPaths['prefix']) and isset($prefixPaths['path'])) {
self::addAdapterPrefixPath($prefixPaths['prefix'], $prefixPaths['path']);
} else {
foreach ($prefixPaths as $prefix => $path) {
if (is_array($path) and isset($path['prefix']) and isset($path['path'])) {
$prefix = $path['prefix'];
$path = $path['path'];
}
self::addAdapterPrefixPath($prefix, $path);
}
}
}
/**
* Adds a scrolling style prefix path to the plugin loader.
*
* @param string $prefix
* @param string $path
*/
public static function addScrollingStylePrefixPath($prefix, $path)
{
self::getScrollingStyleLoader()->addPrefixPath($prefix, $path);
}
/**
* Adds an array of scrolling style prefix paths to the plugin
* loader.
*
* <code>
* $prefixPaths = array(
* 'My_Paginator_ScrollingStyle' => 'My/Paginator/ScrollingStyle/',
* 'Your_Paginator_ScrollingStyle' => 'Your/Paginator/ScrollingStyle/'
* );
* </code>
*
* @param array $prefixPaths
*/
public static function addScrollingStylePrefixPaths(array $prefixPaths)
{
if (isset($prefixPaths['prefix']) and isset($prefixPaths['path'])) {
self::addScrollingStylePrefixPath($prefixPaths['prefix'], $prefixPaths['path']);
} else {
foreach ($prefixPaths as $prefix => $path) {
if (is_array($path) and isset($path['prefix']) and isset($path['path'])) {
$prefix = $path['prefix'];
$path = $path['path'];
}
self::addScrollingStylePrefixPath($prefix, $path);
}
}
}
/**
* Factory.
*
* @param mixed $data
* @param string $adapter
* @param array $prefixPaths
* @return Zend_Paginator
*/
public static function factory($data, $adapter = self::INTERNAL_ADAPTER,
array $prefixPaths = null)
{
if ($adapter == self::INTERNAL_ADAPTER) {
if (is_array($data)) {
$adapter = 'Array';
} else if ($data instanceof Zend_Db_Table_Select) {
$adapter = 'DbTableSelect';
} else if ($data instanceof Zend_Db_Select) {
$adapter = 'DbSelect';
} else if ($data instanceof Iterator) {
$adapter = 'Iterator';
} else if (is_integer($data)) {
$adapter = 'Null';
} else {
$type = (is_object($data)) ? get_class($data) : gettype($data);
/**
* @see Zend_Paginator_Exception
*/
require_once 'Zend/Paginator/Exception.php';
throw new Zend_Paginator_Exception('No adapter for type ' . $type);
}
}
$pluginLoader = self::getAdapterLoader();
if (null !== $prefixPaths) {
foreach ($prefixPaths as $prefix => $path) {
$pluginLoader->addPrefixPath($prefix, $path);
}
}
$adapterClassName = $pluginLoader->load($adapter);
return new self(new $adapterClassName($data));
}
/**
* Returns the adapter loader. If it doesn't exist it's created.
*
* @return Zend_Loader_PluginLoader
*/
public static function getAdapterLoader()
{
if (self::$_adapterLoader === null) {
self::$_adapterLoader = new Zend_Loader_PluginLoader(
array('Zend_Paginator_Adapter' => 'Zend/Paginator/Adapter')
);
}
return self::$_adapterLoader;
}
/**
* Set a global config
*
* @param Zend_Config $config
*/
public static function setConfig(Zend_Config $config)
{
self::$_config = $config;
$adapterPaths = $config->get('adapterpaths');
if ($adapterPaths != null) {
self::addAdapterPrefixPaths($adapterPaths->adapterpath->toArray());
}
$prefixPaths = $config->get('prefixpaths');
if ($prefixPaths != null) {
self::addScrollingStylePrefixPaths($prefixPaths->prefixpath->toArray());
}
$scrollingStyle = $config->get('scrollingstyle');
if ($scrollingStyle != null) {
self::setDefaultScrollingStyle($scrollingStyle);
}
}
/**
* Returns the default scrolling style.
*
* @return string
*/
public static function getDefaultScrollingStyle()
{
return self::$_defaultScrollingStyle;
}
/**
* Get the default item count per page
*
* @return int
*/
public static function getDefaultItemCountPerPage()
{
return self::$_defaultItemCountPerPage;
}
/**
* Set the default item count per page
*
* @param int $count
*/
public static function setDefaultItemCountPerPage($count)
{
self::$_defaultItemCountPerPage = (int) $count;
}
/**
* Sets a cache object
*
* @param Zend_Cache_Core $cache
*/
public static function setCache(Zend_Cache_Core $cache)
{
self::$_cache = $cache;
}
/**
* Sets the default scrolling style.
*
* @param string $scrollingStyle
*/
public static function setDefaultScrollingStyle($scrollingStyle = 'Sliding')
{
self::$_defaultScrollingStyle = $scrollingStyle;
}
/**
* Returns the scrolling style loader. If it doesn't exist it's
* created.
*
* @return Zend_Loader_PluginLoader
*/
public static function getScrollingStyleLoader()
{
if (self::$_scrollingStyleLoader === null) {
self::$_scrollingStyleLoader = new Zend_Loader_PluginLoader(
array('Zend_Paginator_ScrollingStyle' => 'Zend/Paginator/ScrollingStyle')
);
}
return self::$_scrollingStyleLoader;
}
/**
* Constructor.
*/
public function __construct(Zend_Paginator_Adapter_Interface $adapter)
{
$this->_adapter = $adapter;
$config = self::$_config;
if ($config != null) {
$setupMethods = array('ItemCountPerPage', 'PageRange');
foreach ($setupMethods as $setupMethod) {
$value = $config->get(strtolower($setupMethod));
if ($value != null) {
$setupMethod = 'set' . $setupMethod;
$this->$setupMethod($value);
}
}
}
}
/**
* Serializes the object as a string. Proxies to {@link render()}.
*
* @return string
*/
public function __toString()
{
try {
$return = $this->render();
return $return;
} catch (Exception $e) {
trigger_error($e->getMessage(), E_USER_WARNING);
}
return '';
}
/**
* Enables/Disables the cache for this instance
*
* @param bool $enable
* @return Zend_Paginator
*/
public function setCacheEnabled($enable)
{
$this->_cacheEnabled = (bool)$enable;
return $this;
}
/**
* Returns the number of pages.
*
* @return integer
*/
public function count()
{
if (!$this->_pageCount) {
$this->_pageCount = $this->_calculatePageCount();
}
return $this->_pageCount;
}
/**
* Returns the total number of items available.
*
* @return integer
*/
public function getTotalItemCount()
{
return count($this->_adapter);
}
/**
* Clear the page item cache.
*
* @param int $pageNumber
* @return Zend_Paginator
*/
public function clearPageItemCache($pageNumber = null)
{
if (!$this->_cacheEnabled()) {
return $this;
}
if (null === $pageNumber) {
$cleanTags = self::CACHE_TAG_PREFIX;
foreach (self::$_cache->getIdsMatchingTags(array($this->_getCacheInternalId())) as $id) {
if (preg_match('|'.self::CACHE_TAG_PREFIX."(\d+)_.*|", $id, $page)) {
self::$_cache->remove($this->_getCacheId($page[1]));
}
}
} else {
$cleanId = $this->_getCacheId($pageNumber);
self::$_cache->remove($cleanId);
}
return $this;
}
/**
* Returns the absolute item number for the specified item.
*
* @param integer $relativeItemNumber Relative item number
* @param integer $pageNumber Page number
* @return integer
*/
public function getAbsoluteItemNumber($relativeItemNumber, $pageNumber = null)
{
$relativeItemNumber = $this->normalizeItemNumber($relativeItemNumber);
if ($pageNumber == null) {
$pageNumber = $this->getCurrentPageNumber();
}
$pageNumber = $this->normalizePageNumber($pageNumber);
return (($pageNumber - 1) * $this->getItemCountPerPage()) + $relativeItemNumber;
}
/**
* Returns the adapter.
*
* @return Zend_Paginator_Adapter_Interface
*/
public function getAdapter()
{
return $this->_adapter;
}
/**
* Returns the number of items for the current page.
*
* @return integer
*/
public function getCurrentItemCount()
{
if ($this->_currentItemCount === null) {
$this->_currentItemCount = $this->getItemCount($this->getCurrentItems());
}
return $this->_currentItemCount;
}
/**
* Returns the items for the current page.
*
* @return Traversable
*/
public function getCurrentItems()
{
if ($this->_currentItems === null) {
$this->_currentItems = $this->getItemsByPage($this->getCurrentPageNumber());
}
return $this->_currentItems;
}
/**
* Returns the current page number.
*
* @return integer
*/
public function getCurrentPageNumber()
{
return $this->normalizePageNumber($this->_currentPageNumber);
}
/**
* Sets the current page number.
*
* @param integer $pageNumber Page number
* @return Zend_Paginator $this
*/
public function setCurrentPageNumber($pageNumber)
{
$this->_currentPageNumber = (integer) $pageNumber;
$this->_currentItems = null;
$this->_currentItemCount = null;
return $this;
}
/**
* Get the filter
*
* @return Zend_Filter_Interface
*/
public function getFilter()
{
return $this->_filter;
}
/**
* Set a filter chain
*
* @param Zend_Filter_Interface $filter
* @return Zend_Paginator
*/
public function setFilter(Zend_Filter_Interface $filter)
{
$this->_filter = $filter;
return $this;
}
/**
* Returns an item from a page. The current page is used if there's no
* page sepcified.
*
* @param integer $itemNumber Item number (1 to itemCountPerPage)
* @param integer $pageNumber
* @return mixed
*/
public function getItem($itemNumber, $pageNumber = null)
{
$itemNumber = $this->normalizeItemNumber($itemNumber);
if ($pageNumber == null) {
$pageNumber = $this->getCurrentPageNumber();
}
$page = $this->getItemsByPage($pageNumber);
$itemCount = $this->getItemCount($page);
if ($itemCount == 0) {
/**
* @see Zend_Paginator_Exception
*/
require_once 'Zend/Paginator/Exception.php';
throw new Zend_Paginator_Exception('Page ' . $pageNumber . ' does not exist');
}
if ($itemNumber > $itemCount) {
/**
* @see Zend_Paginator_Exception
*/
require_once 'Zend/Paginator/Exception.php';
throw new Zend_Paginator_Exception('Page ' . $pageNumber . ' does not'
. ' contain item number ' . $itemNumber);
}
return $page[$itemNumber - 1];
}
/**
* Returns the number of items per page.
*
* @return integer
*/
public function getItemCountPerPage()
{
if ($this->_itemCountPerPage === null) {
$this->_itemCountPerPage = self::getDefaultItemCountPerPage();
}
return $this->_itemCountPerPage;
}
/**
* Sets the number of items per page.
*
* @param integer $itemCountPerPage
* @return Zend_Paginator $this
*/
public function setItemCountPerPage($itemCountPerPage)
{
$this->_itemCountPerPage = (integer) $itemCountPerPage;
if ($this->_itemCountPerPage == 0) {
$this->_itemCountPerPage = 1;
}
$this->_pageCount = $this->_calculatePageCount();
$this->_currentItems = null;
$this->_currentItemCount = null;
return $this;
}
/**
* Returns the number of items in a collection.
*
* @param mixed $items Items
* @return integer
*/
public function getItemCount($items)
{
$itemCount = 0;
if (is_array($items) or $items instanceof Countable) {
$itemCount = count($items);
} else { // $items is something like LimitIterator
$itemCount = iterator_count($items);
}
return $itemCount;
}
/**
* Returns the items for a given page.
*
* @return Traversable
*/
public function getItemsByPage($pageNumber)
{
$pageNumber = $this->normalizePageNumber($pageNumber);
if ($this->_cacheEnabled()) {
$data = self::$_cache->load($this->_getCacheId($pageNumber));
if ($data !== false) {
return $data;
}
}
$offset = ($pageNumber - 1) * $this->_itemCountPerPage;
$items = $this->_adapter->getItems($offset, $this->_itemCountPerPage);
$filter = $this->getFilter();
if ($filter !== null) {
$items = $filter->filter($items);
}
if (!$items instanceof Traversable) {
$items = new ArrayIterator($items);
}
if ($this->_cacheEnabled()) {
self::$_cache->save($items, $this->_getCacheId($pageNumber), array($this->_getCacheInternalId()));
}
return $items;
}
/**
* Returns a foreach-compatible iterator.
*
* @return Traversable
*/
public function getIterator()
{
return $this->getCurrentItems();
}
/**
* Returns the page range (see property declaration above).
*
* @return integer
*/
public function getPageRange()
{
return $this->_pageRange;
}
/**
* Sets the page range (see property declaration above).
*
* @param integer $pageRange
* @return Zend_Paginator $this
*/
public function setPageRange($pageRange)
{
$this->_pageRange = (integer) $pageRange;
return $this;
}
/**
* Returns the page collection.
*
* @param string $scrollingStyle Scrolling style
* @return array
*/
public function getPages($scrollingStyle = null)
{
if ($this->_pages === null) {
$this->_pages = $this->_createPages($scrollingStyle);
}
return $this->_pages;
}
/**
* Returns a subset of pages within a given range.
*
* @param integer $lowerBound Lower bound of the range
* @param integer $upperBound Upper bound of the range
* @return array
*/
public function getPagesInRange($lowerBound, $upperBound)
{
$lowerBound = $this->normalizePageNumber($lowerBound);
$upperBound = $this->normalizePageNumber($upperBound);
$pages = array();
for ($pageNumber = $lowerBound; $pageNumber <= $upperBound; $pageNumber++) {
$pages[$pageNumber] = $pageNumber;
}
return $pages;
}
/**
* Returns the page item cache.
*
* @return array
*/
public function getPageItemCache()
{
$data = array();
if ($this->_cacheEnabled()) {
foreach (self::$_cache->getIdsMatchingTags(array($this->_getCacheInternalId())) as $id) {
if (preg_match('|'.self::CACHE_TAG_PREFIX."(\d+)_.*|", $id, $page)) {
$data[$page[1]] = self::$_cache->load($this->_getCacheId($page[1]));
}
}
}
return $data;
}
/**
* Retrieves the view instance. If none registered, attempts to pull f
* rom ViewRenderer.
*
* @return Zend_View_Interface|null
*/
public function getView()
{
if ($this->_view === null) {
/**
* @see Zend_Controller_Action_HelperBroker
*/
require_once 'Zend/Controller/Action/HelperBroker.php';
$viewRenderer = Zend_Controller_Action_HelperBroker::getStaticHelper('viewRenderer');
if ($viewRenderer->view === null) {
$viewRenderer->initView();
}
$this->_view = $viewRenderer->view;
}
return $this->_view;
}
/**
* Sets the view object.
*
* @param Zend_View_Interface $view
* @return Zend_Paginator
*/
public function setView(Zend_View_Interface $view = null)
{
$this->_view = $view;
return $this;
}
/**
* Brings the item number in range of the page.
*
* @param integer $itemNumber
* @return integer
*/
public function normalizeItemNumber($itemNumber)
{
if ($itemNumber < 1) {
$itemNumber = 1;
}
if ($itemNumber > $this->_itemCountPerPage) {
$itemNumber = $this->_itemCountPerPage;
}
return $itemNumber;
}
/**
* Brings the page number in range of the paginator.
*
* @param integer $pageNumber
* @return integer
*/
public function normalizePageNumber($pageNumber)
{
if ($pageNumber < 1) {
$pageNumber = 1;
}
$pageCount = $this->count();
if ($pageCount > 0 and $pageNumber > $pageCount) {
$pageNumber = $pageCount;
}
return $pageNumber;
}
/**
* Renders the paginator.
*
* @param Zend_View_Interface $view
* @return string
*/
public function render(Zend_View_Interface $view = null)
{
if (null !== $view) {
$this->setView($view);
}
$view = $this->getView();
return $view->paginationControl($this);
}
/**
* Returns the items of the current page as JSON.
*
* @return string
*/
public function toJson()
{
$currentItems = $this->getCurrentItems();
if ($currentItems instanceof Zend_Db_Table_Rowset_Abstract) {
return Zend_Json::encode($currentItems->toArray());
} else {
return Zend_Json::encode($currentItems);
}
}
/**
* Tells if there is an active cache object
* and if the cache has not been desabled
*
* @return bool
*/
protected function _cacheEnabled()
{
return ((self::$_cache !== null) && $this->_cacheEnabled);
}
/**
* Makes an Id for the cache
* Depends on the adapter object and the page number
*
* Used to store item in cache from that Paginator instance
* and that current page
*
* @param int $page
* @return string
*/
protected function _getCacheId($page = null)
{
if ($page === null) {
$page = $this->getCurrentPageNumber();
}
return self::CACHE_TAG_PREFIX . $page . '_' . $this->_getCacheInternalId();
}
/**
* Get the internal cache id
* Depends on the adapter and the item count per page
*
* Used to tag that unique Paginator instance in cache
*
* @return string
*/
protected function _getCacheInternalId()
{
return md5(serialize($this->_adapter).$this->_itemCountPerPage);
}
/**
* Calculates the page count.
*
* @return integer
*/
protected function _calculatePageCount()
{
return (integer) ceil($this->_adapter->count() / $this->_itemCountPerPage);
}
/**
* Creates the page collection.
*
* @param string $scrollingStyle Scrolling style
* @return stdClass
*/
protected function _createPages($scrollingStyle = null)
{
$pageCount = $this->count();
$currentPageNumber = $this->getCurrentPageNumber();
$pages = new stdClass();
$pages->pageCount = $pageCount;
$pages->itemCountPerPage = $this->getItemCountPerPage();
$pages->first = 1;
$pages->current = $currentPageNumber;
$pages->last = $pageCount;
// Previous and next
if ($currentPageNumber - 1 > 0) {
$pages->previous = $currentPageNumber - 1;
}
if ($currentPageNumber + 1 <= $pageCount) {
$pages->next = $currentPageNumber + 1;
}
// Pages in range
$scrollingStyle = $this->_loadScrollingStyle($scrollingStyle);
$pages->pagesInRange = $scrollingStyle->getPages($this);
$pages->firstPageInRange = min($pages->pagesInRange);
$pages->lastPageInRange = max($pages->pagesInRange);
// Item numbers
if ($this->getCurrentItems() !== null) {
$pages->currentItemCount = $this->getCurrentItemCount();
$pages->itemCountPerPage = $this->getItemCountPerPage();
$pages->totalItemCount = $this->getTotalItemCount();
$pages->firstItemNumber = (($currentPageNumber - 1) * $this->_itemCountPerPage) + 1;
$pages->lastItemNumber = $pages->firstItemNumber + $pages->currentItemCount - 1;
}
return $pages;
}
/**
* Loads a scrolling style.
*
* @param string $scrollingStyle
* @return Zend_Paginator_ScrollingStyle_Interface
*/
protected function _loadScrollingStyle($scrollingStyle = null)
{
if ($scrollingStyle === null) {
$scrollingStyle = self::$_defaultScrollingStyle;
}
switch (strtolower(gettype($scrollingStyle))) {
case 'object':
if (!$scrollingStyle instanceof Zend_Paginator_ScrollingStyle_Interface) {
/**
* @see Zend_View_Exception
*/
require_once 'Zend/View/Exception.php';
throw new Zend_View_Exception('Scrolling style must implement ' .
'Zend_Paginator_ScrollingStyle_Interface');
}
return $scrollingStyle;
case 'string':
$className = self::getScrollingStyleLoader()->load($scrollingStyle);
return new $className();
case 'null':
// Fall through to default case
default:
/**
* @see Zend_View_Exception
*/
require_once 'Zend/View/Exception.php';
throw new Zend_View_Exception('Scrolling style must be a class ' .
'name or object implementing Zend_Paginator_ScrollingStyle_Interface');
}
}
}