table.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534
  1. <template>
  2. <div v-loading="loading" element-loading-text="拼命加载中" element-loading-spinner="el-icon-loading" element-loading-background="rgba(0, 0, 0, 0.8)" class="newQueryTable">
  3. <template v-if="istableChild">
  4. <el-table v-el-table-infinite-scroll="load" :data="dealedTableData" :summary-method="getSummaries" :span-method="tableSpanMethod" stripe :show-summary="showSummary" border @cell-click="cellClick" :cell-class-name="cellClass" ref="table" height="100%" class="table infinite-list">
  5. <el-table-column v-for="col in tableColsCopy" :key="col.columnName" :prop="col.columnName" :label="col.groupName" align="center">
  6. <el-table-column v-for="childCol in col.children" :key="childCol.columnName" :sortable="childCol.needSort ? true : false" :prop="childCol.columnName" :label="childCol.columnLabel" :formatter="formatter">
  7. <template #header>
  8. <el-tooltip :content="childCol.columnDescribe || childCol.columnLabel" placement="top">
  9. <template v-if="childCol.needFilters">
  10. <TableHeaderCell :label="childCol.columnLabel" :filter-options="tableDataFilters[childCol.columnName]" :filter-values.sync="filterValues[childCol.columnName]" />
  11. </template>
  12. <template v-else>
  13. <div>{{ childCol.columnLabel }}</div>
  14. </template>
  15. </el-tooltip>
  16. </template>
  17. </el-table-column>
  18. </el-table-column>
  19. </el-table>
  20. </template>
  21. <template v-else>
  22. <el-table v-el-table-infinite-scroll="load" :data="dealedTableData" :summary-method="getSummaries" :span-method="tableSpanMethod" stripe :show-summary="showSummary" border @cell-click="cellClick" :cell-class-name="cellClass" ref="table" height="100%" class="table infinite-list">
  23. <el-table-column v-for="(item, index) in tableColsCopy" :sortable="item.needSort ? true : false" :key="index" :prop="item.columnName" :label="item.columnLabel" :show-overflow-tooltip="showOverflowTooltip">
  24. <template #header>
  25. <el-tooltip :content="item.columnDescribe || item.columnLabel" placement="top">
  26. <template v-if="item.needFilters">
  27. <TableHeaderCell :label="item.columnLabel" :filter-options="tableDataFilters[item.columnName]" :filter-values.sync="filterValues[item.columnName]" />
  28. </template>
  29. <template v-else>
  30. <div>{{ item.columnLabel }}</div>
  31. </template>
  32. </el-tooltip>
  33. </template>
  34. </el-table-column>
  35. </el-table>
  36. </template>
  37. <template v-if="istableCol">
  38. <div :style="btnStyle" class="btns">
  39. <img class="btn-square btn-shadow" src="@/assets/baggage/ic_setting.png" title="列设置" @click="show">
  40. </div>
  41. <Dialog :flag="dialogFlag" width="600px" class="dialog-check-group">
  42. <div class="dialog-wrapper">
  43. <div class="title">列设置</div>
  44. <div style="height: 600px;" class="content">
  45. <el-scrollbar style="height: 100%">
  46. <el-tree ref="columnSetTree" :data="tableCols" :class="colsCheckClass" show-checkbox node-key="index" :default-expand-all="true" :props="{
  47. label: 'columnLabel',
  48. children: 'children',
  49. }" :default-checked-keys="checkedKeysTemp" @check="handleCheck" />
  50. </el-scrollbar>
  51. </div>
  52. <div class="foot right t30">
  53. <el-button size="medium" class="r24" type="primary" @click="onCheck('baggageTableData')">确定</el-button>
  54. <el-button size="medium" @click="hide">取消</el-button>
  55. </div>
  56. </div>
  57. </Dialog>
  58. </template>
  59. </div>
  60. </template>
  61. <script>
  62. import pf from '@/layout/mixin/publicFunc';
  63. import { getToken } from '@/utils/auth';
  64. import { getAuthData, formatOrder } from "@/utils/validate";
  65. import { throttledExportToExcel } from '@/utils/table';
  66. import tableColsMixin from '../mix/tableCols';
  67. import Dialog from '@/layout/components/Dialog/index.vue';
  68. import TableHeaderCell from "@/components/TableHeaderCell";
  69. export default {
  70. name: 'NewQueryTable',
  71. mixins: [pf, tableColsMixin],
  72. components: { Dialog, TableHeaderCell },
  73. props: {
  74. // 不换行,溢出隐藏
  75. showOverflowTooltip: {
  76. type: Boolean,
  77. default: true,
  78. },
  79. tableTag: {
  80. type: Object,
  81. default: () => { }
  82. },
  83. tableName: {
  84. type: String,
  85. default: ''
  86. },
  87. istableCol: {
  88. type: Boolean,
  89. default: false,
  90. },
  91. istableChild: {
  92. type: Boolean,
  93. default: false,
  94. },
  95. btnStyle: {
  96. type: Object,
  97. default: () => { }
  98. },
  99. pageSize: {
  100. type: Number,
  101. default: 20
  102. }
  103. },
  104. data () {
  105. return {
  106. page: 0,
  107. queryId: '',
  108. noMore: false,
  109. loading: false,
  110. tableCols: [], //表头数据
  111. tableData: [], //表格数据
  112. tableColsCopy: [], //表头数据缓存
  113. tableDataFilters: {}, //表头-下拉数据
  114. filterValues: {}, //表头-下拉-选中数据
  115. tableDataCopy: [], //缓存table数据
  116. tableGroups: [], //表格分组数据
  117. colShowFilter: "", //表头-下拉-箭头
  118. spanArr: [], //表格分组数据缓存
  119. pos: 0, //表格分组计数
  120. dataContent: [],
  121. authBtns: [],
  122. authBtnCol: [],
  123. authBtnColName: [],
  124. tableArrs: [], //重组table-表头下拉
  125. tableOptions: {}, //弹框-下来数据缓存
  126. showSummary: false, //是否显示统计
  127. tableDataSortRules: {},
  128. }
  129. },
  130. computed: {
  131. //设置表头-下拉-选中数据
  132. dealedTableData () {
  133. const filtered = this.tableData.filter(item => {
  134. let flag = true
  135. Object.entries(this.filterValues).forEach(([key, arr]) => {
  136. if (arr.length && !arr.includes(String(item[key]))) {
  137. flag = false
  138. }
  139. })
  140. return flag
  141. })
  142. const sortRules = Object.entries(this.tableDataSortRules).reduce(
  143. (pre, [key, value]) => {
  144. if (value) {
  145. pre[0].push(key)
  146. value = value === 'ascending' ? 'asc' : 'desc'
  147. pre[1].push(value)
  148. }
  149. return pre
  150. },
  151. [[], []]
  152. )
  153. return this._.orderBy(filtered, sortRules[0], sortRules[1])
  154. },
  155. tableHeight () {
  156. return 100
  157. },
  158. fromDataType () {
  159. return function (type) {
  160. if (type) {
  161. return type.replace(/\([^\)]*\)/g, "");
  162. }
  163. }
  164. },
  165. },
  166. watch: {
  167. tableTag: {
  168. handler (val) {
  169. this.dataContent = val
  170. this.restTable()
  171. this.load()
  172. },
  173. deep: true
  174. }
  175. },
  176. created () {
  177. if (this.AauthTable && this.AauthTable.length) {
  178. const t = this.AauthTable.filter(item => item.auth_name == this.tableName)
  179. const obj = t[0]
  180. this.AqueryId = obj.queryTemplateID
  181. this.AauthId = obj.auth_id
  182. }
  183. this.getColumnData()
  184. },
  185. mounted () {
  186. this.setCellClick()
  187. },
  188. updated () {
  189. this.$refs['table']?.doLayout()
  190. },
  191. methods: {
  192. //获取表头数据
  193. async getColumnData () {
  194. try {
  195. const { code, returnData } = await this.getQueryList(SERVICE_ID.sysUserAuthId, [{
  196. user_id: getToken('userid'),
  197. auth_id: this.AauthId
  198. }]);
  199. if (code == 0) {
  200. if (returnData && returnData.length) {
  201. if (!this.istableCol) {
  202. sessionStorage.setItem('tableColumns', JSON.stringify(returnData))
  203. this.$store.dispatch('auth/changeAuthMsg', returnData)
  204. }
  205. const msgDatas = returnData.filter((item) => item.needShow);
  206. const msgCounts = msgDatas.filter((item) => item.needCount);
  207. if (msgCounts.length) {
  208. this.showSummary = true
  209. }
  210. const msgDatasShows = formatOrder(msgDatas);
  211. if (this.istableChild) {
  212. const datas = _.cloneDeep(msgDatasShows)
  213. const cache = {};
  214. const indices = [];
  215. const others = []
  216. datas.forEach((item) => {
  217. if (!cache[item.groupName] && item.groupName) {
  218. cache[item.groupName] = item.groupName
  219. indices.push(item)
  220. } else {
  221. const newItem = _.cloneDeep(item)
  222. others.push(newItem)
  223. }
  224. })
  225. indices.map((item, index) => {
  226. item.tabIndex = index
  227. })
  228. others.forEach(item => {
  229. indices.forEach(p => {
  230. if (item.groupName == p.groupName && item.groupName) {
  231. item.tabIndex = p.tabIndex
  232. }
  233. })
  234. })
  235. indices.forEach((item) => {
  236. item.children = [_.cloneDeep(item), ...this.formatCaps(item.tabIndex, others)]
  237. })
  238. this.tableCols = _.cloneDeep(indices);
  239. } else {
  240. this.tableCols = _.cloneDeep(msgDatasShows);
  241. }
  242. this.tableColsCopy = _.cloneDeep(this.tableCols);
  243. this.initTableCols();
  244. }
  245. } else {
  246. this.$message.error("获取表头数据失败");
  247. }
  248. } catch (error) {
  249. console.log(error)
  250. }
  251. },
  252. formatCaps (order, arr) {
  253. const datas = []
  254. for (let i = 0; i < arr.length; i++) {
  255. const element = arr[i];
  256. if (element['tabIndex'] == order) {
  257. datas.push(element)
  258. }
  259. }
  260. return datas
  261. },
  262. //获取表格数据
  263. async getQuery (id, dataContent = this.dataContent, page, pageSize) {
  264. try {
  265. this.loading = true;
  266. const { code, returnData } = await this.getQueryListAuth(id, dataContent, page, pageSize);
  267. if (code == 0) {
  268. if (returnData.length === 0) {
  269. this.page--;
  270. this.noMore = true;
  271. this.loading = false;
  272. }
  273. this.tableData.push(...returnData);
  274. setTimeout(() => {
  275. this.initTableData();
  276. this.loading = false;
  277. }, 100);
  278. } else {
  279. this.page--;
  280. this.loading = false;
  281. this.$message.error("获取表格数据失败");
  282. }
  283. } catch (error) {
  284. this.page--;
  285. this.loading = false;
  286. }
  287. },
  288. restTable () {
  289. this.loading = false
  290. this.noMore = false
  291. this.page = 0
  292. this.tableData = []
  293. },
  294. load () {
  295. if (this.tableTag && Object.keys(this.tableTag).length) {
  296. if (this.noMore || this.loading) {
  297. return;
  298. }
  299. this.getQuery(this.AqueryId, this.dataContent, ++this.page, this.pageSize);
  300. }
  301. },
  302. setCellClick () {
  303. const { auth_id } = this.$route.meta
  304. const { arrs } = getAuthData(auth_id)
  305. const table = arrs.filter(item => item.auth_type == 4)
  306. if (table && table.length) {
  307. const obj = table[0]
  308. const { arrs } = getAuthData(obj.auth_id)
  309. this.authBtnCol = arrs.filter(item => Number(item.service_type) == 6)
  310. if (this.authBtnCol.length) {
  311. this.authBtnCol.forEach(item => {
  312. this.authBtnColName.push(item.relation_data)
  313. })
  314. }
  315. }
  316. },
  317. //初始化表格
  318. initTableData () {
  319. this.tableDataCopy = _.cloneDeep(this.tableData);
  320. const datas = _.cloneDeep(this.tableColsCopy);
  321. // const reqUts = [];
  322. datas.forEach(async (item) => {
  323. this.tableDataFilters[item.columnName] = [];
  324. if (item.needGroup) {
  325. this.tableGroups.push(item.columnName);
  326. }
  327. // if (item.listqueryTemplateID || item.listqueryTemplateID == 0) {
  328. // this.tableArrs.push(item.columnName);
  329. // if (!this.tableOptions[item.columnName]) {
  330. // this.tableOptions[item.columnName] = await this.getSelectData(item.listqueryTemplateID);
  331. // }
  332. // }
  333. });
  334. const dats = this.setTableFilters(this.tableData, this.tableDataFilters);
  335. this.tableDataFilters = _.cloneDeep(dats);
  336. this.tableGroup(this.tableData);
  337. },
  338. setTableFilters (tableData, filters) {
  339. const tempSets = {}
  340. Object.keys(filters).forEach(key => {
  341. tempSets[key] = new Set()
  342. })
  343. tableData.forEach(item => {
  344. Object.keys(tempSets).forEach(key => {
  345. (item[key] ?? '') !== '' && tempSets[key].add(String(item[key]))
  346. })
  347. })
  348. Object.keys(tempSets).forEach(key => {
  349. filters[key] = _.orderBy(
  350. [...tempSets[key]].map(value => ({
  351. text: value,
  352. value
  353. })),
  354. o => o.value
  355. )
  356. })
  357. return filters
  358. },
  359. //分组
  360. tableGroup (tableData) {
  361. const spanArr = [];
  362. let pos = 0;
  363. let ifYj = this.tableGroups[0];
  364. for (let i = 0; i < tableData.length; i++) {
  365. if (i === 0) {
  366. spanArr.push(1);
  367. } else {
  368. if (tableData[i][ifYj] === tableData[i - 1][ifYj]) {
  369. spanArr[pos] += 1;
  370. spanArr.push(0);
  371. } else {
  372. spanArr.push(1);
  373. pos = i;
  374. }
  375. }
  376. }
  377. this.spanArr = spanArr;
  378. this.pos = pos;
  379. },
  380. popoverShowHandler (prop) {
  381. this.colShowFilter = prop;
  382. },
  383. popoverHideHandler () {
  384. this.colShowFilter = "";
  385. },
  386. //获取弹框-下拉数据
  387. async getSelectData (id) {
  388. // name ? [name] : name === null ? [null]: [],
  389. const { code, returnData } = await this.getQueryList(id, {});
  390. console.log(returnData)
  391. if (code == 0) {
  392. return returnData;
  393. } else {
  394. return [];
  395. }
  396. },
  397. //设置表头-下拉-箭头样式
  398. arrowClass () {
  399. return function (prop) {
  400. let classString = "";
  401. if (this.colShowFilter === prop) {
  402. return "arrow-active";
  403. }
  404. if (
  405. Object.entries(this.tableDataFilters).find(
  406. ([key, arr]) => this.filterValues[prop]
  407. )
  408. ) {
  409. classString += "arrow-blue";
  410. }
  411. return classString;
  412. };
  413. },
  414. //合计
  415. getSummaries (param) {
  416. const { columns, data } = param;
  417. const sums = [];
  418. columns.forEach((column, index) => {
  419. if (index === 0) {
  420. sums[index] = '合计:' + this.tableData.length
  421. }
  422. this.tableColsCopy.forEach((p) => {
  423. if (column.property == p.columnName && p.needCount) {
  424. const values = data.map((item) => Number(item[column.property]));
  425. if (!values.every((value) => isNaN(value))) {
  426. sums[index] = values.reduce((prev, curr) => {
  427. const value = Number(curr);
  428. if (!isNaN(value)) {
  429. return prev + curr;
  430. } else {
  431. return prev;
  432. }
  433. }, 0);
  434. sums[index] += "";
  435. }
  436. }
  437. });
  438. });
  439. return sums;
  440. },
  441. //分组
  442. tableSpanMethod ({ row, column, rowIndex, columnIndex }) {
  443. if (this.tableGroups.includes(column["property"])) {
  444. const _row = this.spanArr[rowIndex];
  445. const _col = _row > 0 ? 1 : 0;
  446. return {
  447. rowspan: _row,
  448. colspan: _col,
  449. };
  450. }
  451. },
  452. //表格-设置单元格样式
  453. cellClass ({ row, column, rowIndex, columnIndex }) {
  454. if (this.authBtnColName.includes(column.property)) {
  455. return 'is-click-btn'
  456. }
  457. },
  458. //表格-单元格点击
  459. cellClick (row, column) {
  460. const dataBtns = this.authBtnCol
  461. if (dataBtns && dataBtns.length) {
  462. const clickBtn = dataBtns.filter(item => item.relation_data == column.property)[0]
  463. if (clickBtn) {
  464. const { open_method, route_info, pass_parameters } = clickBtn
  465. if (open_method == 2) {
  466. if (pass_parameters) {
  467. const query = pass_parameters.split(',')
  468. const obj = {}
  469. query.forEach(item => {
  470. obj[item] = row[item]
  471. })
  472. this.$router.push({
  473. path: route_info,
  474. query: obj
  475. })
  476. }
  477. }
  478. }
  479. }
  480. },
  481. //导出
  482. exportHandler () {
  483. const table = this.$refs['table'].$el.cloneNode(true)
  484. // const { luggageNum, flightNo, flightDate } = this.query
  485. const fileName = `高级查询.xlsx`
  486. throttledExportToExcel(table, '高级查询', fileName)
  487. },
  488. formatter (row, column, cellValue, index) {
  489. const sameColumn = this.tableCols.find(col => col.columnName === column.property)
  490. if (sameColumn && this.fromDataType(sameColumn.dataType) === 'datetime') {
  491. return (cellValue ?? '').replace('T', ' ')
  492. }
  493. return cellValue
  494. }
  495. }
  496. }
  497. </script>
  498. <style lang="scss" scoped>
  499. .newQueryTable {
  500. height: 100%;
  501. position: relative;
  502. ::v-deep .table {
  503. .is-click-btn {
  504. .cell {
  505. color: #409eff;
  506. cursor: pointer;
  507. position: relative;
  508. }
  509. }
  510. .cell {
  511. color: #000;
  512. text-align: center;
  513. white-space: nowrap;
  514. display: flex;
  515. justify-content: center;
  516. // display: flex;
  517. .el-tooltip {
  518. white-space: nowrap;
  519. text-overflow: ellipsis;
  520. overflow: hidden;
  521. line-height: 34px;
  522. }
  523. }
  524. }
  525. .btns {
  526. position: absolute;
  527. top: -50px;
  528. right: 32px;
  529. z-index: 10;
  530. }
  531. }
  532. </style>