Twitter Bootstrap具有非常好的分页UI ,在这里我将向您展示如何使用Spring Data Web分页功能和Thymeleaf条件评估功能来实现它。

引导程序中的标准分页

受Rdio启发的简单分页,非常适合应用程序和搜索结果。 大块很难错过,易于扩展,并提供较大的点击区域。

从Bootstrap文档显示分页的原始源代码非常简单:

<div class='pagination'><ul><li><a href='#'>Prev</a></li><li><a href='#'>1</a></li><li><a href='#'>2</a></li><li><a href='#'>3</a></li><li><a href='#'>4</a></li><li><a href='#'>5</a></li><li><a href='#'>Next</a></li></ul>
</div>

您可以看到这只是一个模拟代码,要使其通过正确的超链接URL动态显示页码,我需要对现有代码进行很多更改。 因此,让我们从头开始,先更改域层,然后再更改应用程序服务层,表示层。 最后是将它们粘合在一起的配置。

域层更改

域层的唯一更改是BlogPostRepository 。 在具有检索按publishedTime排序的publishedTime发布BlogPost列表的方法之前:

public interface BlogPostRepository extends MongoRepository<BlogPost, String>{
...List<BlogPost> findByPublishedIsTrueOrderByPublishedTimeDesc();
...
}

现在我们需要获取分页结果列表。 使用Spring Data Page ,我们将返回Page<BlogPost>而不是List<BlogPost> ,并传递Pageable参数:

public interface BlogPostRepository extends MongoRepository<BlogPost, String>{
...Page<BlogPost> findByPublishedIsTrueOrderByPublishedTimeDesc(Pageable pageable);
...
}

应用程序服务层更改:

只需使用BlogPostRepository新功能,应用程序服务层的更改也非常简单:

BlogService界面

public interface BlogService {
...Page<BlogPost> getAllPublishedPosts(Pageable pageable);
...
}

BlogServiceImpl类

public class BlogServiceImpl implements BlogService {
...private final BlogPostRepository blogPostRepository;
...@Overridepublic Page<BlogPost> getAllPublishedPosts(Pageable pageable) {Page<BlogPost> blogList = blogPostRepository.findByPublishedIsTrueOrderByPublishedTimeDesc(pageable);return blogList;}
...
}

表示层更改:

Spring Data Page界面具有许多不错的功能来获取当前页码,获取总页数等。但是仍然缺少让我仅显示总分页的部分页面范围的方法。 因此,我创建了一个适配器类,以使用其他功能包装Sprng数据页面接口。

public class PageWrapper<T> {public static final int MAX_PAGE_ITEM_DISPLAY = 5;private Page<T> page;private List<PageItem> items;private int currentNumber;private String url;public String getUrl() {return url;}public void setUrl(String url) {this.url = url;}public PageWrapper(Page<T> page, String url){this.page = page;this.url = url;items = new ArrayList<PageItem>();currentNumber = page.getNumber() + 1; //start from 1 to match page.pageint start, size;if (page.getTotalPages() <= MAX_PAGE_ITEM_DISPLAY){start = 1;size = page.getTotalPages();} else {if (currentNumber <= MAX_PAGE_ITEM_DISPLAY - MAX_PAGE_ITEM_DISPLAY/2){start = 1;size = MAX_PAGE_ITEM_DISPLAY;} else if (currentNumber >= page.getTotalPages() - MAX_PAGE_ITEM_DISPLAY/2){start = page.getTotalPages() - MAX_PAGE_ITEM_DISPLAY + 1;size = MAX_PAGE_ITEM_DISPLAY;} else {start = currentNumber - MAX_PAGE_ITEM_DISPLAY/2;size = MAX_PAGE_ITEM_DISPLAY;}}for (int i = 0; i<size; i++){items.add(new PageItem(start+i, (start+i)==currentNumber));}}public List<PageItem> getItems(){return items;}public int getNumber(){return currentNumber;}public List<T> getContent(){return page.getContent();}public int getSize(){return page.getSize();}public int getTotalPages(){return page.getTotalPages();}public boolean isFirstPage(){return page.isFirstPage();}public boolean isLastPage(){return page.isLastPage();}public boolean isHasPreviousPage(){return page.hasPreviousPage();}public boolean isHasNextPage(){return page.hasNextPage();}public class PageItem {private int number;private boolean current;public PageItem(int number, boolean current){this.number = number;this.current = current;}public int getNumber(){return this.number;}public boolean isCurrent(){return this.current;}}
}

使用此PageWrapper ,我们可以包装从BlogService返回的Page<BlogPost>并将其放入SpringMVC UI模型。 请参阅博客页面的控制器代码:

@Controller
public class BlogController
...@RequestMapping(value = '/blog', method = RequestMethod.GET)public String blog(Model uiModel, Pageable pageable) {PageWrapper<BlogPost> page = new PageWrapper<BlogPost>(blogService.getAllPublishedPosts(pageable), '/blog');uiModel.addAttribute('page', page);return 'blog';}
...
}

