GridView 小部件在开发中常用的功能及技巧。持续更新中....
数据网格或者说 GridView 小部件是Yii中最强大的部件之一。它有一个属性名叫 dataProvider ,这个属性能够提供一个数据提供者的示例并且可以显示所提供的数据,即使用 yii\grid\GridView::columns 属性的一组列配置,在一个表格中渲染每一行数据。
例如,
- use yii\grid\GridView;
- echo yii\grid\GridView::widget([
- 'dataProvider' => $dataProvider,
- ]);
GridView去掉排序,去掉搜索
- public function actionIndex()
- {
- $searchModel = new SupplierManageSearch();
- $dataProvider = $searchModel->search(Yii::$app->request->queryParams);
- $dataProvider->setSort(false); //去掉排序
- return $this->render('index', [
- 'searchModel' => $searchModel,
- 'dataProvider' => $dataProvider,
- ]);
- }
- <?= GridView::widget([
- 'dataProvider' => $dataProvider,
- //去掉搜索 'filterModel' => $searchModel,
- 'columns' => [
- // 'id',
- 'cod',
- 'nme',
- ['class' => 'yii\grid\ActionColumn'],
- ],
- ]); ?>
一、表格列
表格的列是通过 GridView 配置项中的 yii\grid\GridView::columns 属性配置的.
- <?php
- use yii\grid\GridView;
- echo GridView::widget([
- 'dataProvider' => $dataProvider,
- //表格列值搜索功能,注意一定要配合attribute才会显示
- //$searchModel = new ArticleSearch();
- 'filterModel' => $searchModel,
- //重新定义分页样式
- 'layout'=> '{items}<div class="text-right tooltip-demo">{pager}</div>',
- 'pager'=>[
- //'options'=>['class'=>'hidden']//关闭分页
- 'firstPageLabel'=>"First",
- 'prevPageLabel'=>'Prev',
- 'nextPageLabel'=>'Next',
- 'lastPageLabel'=>'Last',
- ]
- 'columns' => [
- ['class' => 'yii\grid\SerialColumn'],//序列号从1自增长
- // 数据提供者中所含数据所定义的简单的列
- // 使用的是模型的列的数据
- 'id',
- 'username',
- // 更复杂的列数据
- [
- 'class' => 'yii\grid\DataColumn', //由于是默认类型,可以省略
- 'value' => function ($data) {
- return $data->name;
- // 如果是数组数据则为 $data['name'] ,
- 例如,使用 SqlDataProvider 的情形。
- },
- ],
- ['label'=>'标题','value' => 'title'],
- ['label'=>'文章内容','format' => 'html','value' => 'content'],
- [
- 'label'=>'文章类别',
- /*'attribute' => 'cid',产生一个a标签,点击可排序*/
- 'value' => 'cate.cname' //关联表
- ],
- [
- //动作列yii\grid\ActionColumn
- //用于显示一些动作按钮,如每一行的更新、删除操作。
- 'class' => 'yii\grid\ActionColumn',
- 'header' => '操作',
- 'template' => '{delete} {update}',//只需要展示删除和更新
- 'headerOptions' => ['width' => '240'],
- 'buttons' => [
- 'delete' => function($url, $model, $key){
- return Html::a('<i class="fa fa-ban"></i> 删除',
- ['del', 'id' => $key],
- [
- 'class' => 'btn btn-default btn-xs',
- 'data' => ['confirm' => '你确定要删除文章吗?',]
- ]
- );
- },
- ],
- ],
- ],
- ]);
- ?>
1. 处理时间
数据列的主要配置项是 yii\grid\DataColumn::format 属性。它的值默认是使用 \yii\i18n\Formatter 应用组件。
- [
- 'label'=>'更新日期',
- 'format' => ['date', 'php:Y-m-d'],
- 'value' => 'updated_at'
- ],
- //or
- [
- //'attribute' => 'created_at',
- 'label'=>'更新时间',
- 'value'=>function($model){
- return date('Y-m-d H:i:s',$model->created_at);
- },
- 'headerOptions' => ['width' => '170'],
- ],
- 'start_time:datetime',
- [
- 'label'=>'创建日期',
- 'attribute' => 'created_at',
- 'filter' => false, //不显示搜索框
- 'value' => function($data) {
- return date('Y-m-d H:i:s',$data->created_at); }
- ],
- [
- 'label'=>'创建日期',
- 'attribute' => 'created_at',
- 'format' => ['date', 'php:Y-m-d H:i:s'],
- ],
2. 处理图片
- [
- 'label'=>'封面图',
- 'format'=>'raw',
- 'value'=>function($m){
- return Html::img($m->cover,
- ['class' => 'img-circle',
- 'width' => 30]
- );
- }
- ],
3. 数据列有链接
- [
- 'attribute' => 'title',
- 'value' => function ($model, $key, $index, $column) {
- return Html::a($model->title,
- ['article/view', 'id' => $key]);
- },
- 'format' => 'raw',
- ],
- [
- 'attribute'=>'title’,
- 'format'=>'raw’,
- 'value'=> function($data){return Html::a($data->title,['exam/index',
- 'id' => $data->_id],['title' => '审核']);}
- ],
- [
- 'label'=>'更多操作’,
- 'format'=>'raw’,
- 'value' => function($data) {
- $url = "http://www.baidu.com”;
- return Html::a('添加权限组', $url, ['title' => '审核']); }
- ],
4. 数据列显示枚举值(男/女)
- [
- 'attribute' => 'sex',
- 'value'=>function ($model,$key,$index,$column){
- return $model->sex==1?'男':'女';
- },
- //在搜索条件(过滤条件)中使用下拉框来搜索
- 'filter' => ['1'=>'男','0'=>'女'],
- //or
- 'filter' => Html::activeDropDownList($searchModel,
- 'sex',['1'=>'男','0'=>'女'],
- ['prompt'=>'全部']
- )
- ],
- [
- 'label'=>'产品状态',
- 'attribute' => 'pro_name',
- 'value' => function ($model) {
- $state = [
- '0' => '未发货',
- '1' => '已发货',
- '9' => '退货,已处理',
- ];
- return $state[$model->pro_name];
- },
- 'headerOptions' => ['width' => '120']
- ]
- [
- 'attribute' => 'type',
- 'label' => '进/出场',
- 'value' => function ($model) {
- if ($model->type == 1) {
- $txt = "进场";
- if ($model->FlagPark) {
- $txt .= " - <span class='text-danger'>已出场</span>";
- }
- return $txt;
- } else if ($model->type == 2) {
- return "出场";
- } else {
- return "";
- }
- },
- 'filter' => Html::activeDropDownList($searchModel, 'type', ['1' => '进场', '0' => '出场'], ['prompt' => '全部', 'class' => 'form-control']),
- 'format' => 'raw',
- ],
5、下拉菜单搜索
- [
- 'label' => '当前状态',
- 'attribute' => 'status',
- 'filter' => Html::dropDownList('AppBaseSearch[status
- ]',$searchModel-> status,
- [ '' => '请选择',
- ‘0' => '审核拒绝',
- ‘1' => '审核通过',
- ]),
- 'value' => function($date) {
- switch ($date-> status) {
- case 1';
- return '审核通过';
- break;
- case ‘0';
- return '审核拒绝';
- break;
- default:
- return '未知状态';
- break;
- }
- }
- ],
- use yii\grid\GridView;
- use yii\helpers\Html;
- GridView::widget([
- 'dataProvider' => $dataProvider,
- 'filterModel' => $searchModel,
- 'columns' => [
- 'tag_id',
- 'tag_name',
- [
- 'attribute' => 'tag_style',
- //'filter' => $model->tagStyle(), //下拉选择
- 'filter' => Html::activeDropDownList($searchModel,
- 'tag_style', $model->tagStyle(),
- ['class' => 'form-control',] //'prompt'=>'全部'
- )
- ],
- ]
- ]);
- use yii\grid\GridView;
- use yii\helpers\Html;
- GridView::widget([
- 'dataProvider' => $dataProvider,
- 'filterModel' => $searchModel,
- 'columns' => [
- 'tag_id',
- 'tag_name',
- [
- 'attribute' => 'tag_style',
- //'filter' => $model->tagStyle(), //下拉选择
- 'filter' => Html::activeDropDownList($searchModel,
- 'tag_style', $model->tagStyle(),
- ['class' => 'form-control','prompt'=>'全部']
- )
- ],
- ]
- ]);
1、是否显示某列
我们举一个简单的案例
条件:有一个get形参数type
需求:仅且type的值等于1的时候,列name才显示,否则该列不显示。
- [
- 'attribute' => 'name',
- 'value' => $model->name,
- 'visible' => intval(Yii::$app->request->get('type')) == 1,
- ],
2、链接可点击跳转案例
这个跟接下来我们要说的html渲染的效果十分类似,这里要说的是列的属性值 format,具体都有哪些格式可查看文件 yii\i18n\Formatter.php,各种format都可以解决
- [
- 'attribute' => 'order_id',
- 'value' => function ($model) {
- return Html::a($model->order_id, "/order?id={$model->order_id}", ['target' => '_blank']);
- },
- 'format' => 'raw',
- ],
3、显示图片案例
同上,这里只需要指定format格式为image即可,format第二个参数可设定图片大小,可参考下面的代码
- [
- 'label' => '头像',
- 'format' => [
- 'image',
- [
- 'width'=>'84',
- 'height'=>'84'
- ]
- ],
- 'value' => function ($model) {
- return $model->image;
- }
- ],
4、html渲染案例
title含有html标签,我们不想在页面上展示<p>title123<p>这种形式,我们想要title123以p标签的形式展示,代码可参考如下,只需要指定format为raw形式即可:
- [
- 'attribute' => 'title',
- 'value' => function ($model) {
- return Html::encode($model->title);
- },
- 'format' => 'raw',
- ],
5、自定义按钮案例
往往列表页我们不想要删除按钮,想在增加一个比如获取xxx按钮,怎么搞呢?这里需要设置ActionColumn类,修改配置项template并在buttons项增加template里增加的get-xxx即可
- [
- 'class' => 'yii\grid\ActionColumn',
- 'template' => '{get-xxx} {view} {update}',
- 'header' => '操作',
- 'buttons' => [
- 'get-xxx' => function ($url, $model, $key) {
- return Html::a('获取xxx', $url, ['title' => '获取xxx'] );
- },
- ],
- ],
6、设定宽度案例
举个简单的例子,我们的title列,太宽了,定下这一列的宽度
- [
- 'attribute' => 'title',
- 'value' => 'title',
- 'headerOptions' => ['width' => '100'],
- ],
只需要指定配置项headerOptions即可。
7、自定义字段案例
啥时自定义?这里我们是指在表格里增加一列且数据库中不存在对应的列。假如我们新增一列 订单消费金额money且该表不存在该字段
- [
- 'attribute' => '消费金额',
- 'value' => function ($model) {
- // 这里可以根据该表的其他字段进行关联获取
- }
- ],
8、自定义行样式
gii生成的这个gridview表格,行跟行的颜色不明显,我们来看看怎么定义行样式
- <?= GridView::widget([
- // ......
- 'dataProvider' => $dataProvider,
- 'rowOptions' => function($model, $key, $index, $grid) {
- return ['class' => $index % 2 ==0 ? 'label-red' : 'label-green'];
- },
- // ......
- ]); ?>
前面的操作我们都是依据列column的,这里因为是对行的控制,所以我们配置rowOptions要稍微注意一下。此外,自定义的label-red和label-green需要有对应的样式实现,这里我们看一下页面的实际效果
9、增加按钮调用js操作案例
如果修改状态的功能很频繁,每次都要先点进详情页才能修改,加载一个异步请求操作了当前行的状态,我们来看看gridview里面是怎么实现的。
- [
- 'class' => 'yii\grid\ActionColumn',
- 'header' => '操作',
- 'template' => '{view} {update} {update-status}',
- 'buttons' => [
- 'update-status' => function ($url, $model, $key) {
- return Html::a('更新状态', 'javascript:;', ['onclick'=>'update_status(this, '.$model->id.');']); },
- ],
- ],
我们需要在页面写js实现方法 update_status
一、
1、model,对当前model的category_id关联分类表的id:
- public function getCategory()
- {
- return $this->hasOne(lifeCategory::className(), ['id' => 'category_id']);
- }
2、searchModel关联一下定义好的get方法:
3、view中调用:
- 'category.name', //category.字段名
二、
1、定义hasone方法,
2、view中调用:
- [
- 'attribute' => 'category_id',
- 'enableSorting' => true,
- 'value' => function ($model) {
- return $model->category->name;
- },
- ],
更改checkbox的全选框加文字参数为head
- [
- 'class' => 'yii\grid\CheckboxColumn',
- 'header' => Html::checkBox('selection_all', false, [
- 'class' => 'select-on-check-all',
- 'label' => 'Check All',
- ]),
- 'checkboxOptions' => function($model) {
- return [
- 'class' => "selection",
- 'value' => $model['id']
- ];
- },
- ],
结合GridView加全选:
- <?= GridView::widget([
- 'dataProvider' => $dataProvider,
- // id是必须要加的,获取#grid下的select项
- 'options' => ['class' => 'table table-bordered table-hover table-responsive','id' => 'grid'],
- 'columns' => [
- // 'id',
- [
- // 信息id
- 'class' => 'yii\grid\CheckboxColumn',
- 'name' => 'id',
- ],
- ],
- ]); ?>
显示删除按钮,并加点击事件:
- <?= Html::a('批量删除', "javascript:void(0);", ['class' => 'btn btn-success gridview']) ?>
- <?php
- $this->registerJs('
- $(".gridview").on("click", function () {
- //注意这里的$("#grid"),要跟我们第一步设定的options id一致
- var keys = $("#grid").yiiGridView("getSelectedRows");
- console.log(keys);
- });
- ');
- ?>
Yii2 使用gii生成 gridview分页时候 显示第一页和最后一页 方法
在Yii2 使用gii生成 gridview分页时候 model中是ActiveDataProvider ,默认是没有第一页,最后一页的, 要想显示第一页和最后一页,不需要加载其他东西,只需要在view中增加个pager属性
- <?php
- use yii\grid\GridView;
- echo GridView::widget([
- 'dataProvider' => $dataProvider,
- //每列都有搜索框 控制器传过来$searchModel = new ArticleSearch();
- //'filterModel' => $searchModel,
- 'layout'=> '{items}<div class="text-right tooltip-demo">{pager}</div>',
- 'pager'=>[
- //'options'=>['class'=>'hidden']//关闭自带分页
- 'firstPageLabel'=>"First",
- 'prevPageLabel'=>'Prev',
- 'nextPageLabel'=>'Next',
- 'lastPageLabel'=>'Last',
- ],
- 'columns' => [
- //['class' => 'yii\grid\SerialColumn'],//序列号从1开始
- // 数据提供者中所含数据所定义的简单的列
- // 使用的是模型的列的数据
- 'id',
- 'username',
- ['label'=>'文章类别', /*'attribute' => 'cid',产生一个a标签,点击可排序*/ 'value' => 'cate.cname' ],
- ['label'=>'发布日期','format' => ['date', 'php:Y-m-d'],'value' => 'created_at'],
- // 更复杂的列数据
- ['label'=>'封面图','format'=>'raw','value'=>function($m){
- return Html::img($m->cover,['class' => 'img-circle','width' => 30]);
- }],
- [
- 'class' => 'yii\grid\DataColumn', //由于是默认类型,可以省略
- 'value' => function ($data) {
- return $data->name;
- // 如果是数组数据则为 $data['name'] ,例如,使用
- SqlDataProvider 的情形。
- },
- ],
- [
- 'class' => 'yii\grid\ActionColumn',
- 'header' => '操作',
- 'template' => '{delete} {update}',//只需要展示删除和更新
- /*'headerOptions' => ['width' => '80'],*/
- 'buttons' => [
- 'delete' => function($url, $model, $key){
- return Html::a('<i class="glyphicon glyphicon-trash"></i> 删除',
- ['artdel', 'id' => $key],
- ['class' => 'btn btn-default btn-xs',
- 'data' => ['confirm' => '你确定要删除文章吗?',]
- ]);
- },
- 'update' => function($url, $model, $key){
- return Html::a('<i class="fa fa-file"></i> 更新',
- ['artedit', 'id' => $key],
- ['class' => 'btn btn-default btn-xs']);
- },
- ],
- ],
- ],
- ]);
- ?>
gridview 上面加搜索表单
- <?php $form = ActiveForm::begin([
- 'action' => ['index'],
- 'method' => 'get',
- 'id' => 'cateadd-form',
- 'options' => ['class' => 'form-horizontal'],
- ]); ?>
- <?= $form->field($searchModel, 'tag_name', [
- 'options' => ['class' => ''],
- 'inputOptions' => ['placeholder' => '文章搜索', 'class' => 'input-sm form-control'],
- ])->label(false) ?>
- <span class="input-group-btn">
- <?= Html::submitButton('Go!', ['class' => 'btn btn-sm btn-primary']) ?>
- </span>
- <?php ActiveForm::end(); ?>