Symfony2.x EasyUI Ajax 异步加载DataGrid 分页

原文地址:http://blog.csdn.net/nuaazdh/article/details/37660377,转载请勿标明,多谢~

前言

  Symfony2.x本来应该许配给Bootstrap的,可是考虑到本人前端基础实在太差,使用Bootstrap还要使用DIV来布局,前端部分又得使用jQuery,因此就委屈求全使用了EasyUI。EasyUI界面不够炫,但确实方便。不过,之前没有用过,开始做还是遇到不少问题。闲话少叙,文章的主要内容是Symfony2.x作为框架,Doctrine从后台获取数据,并以json格式返回给前端。前端使用EasyUI的界面。其中重点是解决datagrid的异步加载的分页问题,如果没有分页需求可参见另一篇博文:Symfony2.x + EasyUI datagrid Ajax方式实现数据交互。

正文

一、分页问题的出现

   首先明确一下为何会有分页问题。前一篇博文中给出的解决方案虽然是ajax异步加载,但是采用的是自动分页,即数据从服务端传来,直接赋值给datagrid控件。如果你的datagrid控件设置了pagination = true,它会自动加载一个分页条,实现自动分页。既然都自动分页了,那还有讨论这个问题干嘛? 假设你有这样的需求,后台的数据量很大,即使使用ajax数据量也太多,这样前端其实响应还是比较慢的,我们需要后端分页后每次只传给一页的数据量。其次,如果你的数据太多,它自动分页了,你要添加“编辑”、“删除”的按钮,编辑后需要更新显示,那么前面的做法就不行了,只能刷新整个页面,但是刷新后会自动显示第一页,不是你当时编辑的页面,怎么办?

二、 分页问题解决方案

  解决方案: 将datagrid 控件和 分页pagination控件分开,datagrid 不再设置 pagination = true, 手动实现点击下一页的js代码。当用户点击”下一页“时,发出ajax请求,服务端根据传入的page (页的序号)和 limits(每页的记录个数)查询数据库,返回json结果。前端将返回的结果绑定到datagrid控件中。具体有以下几点:

① 前端(模板)中将 datagrid 和 pagination 控件分开,设置 手动完成切换页面的触发onSelectPage()函数;

② 在 onSelectPage()函数中,发出ajax请求,将页号(page)和每页记录数(limits或者rows)发送给服务端;

③ 服务端根据page和limits参数,做查询操作,返回从 (page-1)*limits 开始的limits个记录;

④ 将后端返回的数据绑定到前端的datagrid中,异步刷新;

⑤ 当前端发生单个记录的编辑操作时,编辑完成后,调用$('#pp').pagination('select'); (其中pp就是pagination控件的id),自动刷新当前页;

三、代码

  路由文件和布局模板直接省略,直接看功能页面的twig文件和后端的获取分页数据的代码:

1. 前端twig模板代码

{% extends 'AcmeStoreBundle::layout.html.twig' %}

{% block content %}
	
	
ID 名称
{% endblock %}

2. 后端更新操作和显示一页的php代码

isXmlHttpRequest()){
			// 获取请求参数
			$rows = $request->request->get('rows');
			$page = $request->request->get('page');
			// 设置请求参数的默认值,若前端js中处理了,此处可以省略
			isset($rows)?($rows):($rows=10);
			isset($page)?($page):($page=1);
			// 这是是使用Doctrine获取分页的关键代码
			$em = $this->getDoctrine()->getManager();
			$dql = 'SELECT p FROM AcmeStoreBundle:XXXEntity p ORDER BY p.id ASC';
			$query = $em->createQuery($dql)
						->setFirstResult(($page-1)*$rows)
						->setMaxResults($rows);
			$paginator  = new Paginator($query,true);
			// json 字符串
			$json_str = "";
			$i = 0; 
			$len = count($paginator);
			foreach ($paginator as  $item)
			{
				$i++;
				// 这里提供日期和时间类型转换函数,供需要的参考
				//$publishDateStr = date_format($item->getPublishDate(), 'Y-m-d');
				//$signitureTimeStr = date_format($item->getSignitureTime(), 'Y-m-d H:i:s');
				$publishTypeStr = addslashes($item->getPublishType());
				$json_str.="{id:"."\"{$item->getId()}\","
				."name:"."\"{$item->getName()}\",";
				if($i!=$len){// 对最后一个记录做特殊处理:省略右花括号}后的逗号,
					$json_str .= "lastColumnName:"."\"{$cutsizeStr}\"},";
				}else{
					$json_str .= "lastColumnName:"."\"{$cutsizeStr}\"}";
				}
			}
			// 返回记录的总数,和当前页的所有记录数据,记录总数会在分页控件中用到,"共xxx项,当前显示xxx-xxx项"
			return new Response(json_encode(array('count' => $len,'data' => $json_str)));
			
		}else{
			return new Response('Illigal request');
		}
	}
	
	/**
	*@Route("/XXX/",name="XXX_index")
	*@Template("AcmeStoreBundle:XXXEntity:index.html.twig")
	*/
	public function updateAction(Request $request)
	{
	if ($request->isXmlHttpRequest()) {
			$json_data =  $request->request->get('data_p');
			$data = json_decode($json_data);	// 解析 json数据
 			// 获取各个属性值
			$id = (int)$data[0]->{"value"};
			$name = $data[1]->{"value"};
			// 提供两个日期和时间的处理函数,供需要参考
			//$publishDate = date_create_from_format('Y/m/j',$data[2]->{"value"});
			//$signitureTime = date_create_from_format('Y/m/j H:i:s',$data[3]->{"value"});			
			// 在 Repository中根据id查询记录	
			$em = $this->getDoctrine()->getEntityManager();
			$item = $em->getRepository('AcmeStoreBundle:XXXEntity')->find($id);
			
			if(!$item){ // 记录不存在,可以创建,这样一个updateAction完成了前端“新建”和“编辑”两个功能
				$item = new XXXEntity();				
				$item->setName($name);
				//$item->setpublishDate($publishDate);
				//$item->setSignitureTime($signitureTime);
				$em->persist($item);
				$em->flush();
				return  new Response(json_encode('创建项目成功,ID:'.$item->getId()));
			}else{ // 记录存在,更新值
				$item->setName($name);
				//$item->setpublishDate($publishDate);
				//$item->setSignitureTime($signitureTime);
				$em->flush();
				// 注意返回的数据格式也是json_encode()处理过的,否则前端ajax获取失败
 				return  new Response(json_encode('更新 ID = '.$id.' 文件信息成功!'));
			return $response;
			}
		}else{
			return  new Response(json_encode('非法的更新请求!'),400);
		}
	}
}
?>

3. 页面效果

Firefox的Firebug大家都比较熟悉,不用推荐了,贴一下效果,easyui的界面都差不多,呵

Symfony2.x EasyUI Ajax 异步加载DataGrid 分页_第1张图片


你可能感兴趣的:(PHP)