IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> JavaScript知识库 -> Vue2 Element Pagination组件 每页数据量不同的解决方案 -> 正文阅读

[JavaScript知识库]Vue2 Element Pagination组件 每页数据量不同的解决方案


前言

Element pagination, 每页数据量不同的解决方案, 单页数据不足会空出空间.
我知道Pagination组件有page-count属性, 但是如果需要支持pageSize的更改, 就只能给出pageSizetotal让组件自己计算.

我在项目中使用到了被二次封装的el-pagination组件, 同上, 为了能支持pageSize的更改, 组件作者也采用了pageSize+total设置页数的方式, 需要传入总数据量total和单页最大数据量pageSize, 虽然我是围绕这个组件说的, 但是不用担心, 这个组件封装的很浅, 而且也通过同el-paginationpageSizetotal方法来分页.

对于全部数据无差别的平均分页, 这个方案是比较理想的, 但是举例pageSize为10, 第一页只需要展示3条记录, 第二页需要6条, 在前端分页的情况下就不太好办了.

首先pageSize没法给, 固定值给上去就变成数据平均分页. pageSize也不能实时计算, pageSize一旦不固定, Element计算的总页码也是随着翻页不断变化, 这样页码的意义又在何处, 而且容易在末页产生空页情况.


一、情景

1.前端情况

这种情况后端分页会好办一些, 额, 采用前端分页, 切页不做请求.
Element多选表格, 选中多项后提交, 前端拿所有受选项去后端查询, 每个受选项查询到不定数量结果返回给前端, 后端一次性返回分隔好的所有受选项查询结果, 前端每页仅展示一个受选项的所有查询结果.

所以, 每页数据量不固定, 也不允许均分数据展示.


2.后端数据

data[i].ezglist数组为单页数据, 其内部一个对象为一条记录.
data[i].ezglist内部记录数量不定.

"data": [
  {
    "ezglist": [
      {
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx"
      }
    ]
  },
  {
    "ezglist": [
      {
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx"
      },
      {
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx"
      }
    ]
  },
  ...
]

这个就是第二章的res, 虽然不会这么短.


二、解决方案

这个二次封装的组件已经在多处使用, 我绝对不该乱动, 最好是从我的数据下手, 然后可以在渲染的时候做一点小手脚, 基本原理是用空数据占据单页内的空缺位, 防止分页组件平均分数据的时候下一页的数据压上来, 不过你放心也不会展示空行的.
这里用到了Vuex, 额, 那些指令都是SET开头, 你完全可以简单理解, 向Vuex中的某某set一个新的值.


1.处理数据

各个步骤的方法我已经全部抽离, 那么这个函数或许可以作为一个流程图?

async initTableData({ commit, state, dispatch }, queryParams) { // 初始化报表数据
  commit('SET_TABLE_LOADING', true); // 启动表格加载
  
  commit('SET_PAGE_SIZE', 9); // 固定pageSize
  
  const res = await EBEC030IBillPrint(queryParams); // 请求所有页面的、全部的数据
  commit('SET_PAGE_TOTAL', res.length); // 初始化总页数

  await dispatch('stabilizeTableAllData', res); // 生成稳定化数据
  await dispatch('getDataTotal', state.reportTableAllData); // 获取稳定化数据内所有记录构成的数组
  //必须在调节数据条数进行dataTotal获取
  
  commit('SET_REPORT_TABLE_DATA', state.reportTableAllData[0].ezglist); // 初始化第一页数据
  
  commit('SET_TABLE_LOADING', false); // 关闭表格加载
  },

  1. 固定pageSize
    如果你需要完全的隔离, 甚至都不允许有一点的本页数据流入其他页面, 就写大一点, 大到你觉得本页数据量不可能达到的值.
    我没有这样, 我允许单页内多出的数据再独占几页. 不过你依然可以按照这个方法做, 我们的区别只会是pageSize大小, 只要你的pageSize够大, 就会是你希望的结果.
commit('SET_PAGE_SIZE', 9); // 固定pageSize

  1. 请求原始数据
    也就是第一章提到的那种数据格式的数据. 并以原始数据中的对象数量作为总页数:
const res = await EBEC030IBillPrint(queryParams); // 请求所有页面的、全部的数据
commit('SET_PAGE_TOTAL', res.length); // 初始化总页数

  1. 稳定化处理
    基于pageSize对待渲染数据进行稳定化处理
await dispatch('stabilizeTableAllData', res); // 生成稳定化数据
stabilizeTableAllData({ commit, state }, data) {
  data.forEach((item) => {
    if (item.ezglist.length < state.pageSize) { // 判定:本页数据不足以填充整个页面
      const temNum = state.pageSize - item.ezglist.length; // 计算空缺位
      for (let i = 0; i <= temNum - 1; i++) { // 空缺位补齐
        item.ezglist.push(''); // 加空字符串性能大概会好些...
      }
    }
    
    if (item.ezglist.length > state.pageSize) { // 判定: 本页数据将超出本页面
      const temNum = state.pageSize - Math.ceil(item.ezglist.length % state.pageSize); //计算末页空缺位
      for (var i = 0; i <= temNum - 1; i++) { // 整页不处理, 末页空缺位补齐
        item.ezglist.push('');
      }
    }
    // 刚刚好: 不处理
  })
  commit('SET_REPORT_TABLE_ALL_DATA', data) // 稳定化数据, 存入reportTableAllData
}

