全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

Spring Data + Thymeleaf 3 + Bootstrap 4 实现分页器实例代码

实际上分页器或者分页组件在现实中都有广泛着的应用,照理来说老卫没有必要单独撰文来提这茬。事实是,我近期刚好在写一门关于Spring Data、Thymeleaf 3、Bootstrap 4 的应用课程,所以用了Bootstrap 4的样式,结果之前很多例如 Bootstrap 3 的表格、分页器啊之类的插件都不能很好的兼容,百度谷歌无果,而且 Bootstrap 4 还没有出稳定版本,官网的示例也是少的可怜,最终下决心要自己写个分页器了,所用到的技术就是 Spring Data、Thymeleaf 3、Bootstrap 4 。

分页器有哪些需求

中国式报表从来都是最复杂的,随之衍生而来的分页器要求也是错综复杂。本例为求把分页器原理告诉给大家,所以,将分页组件的抽象为以下通用的内容:

  1. 显示页码的列表;
  2. 该列表的第一项是“上一页”,最后一项是“下一页”;
  3. 当前选中的页码要高亮;
  4. 当当前页的上一页没有页码可选时,则“上一页”置为不可点击的状态;
  5. 当当前页的下一页没有页码可选时,则“下一页”置为不可点击的状态;

我们很容易就能找到一个 Bootstrap 分页器的设计原型,如下图:

你可以参考 Bootstrap 官网的介绍 http://getbootstrap.com/components/#pagination,但建议你不要直接用上面的样式,因为这个样式是 Bootstrap 3版本的。 最后,我找到的了Bootstrap 4 里面的样式,却不在官网 http://www.quackit.com/bootstrap/bootstrap_4/tutorial/bootstrap_pagination.cfm。感谢 books-collection 项目带给程序员的开源、免费图书集合!

Spring Data 能做什么

org.springframework.data.domain.Page 是 Spring Data 提供的一个分页器接口,提供了常用的方法,比如:

List getContent(); // 返回分页后的数据的列表
int getTotalPages(); // 总页数
long getTotalElements(); // 总数据量
boolean isFirst(); // 是否是第一个数据;
boolean isLast(); // 是否是最后一个数据;
int getNumber(); // 当前页面索引

构造一个 Page,通常需要传入一个 org.springframework.data.domain.PageRequest.PageRequest对象,所需参数为 (int page, int size),其中 page 就是 要请求的页面的索引,size 是页面的大小(一页最多有多少个数据)。

Spring Data 可以说提供了我们前端分页器所需要的所有元素了。

Thymeleaf 牛刀小试

Thymeleaf 作为模版引擎,其好处就是可以绑定数据源,并且根据数据源来渲染页面。最爽的莫过于根据绑定的数据列表来遍历生成页面元素,比如:

<ul class="pagination" >
 <!-- 上一页 -->
 <li class="page-item" data-th-classappend="*{first} ? 'disabled' : ''">
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link" data-th-attr="pageIndex=${page.number} - 1" aria-label="Previous">
  <span aria-hidden="true">«</span>
 </a>
 </li>
 
 <!-- 迭代生成页码 -->
 <li class="page-item" data-th-each="i : ${#numbers.sequence(1, page.totalPages)}" 
 data-th-classappend="${(page.number + 1) eq i} ? 'active' : ''" >
 <a class="page-link" data-th-attr="pageIndex=${i} - 1" href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >
       <span data-th-text="${i}"></span>
    </a>
 </li>
 
 <!-- 下一页 -->
 <li class="page-item" data-th-classappend="*{last} ? 'disabled' : ''">
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link" data-th-attr="pageIndex=${page.number} + 1" aria-label="Next">
  <span aria-hidden="true">»</span>
 </a>
 </li>
</ul>

这个就是简单版本的分页器了,可以看到我们的分页器的“上一页”和“下一页”是固定不变的,中间根据 totalPages(总页数)来动态生成页面。同时,我们根据是否是当前页(number + 1)来设置样式是否高亮(active)。“上一页”和“下一页”是需要做一下判断的,若当前页是第一页(first)则“上一页”不能点击(disabled);如果当前页是最后一页(last)则“下一页”不能点击(disabled)。

考虑的再多一点

实际上,上面版本可以应付大多数的应用场景了。但是,可能会有点不完美,比如,我的页数很多怎么办?那么我们的分页列表可能被拉得很长了,领导们可能会不满意的!绝对要把这种不满意的情绪扼杀在摇篮里。

