11

我有一个类别表。每个类别都可以有一个可选的父级(如果没有父级,则默认为 0)。

我想要做的是构建一个简单的 html 列表树,其中包含类别的级别。

示例日期:

Foods
-- Fruit
---- Apple
---- Banana
---- Orange
-- Veg
---- Cucumber
---- Lettuce
Drinks
-- Alcoholic
---- Beer
---- Vodka
Misc
-- Household Objects
---- Kitchen
------ Electrical
-------- Cooking
---------- Stove
---------- Toaster
---------- Microwave

请注意,这需要大约 10 个“级别”。我希望它是无限的,但我真的不想走使用嵌套集合模型的路线,因为它会导致这个项目的巨大延迟。

laravel 的文档很糟糕,甚至没有真正参考从哪里开始。几天来,我一直在玩它,试图弄清楚该怎么做,如果没有一个巨大的混乱块,彼此之间的每个循环 10 次,我似乎一无所获。

我在模型中使用以下内容获得了数据树:

<?php
class Item extends Eloquent {
    public function parent()
    {
        return $this->hasOne('Item', 'id', 'parent_id');
    }

    public function children()
    {
        return $this->hasMany('Item', 'parent_id', 'id');
    }

    public function tree()
    {
        return static::with(implode('.', array_fill(0,10, 'children')))->where('parent_id', '=', '0')->get();
    }
}

这使所有父级和子级都达到了 10 级。这工作正常,但是如果没有手动在彼此之间设置 10 个 foreach 循环,您就不能真正对子数据执行任何操作。

我在这里做错了什么?当然这不应该这么难/执行不力吗?我想要做的就是获得一个简单的 html 列表,其中包含树结构中的项目。

我已经汇总了上面使用的虚拟数据的快速 SQLFiddle 示例:http ://sqlfiddle.com/#!2/e6d18/1

4

3 回答 3

39

这比我平常早上的填字游戏有趣得多。:)

这是一个 ItemsHelper 类,它将完成您正在寻找的事情,而且更好的是,它会尽可能地递归下去。

app/models/ItemsHelper.php:

<?php

  class ItemsHelper {

    private $items;

    public function __construct($items) {
      $this->items = $items;
    }

    public function htmlList() {
      return $this->htmlFromArray($this->itemArray());
    }

    private function itemArray() {
      $result = array();
      foreach($this->items as $item) {
        if ($item->parent_id == 0) {
          $result[$item->name] = $this->itemWithChildren($item);
        }
      }
      return $result;
    }

    private function childrenOf($item) {
      $result = array();
      foreach($this->items as $i) {
        if ($i->parent_id == $item->id) {
          $result[] = $i;
        }
      }
      return $result;
    }

    private function itemWithChildren($item) {
      $result = array();
      $children = $this->childrenOf($item);
      foreach ($children as $child) {
        $result[$child->name] = $this->itemWithChildren($child);
      }
      return $result;
    }

    private function htmlFromArray($array) {
      $html = '';
      foreach($array as $k=>$v) {
        $html .= "<ul>";
        $html .= "<li>".$k."</li>";
        if(count($v) > 0) {
          $html .= $this->htmlFromArray($v);
        }
        $html .= "</ul>";
      }
      return $html;
    }
  }

我刚刚使用了新安装的 Laravel 4 和基本hello.php视图。

这是我的路线app/routes.php

Route::get('/', function()
{
  $items = Item::all();
  $itemsHelper = new ItemsHelper($items);
  return View::make('hello',compact('items','itemsHelper'));
});

尽管我的视图不使用 items 变量,但我将它传递到这里是因为您可能也想对它们做其他事情。

最后,我app/views/hello.php只有一行:

<?= $itemsHelper->htmlList(); ?>

输出如下所示:

  • 食品
    • 水果
      • 苹果
      • 香蕉
      • 橙子
    • 蔬菜
      • 黄瓜
      • 莴苣
  • 饮料
    • 酒鬼
      • 啤酒
      • 伏特加酒
  • 杂项
    • 家居用品
      • 厨房
        • 电气
          • 烹饪
            • 火炉
            • 烤面包机
            • 微波

注意:您的 SQL Fiddle 有 5(“Orange”)作为 Cucumber 和 Lettuce 的 parent_id,我必须将其更改为 6(“Veg”)。

于 2013-05-14T16:04:07.780 回答
2

我已经扩展了 Mark Smith 接受的答案,以允许生成的列表引用传递给类的附加数据。

该类的工作方式几乎相同,但我已将其打包,因此希望它可以轻松使用。

只需在控制器中引用帮助程序类:

use App\Helpers\CategoryHierarchy;

然后你可以手动实例化这个类,或者使用 Laravel 5 的方法注入,事情会变得更好:

$products = $product->getAllProducts();
$hierarchy->setupItems($products);
return $hierarchy->render();

这可以输出以下内容:

<ul class='simple-list'>
   <li><input type="checkbox" name="hierarchy-checkboxes[]" value="1" >Home delivery</li>
      <ul>
        <li><input type="checkbox" name="hierarchy-checkboxes[]" value="2" >Italian</li>
          <ul>
            <li><input type="checkbox" name="hierarchy-checkboxes[]" value="3" >Pizza</li>
            <li><input type="checkbox" name="hierarchy-checkboxes[]" value="4" >Pasta</li>
          </ul>
        <li><input type="checkbox" name="hierarchy-checkboxes[]" value="5" >Burgers</li>
      </ul>
</ul>

有一个 repo 可用:https ://github.com/amochohan/CategoryHierarchy更详细地解释了。

于 2015-01-19T13:39:54.823 回答
1

我使用此功能使其工作。

 //Returns Root elements
 public function scopeRoot($query) {
        $all = $query->whereParent(0)->get();
        $collection = $all->filter(function($single) {
            if ($single->ModelFilter('GET')) {
                return $single;
            }
        });
        return $collection;
    }
    //Recursive call
    public function traverse() {
        self::_traverse($this->Children, $array, $this);
        return $array;
    }

    //This function build a multidimensional array based on a collection of elements
    private static function _traverse($collection, &$array, $object) {
        $new_array = array();
        foreach ($collection as $element) {
            self::_traverse($element->Children, $new_array, $element);
        }
        $array[] = $object;
        if (count($new_array) > 0) {
            $array[] = $new_array;
        }
    }

首先,我得到一个根元素的集合,这些元素是我传递给我想要列出树的视图的那些......

那我做...

 <ul class="bg-info cat-list">
        @foreach($categories as $category)
        <?php
        $array = $category->traverse();
        list_view($array);
        ?>
        @endforeach
    </ul>

使用这个功能...

//Prints a multidimensional array as a nested HTML list
function list_view($element, $ul = true) {
    foreach ($element as $value) {
        if (!is_array($value)) {
            echo "<li>";
            echo $value->name;
        } else {
            echo ($ul) ? "<ul>" : "<ol>";
            list_view($valuce, $ul);
            echo "</li>";
            echo ($ul) ? "</ul>" : "</ol>";
        }
    }
}

输出

希望能帮助到你

于 2014-07-24T18:38:18.993 回答