|
|
@@ -0,0 +1,447 @@
|
|
|
+<?php
|
|
|
+ /* SVN FILE: $Id: tree.php 205 2008-08-13 14:13:32Z ad7six $ */
|
|
|
+/**
|
|
|
+ * Tree Helper.
|
|
|
+ *
|
|
|
+ * Used the generate nested representations of hierarchial data
|
|
|
+ *
|
|
|
+ * PHP versions 4 and 5
|
|
|
+ *
|
|
|
+ * Copyright (c) 2008, Andy Dawson
|
|
|
+ *
|
|
|
+ * Licensed under The MIT License
|
|
|
+ * Redistributions of files must retain the above copyright notice.
|
|
|
+ *
|
|
|
+ * @filesource
|
|
|
+ * @copyright Copyright (c) 2008, Andy Dawson
|
|
|
+ * @link www.ad7six.com
|
|
|
+ * @package cake-base
|
|
|
+ * @subpackage cake-base.app.views.helpers
|
|
|
+ * @since v 1.0
|
|
|
+ * @version $Revision: 205 $
|
|
|
+ * @modifiedBy $LastChangedBy: ad7six $
|
|
|
+ * @lastModified $Date: 2008-08-13 16:13:32 +0200 (Wed, 13 Aug 2008) $
|
|
|
+ * @license http://www.opensource.org/licenses/mit-license.php The MIT License
|
|
|
+ */
|
|
|
+/**
|
|
|
+ * Tree helper
|
|
|
+ *
|
|
|
+ * Helper to generate tree representations of MPTT or recursively nested data
|
|
|
+ */
|
|
|
+class TreeHelper extends AppHelper {
|
|
|
+
|
|
|
+/**
|
|
|
+ * settings property
|
|
|
+ *
|
|
|
+ * @var array
|
|
|
+ * @access private
|
|
|
+ */
|
|
|
+ public $_settings = array();
|
|
|
+/**
|
|
|
+ * typeAttributes property
|
|
|
+ *
|
|
|
+ * @var array
|
|
|
+ * @access private
|
|
|
+ */
|
|
|
+ public $_typeAttributes = array();
|
|
|
+/**
|
|
|
+ * typeAttributesNext property
|
|
|
+ *
|
|
|
+ * @var array
|
|
|
+ * @access private
|
|
|
+ */
|
|
|
+ public $_typeAttributesNext = array();
|
|
|
+/**
|
|
|
+ * itemAttributes property
|
|
|
+ *
|
|
|
+ * @var array
|
|
|
+ * @access private
|
|
|
+ */
|
|
|
+ public $_itemAttributes = array();
|
|
|
+/**
|
|
|
+ * helpers variable
|
|
|
+ *
|
|
|
+ * @var array
|
|
|
+ * @access public
|
|
|
+ */
|
|
|
+ public $helpers = array ('Html');
|
|
|
+/**
|
|
|
+ * Tree generation method.
|
|
|
+ *
|
|
|
+ * Accepts the results of
|
|
|
+ * find('all', array('fields' => array('lft', 'rght', 'whatever'), 'order' => 'lft ASC'));
|
|
|
+ * children(); // if you have the tree behavior of course!
|
|
|
+ * or findAllThreaded(); and generates a tree structure of the data.
|
|
|
+ *
|
|
|
+ * Settings (2nd parameter):
|
|
|
+ * 'model' => name of the model (key) to look for in the data array. defaults to the first model for the current
|
|
|
+ * controller. If set to false 2d arrays will be allowed/expected.
|
|
|
+ * 'alias' => the array key to output for a simple ul (not used if element or callback is specified)
|
|
|
+ * 'type' => type of output defaults to ul
|
|
|
+ * 'itemType => type of item output default to li
|
|
|
+ * 'id' => id for top level 'type'
|
|
|
+ * 'class' => class for top level 'type'
|
|
|
+ * 'element' => path to an element to render to get node contents.
|
|
|
+ * 'callback' => callback to use to get node contents. e.g. array(&$anObject, 'methodName') or 'floatingMethod'
|
|
|
+ * 'autoPath' => array($left, $right [$classToAdd = 'active']) if set any item in the path will have the class $classToAdd added. MPTT only.
|
|
|
+ * 'left' => name of the 'lft' field if not lft. only applies to MPTT data
|
|
|
+ * 'right' => name of the 'rght' field if not lft. only applies to MPTT data
|
|
|
+ * 'depth' => used internally when running recursively, can be used to override the depth in either mode.
|
|
|
+ * 'firstChild' => used internally when running recursively.
|
|
|
+ * 'splitDepth' => if multiple "parallel" types are required, instead of one big type, nominate the depth to do so here
|
|
|
+ * example: useful if you have 30 items to display, and you'd prefer they appeared in the source as 3 lists of 10 to be able to
|
|
|
+ * style/float them.
|
|
|
+ * 'splitCount' => the number of "parallel" types. defaults to 3
|
|
|
+ *
|
|
|
+ * @param array $data data to loop on
|
|
|
+ * @param array $settings
|
|
|
+ * @return string html representation of the passed data
|
|
|
+ * @access public
|
|
|
+ */
|
|
|
+ public function generate($data, $settings = array ()) {
|
|
|
+ $this->_settings = array_merge(array(
|
|
|
+ 'model' => null,
|
|
|
+ 'alias' => 'name',
|
|
|
+ 'type' => 'ul',
|
|
|
+ 'itemType' => 'li',
|
|
|
+ 'id' => false,
|
|
|
+ 'class' => false,
|
|
|
+ 'element' => false,
|
|
|
+ 'callback' => false,
|
|
|
+ 'autoPath' => false,
|
|
|
+ 'left' => 'lft',
|
|
|
+ 'right' => 'rght',
|
|
|
+ 'depth' => 0,
|
|
|
+ 'firstChild' => true,
|
|
|
+ 'indent' => null,
|
|
|
+ 'splitDepth' => false,
|
|
|
+ 'splitCount' => 3,
|
|
|
+ ), (array)$settings);
|
|
|
+ if ($this->_settings['autoPath'] && !isset($this->_settings['autoPath'][2])) {
|
|
|
+ $this->_settings['autoPath'][2] = 'active';
|
|
|
+ }
|
|
|
+ extract($this->_settings);
|
|
|
+ if ($indent === null && Configure::read('debug')) {
|
|
|
+ $indent = true;
|
|
|
+ }
|
|
|
+ if ($model === null) {
|
|
|
+ //$model = Inflector::classify($this->_View->params['models'][0]);
|
|
|
+ }
|
|
|
+ if (!$model) {
|
|
|
+ $model = '_NULL_';
|
|
|
+ }
|
|
|
+ $stack = array();
|
|
|
+ if ($depth == 0) {
|
|
|
+ if ($class) {
|
|
|
+ $this->addTypeAttribute('class', $class, null, 'previous');
|
|
|
+ }
|
|
|
+ if ($id) {
|
|
|
+ $this->addTypeAttribute('id', $id, null, 'previous');
|
|
|
+ }
|
|
|
+ }
|
|
|
+ $return = '';
|
|
|
+ if ($indent) {
|
|
|
+ $return = "\r\n";
|
|
|
+ }
|
|
|
+ $_addType = true;
|
|
|
+ foreach ($data as $i => $result) {
|
|
|
+ /* Allow 2d data arrays */
|
|
|
+ if ($model == '_NULL_') {
|
|
|
+ $_result = $result;
|
|
|
+ $result[$model] = $_result;
|
|
|
+ }
|
|
|
+ /* BulletProof */
|
|
|
+ if (!isset($result[$model][$left]) && !isset($result['children'])) {
|
|
|
+ $result['children'] = array();
|
|
|
+ }
|
|
|
+ /* Close open items as appropriate */
|
|
|
+ while ($stack && ($stack[count($stack)-1] < $result[$model][$right])) {
|
|
|
+ array_pop($stack);
|
|
|
+ if ($indent) {
|
|
|
+ $whiteSpace = str_repeat("\t",count($stack));
|
|
|
+ $return .= "\r\n" . $whiteSpace . "\t";
|
|
|
+ }
|
|
|
+ if ($type) {
|
|
|
+ $return .= '</' . $type . '>';
|
|
|
+ }
|
|
|
+ if ($itemType) {
|
|
|
+ $return .= '</' . $itemType . '>';
|
|
|
+ }
|
|
|
+ }
|
|
|
+ /* Some useful vars */
|
|
|
+ $hasChildren = $firstChild = $lastChild = $hasVisibleChildren = false;
|
|
|
+ $numberOfDirectChildren = $numberOfTotalChildren = null;
|
|
|
+ if (isset($result['children'])) {
|
|
|
+ if ($result['children']) {
|
|
|
+ $hasChildren = $hasVisibleChildren = true;
|
|
|
+ $numberOfDirectChildren = count($result['children']);
|
|
|
+ }
|
|
|
+ $prevRow = prev($data);
|
|
|
+ if (!$prevRow) {
|
|
|
+ $firstChild = true;
|
|
|
+ }
|
|
|
+ next($data);
|
|
|
+ $nextRow = next($data);
|
|
|
+ if (!$nextRow) {
|
|
|
+ $lastChild = true;
|
|
|
+ }
|
|
|
+ prev($data);
|
|
|
+ } elseif (isset($result[$model][$left])) {
|
|
|
+ if ($result[$model][$left] != ($result[$model][$right] - 1)) {
|
|
|
+ $hasChildren = true;
|
|
|
+ $numberOfTotalChildren = ($result[$model][$right] - $result[$model][$left] - 1) / 2;
|
|
|
+ if (isset($data[$i + 1]) && $data[$i + 1][$model][$right] < $result[$model][$right]) {
|
|
|
+ $hasVisibleChildren = true;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if (!isset($data[$i - 1]) || ($data[$i - 1][$model][$left] == ($result[$model][$left] - 1))) {
|
|
|
+ $firstChild = true;
|
|
|
+ }
|
|
|
+ if (!isset($data[$i + 1]) || ($stack && $stack[count($stack) - 1] == ($result[$model][$right] + 1))) {
|
|
|
+ $lastChild = true;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ $elementData = array(
|
|
|
+ 'data' => $result,
|
|
|
+ 'depth' => $depth?$depth:count($stack),
|
|
|
+ 'hasChildren' => $hasChildren,
|
|
|
+ 'numberOfDirectChildren' => $numberOfDirectChildren,
|
|
|
+ 'numberOfTotalChildren' => $numberOfTotalChildren,
|
|
|
+ 'firstChild' => $firstChild,
|
|
|
+ 'lastChild' => $lastChild,
|
|
|
+ 'hasVisibleChildren' => $hasVisibleChildren
|
|
|
+ );
|
|
|
+ $this->_settings = array_merge($this->_settings, $elementData);
|
|
|
+ /* Main Content */
|
|
|
+ if ($element) {
|
|
|
+ $content = $this->_View->element($element, $elementData);
|
|
|
+ } elseif ($callback) {
|
|
|
+ list($content) = array_map($callback, array($elementData));
|
|
|
+ } else {
|
|
|
+ $content = $result[$model][$alias];
|
|
|
+ }
|
|
|
+ if (!$content) {
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ $whiteSpace = str_repeat("\t", $depth);
|
|
|
+ if ($indent && strpos($content, "\r\n", 1)) {
|
|
|
+ $content = str_replace("\r\n", "\n" . $whiteSpace . "\t", $content);
|
|
|
+ }
|
|
|
+ /* Prefix */
|
|
|
+ if ($_addType) {
|
|
|
+ if ($indent) {
|
|
|
+ $return .= "\r\n" . $whiteSpace;
|
|
|
+ }
|
|
|
+ if ($type) {
|
|
|
+ $typeAttributes = $this->_attributes($type, array('data' => $elementData));
|
|
|
+ $return .= '<' . $type . $typeAttributes . '>';
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if ($indent) {
|
|
|
+ $return .= "\r\n" . $whiteSpace . "\t";
|
|
|
+ }
|
|
|
+ if ($itemType) {
|
|
|
+ $itemAttributes = $this->_attributes($itemType, $elementData);
|
|
|
+ $return .= '<' . $itemType . $itemAttributes . '>';
|
|
|
+ }
|
|
|
+ $return .= $content;
|
|
|
+ /* Suffix */
|
|
|
+ $_addType = false;
|
|
|
+ if ($hasVisibleChildren) {
|
|
|
+ if ($numberOfDirectChildren) {
|
|
|
+ $settings['depth'] = $depth + 1;
|
|
|
+ $return .= $this->_suffix();
|
|
|
+ $return .= $this->generate($result['children'], $settings);
|
|
|
+ if ($itemType) {
|
|
|
+ $return .= '</' . $itemType . '>';
|
|
|
+ }
|
|
|
+ } elseif ($numberOfTotalChildren) {
|
|
|
+ $_addType = true;
|
|
|
+ $stack[] = $result[$model][$right];
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ if ($itemType) {
|
|
|
+ $return .= '</' . $itemType . '>';
|
|
|
+ }
|
|
|
+ $return .= $this->_suffix();
|
|
|
+ }
|
|
|
+ }
|
|
|
+ /* Cleanup */
|
|
|
+ while ($stack) {
|
|
|
+ array_pop($stack);
|
|
|
+ if ($indent) {
|
|
|
+ $whiteSpace = str_repeat("\t",count($stack));
|
|
|
+ $return .= "\r\n" . $whiteSpace . "\t";
|
|
|
+ }
|
|
|
+ if ($type) {
|
|
|
+ $return .= '</' . $type . '>';
|
|
|
+ }
|
|
|
+ if ($itemType) {
|
|
|
+ $return .= '</' . $itemType . '>';
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if ($indent) {
|
|
|
+ $return .= "\r\n";
|
|
|
+ }
|
|
|
+ if ($type) {
|
|
|
+ $return .= '</' . $type . '>';
|
|
|
+ if ($indent) {
|
|
|
+ $return .= "\r\n";
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return $return;
|
|
|
+ }
|
|
|
+/**
|
|
|
+ * addItemAttribute function
|
|
|
+ *
|
|
|
+ * Called to modify the attributes of the next <item> to be processed
|
|
|
+ * Note that the content of a 'node' is processed before generating its wrapping <item> tag
|
|
|
+ *
|
|
|
+ * @param string $id
|
|
|
+ * @param string $key
|
|
|
+ * @param mixed $value
|
|
|
+ * @access public
|
|
|
+ * @return void
|
|
|
+ */
|
|
|
+ public function addItemAttribute($id = '', $key = '', $value = null) {
|
|
|
+ if (!is_null($value)) {
|
|
|
+ $this->_itemAttributes[$id][$key] = $value;
|
|
|
+ } elseif (!(isset($this->_itemAttributes[$id]) && in_array($key, $this->_itemAttributes[$id]))) {
|
|
|
+ $this->_itemAttributes[$id][] = $key;
|
|
|
+ }
|
|
|
+ }
|
|
|
+/**
|
|
|
+ * addTypeAttribute function
|
|
|
+ *
|
|
|
+ * Called to modify the attributes of the next <type> to be processed
|
|
|
+ * Note that the content of a 'node' is processed before generating its wrapping <type> tag (if appropriate)
|
|
|
+ * An 'interesting' case is that of a first child with children. To generate the output
|
|
|
+ * <ul> (1)
|
|
|
+ * <li>XYZ (3)
|
|
|
+ * <ul> (2)
|
|
|
+ * <li>ABC...
|
|
|
+ * ...
|
|
|
+ * </ul>
|
|
|
+ * ...
|
|
|
+ * The processing order is indicated by the numbers in brackets.
|
|
|
+ * attributes are allways applied to the next type (2) to be generated
|
|
|
+ * to set properties of the holding type - pass 'previous' for the 4th param
|
|
|
+ * i.e.
|
|
|
+ * // Hide children (2)
|
|
|
+ * $this->Tree->addTypeAttribute('style', 'display', 'hidden');
|
|
|
+ * // give top level type (1) a class
|
|
|
+ * $this->Tree->addTypeAttribute('class', 'hasHiddenGrandChildren', null, 'previous');
|
|
|
+ *
|
|
|
+ * @param string $id
|
|
|
+ * @param string $key
|
|
|
+ * @param mixed $value
|
|
|
+ * @access public
|
|
|
+ * @return void
|
|
|
+ */
|
|
|
+ public function addTypeAttribute($id = '', $key = '', $value = null, $previousOrNext = 'next') {
|
|
|
+ $var = '__typeAttributes';
|
|
|
+ $firstChild = isset($this->_settings['firstChild'])?$this->_settings['firstChild']:true;
|
|
|
+ if ($previousOrNext == 'next' && $firstChild) {
|
|
|
+ $var = '__typeAttributesNext';
|
|
|
+ }
|
|
|
+ if (!is_null($value)) {
|
|
|
+ $this->{$var}[$id][$key] = $value;
|
|
|
+ } elseif (!(isset($this->{$var}[$id]) && in_array($key, $this->{$var}[$id]))) {
|
|
|
+ $this->{$var}[$id][] = $key;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+/**
|
|
|
+ * supressChildren method
|
|
|
+ *
|
|
|
+ * @return void
|
|
|
+ * @access public
|
|
|
+ */
|
|
|
+ public function supressChildren() {
|
|
|
+ }
|
|
|
+/**
|
|
|
+ * suffix method
|
|
|
+ *
|
|
|
+ * Used to close and reopen a ul/ol to allow easier listings
|
|
|
+ *
|
|
|
+ * @access private
|
|
|
+ * @return void
|
|
|
+ */
|
|
|
+ public function _suffix() {
|
|
|
+ static $_splitCount = 0;
|
|
|
+ static $_splitCounter = 0;
|
|
|
+ extract($this->_settings);
|
|
|
+ if ($splitDepth) {
|
|
|
+ if ($depth == $splitDepth -1) {
|
|
|
+ $total = $numberOfDirectChildren?$numberOfDirectChildren:$numberOfTotalChildren;
|
|
|
+ if ($total) {
|
|
|
+ $_splitCounter = 0;
|
|
|
+ $_splitCount = $total / $splitCount;
|
|
|
+ $rounded = (int)$_splitCount;
|
|
|
+ if ($rounded < $_splitCount) {
|
|
|
+ $_splitCount = $rounded + 1;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if ($depth == $splitDepth) {
|
|
|
+ $_splitCounter++;
|
|
|
+ if ($type && ($_splitCounter % $_splitCount) == 0) {
|
|
|
+ return '</' . $type . '><' . $type . '>';
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return;
|
|
|
+ }
|
|
|
+/**
|
|
|
+ * attributes function
|
|
|
+ *
|
|
|
+ * Logic to apply styles to tags.
|
|
|
+ *
|
|
|
+ * @param mixed $rType
|
|
|
+ * @param array $elementData
|
|
|
+ * @access private
|
|
|
+ * @return void
|
|
|
+ */
|
|
|
+ public function _attributes($rType, $elementData = array(), $clear = true) {
|
|
|
+ extract($this->_settings);
|
|
|
+ if ($rType == $type) {
|
|
|
+ $attributes = $this->_typeAttributes;
|
|
|
+ if ($clear) {
|
|
|
+ $this->_typeAttributes = $this->_typeAttributesNext;
|
|
|
+ $this->_typeAttributesNext = array();
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ $attributes = $this->_itemAttributes;
|
|
|
+ $this->_itemAttributes = array();
|
|
|
+ if ($clear) {
|
|
|
+ $this->_itemAttributes = array();
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if ($autoPath && $depth) {
|
|
|
+ if ($this->_settings['data'][$model][$left] < $autoPath[0] && $this->_settings['data'][$model][$right] > $autoPath[1]) {
|
|
|
+ $attributes['class'][] = $autoPath[2];
|
|
|
+ } elseif (isset($autoPath[3]) && $this->_settings['data'][$model][$left] == $autoPath[0]) {
|
|
|
+ $attributes['class'][] = $autoPath[3];
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if ($attributes) {
|
|
|
+ foreach ($attributes as $type => $values) {
|
|
|
+ foreach ($values as $key => $val) {
|
|
|
+ if (is_array($val)) {
|
|
|
+ $attributes[$type][$key] = '';
|
|
|
+ foreach ($val as $vKey => $v) {
|
|
|
+ $attributes[$type][$key][$vKey] .= $vKey . ':' . $v;
|
|
|
+ }
|
|
|
+ $attributes[$type][$key] = implode(';', $attributes[$type][$key]);
|
|
|
+ }
|
|
|
+ if (is_string($key)) {
|
|
|
+ $attributes[$type][$key] = $key . ':' . $val . ';';
|
|
|
+ }
|
|
|
+ }
|
|
|
+ $attributes[$type] = $type . '="' . implode(' ', $attributes[$type]) . '"';
|
|
|
+ }
|
|
|
+ return ' ' . implode(' ', $attributes);
|
|
|
+ }
|
|
|
+ return '';
|
|
|
+ }
|
|
|
+}
|