Pageable是从PageableArgumentResolver传入的,我将在后面解释。 另一个技巧是我还将视图URL传递给PageWrapper ,它可用于在分页栏中构造Thymeleaf超链接。

由于我的PageWrapper非常通用,因此我为分页栏创建了一个html片段,因此当需要分页时,可以将其用于应用程序页面中的任何位置。 该片段html使用Thymeleaf th:if根据链接是否被禁用来在静态文本或超链接之间动态切换。 并且它使用th:href构造具有正确页码和页面大小的URL。

<!-- Pagination Bar -->
<div th:fragment='paginationbar'><div class='pagination pagination-centered'><ul><li th:class='${page.firstPage}? 'disabled' : '''><span th:if='${page.firstPage}'>← First</span><a th:if='${not page.firstPage}' th:href='@{${page.url}(page.page=1,page.size=${page.size})}'>← First</a></li><li th:class='${page.hasPreviousPage}? '' : 'disabled''><span th:if='${not page.hasPreviousPage}'>«</span><a th:if='${page.hasPreviousPage}' th:href='@{${page.url}(page.page=${page.number-1},page.size=${page.size})}' title='Go to previous page'>«</a></li><li th:each='item : ${page.items}' th:class='${item.current}? 'active' : '''><span th:if='${item.current}' th:text='${item.number}'>1</span><a th:if='${not item.current}' th:href='@{${page.url}(page.page=${item.number},page.size=${page.size})}'><span th:text='${item.number}'>1</span></a></li><li th:class='${page.hasNextPage}? '' : 'disabled''><span th:if='${not page.hasNextPage}'>»</span><a th:if='${page.hasNextPage}' th:href='@{${page.url}(page.page=${page.number+1},page.size=${page.size})}' title='Go to next page'>»</a></li><li th:class='${page.lastPage}? 'disabled' : '''><span th:if='${page.lastPage}'>Last →</span><a th:if='${not page.lastPage}' th:href='@{${page.url}(page.page=${page.totalPages},page.size=${page.size})}'>Last →</a></li></ul></div>
</div>

Spring配置变更

最后一步是将它们放在一起。 幸运的是,在更新代码之前,我做了一些研究。 Doug Haber 撰写了一篇非常不错的博客文章, 其中介绍了Spring MVC,Spring Data和Java Config 。 Doug在他的博客中提到了一些陷阱,尤其是Pageable参数需要一些配置技巧:

为了让Spring知道如何将参数转换为Pageable对象,您需要配置HandlerMethodArgumentResolver。 Spring Data提供了一个PageableArgumentResolver,但是它使用了旧的ArgumentResolver接口,而不是新的(Spring 3.1)HandlerMethodArgumentResolver接口。 XML config可以为我们解决这种差异,但是由于我们使用的是Java Config,因此我们必须手动进行一些操作。 幸运的是,如果您知道正确的魔术咒语,就可以轻松解决此问题……
道格·哈伯(Doug Haber)

在Doug的帮助下,我将此参数解析器添加到了WebConfig类中:

@Configuration
@EnableWebMvc
@ComponentScan(basePackages = 'com.jiwhiz.blog.web')
public class WebConfig extends WebMvcConfigurerAdapter {
...@Overridepublic void addArgumentResolvers(List<HandlerMethodArgumentResolver> argumentResolvers) {PageableArgumentResolver resolver = new PageableArgumentResolver();resolver.setFallbackPagable(new PageRequest(1, 5));argumentResolvers.add(new ServletWebArgumentResolverAdapter(resolver));}
...
}

完成所有这些更改后,我的博客列表的页面顶部和底部将具有分页栏,并且它始终最多具有5个页码,中间​​是当前编号,并且已禁用。 分页栏还具有第一和开头以前的链接, 然后在年底最后环节。 我还在管理页面,用户列表和评论列表中使用了它,并且效果很好。

参考:来自Jiwhiz博客的JCG合作伙伴 Yuan Ji的Spring Data和Thymeleaf实现Bootstrap分页 。

翻译自: https://www.javacodegeeks.com/2013/03/implement-bootstrap-pagination-with-spring-data-and-thymeleaf.html

