Yii2:在网格视图中显示和过滤关系表


Yii2: Show and filter relational table in gridview

>我有两个表:People & Categories。

- id
- name
- ...

类别

- id
- name
- description

另一个表,其中包含两者之间的关系,因为一个人可以分配多个类别:

人物类别协会

- peopleID
- categoryID

我可以在我的网格视图(视图/人员/索引.php)中显示类别 ID:

<?= GridView::widget([
    'dataProvider' => $dataProvider,
    'filterModel' => $searchModel,
    'columns' => [
        ...,
        [
            'label' => 'Category',
            'value' => function ($data) {
                $output = '';
                foreach($data->peopleCategoriesAssns as $request) {
                    $output .= $request->talentCategoryID.'<br>';
                }
                return $output;
            },
        ],
        ...

它从模型/人员中使用它.php

public function getPeopleCategoriesAssns() 
{ 
    return $this->hasMany(PeopleCategoriesAssn::className(), ['peopleID' => 'id']); 
} 

如何在网格视图中显示类别名称而不是分类 ID,以及我应该在/models/PeopleSearch.php 上做什么才能允许搜索/筛选该字段?

提前谢谢你,

要回答第一部分,您需要向 People 类添加另一个方法:

public function getCategories() {
    return $this->hasMany( Categories::className(), [ 'id' => 'categoryID' ] )
                ->via('PeopleCategoriesAssn');
}

参见: http://www.yiiframework.com/doc-2.0/yii-db-activerelationtrait.html#via()-detail

这将允许您执行以下操作:

foreach( $data->categories as $category) {
    $output .= $category->name.'<br>';
}

至于搜索,我不确定,但是将公共$categoryName属性添加到搜索类将是一个很好的起点。

您的问题涉及两个部分。

1. 显示值

这只是对GridView列的重新规范:

'columns'=>[
    //...
    [
        'attribute'=>'catNameSearch',
        'value'=>function ($model, $key, $index, $column) {
            $cats = $model->peopleCategoriesAssns;
            if (empty($cats)) {
                return null;
            } else {
                return implode(', ', ArrayHelper::getColumn($cats, 'name');
            }
        },
    ],
    //...
]

2. 搜索猫名

现在要解决搜索它们的问题,它变得有点复杂。如您所见,我将上面的属性设置为 catNameSearch .您需要将其作为公共 var 添加到您的 PeopleSearch -class 中,并将其声明为搜索类的验证器中的safe

class PeopleSearch extends 'app'models'People
{
    public $catNameSearch;
    public function rules()
    {
        return [
            //...
            [['catNameSearch'], 'safe'],
            //...
        ];
    }
 }

在搜索方法本身中,您可以添加一个子查询以仅显示分配给相应类别的人员。这可以像这样完成:

if (!empty($this->catNameSearch)) {
    $peopleIdsMatching = PeopleCategoriesAssn::find()
         ->select('peopleID')
         ->distinct(true)
         ->joinWidth('category')
         ->andWhere(['like', ['category.name'=>$this->catNameSearch]])
        ->column();
     $this->andWhere(['id'=>$peopleIdsMatching]);
}

现在,仅显示与给定类别相关的人员。这可以通过在人员搜索的实际子查询中执行比较而不是执行单独的查询来进一步优化,但这应该为您提供基本概念!