可以看到,假如要做得更加完美,则还需要考虑,当页数太多时,应该将某些用省略号。这就涉及到三种情况了:

  1. 当当前页页码接近首页时,省略号在后部出现;
  2. 当当前页页码接最后页时,省略号在前部出现;
  3. 最烦的要属于,当当前页在中部时,前部、后部都需要省略号;

带省略号的分页器

聪明的工程师们应该马上行动起来,大致的把算法画了个图:

为求简单,我们预设页码的列表最多在 7 页(你也可以根据需要来改),也就是说,当 totalPages(总页数)超过 7时,我们才需要考虑省略号的事情。

  1. “上一页”和“下一页”的算法于我们上面的简单版本类似,这里就不赘述了。
  2. 当前页面页码小于等于4时,省略号在列表后部的倒数第二个出现;
  3. 最后一页与当前页面之差小于等于3时,省略号在列表前部的第二个位置出现;
  4. 其余情况,则当前页适中处于中间位置,省略号同时在列表第二个位置及倒数第二个位置出现。

实现方式如下:

<!-- 处理页数大于7 的情况 --> 
<ul class="pagination" data-th-if="${page.totalPages gt 7}" >
 <!-- 上一页 -->
 <li class="page-item" data-th-classappend="*{first} ? 'disabled' : ''">
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link" data-th-attr="pageIndex=${page.number} - 1" aria-label="Previous">
  <span aria-hidden="true">«</span>
 </a>
 </li>
 
 <!-- 首页 -->
 <li class="page-item" data-th-classappend="${(page.number + 1) eq 1} ? 'active' : ''" >
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link" data-th-attr="pageIndex=0">1</a>
 </li>
 
 
 <!-- 当前页面小于等于4 -->
 <li class="page-item" data-th-if="${(page.number + 1) le 4}" data-th-each="i : ${#numbers.sequence(2,5)}" 
 data-th-classappend="${(page.number + 1) eq i} ? 'active' : ''" >
 <a class="page-link" href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" data-th-attr="pageIndex=${i} - 1">
      <span data-th-text="${i}"></span>
    </a>
 </li>

 <li class="page-item disabled" data-th-if="${(page.number + 1) le 4}">
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link">
  <span aria-hidden="true">...</span>
 </a>
 </li>
 
 <!-- 最后一页与当前页面之差,小于等于3 -->
 <li class="page-item disabled" data-th-if="${(page.totalPages-(page.number + 1)) le 3}">
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link">
  <span aria-hidden="true">...</span>
 </a>
 </li> 
 <li class="page-item" data-th-if="${(page.totalPages-(page.number + 1)) le 3}" data-th-each="i : ${#numbers.sequence(page.totalPages-4, page.totalPages-1)}" 
 data-th-classappend="${(page.number + 1) eq i} ? 'active' : ''" >
 <a class="page-link" href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" data-th-attr="pageIndex=${i} - 1">
      <span data-th-text="${i}"></span>
    </a>
 </li>
 
 <!-- 最后一页与当前页面之差大于3,且 当前页面大于4-->
 
 <li class="page-item disabled" data-th-if="${((page.number + 1) gt 4) && ((page.totalPages-(page.number + 1)) gt 3 )}">
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link">
  <span aria-hidden="true">...</span>
 </a>
 </li> 
 <li class="page-item" data-th-if="${((page.number + 1) gt 4) && ((page.totalPages-(page.number + 1)) gt 3 )}" >
  <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link" data-th-attr="pageIndex=${page.number}">[[${page.number}]]</a>
 </li>
 <li class="page-item active" data-th-if="${((page.number + 1) gt 4) && ((page.totalPages-(page.number + 1)) gt 3 )}">
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link" data-th-attr="pageIndex=${page.number} + 1">[[${page.number + 1}]]</a>
 </li>
 <li class="page-item" data-th-if="${((page.number + 1) gt 4) && ((page.totalPages-(page.number + 1)) gt 3 )}">
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link" data-th-attr="pageIndex=${page.number} + 2">[[${page.number + 2}]]</a>
 </li>
 
 <li class="page-item disabled" data-th-if="${((page.number + 1) gt 4) && ((page.totalPages-(page.number + 1)) gt 3 )}">
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link">
  <span aria-hidden="true">...</span>
 </a>
 </li>

 <!-- 最后一页 -->
 <li class="page-item" data-th-classappend="${(page.number + 1) eq page.totalPages} ? 'active' : ''" >
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link" data-th-attr="pageIndex=${page.totalPages} - 1">[[${page.totalPages}]]</a>
 </li>

 <!-- 下一页 -->
 <li class="page-item" data-th-classappend="*{last} ? 'disabled' : ''">
 <a href="javascript:void(0);" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="page-link" data-th-attr="pageIndex=${page.number} + 1" aria-label="Next">
  <span aria-hidden="true">»</span>
 </a>
 </li>