使用Spring数据和Thymeleaf实现Bootstrap分页相关推荐

  1. SpringBoot+Mybatis-Plus+Thymeleaf+Bootstrap分页查询(前后端完整版开源学习)图书管理系统

    目录 分页主要逻辑,在3.7和3.8 1.准备工作 1.1 参考博客 1.2 项目结构 2. 数据库 3. 详细代码部分 3.1 pom依赖 3.2 application.yml 3.3 BookM ...

  2. Spring MVC+layui(基于bootstrap)+t 新增功能(页面和数据传递)

    为开发的规范,更易于扩充,将新增与列表页分开为:list.html.add.html list.html: <!DOCTYPE html> <html xmlns:th=" ...

  3. Thymeleaf的Spring数据

    介绍 今天,我将讨论更具体的问题. 这次没有设计模式或算法:-). 我们并不总是从头开始设计软件组件. 通常,我们必须尝试使现有软件组件协同工作. Spring Boot是Java世界上最好的免费软件 ...

  4. Spring MVC和Thymeleaf:如何从模板访问数据

    在典型的Spring MVC应用程序中, @Controller类负责使用数据准备模型映射并选择要呈现的视图. 该model map允许视图技术的完整抽象,对于Thymeleaf而言,它被转换为Thy ...

  5. spring boot集成mybatis-plus——Mybatis Plus 分页查询数据(图解)

    Mybatis Plus 分页查询数据(图解)  更新时间 2023-01-01 11:58:32 大家好,我是小哈. 本小节中,我们将学习如何通过 Mybatis Plus 分页查询数据库表中的数据 ...

  6. Spring MVC 5 + Thymeleaf 基于Java配置和注解配置

    Spring MVC 5 + Thymeleaf 注解配置 Spring的配置方式一般为两种:XML配置和注解配置 Spring从3.0开始以后,推荐使用注解配置,这两种配置的优缺点说的人很多,我就不 ...

  7. 《5K入门级项目实战:好来屋在线影院》之第 11 战 —— 用户端首页、BootStrap 分页(中)

    OK,这一节我们开发 API 接口,把需要的数据返回给客户端. 我们先来解决: 1.indexMovie.html 里的热门电影接口数据 2.index.html 文件里的相关接口,如截图: 实际的流 ...

  8. Spring Boot (十五): Spring Boot + Jpa + Thymeleaf 增删改查示例

    <p>这篇文章介绍如何使用 Jpa 和 Thymeleaf 做一个增删改查的示例.</p> 先和大家聊聊我为什么喜欢写这种脚手架的项目,在我学习一门新技术的时候,总是想快速的搭 ...

  9. java 模板引擎_极简 Spring Boot 整合 Thymeleaf 页面模板

    点击"牧码小子"关注,和众多大牛一起成长! 关注后,后台回复 java ,领取松哥为你精心准备的技术干货! 虽然现在慢慢在流行前后端分离开发,但是据松哥所了解到的,还是有一些公司在 ...

最新文章

  1. Qt5.3.1 MinGW482 release静态版编译结果、过程及QtCreator配置(转)
  2. 计算机基础1模拟题,计算机基础模拟题1(有答案).doc
  3. 【ESP8266】NONOS SDK开发,串口发送、接收与中断
  4. BZOJ 1565 Luogu P2805 [NOI2009]植物大战僵尸 (Tarjan判环、最小割)
  5. 每日程序C语言45-连接两个链表
  6. node源码详解(五)
  7. 开源开放 | 欢迎选修浙江大学《知识图谱》开放共享慕课
  8. react-native 自定义 下拉刷新 / 上拉加载更多 组件
  9. UVA 10917 Walk Through the Forest 最短路 + DP
  10. C++ 默认构造函数的构建
  11. MIDIPLUS STUDIO M S 2 4声卡安装调试教程
  12. linux 进程 清理,Linux僵尸进程清除方法
  13. 计算机网络毕业设计选题背景,毕业设计论文选题依据与背景
  14. 明月当空照,python3D编程初阶:坐标、模型、纹理与光
  15. 亨利福特真的说过“faster horse”么?
  16. springtask Cron表达式与fixedDelay的区别
  17. 数据结构和算法学习网站
  18. HTML中利用JS调用PHP (以登录为例)
  19. antv x6踩坑记录二
  20. myCobot pro 机械臂(5)Robotics Toolbox for MATLA(开发环境:matlab)

热门文章

  1. internet地址java表示
  2. 如何不用 List.clear() 方法 就清空 list 中的 所有元素(中兴面试)
  3. hibernate左连接查询时在easyUI的dataGrid中有些行取值为空的解决办法
  4. json解析适配模板_认识适配器:JSON绑定概述系列
  5. 第一个JDK 10(18.3)候选版本(内部版本43)展示了新的版本控制方案
  6. java登录界面命令_Java命令行界面(第29部分):自己动手
  7. apache ignite_使用Apache Storm和Apache Ignite进行复杂事件处理(CEP)
  8. 卡夫卡如何分区_通过分区在卡夫卡实现订单担保人
  9. 通用编程_通用编程准则
  10. 自定义ui_如何允许用户自定义UI