原文 http://cuisuqiang.iteye.com/blog/1582942
说明:实现方法是用java代码输出前段js(结合第三方表格控件)代码来实现。
如果你用公司的平台进行开发的话,许多时候向按钮,输入框,树,菜单等都是直接用一个标签设置几个属性就可以了。全局上样式是统一的,而且容易维护。
之前我已经发使用自定义标签来做数据字典的示例,也就是说自定义标签并不是你想的那么难,今天就再来作一个自定标标签实现的表格控件。当然你别较真,麻雀虽小五脏俱全的,你懂就行了。
我先来简单介绍一下:
效果如下图
这个控件包含了列表显示,选择框,隔行染色,分页,排序一些基本功能!
原来很简单,后台查询List数据,进行JSON格式转换为JSON字符串,前台使用JS进行表格生成。
List转字符串我们直接使用json-lib即可,很简单的使用。
前台我采用 tableView ,网址是 http://www.ideawu.net/person/tableview/v1.1/
然后采用自定义标签将两者融合展示
首先来看一下标签的定义:
- <?xml version="1.0" encoding="UTF-8"?>
- <!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN"
- "http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd">
- <!--
- 自定义标签
- 崔素强
- V 1.0
- -->
- <taglib>
- <tlib-version>1.0</tlib-version>
- <jsp-version>1.2</jsp-version>
- <short-name>html</short-name>
- <tag>
- <name>grid</name>
- <tag-class>com.cui.GridTable_Tag</tag-class>
- <body-content>JSP</body-content>
- <attribute>
- <name>colunmName</name><!-- 列名称,以 , 分割。第一列必须为主键列,该列不显示,但是在返回选择的列时返回主键值 -->
- <rtexprvalue>true</rtexprvalue>
- </attribute>
- <attribute>
- <name>colunmShow</name><!-- 列显示的文字,以 , 分割 -->
- <rtexprvalue>true</rtexprvalue>
- </attribute>
- <attribute>
- <name>isPager</name><!-- 是否显示分页 -->
- <rtexprvalue>true</rtexprvalue>
- </attribute>
- <attribute>
- <name>isTablePager</name><!-- 是否显示分页 -->
- <rtexprvalue>true</rtexprvalue>
- </attribute>
- <attribute>
- <name>isShowCheck</name>
- <rtexprvalue>true</rtexprvalue>
- </attribute>
- <attribute>
- <name>showAllColunm</name>
- <rtexprvalue>true</rtexprvalue>
- </attribute>
- </tag>
- </taglib>
介绍一下属性:
colunmName:要显示列的名称,因为有的人查询是查询了对象所有的属性出来,所以这里我们指定要显示那写列
colunmShow:显示列的列名,这个是在页面看的,比如你列是name,那么显示到页面的列应该是 名字
isPager:是否分页,其实分页控件也就是做一个显示,如果你不用分页设置为false就不会显示分页内容了
isTablePager:这个和上面的分页是分开的不能并行的,这个就是在不进行分页时,页面进行分页的意思。比如说你要显示100条数据,不进行分页查询,那么一次性返回到页面100条数据,你又要分页显示,就可以设置这个属性
isShowCheck:这个是控制是否显示多选框
showAllColunm:因为我们不可能把主键也显示出去的,但是当我们操作一条数据时又必须得到主键,所以我对tableView进行了修改,不显示主键列,当然是默认不显示,如果你设置了这个属性为true,那么会显示主键列
后代的标签代码我们来看一下,主要是通过标签属性获得要展示的数据内容,然后拼装动态JS内容块
- import javax.servlet.http.HttpServletRequest;
- import javax.servlet.jsp.JspException;
- import javax.servlet.jsp.PageContext;
- import javax.servlet.jsp.tagext.BodyTagSupport;
- /**
- * @说明 用于显示表格
- * @author cuisuqiang
- * @version 1.0
- * @since
- */
- @SuppressWarnings("serial")
- public class GridTable_Tag extends BodyTagSupport{
- private HttpServletRequest request ;
- @Override
- public void setPageContext(PageContext pageContext) {
- request = (HttpServletRequest)pageContext.getRequest();
- super.setPageContext(pageContext);
- }
- @Override
- public int doEndTag() throws JspException {
- return EVAL_PAGE;
- }
- @Override
- public int doStartTag() throws JspException {
- String path = request.getContextPath();
- String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
- try {
- StringBuffer results = new StringBuffer("");
- // 引入文件
- results.append("<link rel='stylesheet' type='text/css' href='" + basePath + "js/style.css' />");
- results.append("<script type='text/javascript' src='" + basePath + "js/jquery.js'></script>");
- results.append("<script type='text/javascript' src='" + basePath + "js/TableView.js'></script>");
- results.append("<script type='text/javascript' src='" + basePath + "js/PagerView.js'></script>");
- results.append("<script type='text/javascript' src='" + basePath + "js/SortView.js'></script>");
- results.append("<script type='text/javascript' src='" + basePath + "js/commonAttribute.js'></script>");
- String[] cn = colunmName.split(",");
- String[] cs = colunmShow.split(",");
- // 开始绘制表格
- results.append("<script type='text/javascript'>");
- results.append("document.getElementById('table_div').className='tableShow';");
- results.append("var index=1;");
- results.append("var table = new TableView('table_div');");
- results.append("table.header={");
- results.append("index:'序号',");
- for(int i=0;i<cn.length;i++){
- results.append(cn[i] + ":'" + cs[i] + "',");
- }
- results.delete(results.length() - 1, results.length());
- results.append("};");
- results.append("table.dataKey='" + cn[0] + "';");
- String date = "";
- date = null == request.getAttribute("currentJsonDate") ? "" : (String)request.getAttribute("currentJsonDate");
- if(null != date && !"".equals(date))
- date = date.replace("\\", "/"); // 部分数据可能会引起JS解析错误
- results.append("var json='" + date + "';");
- results.append("var objs=eval(json);"); // EVAL函数的使用
- results.append("if(null!=objs){");
- results.append("for(var i=0;i<objs.length;i++){");
- results.append("var u = objs[i];");
- results.append("table.add({");
- results.append("index:index++,");
- for(int i=0;i<cn.length;i++){
- if(i==0){
- results.append(cn[i] + ":u." + cn[i] + ",");
- }else{
- results.append(cn[i] + ":(u." + cn[i] + " + '').substr(0, 20),"); // 这里做了数据截取
- }
- }
- results.delete(results.length() - 1, results.length());
- results.append("});");
- results.append("}");
- results.append("}");
- results.append("table.display.sort=true;");
- // 是否显示所有列
- if(null != showAllColunm && "true".equals(showAllColunm)){
- results.append("table.display.showAllColunm=true;");
- }
- // 是否显示多选框列
- if(null != isShowCheck && "false"==isShowCheck){
- results.append("table.display.multiple=false;");
- results.append("table.display.marker=false;");
- }
- // 必须显示的进行表格分页,且页面分页不能同时进行
- if("true".equals(isTablePager) && (null == isPager || "false".equals(isPager))){
- results.append("table.display.pager=true;");
- results.append("table.pager.size=pageSize;"); // 分页页数是JS文件里面定义的
- }else{
- results.append("table.display.pager=false;");
- }
- results.append("table.render();");
- // 开始显示分页,表格自动分页和显示分页标签不能一同进行
- if(!"false".equals(isPager) && (null == isTablePager || "false".equals(isTablePager))){
- results.append("var pager=new PagerView('pager_div');");
- results.append("pager.itemCount=" + (null != request.getAttribute("currentPageCount") ? request.getAttribute("currentPageCount").toString() : "0") + ";");
- results.append("pager.size=" + (null != request.getAttribute("currentPageSize") ? request.getAttribute("currentPageSize").toString() : "10") + ";");
- results.append("pager.index=" + (null != request.getAttribute("currentPageIndex") ? request.getAttribute("currentPageIndex").toString() : "0") + ";");
- results.append("pager.render();");
- }
- // 增加Ajax数据变换时的JS方法
- results.append("function setJsonData(data){");
- results.append("if(null!=data&&''!=data){");
- results.append("index=1;");
- results.append("table.clear();"); // 清除原有数据
- results.append("var json=data;");
- results.append("var objs=eval(json);");
- results.append("for(var i=0;i<objs.length;i++){");
- results.append("var u = objs[i];");
- results.append("table.add({");
- results.append("index:index++,");
- for(int i=0;i<cn.length;i++){
- if(i==0){
- results.append(cn[i] + ":u." + cn[i] + ",");
- }else{
- results.append(cn[i] + ":(u." + cn[i] + " + '').substr(0, 20),");
- }
- }
- results.delete(results.length() - 1, results.length());
- results.append("});");
- results.append("table.render();");
- results.append("}");
- results.append("}");
- results.append("}");
- results.append("</script>");
- results.append("<input type='hidden' id='colunmName' name='colunmName' value='" + colunmName + "'>");
- results.append("<input type='hidden' id='colunmShow' name='colunmShow' value='" + colunmShow + "'>");
- pageContext.getOut().write(results.toString());
- } catch (Exception e) {
- e.printStackTrace();
- }
- return EVAL_BODY_INCLUDE;
- }
- protected String colunmName;
- protected String colunmShow;
- protected String isPager;
- protected String showAllColunm;
- protected String isTablePager;
- protected String currentJsonDate;
- protected String currentPageCount;
- protected String currentPageSize;
- protected String isShowCheck;
- protected String currentPageIndex;
- // get和set方法省略
- }
如果你到tableView网站上看看他的使用,就明白,这个类的功能就是做一些JS拼装的工作,然后我增加了一个JS方法,
- function setJsonData(data){}
用这个来动态展示数据
但是这里我特别声明一下:如果你要做的是监控类数据项目一直在变且长时间运行的话,那么你就不要用了,因为每次刷新浏览器的内存都会增加特别多,有内存泄漏问题!
请求一个Servlet或者Struts的Action,在业务逻辑里面增加要展示的内容
- public void doGet(HttpServletRequest request, HttpServletResponse response)
- throws ServletException, IOException {
- // 当前是第几页
- String index = request.getParameter("pageIndex");
- int itindex = 1;
- if(null != index){
- try {
- itindex = Integer.parseInt(index);
- } catch (Exception e) {
- }
- }
- List<Users> list = new ArrayList<Users>();
- // 模拟数据
- for(int i=(itindex - 1) * 10;i<(itindex - 1) * 10 + 10;i++){
- Users user = new Users();
- user.setId(i);
- user.setEndTime("endTime_" + i);
- user.setName("name_" + i);
- user.setPass("pass_" + i);
- user.setUpdateTime("updateTime_" + i);
- list.add(user);
- }
- <span style="background-color: #ffff00;">JSONArray jsonArr = JSONArray.fromObject(list); // 使用工具生成JSON字符串</span>
- request.setAttribute("currentJsonDate", jsonArr.toString());
- request.setAttribute("currentPageCount", 55); // 数据量是从数据库统计出来的
- request.setAttribute("currentPageSize", "10"); // 页数一般是定死的
- request.setAttribute("currentPageIndex", itindex); // 当前页是传递的参数
- request.getRequestDispatcher("/grid.jsp").forward(request, response);
- }
JSON字符串的生成你不用担心,这样只要从数据库获得一个List到前面即可
这样就做好了一个自定义标签的表格控件,JSP页面的使用很简单
- <html:grid colunmName="id,name,endTime,updateTime" colunmShow="主键,用户名,有效期,更新时间"></html:grid>
这样就可以生成一个表格信息了
当然你可以设置一些属性来控制表格的显示,也可以通过提供的JS方法来进行一些操作
- <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
- <%@ taglib uri="/tld/web-html" prefix="html"%>
- <%
- String path = request.getContextPath();
- String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
- %>
- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
- <html>
- <head>
- <base href="<%=basePath%>">
- <title>My JSP 'grid.jsp' starting page</title>
- </head>
- <body>
- <form action="" id="form1" name="form1">
- <input type="hidden" id="pageIndex" name="pageIndex" value="">
- </form>
- <p>
- <input type="button" name="fordelete" id="fordelete" onclick="fordelete()" value="删除数据">
- </p>
- <div align="left">
- <div id="table_div" align="left"></div>
- <br>
- <div id="pager_div" align="left"></div>
- </div>
- <html:grid colunmName="id,name,endTime,updateTime" colunmShow="主键,用户名,有效期,更新时间"></html:grid>
- <script type="text/javascript">
- pager.onclick = function(index){
- document.getElementById("pageIndex").value=index;
- document.form1.action="<%=basePath %>MySevlet";
- document.form1.submit();
- };
- function fordelete(){
- <span style="background-color: #ffff00;">var pk = table.getSelectedKeys();</span>
- if(""==pk){
- alert("请选择一条记录!");
- return false;
- }
- if(pk.length > 1){
- alert("请选择一条记录!");
- return false;
- }
- alert("您选择了ID:" + pk + "!");
- }
- </script>
- </body>
- </html>
获得选择的数据项是tableView提供的方法,我们无需关心
这样后台和前台的展示也有了分离,因为后台传递的是JSON,如果你前台的表格要换,那么很简单。
请您到ITEYE看我的原创:http://cuisuqiang.iteye.com
或支持我的个人博客,地址:http://www.javacui.com
相关推荐
在开发这些应用时,数据展示是非常关键的一环,而“酷牛逼”的自定义表格控件则能极大地提升用户体验和开发效率。这里我们将深入探讨C#中的dataGridView控件以及如何创建自定义表格控件。 dataGridView控件是...
在这种情况下,"ios 自定义表格控件 gridview" 提供了一种解决方案,特别是对于那些希望实现类似网格布局,同时支持首列固定和左右滑动效果的应用。这种自定义表格控件通常被称为GridView,它在iPhone平台上特别有用...
本文将深入探讨如何自定义一个表格控件,以满足特定的业务需求,如动态添加、删除行以及支持编辑功能,并且能够与外部的DataTable对象进行数据交互。我们将基于提供的文件`TableControl.xaml.cs`和`TableControl....
不能访问源代码(译注:因为商业目的)使得这非常令人沮丧,因此我想倒不如我写篇CodeProject文章,看看我是否可以拿出一个自定义解决方案。 因为网格比列表更加灵活,我决定实现一个可以一起分组的网格控件,正如...
本资源提供的是一个关于PPC上的自定义表格控件的源代码,名为"OwnerDrawnGridSample"。这个控件允许开发者在界面上创建具有高度定制能力的表格,以展示复杂的数据结构或实现独特的用户交互体验。 首先,"OwnerDrawn...
在.NET框架中,自定义表格控件是一种常见的需求,它允许开发者根据项目特性和用户界面设计,创建符合特定功能和视觉样式的数据展示组件。FormView控件是ASP.NET中一个非常灵活的表格控件,主要用于展示单个数据记录...
本文将深入探讨著名的自定义表格控件——GridCtrl,它在MFC(Microsoft Foundation Classes)项目中被广泛应用,因其轻量级且易用的特性,受到了众多开发者的青睐。 GridCtrl是一款高度可定制的表格控件,它提供了...
Grid控件是用于显示二维数据的常见组件,它提供了直观的方式来展示表格形式的信息。在本篇文章中,我们将深入探讨如何在DELPHI7.0下创建一个具有自定义功能的Grid控件,包括自定义查询、点击标题排序、打印内容以及...
在Android开发中,为了满足用户对独特视觉效果和个性化界面的需求,开发者经常需要...自定义表格控件只是一个起点,你可以在此基础上进一步扩展,比如添加触摸事件处理、动态调整行数和列数,甚至实现数据绑定等功能。
总结起来,创建WinForm的自定义控件,实现无边框效果和透明背景,以及自定义表格控件的背景,是一项涉及图形绘制、事件处理和控件扩展的综合任务。通过以上步骤,我们可以创建出更符合设计需求的用户界面,提升应用...
笔者在使用qt的qml语言开发产品时候,经常需要用到一些表格类的控件,...这一表格控件采用最基本qml元素编写,兼容性非常好,该表格控件还支持qt接口调用,动态显示。在文章的最后会公开源代码,仅供读者们参考学习。
C# WinForm 开发的自定义水晶控件!代码也是我在网上偶然发现的,特0分分享出来,让CS的童鞋们知道客户端程序也不再单一!
图表控件的应用与C#自带的可视化控件相同; 可以在控件的设计期对图表样式进行相应的设置; 图表可以显示条形图、面形图、线形图和饼形图; 可以在图表控件中动态显示数据的标签说明; 实现条形图和面形图的上下...
2. **控件的封装与扩展**:自定义控件类,扩展.NET Framework提供的基础控件,增加拖放和属性编辑功能。 3. **代码生成逻辑**:查看源码中的代码生成部分,理解如何将设计时的表单布局和事件转换为运行时的代码。 ...
本文将深入探讨如何在QML中自定义表格控件,以满足特定的展示或交互需求。 首先,QML中的表格控件通常使用`TableColumn`和`TableView`来实现。然而,标准的`TableView`可能无法满足所有复杂的应用场景,例如自定义...
自定义表格控件本质上就是通过组合这些基础布局,以及利用Java代码进行动态操作,来实现特定功能的组件。 创建自定义表格的第一步是设计数据模型。你需要定义一个类来存储表格每一行的数据,这通常包括列的值和其他...
在实际应用中,这样的自定义分页控件可以广泛应用于各种数据展示场景,如网格视图(GridView)或其他表格组件。通过解耦数据源和控件,开发人员可以轻松地将分页功能集成到任何需要的地方,提高代码的复用性和项目...
这个表格类就是这样一个自定义控件,它可能继承自MFC(Microsoft Foundation Classes)的CWnd或者CControlBar类,并扩展了标准表格控件的功能。 2. **翻译和注释**:为了提高代码的可读性和易用性,开发者为大部分...
3. 自定义表格控件(QQTTableWidget)封装了对数据库表的查询、分页显示、选择、删除等功能,适用于展示和操作数据库表格的场景。 4. 自定义树形视图(QQTTreeView)提供树形结构的数据展示,支持自定义绘制和事件...