CommonTable.vue 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. <template>
  2. <div class="tablePaganations">
  3. <!-- <a-config-provider :locale="locale"> -->
  4. <a-table
  5. id="commonTable"
  6. class="ant-table-striped"
  7. bordered
  8. size="small"
  9. height="1000px"
  10. :loading="isLoading"
  11. :data-source="dataSource"
  12. :columns="columns"
  13. :row-key="(record) => record.id"
  14. :scroll="{ y: yScroll }"
  15. :pagination="havePage ? pagination : false"
  16. :row-class-name="
  17. (_record, index) => (index % 2 === 1 ? 'table-striped' : null)
  18. "
  19. :row-selection="
  20. isSelect
  21. ? {
  22. selectedRowKeys: state.selectedRowKeys,
  23. onSelect: selectEvent,
  24. onSelectAll: selectAllEvent,
  25. }
  26. : null
  27. "
  28. @change="tableChange"
  29. @resize-column="handleResizeColumn"
  30. >
  31. <template
  32. v-for="(item, index) in renderArr"
  33. #[item]="scope"
  34. :key="index"
  35. >
  36. <slot :name="item" :scope="scope" v-bind="scope || {}" />
  37. </template>
  38. </a-table>
  39. <!-- </a-config-provider> -->
  40. </div>
  41. </template>
  42. <script setup>
  43. import {
  44. useSlots,
  45. ref,
  46. reactive,
  47. defineProps,
  48. defineEmits,
  49. defineExpose,
  50. watch,
  51. onMounted,
  52. } from 'vue';
  53. import { getTableScroll } from '../common/tableScroll.js';
  54. // import zhCN from 'ant-design-vue/lib/locale-provider/zh_CN';
  55. // const locale = ref(zhCN);
  56. const props = defineProps({
  57. // 表格数据
  58. dataSource: {
  59. type: Object,
  60. required: true,
  61. },
  62. // 表头数据
  63. columns: {
  64. type: Object,
  65. required: true,
  66. },
  67. // 是否可选择
  68. isSelect: {
  69. type: Boolean,
  70. },
  71. // 表格loading
  72. isLoading: {
  73. type: Boolean,
  74. },
  75. // 数据总数
  76. total: {
  77. type: Number,
  78. default: 0,
  79. },
  80. // 是否分页
  81. havePage: {
  82. type: Boolean,
  83. default: true,
  84. },
  85. // 表格距底部高度
  86. extraHeight: {
  87. type: Number,
  88. default: undefined,
  89. },
  90. // 选择的key值
  91. selectedKeys: {
  92. type: Array,
  93. default: () => [],
  94. },
  95. });
  96. const emit = defineEmits(['getPager', 'getSorter', 'getSelected']);
  97. // 分页配置
  98. const pagination = reactive({
  99. showQuickJumper: true,
  100. current: 1,
  101. pageSize: 20, // 默认每页显示数量
  102. showSizeChanger: true, // 显示可改变每页数量
  103. pageSizeOptions: ['10', '20', '50', '100', '200', '500'], // 每页数量选项值
  104. showTotal: (total, range) =>
  105. range[0] + '-' + range[1] + '条' + ' 共' + total + '条', // 显示总数
  106. onShowSizeChange: (current, pageSize) => showSizeChange(current, pageSize),
  107. onChange: (current, pageSize) => changePage(current, pageSize), //点击页码事件
  108. total: props.total,
  109. });
  110. const yScroll = ref(400); //默认滚动高度
  111. const extraHeight = ref(undefined); //表格距离底部值
  112. // 最后一次排序信息
  113. const lastSorter = reactive({ field: '', order: '' });
  114. // 选择的数据
  115. const state = reactive({
  116. selectedRows: [],
  117. selectedRowKeys: [],
  118. });
  119. onMounted(() => {
  120. if (!props.havePage) {
  121. extraHeight.value = 30;
  122. } else {
  123. extraHeight.value = props.extraHeight;
  124. }
  125. onResizeTable();
  126. window.onresize = () => {
  127. onResizeTable();
  128. };
  129. });
  130. // 表格位置
  131. const onResizeTable = () => {
  132. yScroll.value = getTableScroll({
  133. extraHeight: extraHeight.value,
  134. id: 'commonTable',
  135. });
  136. };
  137. //点击页码事件
  138. const changePage = (current, size) => {
  139. pagination.current = current;
  140. emit('getPager', pagination.current, size);
  141. };
  142. // 改变每页数量时更新显示
  143. const showSizeChange = (current, pageSize) => {
  144. setTimeout(() => {
  145. pagination.current = 1;
  146. emit('getPager', pagination.current, pageSize);
  147. });
  148. pagination.pageSize = pageSize;
  149. };
  150. // 回到第一页
  151. const backFirstPage = () => {
  152. pagination.current = 1;
  153. emit('getPager', pagination.current, pagination.pageSize);
  154. };
  155. // 伸缩列
  156. const handleResizeColumn = (w, col) => {
  157. col.width = w;
  158. };
  159. // 选择每一项操作
  160. const selectEvent = (record, selected) => {
  161. if (selected) {
  162. state.selectedRows.push(record);
  163. state.selectedRowKeys.push(record.id);
  164. } else {
  165. let index = state.selectedRowKeys.indexOf(record.id);
  166. if (index >= 0) {
  167. state.selectedRows.splice(index, 1);
  168. state.selectedRowKeys.splice(index, 1);
  169. }
  170. }
  171. emit('getSelected', state);
  172. };
  173. // 点击以后全选当前分页数据
  174. const selectAllEvent = (selected, selectedRows, changeRows) => {
  175. if (selected) {
  176. changeRows.forEach(item => {
  177. state.selectedRows.push(item);
  178. state.selectedRowKeys.push(item.id);
  179. });
  180. } else {
  181. changeRows.forEach(item => {
  182. let index = state.selectedRowKeys.indexOf(item.id);
  183. if (index >= 0) {
  184. state.selectedRows.splice(index, 1);
  185. state.selectedRowKeys.splice(index, 1);
  186. }
  187. });
  188. }
  189. emit('getSelected', state);
  190. };
  191. // 清空选择
  192. const clear = () => {
  193. state.selectedRowKeys = [];
  194. state.selectedRows = [];
  195. emit('getSelected', state);
  196. };
  197. // 获取排序信息
  198. const tableChange = (pagination, filters, sorter) => {
  199. // pagination, filters 变化时也会触发所以对sorter进行判断限制执行
  200. if (Object.keys(sorter).length > 0) {
  201. if (sorter.field != lastSorter.field && sorter.order != lastSorter.order) {
  202. lastSorter.field = sorter.field;
  203. lastSorter.order = sorter.order;
  204. emit('getSorter', sorter);
  205. }
  206. if (sorter.field != lastSorter.field && sorter.order == lastSorter.order) {
  207. lastSorter.field = sorter.field;
  208. lastSorter.order = sorter.order;
  209. emit('getSorter', sorter);
  210. }
  211. if (sorter.field == lastSorter.field && sorter.order != lastSorter.order) {
  212. lastSorter.field = sorter.field;
  213. lastSorter.order = sorter.order;
  214. emit('getSorter', sorter);
  215. }
  216. }
  217. };
  218. // 暴露出方法
  219. defineExpose({ backFirstPage, clear });
  220. // 监听total变化
  221. watch(
  222. props,
  223. newData => {
  224. pagination.total = newData.total;
  225. extraHeight.value = newData.extraHeight;
  226. },
  227. { immediate: true, deep: true },
  228. );
  229. // 插槽的实例
  230. const slots = useSlots();
  231. const renderArr = Object.keys(slots);
  232. </script>
  233. <style scoped>
  234. .tablePaganations {
  235. width: 100%;
  236. margin-top: 8px;
  237. }
  238. .ant-table-striped :deep(.table-striped) td {
  239. background-color: #fafafa;
  240. }
  241. </style>