vendor/pimcore/pimcore/lib/Navigation/Container.php line 548

Open in your IDE?
  1. <?php
  2. /**
  3.  * Pimcore
  4.  *
  5.  * This source file is available under two different licenses:
  6.  * - GNU General Public License version 3 (GPLv3)
  7.  * - Pimcore Commercial License (PCL)
  8.  * Full copyright and license information is available in
  9.  * LICENSE.md which is distributed with this source code.
  10.  *
  11.  *  @copyright  Copyright (c) Pimcore GmbH (http://www.pimcore.org)
  12.  *  @license    http://www.pimcore.org/license     GPLv3 and PCL
  13.  */
  14. /**
  15.  * ----------------------------------------------------------------------------------
  16.  * based on @author ZF1 Zend_Navigation_Container
  17.  * ----------------------------------------------------------------------------------
  18.  */
  19. /**
  20.  * Zend Framework
  21.  *
  22.  * LICENSE
  23.  *
  24.  * This source file is subject to the new BSD license that is bundled
  25.  * with this package in the file LICENSE.txt.
  26.  * It is also available through the world-wide-web at this URL:
  27.  * http://framework.zend.com/license/new-bsd
  28.  * If you did not receive a copy of the license and are unable to
  29.  * obtain it through the world-wide-web, please send an email
  30.  * to license@zend.com so we can send you a copy immediately.
  31.  *
  32.  * @copyright  Copyright (c) 2005-2015 Zend Technologies USA Inc. (http://www.zend.com)
  33.  * @license   http://framework.zend.com/license/new-bsd     New BSD License
  34.  */
  35. namespace Pimcore\Navigation;
  36. class Container implements \RecursiveIterator\Countable
  37. {
  38.     /**
  39.      * Contains sub pages
  40.      *
  41.      * @var Page[]
  42.      */
  43.     protected $_pages = [];
  44.     /**
  45.      * An index that contains the order in which to iterate pages
  46.      *
  47.      * @var array
  48.      */
  49.     protected $_index = [];
  50.     /**
  51.      * Whether index is dirty and needs to be re-arranged
  52.      *
  53.      * @var bool
  54.      */
  55.     protected $_dirtyIndex false;
  56.     // Internal methods:
  57.     /**
  58.      * Sorts the page index according to page order
  59.      *
  60.      * @return void
  61.      */
  62.     protected function _sort()
  63.     {
  64.         if ($this->_dirtyIndex) {
  65.             $newIndex = [];
  66.             $index 0;
  67.             foreach ($this->_pages as $hash => $page) {
  68.                 $order $page->getOrder();
  69.                 if ($order === null) {
  70.                     $newIndex[$hash] = $index;
  71.                     $index++;
  72.                 } else {
  73.                     $newIndex[$hash] = $order;
  74.                 }
  75.             }
  76.             asort($newIndex);
  77.             $this->_index $newIndex;
  78.             $this->_dirtyIndex false;
  79.         }
  80.     }
  81.     // Public methods:
  82.     /**
  83.      * Notifies container that the order of pages are updated
  84.      *
  85.      * @return void
  86.      */
  87.     public function notifyOrderUpdated()
  88.     {
  89.         $this->_dirtyIndex true;
  90.     }
  91.     /**
  92.      * Adds a page to the container
  93.      *
  94.      * This method will inject the container as the given page's parent by
  95.      * calling {@link Page::setParent()}.
  96.      *
  97.      * @param  Page|array $page  page to add
  98.      *
  99.      * @return Container fluent interface, returns self
  100.      *
  101.      * @throws \Exception if page is invalid
  102.      */
  103.     public function addPage($page)
  104.     {
  105.         if ($page === $this) {
  106.             throw new \Exception('A page cannot have itself as a parent');
  107.         }
  108.         if (is_array($page)) {
  109.             $page Page::factory($page);
  110.         } elseif (!$page instanceof Page) {
  111.             throw new \Exception('Invalid argument: $page must be an instance of \Pimcore\Navigation\Page or an array');
  112.         }
  113.         $hash $page->hashCode();
  114.         if (array_key_exists($hash$this->_index)) {
  115.             // page is already in container
  116.             return $this;
  117.         }
  118.         // adds page to container and sets dirty flag
  119.         $this->_pages[$hash] = $page;
  120.         $this->_index[$hash] = $page->getOrder();
  121.         $this->_dirtyIndex true;
  122.         // inject self as page parent
  123.         $page->setParent($this);
  124.         return $this;
  125.     }
  126.     /**
  127.      * Adds several pages at once
  128.      *
  129.      * @param  Page[]|Container  $pages  pages to add
  130.      *
  131.      * @return Container fluent interface, returns self
  132.      *
  133.      * @throws \Exception if $pages is not array or Container
  134.      */
  135.     public function addPages($pages)
  136.     {
  137.         if ($pages instanceof self) {
  138.             $pages iterator_to_array($pages);
  139.         }
  140.         if (!is_array($pages)) {
  141.             throw new \Exception('Invalid argument: $pages must be an array  or an instance of Container');
  142.         }
  143.         foreach ($pages as $page) {
  144.             $this->addPage($page);
  145.         }
  146.         return $this;
  147.     }
  148.     /**
  149.      * Sets pages this container should have, removing existing pages
  150.      *
  151.      * @param  Page[] $pages pages to set
  152.      *
  153.      * @return Container  fluent interface, returns self
  154.      */
  155.     public function setPages(array $pages)
  156.     {
  157.         $this->removePages();
  158.         return $this->addPages($pages);
  159.     }
  160.     /**
  161.      * Returns pages in the container
  162.      *
  163.      * @return Page[]
  164.      */
  165.     public function getPages()
  166.     {
  167.         return $this->_pages;
  168.     }
  169.     /**
  170.      * Removes the given page from the container
  171.      *
  172.      * @param  Page|int $page page to remove, either a page instance or a specific page order
  173.      * @param  bool $recursive [optional] whether to remove recursively
  174.      *
  175.      * @return bool whether the removal was successful
  176.      */
  177.     public function removePage($page$recursive false)
  178.     {
  179.         if ($page instanceof Page) {
  180.             $hash $page->hashCode();
  181.         } elseif (is_int($page)) {
  182.             $this->_sort();
  183.             if (!$hash array_search($page$this->_index)) {
  184.                 return false;
  185.             }
  186.         } else {
  187.             return false;
  188.         }
  189.         if (isset($this->_pages[$hash])) {
  190.             unset($this->_pages[$hash]);
  191.             unset($this->_index[$hash]);
  192.             $this->_dirtyIndex true;
  193.             return true;
  194.         }
  195.         if ($recursive) {
  196.             foreach ($this->_pages as $childPage) {
  197.                 if ($childPage->hasPage($pagetrue)) {
  198.                     $childPage->removePage($pagetrue);
  199.                     return true;
  200.                 }
  201.             }
  202.         }
  203.         return false;
  204.     }
  205.     /**
  206.      * Removes all pages in container
  207.      *
  208.      * @return Container  fluent interface, returns self
  209.      */
  210.     public function removePages()
  211.     {
  212.         $this->_pages = [];
  213.         $this->_index = [];
  214.         return $this;
  215.     }
  216.     /**
  217.      * Checks if the container has the given page
  218.      *
  219.      * @param  Page $page  page to look for
  220.      * @param  bool $recursive  [optional] whether to search recursively. Default is false.
  221.      *
  222.      * @return bool whether page is in container
  223.      */
  224.     public function hasPage(Page $page$recursive false)
  225.     {
  226.         if (array_key_exists($page->hashCode(), $this->_index)) {
  227.             return true;
  228.         } elseif ($recursive) {
  229.             foreach ($this->_pages as $childPage) {
  230.                 if ($childPage->hasPage($pagetrue)) {
  231.                     return true;
  232.                 }
  233.             }
  234.         }
  235.         return false;
  236.     }
  237.     /**
  238.      * Returns true if container contains any pages
  239.      *
  240.      * @return bool  whether container has any pages
  241.      */
  242.     public function hasPages()
  243.     {
  244.         return count($this->_index) > 0;
  245.     }
  246.     /**
  247.      * Returns true if container contains any visible page
  248.      *
  249.      * @return bool whether container has any visible page
  250.      */
  251.     public function hasVisiblePages()
  252.     {
  253.         if ($this->hasPages()) {
  254.             foreach ($this->getPages() as $page) {
  255.                 if ($page->isVisible()) {
  256.                     return true;
  257.                 }
  258.             }
  259.         }
  260.         return false;
  261.     }
  262.     /**
  263.      * Returns a child page matching $property == $value or
  264.      * preg_match($value, $property), or null if not found
  265.      *
  266.      * @param  string  $property          name of property to match against
  267.      * @param  mixed   $value             value to match property against
  268.      * @param  bool    $useRegex          [optional] if true PHP's preg_match
  269.      *                                    is used. Default is false.
  270.      *
  271.      * @return Page|null  matching page or null
  272.      */
  273.     public function findOneBy($property$value$useRegex false)
  274.     {
  275.         $iterator = new \RecursiveIteratorIterator($this\RecursiveIteratorIterator::SELF_FIRST);
  276.         foreach ($iterator as $page) {
  277.             $pageProperty $page->get($property);
  278.             // Rel and rev
  279.             if (is_array($pageProperty)) {
  280.                 foreach ($pageProperty as $item) {
  281.                     if (is_array($item)) {
  282.                         // Use regex?
  283.                         if (true === $useRegex) {
  284.                             foreach ($item as $item2) {
  285.                                 if (!== preg_match($value$item2)) {
  286.                                     return $page;
  287.                                 }
  288.                             }
  289.                         } else {
  290.                             if (in_array($value$item)) {
  291.                                 return $page;
  292.                             }
  293.                         }
  294.                     } else {
  295.                         // Use regex?
  296.                         if (true === $useRegex) {
  297.                             if (!== preg_match($value$item)) {
  298.                                 return $page;
  299.                             }
  300.                         } else {
  301.                             if ($item == $value) {
  302.                                 return $page;
  303.                             }
  304.                         }
  305.                     }
  306.                 }
  307.                 continue;
  308.             }
  309.             // Use regex?
  310.             if (true === $useRegex) {
  311.                 if (preg_match($value$pageProperty)) {
  312.                     return $page;
  313.                 }
  314.             } else {
  315.                 if ($pageProperty == $value) {
  316.                     return $page;
  317.                 }
  318.             }
  319.         }
  320.         return null;
  321.     }
  322.     /**
  323.      * Returns all child pages matching $property == $value or
  324.      * preg_match($value, $property), or an empty array if no pages are found
  325.      *
  326.      * @param  string $property  name of property to match against
  327.      * @param  mixed  $value     value to match property against
  328.      * @param  bool   $useRegex  [optional] if true PHP's preg_match is used.
  329.      *                           Default is false.
  330.      *
  331.      * @return Page[] array containing only Page instances
  332.      */
  333.     public function findAllBy($property$value$useRegex false)
  334.     {
  335.         $found = [];
  336.         $iterator = new \RecursiveIteratorIterator($this\RecursiveIteratorIterator::SELF_FIRST);
  337.         foreach ($iterator as $page) {
  338.             $pageProperty $page->get($property);
  339.             // Rel and rev
  340.             if (is_array($pageProperty)) {
  341.                 foreach ($pageProperty as $item) {
  342.                     if (is_array($item)) {
  343.                         // Use regex?
  344.                         if (true === $useRegex) {
  345.                             foreach ($item as $item2) {
  346.                                 if (!== preg_match($value$item2)) {
  347.                                     $found[] = $page;
  348.                                 }
  349.                             }
  350.                         } else {
  351.                             if (in_array($value$item)) {
  352.                                 $found[] = $page;
  353.                             }
  354.                         }
  355.                     } else {
  356.                         // Use regex?
  357.                         if (true === $useRegex) {
  358.                             if (!== preg_match($value$item)) {
  359.                                 $found[] = $page;
  360.                             }
  361.                         } else {
  362.                             if ($item == $value) {
  363.                                 $found[] = $page;
  364.                             }
  365.                         }
  366.                     }
  367.                 }
  368.                 continue;
  369.             }
  370.             // Use regex?
  371.             if (true === $useRegex) {
  372.                 if (!== preg_match($value$pageProperty)) {
  373.                     $found[] = $page;
  374.                 }
  375.             } else {
  376.                 if ($pageProperty == $value) {
  377.                     $found[] = $page;
  378.                 }
  379.             }
  380.         }
  381.         return $found;
  382.     }
  383.     /**
  384.      * Returns page(s) matching $property == $value or
  385.      * preg_match($value, $property)
  386.      *
  387.      * @param  string $property  name of property to match against
  388.      * @param  mixed  $value     value to match property against
  389.      * @param  bool   $all       [optional] whether an array of all matching
  390.      *                           pages should be returned, or only the first.
  391.      *                           If true, an array will be returned, even if not
  392.      *                           matching pages are found. If false, null will
  393.      *                           be returned if no matching page is found.
  394.      *                           Default is false.
  395.      * @param  bool   $useRegex  [optional] if true PHP's preg_match is used.
  396.      *                           Default is false.
  397.      *
  398.      * @return Page|array<Page>|null  matching page or null
  399.      */
  400.     public function findBy($property$value$all false$useRegex false)
  401.     {
  402.         if ($all) {
  403.             return $this->findAllBy($property$value$useRegex);
  404.         }
  405.         return $this->findOneBy($property$value$useRegex);
  406.     }
  407.     /**
  408.      * Magic overload: Proxy calls to finder methods
  409.      *
  410.      * Examples of finder calls:
  411.      * <code>
  412.      * // METHOD                         // SAME AS
  413.      * $nav->findByLabel('foo');         // $nav->findOneBy('label', 'foo');
  414.      * $nav->findByLabel('/foo/', true); // $nav->findBy('label', '/foo/', true);
  415.      * $nav->findOneByLabel('foo');      // $nav->findOneBy('label', 'foo');
  416.      * $nav->findAllByClass('foo');      // $nav->findAllBy('class', 'foo');
  417.      * </code>
  418.      *
  419.      * @param  string $method                       method name
  420.      * @param  array  $arguments                    method arguments
  421.      *
  422.      * @return mixed  Pimcore\Navigation|array|null    matching page, array of pages
  423.      *                                              or null
  424.      *
  425.      * @throws \Exception            if method does not exist
  426.      */
  427.     public function __call($method$arguments)
  428.     {
  429.         if (@preg_match('/(find(?:One|All)?By)(.+)/'$method$match)) {
  430.             return $this->{$match[1]}($match[2], $arguments[0], !empty($arguments[1]));
  431.         }
  432.         throw new \Exception(sprintf('Bad method call: Unknown method %s::%s'get_class($this), $method));
  433.     }
  434.     /**
  435.      * Returns an array representation of all pages in container
  436.      *
  437.      * @return array
  438.      */
  439.     public function toArray(): array
  440.     {
  441.         $pages = [];
  442.         $this->_dirtyIndex true;
  443.         $this->_sort();
  444.         $indexes array_keys($this->_index);
  445.         foreach ($indexes as $hash) {
  446.             $pages[] = $this->_pages[$hash]->toArray();
  447.         }
  448.         return $pages;
  449.     }
  450.     /**
  451.      * {@inheritdoc}
  452.      */
  453.     public function current()
  454.     {
  455.         $this->_sort();
  456.         $hash key($this->_index);
  457.         if (isset($this->_pages[$hash])) {
  458.             return $this->_pages[$hash];
  459.         }
  460.         throw new \Exception('Corruption detected in container; invalid key found in internal iterator');
  461.     }
  462.     /**
  463.      * {@inheritdoc}
  464.      */
  465.     public function key()
  466.     {
  467.         $this->_sort();
  468.         return key($this->_index);
  469.     }
  470.     /**
  471.      * {@inheritdoc}
  472.      */
  473.     public function next()
  474.     {
  475.         $this->_sort();
  476.         next($this->_index);
  477.     }
  478.     /**
  479.      * {@inheritdoc}
  480.      */
  481.     public function rewind()
  482.     {
  483.         $this->_sort();
  484.         reset($this->_index);
  485.     }
  486.     /**
  487.      * {@inheritdoc}
  488.      */
  489.     public function valid()
  490.     {
  491.         $this->_sort();
  492.         return current($this->_index) !== false;
  493.     }
  494.     /**
  495.      * {@inheritdoc}
  496.      */
  497.     public function hasChildren()
  498.     {
  499.         return $this->hasPages();
  500.     }
  501.     /**
  502.      * @return Page|\RecursiveIterator|null
  503.      */
  504.     public function getChildren()
  505.     {
  506.         $hash key($this->_index);
  507.         if (isset($this->_pages[$hash])) {
  508.             return $this->_pages[$hash];
  509.         }
  510.         return null;
  511.     }
  512.     /**
  513.      * {@inheritdoc}
  514.      */
  515.     public function count()
  516.     {
  517.         return count($this->_index);
  518.     }
  519. }