热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Yii框架中通过继承CGridView实现完全自定义gridview视图方法详解

Yii框架中通过继承CGridView实现完全自定义gridview视图方法详解

今天在项目中要实现表格的翻页功能,很自然联想到Yii框架后台admin页面的gridview视图,也是表格的形式展示所有内容,然后有分页功能,所以准备借鉴其实现思路来实现。

由于在具体的实现过程中,涉及到很多前台样式及界面的修改,功能的增删,Yii内置的CGridView已经不足以完全满足我的需求,所以首先要开刀的就是继承CGridView实现一个自己的挂件类TGridView,主要是删除一些内置的class及id,在CGridView中渲染表格单元还调用到CDataColumn及CGridColumn,所以一并自定义了TDataColumn和TGridColumn(分别继承自CDataColumn和CGridColumn,其中CDataColumn继承自CGridColumn,这里我的处理方式是TDataColumn继承自TGridColumn,TGridColumn继承自CGridColumn),这样一套下来完成整个gridview生成表格的界面及功能的自定义,具体代码如下:

TGridView.php:

class TGridView extends CGridView{
protected function initColumns()
{
if($this->columns===array())
{
if($this->dataProvider instanceof CActiveDataProvider)
$this->columns=$this->dataProvider->model->attributeNames();
elseif($this->dataProvider instanceof IDataProvider)
{
// use the keys of the first row of data as the default columns
$data=$this->dataProvider->getData();
if(isset($data[0]) && is_array($data[0]))
$this->columns=array_keys($data[0]);
}
}
$id=$this->getId();
foreach($this->columns as $i=>$column)
{
if(is_string($column))
$column=$this->createDataColumn($column);
else
{
if(!isset($column['class']))
$column['class']='CDataColumn';
$column=Yii::createComponent($column, $this);
}
if(!$column->visible)
{
unset($this->columns[$i]);
continue;
}
$this->columns[$i]=$column;
}

foreach($this->columns as $column)
$column->init();
}

public function renderItems()
{
if($this->dataProvider->getItemCount()>0 || $this->showTableOnEmpty)
{
echo "\n";
$this->renderTableHeader();
ob_start();
$this->renderTableBody();
$body=ob_get_clean();
$this->renderTableFooter();
echo $body; // TFOOT must appear before TBODY according to the standard.
echo "
"; } else $this->renderEmptyText(); } public function renderTableHeader() { if(!$this->hideHeader) { if($this->filterPosition===self::FILTER_POS_HEADER) $this->renderFilter(); echo "\n"; foreach($this->columns as $column) $column->renderHeaderCell(); echo "\n"; if($this->filterPosition===self::FILTER_POS_BODY) $this->renderFilter(); } elseif($this->filter!==null && ($this->filterPosition===self::FILTER_POS_HEADER || $this->filterPosition===self::FILTER_POS_BODY)) { $this->renderFilter(); } } public function renderTableBody() { $data=$this->dataProvider->getData(); $n=count($data); if($n>0) { for($row=0;$row<$n;++$row) $this->renderTableRow($row); } else { echo ''; $this->renderEmptyText(); echo "\n"; } } protected function createDataColumn($text) { if(!preg_match('/^([\w\.]+)(:(\w*))?(:(.*))?$/',$text,$matches)) throw new CException(Yii::t('zii','The column must be specified in the format of "Name:Type:Label", where "Type" and "Label" are optional.')); $column=new TDataColumn($this); //自定义TDataColumn渲染表格单元内容及样式 $column->name=$matches[1]; //获取到列名如id,name,type.name之类 if(isset($matches[3]) && $matches[3]!=='') $column->type=$matches[3]; if(isset($matches[5])) $column->header=$matches[5]; return $column; } }

TDataColumn.php:

class TDataColumn extends TGridColumn{
protected function renderHeaderCellContent()
{
if($this->grid->enableSorting && $this->sortable && $this->name!==null)
echo $this->grid->dataProvider->model->getAttributeLabel($this->name);  //去除掉默认生成的Header链接
elseif($this->name!==null && $this->header===null)
{
if($this->grid->dataProvider instanceof CActiveDataProvider)
echo CHtml::encode($this->grid->dataProvider->model->getAttributeLabel($this->name));
else
echo CHtml::encode($this->name);
}
else
parent::renderHeaderCellContent();
}
protected function renderDataCellContent($row,$data)
{
if($this->value!==null)
$value=$this->evaluateExpression($this->value,array('data'=>$data,'row'=>$row));
elseif($this->name!==null)
$value=CHtml::value($data,$this->name);
echo $value===null ? $this->grid->nullDisplay : $this->grid->getFormatter()->format($value,$this->type);
}
}

TGridColumn.php

class TGridColumn extends CGridColumn{
public $name;   //列名对应model的属性名
public $value;   //列值对应model的属性值
public $type='text';
public function renderHeaderCell()
{
if($this->name=='xxx'):
echo '';
elseif($this->name=='xxxx'):
echo '';
elseif($this->name=='xxxxx'):
echo '';
else:
echo '';
endif;
$this->renderHeaderCellContent();
echo "";
}
public function renderDataCell($row)
{
$data=$this->grid->dataProvider->data[$row];
$optiOns=$this->htmlOptions;
if($this->cssClassExpression!==null)
{
$class=$this->evaluateExpression($this->cssClassExpression,array('row'=>$row,'data'=>$data));
if(!empty($class))
{
if(isset($options['class']))
$options['class'].=' '.$class;
else
$options['class']=$class;
}
}
echo CHtml::openTag('td',$options);
$this->renderDataCellContent($row,$data);
echo '';
}
}

这些自定义类都放在/protected/components目录下,供视图调用:

widget('TGridView', array(
'summaryText'=>'',
'dataProvider'=>$dataProvider,
'enableSorting'=>false,
'pager'=>array('class'=>'TPLinkPager',
'maxButtonCount'=>10,
'htmlOptions'=>array('class'=>'paging'),
'header'=>'',
'prevPageLabel' => '上一页',
'nextPageLabel' => '下一页', ),
'columns'=>array(
'id',
'partner',
'pname',
'type.name',
),
)); ?>

这里Yii内置的分页类是CLinkPager,同样因为不满足我的需求,我自定义了一个继承自TLinkPager(继承自CLinkPager)的类(这个类的定义在Yii框架中使用CLinkPager实现分页功能中有详细叙述,TPLinkPager主要去掉了跳转到指定页面的功能)。

Yii后台admin.php页面中dataProvider来自model->search()方法,这里因为我去掉了sort的功能,所以不必要通过search方法,我是通过控制器传递过来的变量$dataProvider提供数据支持:

$dataProvider=new CActiveDataProvider('model名称', array(
'criteria'=>array(
'order'=>'id ASC',
),
'pagination'=>array(
'pageSize'=>10,   //每页显示10条记录
),
));

至此完成了整个的自定义CGridView过程,效果图如下所示:

自定义gridview效果图


推荐阅读
  • 基于函数实现的进制转换工具
    本文介绍了一种利用函数实现不同进制数(二进制、八进制、十进制)之间转换的方法。包括了程序的运行效果展示、所使用的主要函数解析、以及如何验证用户输入的合法性。整个项目仅使用了两个全局变量来存储用户的选项和输入的数值。 ... [详细]
  • ArcGIS技巧:为相邻地块创建指定宽度的隔离带
    在地理信息系统(GIS)的数据处理中,为了满足特定项目的质量检查标准,需要在相邻地块之间创建一定宽度的隔离带。本文将探讨如何使用ArcGIS工具解决这一问题,确保不同地块图斑间保持规定的最小距离。 ... [详细]
  • CGroups: 资源管理和控制
    CGroups(Control Groups)是Linux内核提供的一个功能,旨在限制、记录和隔离进程组使用的物理资源,如CPU、内存和I/O等。它通过精细的资源管理,支持现代容器技术如Docker的资源限制需求。 ... [详细]
  • 深入解析Java中的锁类型及其应用场景
    本文详细介绍了Java中常见的锁类型,包括乐观锁与悲观锁、独占锁与共享锁、互斥锁与读写锁、可重入锁、公平锁与非公平锁、分段锁、偏向锁、轻量级锁、重量级锁以及自旋锁。每种锁的特性、作用及适用场景均有所涉及。 ... [详细]
  • 本文探讨了HDU 4035的问题,涉及一个由n个房间组成的迷宫,这些房间通过n-1条隧道相互连接,形成一棵树结构。任务是从起点1号房间出发,计算到达出口所需经过的平均隧道数量,考虑了在每个房间中可能发生的三种情况及其相应概率。 ... [详细]
  • 本文介绍如何利用Python中的Epoll机制构建一个高效的Web服务器,该服务器能够处理多个并发连接,并向每个连接的客户端返回预定义的响应文本。通过使用Epoll,服务器可以实现高性能的I/O多路复用。 ... [详细]
  • 一、数据更新操作DML语法中主要包括两个内容:查询与更新,更新主要包括:增加数据、修改数据、删除数据。其中这些操作是离不开查询的。1、增加数据语法:INSERTINTO表名称[(字 ... [详细]
  • 本文介绍了如何通过安装 VirtualBox 和 Vagrant 来快速搭建和管理虚拟机环境。我们将详细探讨如何选择合适的 Box 镜像,以及如何高效地下载、添加和管理这些镜像。 ... [详细]
  • HTML中用于创建表单的标签是什么
    本文将详细介绍HTML中用于创建表单的标签及其基本用法,包括表单的主要特性和常用的属性设置。如果您正在学习HTML或需要了解如何在网页中添加表单,这将是一个很好的起点。 ... [详细]
  • SQL注入实验:SqliLabs第38至45关解析
    本文深入探讨了SqliLabs项目中的第38至45关,重点讲解了堆叠注入(Stacked Queries)的应用技巧及防御策略。通过实际案例分析,帮助读者理解如何利用和防范此类SQL注入攻击。 ... [详细]
  • 解决fetch上传图片至微信公众号H5页面的问题
    在近期的一个项目需求中,需要在微信公众号内嵌入H5页面,并实现用户通过该页面上传图片的功能,包括拍摄新照片或从已有相册中选择。前端开发中采用了fetch API进行接口调用,但遇到了上传图片时数据无法正确传递的问题。 ... [详细]
  • 如何将Redis配置为后台服务
    本文介绍了在安装Redis后,如何通过修改配置文件使其以守护进程模式在后台运行,避免因控制台被占用而无法进行其他操作的问题。 ... [详细]
  • Linux 文件系统结构详解
    本文详细介绍了Linux操作系统的文件系统结构,包括其独特的树状目录体系、根目录的作用、目录与磁盘分区的关系等,并对各主要目录的功能进行了深入解析。 ... [详细]
  • KKCMS代码审计初探
    本文主要介绍了KKCMS的安装过程及其基本功能,重点分析了该系统中存在的验证码重用、SQL注入及XSS等安全问题。适合初学者作为入门指南。 ... [详细]
  • 本文探讨了使用Lighttpd与FastCGI实现分布式部署的方法。通过在中心服务器上配置Lighttpd负责请求转发,同时在多个远程服务器上运行FastCGI进程来处理实际业务逻辑,从而提高系统的负载能力和响应速度。 ... [详细]
author-avatar
cfn7831325
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有