</ul>

还要再考虑的多一点?

当然,正如我们开篇所讲,中国式报表的需求千奇百怪,本文也只是从大部分通用需求出发,给个思路, 不一定能满足所有人的需求。如果可能的话,再考虑多一点,比如:

  1. 是否可以选择页面的最大页?
  2. 是否可以选择任意页面的索引?

等等,看了下表快凌晨1点了。顶不顺了,要睡了。各位读者朋友可以继续完善~

最终效果

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# spring data 分页  # thymeleaf 分页  # bootstrap 分页  # 分页  # 下一页  # 上一页  # 前页  # 第二个  # 官网  # 之差  # 最多  # 当前页  # 可以看到  # 可选  # 不满意  # 可以选择  # 首页  # 绑定  # 为求  # 都是  # 都有  # 还没有  # 很好 


相关文章: 车管所网站制作流程,交警当场开简易程序处罚决定书,在交警网站查询不到怎么办?  网站制作多少钱一个,建一个论坛网站大约需要多少钱?  北京专业网站制作设计师招聘,北京白云观官方网站?  如何在阿里云香港服务器快速搭建网站?  韩国网站服务器搭建指南:VPS选购、域名解析与DNS配置推荐  建站主机与虚拟主机有何区别?如何选择最优方案?  网站图片在线制作软件,怎么在图片上做链接?  整蛊网站制作软件,手机不停的收到各种网站的验证码短信,是手机病毒还是人为恶搞?有这种手机病毒吗?  建站之星客服服务时间及联系方式如何?  公司网站的制作公司,企业网站制作基本流程有哪些?  建站主机是什么?如何选择适合的建站主机?  如何快速生成ASP一键建站模板并优化安全性?  大连网站制作公司哪家好一点,大连买房网站哪个好?  我的世界制作壁纸网站下载,手机怎么换我的世界壁纸?  如何通过建站之星自助学习解决操作问题?  网站制作知乎推荐,想做自己的网站用什么工具比较好?  如何在建站之星网店版论坛获取技术支持?  深圳网站制作费用多少钱,读秀,深圳文献港这样的网站很多只提供网上试读,但有些人只要提供试读的文章就能全篇下载,这个是怎么弄的?  如何通过IIS搭建网站并配置访问权限?  家庭服务器如何搭建个人网站?  陕西网站制作公司有哪些,陕西凌云电器有限公司官网?  重庆网站制作公司哪家好,重庆中考招生办官方网站?  XML的“混合内容”是什么 怎么用DTD或XSD定义  建站主机类型有哪些?如何正确选型  ,怎么在广州志愿者网站注册?  企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?  如何在阿里云域名上完成建站全流程?  如何在Golang中使用encoding/gob序列化对象_存储和传输数据  制作充值网站的软件,做人力招聘为什么要自己交端口钱?  北京网站制作网页,网站升级改版需要多久?  高端网站建设与定制开发一站式解决方案 中企动力  如何快速生成高效建站系统源代码?  全景视频制作网站有哪些,全景图怎么做成网页?  广州网站制作的公司,现在专门做网站的公司有没有哪几家是比较好的,性价比高,模板也多的?  北京制作网站的公司,北京铁路集团官方网站?  深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?  高端企业智能建站程序:SEO优化与响应式模板定制开发  建站VPS配置与SEO优化指南:关键词排名提升策略  青岛网站建设如何选择本地服务器?  关于BootStrap modal 在IOS9中不能弹出的解决方法(IOS 9 bootstrap modal ios 9 noticework)  mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?  建站之星如何通过成品分离优化网站效率?  贸易公司网站制作流程,出口贸易网站设计怎么做?  太原网站制作公司有哪些,网约车营运证查询官网?  建站之星CMS建站配置指南:模板选择与SEO优化技巧  武汉外贸网站制作公司,现在武汉外贸前景怎么样啊?  家庭建站与云服务器建站,如何选择更优?  活动邀请函制作网站有哪些,活动邀请函文案?  如何通过VPS搭建网站快速盈利?  西安专业网站制作公司有哪些,陕西省建行官方网站? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。