一、整体效果图
二、选中数据效果
注明:此表格是表头自定义多类型筛选的升级版
- 涉及到的组件代码会以链接的形式贴出来
- 涉及到的Scss变量、样式、接口方法,请自行调试
- 列表的表头及每列数据均由后端返回
- 列表当中数据的状态颜色、tag标签均可配置
- 效果仅针对记录当前项目功能效果的实现
- Zn开头的组件为全局加载,请自行引入加载
- 列表无数据或无权限时prop传入图片路径及提示语即可,(此处可优化)
- 表格第一列序号与复选框hover切换
- 表格上方左侧为视图类别,类似tabs切换,类型不同,展示的数据形式也不同 , 比如下方数据展示有可能是列表, 有可能是card(创建视图功能暂未开发 , 目前只做了类型切换的展示 , 具体样子可去飞书观看一下)
三、页面示例代码
<template>
<div class="workOrder-container">
<zn-query-form>
<zn-query-form-left-panel>
<zn-views-tabs :options="viewsTabsOptions" @upTable="getHeader" />
</zn-query-form-left-panel>
<zn-query-form-right-panel>
<!-- 搜索 -->
<zn-search-btn @up-Search="upSearch" />
<!-- 筛选 -->
<zn-screen-btn
ref="ZnScreenBtn"
:mark="mark"
:views_id="queryForm.views_id"
:options="managementOptions"
@upTable="fetchData"
/>
<!-- 字段管理 -->
<zn-field-management-btn
ref="ZnFieldManagementBtn"
v-if="viewsType == 'table'"
:mark="mark"
:views_id="queryForm.views_id"
:options="managementOptions"
@upTable="upTableData"
/>
<!-- 卡片管理 -->
<zn-card-management-btn
ref="ZnCardManagementBtn"
v-if="viewsType == 'gallery'"
:mark="mark"
:views_id="queryForm.views_id"
:options="managementOptions"
@upCard="upCardData"
/>
<!-- 导出全部 -->
<el-button
type="primary"
size="medium"
plain
@click="reportAll"
v-permissions="{
permission: ['workOrder:workOrder:allWorkOrder:export'],
}"
>
<zn-icon :iconName="'emport'" />
导出
</el-button>
<el-button
class="el-add"
type="primary"
size="medium"
@click="addWorkOrder"
v-permissions="{ permission: ['workOrder:allWorkOrder:add'] }"
>
<zn-icon :iconName="'callcenter-add'" />
新建工单
</el-button>
</zn-query-form-right-panel>
</zn-query-form>
<!-- 相册试图页面 -->
<div
v-if="viewsType === 'gallery'"
class="content-view album-view"
v-infinite-scroll="loadMore"
:infinite-scroll-immediate="false"
>
<zn-filter-album
ref="filterAlbum"
:albumData="tableList"
:finallyColumns="finallyColumns"
:deatilsPath="deatilsPath"
:indexColumn="indexColumn"
@fetch-data="fetchData"
/>
</div>
<!-- 表格试图页面 -->
<div class="content-view table-view" v-else>
<zn-filter-table
v-if="viewsType == 'table'"
ref="filterTable"
:class="total === 0 ? 'table-view-empty' : ''"
:multiple="true"
:tableData="tableList"
:finallyColumns="finallyColumns"
:deatilsPath="deatilsPath"
@selectList="getSelect"
@fetch-data="fetchData"
:defaultPageImgUrl="
require('@/assets/images/default_page/pic-currency.png')
"
:defaultPageTips="'暂无数据'"
>
<el-button type="text" size="medium" @click="reportPart">
<zn-icon :iconName="'emport'" />
<span>导出选中</span>
</el-button>
</zn-filter-table>
<zn-pagination
v-show="total > 0"
:page.sync="queryForm.page"
:limit.sync="queryForm.listRows"
@pagination="fetchData"
:total="total"
:algin="'right'"
class="table-pagination"
/>
</div>
<add-work-order
:modalToBody="false"
:configData="configData"
ref="addWorkOrder"
/>
</div>
</template>
<script>
import addWorkOrder from './components/addWorkOrder'
import {
getList,
getWorkOrderConfig,
exportAllExcel,
exportSelectExcel,
} from '@/api/workOrder'
import { getTableHeader, viewsTabs, filters } from '@/api/index'
export default {
name: 'WorkOrder',
props: {},
components: {
addWorkOrder,
},
data() {
return {
mark: 'WorkOrderList',
configData: null,
viewsType: '',
total: 0,
tableList: [],
viewsTabsOptions: [],
managementOptions: {},
listLoading: false,
queryForm: {
page: 1,
listRows: 10,
keywords: '',
views_id: '',
},
deatilsPath: '/workOrder/workOrderDetail',
columns: [],
checkList: [],
moreList: [],
indexColumn: '',
}
},
computed: {
finallyColumns() {
if (this.checkList.length > 0) {
return this.columns.filter((item) =>
this.checkList.includes(item.label)
)
}
},
},
watch: {},
created() {
this.getConfigData()
this.getViewsTabs()
},
mounted() {},
methods: {
getConfigData() {
getWorkOrderConfig().then((res) => {
this.configData = res.data
})
},
addWorkOrder() {
this.$refs['addWorkOrder'].show()
},
getViewsTabs() {
viewsTabs({ mark: this.mark }).then((res) => {
this.viewsTabsOptions = res.data
if (res.code == 200 && res.data.length > 0) {
this.queryForm.views_id = res.data[0].id
this.viewsType = res.data[0].type
this.getFilters()
this.getHeader()
if (res.data[0].type === 'table') {
this.fetchData()
} else {
this.fetchAlbumData()
}
}
})
},
getFilters() {
filters({ mark: this.mark, views_id: this.queryForm.views_id }).then(
(res) => {
if (this.$refs.ZnScreenBtn) {
this.$refs.ZnScreenBtn.form.filters = res.data.filters
}
}
)
},
async getHeader(obj) {
getTableHeader({
mark: this.mark,
views_id: obj ? obj.id : this.queryForm.views_id,
}).then((res) => {
this.managementOptions = res.data
this.columns = res.data.fields
this.indexColumn = res.data.options.indexColumn
if (obj) {
this.queryForm.views_id = obj.id
this.viewsType = obj.type
this.getFilters()
if (obj.type === 'table') {
this.fetchData()
} else {
this.tableList = []
this.queryForm.page = 1
this.loadMore()
}
}
this.managementOptions.fields.map((item) => {
if (item.isShow == true) {
this.checkList.push(item.label)
}
})
})
},
async fetchData() {
this.listLoading = true
const {
data: { data, total },
} = await getList(this.queryForm)
this.tableList = data
this.total = total
this.listLoading = false
},
async fetchAlbumData() {
this.listLoading = true
const {
data: { data, total, last_page },
} = await getList(this.queryForm)
this.tableList = this.tableList.concat(data)
if (this.queryForm.page >= last_page + 1) {
this.queryForm.page = last_page + 1
} else {
this.queryForm.page = this.queryForm.page + 1
}
this.total = total
this.listLoading = false
},
upSearch(val) {
this.tableList = []
this.queryForm.page = 1
this.queryForm.keywords = val
if (this.viewsType === 'table') {
this.fetchData()
} else {
this.loadMore()
}
},
getSelect(list) {
this.moreList = list
},
upTableData(columns, checkList) {
this.columns = columns
this.checkList = checkList
},
upCardData(columns, checkList, showLabel) {
this.columns = columns
this.checkList = checkList
this.showLabel = showLabel
},
loadMore() {
this.queryForm.listRows = 6
this.fetchAlbumData()
},
reportPart() {
exportSelectExcel({
views_id: this.queryForm.views_id,
ids: this.moreList.map((item) => item.id).join(','),
}).then((res) => {
if (res.code === 200) {
this.$download(res.data)
}
})
},
reportAll() {
exportAllExcel({
views_id: this.queryForm.views_id,
keywords: this.queryForm.keywords,
}).then((res) => {
if (res.code === 200) {
this.$download(res.data)
}
})
},
},
}
</script>
<style lang="scss" scoped>
.workOrder-container {
height: calc(100vh - 60px - 50px - 16px * 2);
::v-deep .add-workorder-drawer {
position: absolute;
}
.el-add {
margin-right: 0 !important;
}
.content-view {
height: calc(100% - 55px);
&.album-view {
overflow-x: hidden;
overflow-y: auto;
padding: 0 16px 16px;
}
&.table-view {
position: relative;
}
.filter-table {
height: calc(100% - 69px);
padding-bottom: 0;
&.table-view-empty {
height: 100%;
}
::v-deep .el-table {
height: 100%;
.el-table__body-wrapper {
height: calc(100% - 48px);
overflow: auto;
}
}
}
}
}
</style>
四、总结
- 功能很强,但是也有局限性,适用于瀑布流开发模式
- 页面调用的这个层级其实还可以再封装一层
- 有些地方是需要再优化的,由于需求一直在变,也无心高度优化(每个版本都要动这里面的东西,心累)
五、github源码链接
|