遍历数据, 判定每页的数据数组中记录数是否足够填满页面, 如果不足以填满则向空缺位填充空字符串. 足以填满则允许分页, 但是末页如果无法占满依旧选择补齐.
这步完成后数据基本是这样:
假设我每页限制4条, 不假设太长了, 写不开…

"data": [
  {
    "ezglist": [
      {
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx"
      },
      '',
      '',
      ''
    ]
  },
  {
    "ezglist": [
      {
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx"
      },
      {
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx",
        "xxx": "xxx"
      },
      '',
      ''
    ]
  },
  ...
]

这种结构已经可以交付分页组件使用, 但是现在还没有dataTotal


  1. 对稳定化数据提取并集中得到新的dataTotal
    我们得把稳定化数据内的所有对象都拿出来组成一个数组, 才能计算出dataTotal, 不过也不用担心分页错乱的问题, 你想想是不是…
    各页的数据中间还是有空字符串占位的, 到时候pageSize切割完还是整页整页的数据, 不过一定要在数据稳定化之后再进行这步.
// 提取稳定化数据内所有记录生成数组
await dispatch('getDataTotal', state.reportTableAllData); 
getDataTotal({ commit }, data) {
  const temArr = [];
  for (var i = 0; i <= data.length; i++) {
    if (data[i]) {
      for (var j = 0; j <= data[i].ezglist.length - 1; j++) {
        temArr.push(data[i].ezglist[j]);
      }
    }
  }
  commit('SET_DATA_TOTAL', temArr.length); // 集中的稳定化数据, 存入dataTotal
},

现在的数据基本是这样, 我如果还是4条一页的话:

[
  {
    "xxx": "xxx",
    "xxx": "xxx",
    "xxx": "xxx",
    "xxx": "xxx"
  },
  '',
  '',
  '',
  // pageSize切割处
  {
    "xxx": "xxx",
    "xxx": "xxx",
    "xxx": "xxx",
    "xxx": "xxx"
  },
  {
    "xxx": "xxx",
    "xxx": "xxx",
    "xxx": "xxx",
    "xxx": "xxx"
  },
  '',
  ''
]

达到这个占位目的就行, 函数不用非要按我的来.


  1. 设置第一页数据
    我会在initTableData把首次进入页面, 即第一页的事情, 还有以后都不用动的东西准备好, 切换页面的时候我只是调用一下updateTableData传入当前页码来切换一下当前呈现的数据就可以了.
commit('SET_REPORT_TABLE_DATA', state.reportTableAllData[0].ezglist); // 初始化第一页数据
updateTableData({ commit, state }, pageIndex) {
  commit('SET_TABLE_LOADING', true); // 开启表格加载
  const temArr = state.reportTableAllData[pageIndex - 1].ezglist;
  commit('SET_REPORT_TABLE_DATA', temArr); // 更新当前页数据
  commit('SET_PAGE_INDEX', pageIndex); // 更新当前页码
  commit("SET_TABLE_LOADING", false); // 关闭表格加载
}

7.更新表格数据
dataTotal在第四步生成, 现在可以从Vuex拿到
page在本页面data
pageSize写死在Vuex

<page-card>
  <template v-slot:pagination>
    <Pagination
      v-if="dataTotal > 0"
      :total="dataTotal"
      :page="pageNumber"
      :limit="pageSize"
      @pagination="changePage"
    />
  </template>
</page-card>
updateTableData({ commit, state }, pageIndex) {
  commit('SET_TABLE_LOADING', true); // 开启表格加载
  const temArr = state.reportTableAllData[pageIndex - 1].ezglist;
  commit('SET_REPORT_TABLE_DATA', temArr); // 更新当前页数据
  commit('SET_PAGE_INDEX', pageIndex); // 更新当前页码
  commit("SET_TABLE_LOADING", false); // 关闭表格加载
}

2.渲染判定

我觉得你肯定不能让el-table去给你呈现你那些空字符串吧, 在el-table-column上加个v-if, 当前item如果不是个对象就不渲染:

<el-table
  :data="reportTableAllData"
>
  <template v-for="(item, index) in reportTableEvent">
    <el-table-column
      :prop="item.prop"
      label="item.label"
      v-if="item !== ''"
    >
    </el-table-column>
    <template v-else></template>
  </template>
<el-table>

记得包裹和空渲染都用template, 这不会打乱el-table的内部表结构, 它本质上还是基于table封装的.


总结

我的pageSize比较小, 9 , 如果这页不足九条, 后面的数据也不会压上来, 想完全一页数据用一页就pageSize大点, 反正不会干扰后面的数据.
多于9条会分页, 末页如果不能填满整个页面, 后面的数据也不会压上来, 改的是数据结构和选择性渲染, 这样的方法比较稳定.

  JavaScript知识库 最新文章
ES6的相关知识点
react 函数式组件 & react其他一些总结
Vue基础超详细
前端JS也可以连点成线(Vue中运用 AntVG6)
Vue事件处理的基本使用
Vue后台项目的记录 (一)
前后端分离vue跨域,devServer配置proxy代理
TypeScript
初识vuex
vue项目安装包指令收集
上一篇文章      下一篇文章      查看所有文章
加:2022-09-15 01:54:56  更:2022-09-15 01:57:15 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 14:42:45-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码