<template>
|
<div class="table-container">
|
<div v-if="!hideTool" class="table-header mb15">
|
<div>
|
<slot name="command"></slot>
|
</div>
|
<div v-loading="state.exportLoading" class="table-footer-tool">
|
<SvgIcon v-if="!config.hideRefresh" name="iconfont icon-shuaxin" :size="22" title="刷新" @click="() => onRefreshTable()" class="tool-icon" />
|
<el-tooltip effect="light" :content="state.switchFixedContent" placement="bottom-start" :show-after="200" v-if="state.haveFixed" >
|
<el-icon :style="{ color: state.fixedIconColor }" @click="switchFixed" class="tool-icon"><ele-Switch /></el-icon>
|
</el-tooltip>
|
<el-dropdown v-if="!config.hideExport" trigger="click">
|
<SvgIcon name="iconfont icon-yunxiazai_o" :size="22" title="导出" class="tool-icon" />
|
<template #dropdown>
|
<el-dropdown-menu>
|
<el-dropdown-item @click="onExportTable">导出本页数据</el-dropdown-item>
|
<el-dropdown-item @click="onExportTableAll">导出全部数据</el-dropdown-item>
|
</el-dropdown-menu>
|
</template>
|
</el-dropdown>
|
<SvgIcon v-if="!config.hidePrint" name="iconfont icon-dayin" :size="19" title="打印" @click="onPrintTable" class="tool-icon" />
|
<el-popover v-if="!config.hideSet" placement="bottom-end" trigger="click" transition="el-zoom-in-top" popper-class="table-tool-popper" :width="180" :persistent="false" @show="onSetTable">
|
<template #reference>
|
<SvgIcon name="iconfont icon-quanjushezhi_o" class="tool-icon" :size="22" title="设置" />
|
</template>
|
<template #default>
|
<div class="tool-box">
|
<el-checkbox v-model="state.checkListAll" :indeterminate="state.checkListIndeterminate" class="ml10 mr1" label="列显示" @change="onCheckAllChange" />
|
<el-checkbox v-model="getConfig.isSerialNo" class="ml12 mr1" label="序号" />
|
<el-checkbox v-if="getConfig.showSelection" v-model="getConfig.isSelection" class="ml12 mr1" label="多选" />
|
<el-tooltip content="拖动进行排序" placement="top-start">
|
<SvgIcon style="float: right; margin-right: 5px; margin-top: 2px" name="fa fa-question-circle-o" :size="17" class="ml11" color="#909399" />
|
</el-tooltip>
|
</div>
|
<el-divider style="margin: 10px 0 10px -5px" />
|
<el-scrollbar>
|
<div ref="toolSetRef" class="tool-sortable">
|
<div class="tool-sortable-item" v-for="v in columns" :key="v.prop" v-show="!v.hideCheck" :data-key="v.prop">
|
<i class="fa fa-arrows-alt handle cursor-pointer"></i>
|
<el-checkbox v-model="v.isCheck" size="default" class="ml12 mr8" :label="v.label" @change="onCheckChange" />
|
</div>
|
</div>
|
</el-scrollbar>
|
</template>
|
</el-popover>
|
</div>
|
</div>
|
<el-table
|
ref="tableRef"
|
:data="state.data"
|
:border="setBorder"
|
:stripe="setStripe"
|
v-bind="$attrs"
|
row-key="id"
|
default-expand-all
|
style="width: 100%"
|
v-loading="state.loading"
|
:default-sort="defaultSort"
|
@selection-change="onSelectionChange"
|
@sort-change="sortChange"
|
>
|
<el-table-column type="selection" :reserve-selection="true" :width="30" v-if="config.isSelection && config.showSelection" />
|
<el-table-column type="index" :fixed="state.currentFixed && state.serialNoFixed" label="序号" align="center" :width="60" v-if="config.isSerialNo" />
|
<el-table-column v-for="(item, index) in setHeader" :key="index" v-bind="item">
|
<template #header v-if="!item.children && $slots[item.prop]">
|
<slot :name="`${item.prop}header`" />
|
</template>
|
<!-- 自定义列插槽,插槽名为columns属性的prop -->
|
<template #default="scope" v-if="!item.children && $slots[item.prop]">
|
<formatter v-if="item.formatter" :fn="item.formatter(scope.row, scope.column, scope.cellValue, scope.index)"> </formatter>
|
<slot v-else :name="item.prop" v-bind="scope"></slot>
|
</template>
|
<template v-else-if="!item.children" v-slot="scope">
|
<formatter v-if="item.formatter" :fn="item.formatter(scope.row, scope.column, scope.cellValue, scope.index)"> </formatter>
|
<template v-else-if="item.type === 'image'">
|
<el-image
|
:style="{ width: `${item.width}px`, height: `${item.height}px` }"
|
:src="scope.row[item.prop]"
|
:zoom-rate="1.2"
|
:preview-src-list="[scope.row[item.prop]]"
|
preview-teleported
|
fit="cover"
|
/>
|
</template>
|
<template v-else>
|
{{ getProperty(scope.row, item.prop) }}
|
</template>
|
</template>
|
<el-table-column v-for="(childrenItem, childrenIndex) in item.children" :key="childrenIndex" v-bind="childrenItem">
|
<!-- 自定义列插槽,插槽名为columns属性的prop -->
|
<template #default="scope" v-if="$slots[childrenItem.prop]">
|
<formatter v-if="childrenItem.formatter" :fn="childrenItem.formatter(scope.row, scope.column, scope.cellValue, scope.index)"> </formatter>
|
<slot v-else :name="childrenItem.prop" v-bind="scope"></slot>
|
</template>
|
<template v-else v-slot="scope">
|
<formatter v-if="childrenItem.formatter" :fn="childrenItem.formatter(scope.row, scope.column, scope.cellValue, scope.index)"> </formatter>
|
<template v-else-if="childrenItem.type === 'image'">
|
<el-image
|
:style="{ width: `${childrenItem.width}px`, height: `${childrenItem.height}px` }"
|
:src="scope.row[childrenItem.prop]"
|
:zoom-rate="1.2"
|
:preview-src-list="[scope.row[childrenItem.prop]]"
|
preview-teleported
|
fit="cover"
|
/>
|
</template>
|
<template v-else>
|
{{ getProperty(scope.row, childrenItem.prop) }}
|
</template>
|
</template>
|
</el-table-column>
|
</el-table-column>
|
<template #empty>
|
<el-empty description="暂无数据" />
|
</template>
|
</el-table>
|
<div v-if="!config.hidePagination && state.showPagination" class="table-footer mt15">
|
<el-pagination
|
v-model:current-page="state.page.page"
|
v-model:page-size="state.page.pageSize"
|
size="small"
|
:pager-count="5"
|
:page-sizes="config.pageSizes"
|
:total="state.total"
|
layout="total, sizes, prev, pager, next, jumper"
|
background
|
@size-change="onHandleSizeChange"
|
@current-change="onHandleCurrentChange"
|
>
|
</el-pagination>
|
</div>
|
</div>
|
</template>
|
|
<script setup lang="ts" name="netxTable">
|
import { reactive, computed, nextTick, ref, onMounted } from 'vue';
|
import { ElMessage } from 'element-plus';
|
import Sortable from 'sortablejs';
|
import { storeToRefs } from 'pinia';
|
import printJs from 'print-js';
|
import { EmptyObjectType } from "/@/types/global";
|
import formatter from '/@/components/table/formatter.vue';
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
import { exportExcel } from '/@/utils/exportExcel';
|
|
// 定义父组件传过来的值
|
const props = defineProps({
|
// 获取数据的方法,由父组件传递
|
getData: {
|
type: Function,
|
required: true,
|
},
|
// 列属性,和elementUI的Table-column 属性相同,附加属性:isCheck-是否默认勾选展示,hideCheck-是否隐藏该列的可勾选和拖拽
|
columns: {
|
type: Array<any>,
|
default: () => [],
|
},
|
// 配置项:isBorder-是否显示表格边框,isSerialNo-是否显示表格序号,showSelection-是否显示表格可多选,isSelection-是否默认选中表格多选,pageSize-每页条数,hideExport-是否隐藏导出按钮,exportFileName-导出表格的文件名,空值默认用应用名称作为文件名
|
config: {
|
type: Object,
|
default: () => ({}),
|
},
|
// 筛选参数
|
param: {
|
type: Object,
|
default: () => ({}),
|
},
|
// 默认排序方式,{prop:"排序字段",order:"ascending or descending"}
|
defaultSort: {
|
type: Object,
|
default: () => ({}),
|
},
|
// 导出报表自定义数据转换方法,不传按字段值导出
|
exportChangeData: {
|
type: Function,
|
},
|
// 打印标题
|
printName: {
|
type: String,
|
default: () => '',
|
},
|
});
|
|
// 定义子组件向父组件传值/事件,pageChange-翻页事件,selectionChange-表格多选事件,可以在父组件处理批量删除/修改等功能,sortHeader-拖拽列顺序事件
|
const emit = defineEmits(['pageChange', 'selectionChange', 'sortHeader']);
|
|
// 定义变量内容
|
const toolSetRef = ref();
|
const tableRef = ref();
|
const storesThemeConfig = useThemeConfig();
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
const state = reactive({
|
data: [] as Array<EmptyObjectType>,
|
loading: false,
|
exportLoading: false,
|
total: 0,
|
page: {
|
page: 1,
|
pageSize: 50,
|
field: '',
|
order: '',
|
},
|
showPagination: true,
|
selectlist: [] as EmptyObjectType[],
|
checkListAll: true,
|
checkListIndeterminate: false,
|
oldColumns: [] as EmptyObjectType[],
|
columns: [] as EmptyObjectType[],
|
haveFixed: false,
|
currentFixed: false,
|
serialNoFixed: false,
|
switchFixedContent: '取消固定列',
|
fixedIconColor: themeConfig.value.primary,
|
});
|
|
const hideTool = computed(() => {
|
return props.config.hideTool ?? false;
|
});
|
|
const getProperty = (obj: any, property: any) => {
|
const keys = property.split('.');
|
let value = obj;
|
for (const key of keys) {
|
value = value[key];
|
}
|
return value;
|
};
|
|
// 设置边框显示/隐藏
|
const setBorder = computed(() => {
|
return props.config.isBorder ? true : false;
|
});
|
// 设置斑马纹显示/隐藏
|
const setStripe = computed(() => {
|
return props.config.isStripe ? true : false;
|
});
|
// 获取父组件 配置项(必传)
|
const getConfig = computed(() => {
|
return props.config;
|
});
|
// 设置 tool header 数据
|
const setHeader = computed(() => {
|
return state.columns.filter((v) => v.isCheck);
|
});
|
// tool 列显示全选改变时
|
const onCheckAllChange = <T,>(val: T) => {
|
if (val) state.columns.forEach((v) => (v.isCheck = true));
|
else state.columns.forEach((v) => (v.isCheck = false));
|
state.checkListIndeterminate = false;
|
};
|
// tool 列显示当前项改变时
|
const onCheckChange = () => {
|
const headers = state.columns.filter((v) => v.isCheck).length;
|
state.checkListAll = headers === state.columns.length;
|
state.checkListIndeterminate = headers > 0 && headers < state.columns.length;
|
};
|
// 表格多选改变时
|
const onSelectionChange = (val: EmptyObjectType[]) => {
|
state.selectlist = val;
|
emit('selectionChange', state.selectlist);
|
};
|
// 分页改变
|
const onHandleSizeChange = (val: number) => {
|
state.page.pageSize = val;
|
onRefreshTable();
|
emit('pageChange', state.page);
|
};
|
// 改变当前页
|
const onHandleCurrentChange = (val: number) => {
|
state.page.page = val;
|
onRefreshTable();
|
emit('pageChange', state.page);
|
};
|
// 列排序
|
const sortChange = (column: any) => {
|
state.page.field = column.prop;
|
state.page.order = column.order;
|
onRefreshTable();
|
};
|
// 重置列表
|
const pageReset = () => {
|
tableRef.value.clearSelection();
|
state.page.page = 1;
|
onRefreshTable();
|
};
|
// 导出当前页
|
const onExportTable = () => {
|
if (setHeader.value.length <= 0) return ElMessage.error('没有勾选要导出的列');
|
exportData(state.data);
|
};
|
// 全部导出
|
const onExportTableAll = async () => {
|
if (setHeader.value.length <= 0) return ElMessage.error('没有勾选要导出的列');
|
state.exportLoading = true;
|
const param = Object.assign({}, props.param, { page: 1, pageSize: 9999999 });
|
const res = await props.getData(param);
|
state.exportLoading = false;
|
const data = res.result?.items ?? [];
|
exportData(data);
|
};
|
// 导出方法
|
const exportData = (data: Array<EmptyObjectType>) => {
|
if (data.length <= 0) return ElMessage.error('没有数据可以导出');
|
state.exportLoading = true;
|
let exportData = JSON.parse(JSON.stringify(data));
|
if (props.exportChangeData) {
|
exportData = props.exportChangeData(exportData);
|
}
|
exportExcel(
|
exportData,
|
`${props.config.exportFileName ? props.config.exportFileName : themeConfig.value.globalTitle}_${new Date().toLocaleString()}`,
|
setHeader.value.filter((item) => {
|
return item.type != 'action';
|
}),
|
'导出数据'
|
);
|
state.exportLoading = false;
|
};
|
// 打印
|
const onPrintTable = () => {
|
// https://printjs.crabbly.com/#documentation
|
// 自定义打印
|
let tableTh = '';
|
let tableTrTd = '';
|
let tableTd: any = {};
|
// 表头
|
setHeader.value.forEach((v: any) => {
|
if (v.prop === 'action') {
|
return;
|
}
|
tableTh += `<th class="table-th">${v.label}</th>`;
|
});
|
// 表格内容
|
state.data.forEach((val: any, key: any) => {
|
if (!tableTd[key]) tableTd[key] = [];
|
setHeader.value.forEach((v: any) => {
|
if (v.prop === 'action') {
|
return;
|
}
|
if (v.type === 'text') {
|
tableTd[key].push(`<td class="table-th table-center">${val[v.prop]}</td>`);
|
} else if (v.type === 'image') {
|
tableTd[key].push(`<td class="table-th table-center"><img src="${val[v.prop]}" style="width:${v.width}px;height:${v.height}px;"/></td>`);
|
} else {
|
tableTd[key].push(`<td class="table-th table-center">${val[v.prop]}</td>`);
|
}
|
});
|
tableTrTd += `<tr>${tableTd[key].join('')}</tr>`;
|
});
|
// 打印
|
printJs({
|
printable: `<div style=display:flex;flex-direction:column;text-align:center><h3>${props.printName}</h3></div><table border=1 cellspacing=0><tr>${tableTh}${tableTrTd}</table>`,
|
type: 'raw-html',
|
css: ['//at.alicdn.com/t/c/font_2298093_rnp72ifj3ba.css', '//unpkg.com/element-plus/dist/index.css'],
|
style: `@media print{.mb15{margin-bottom:15px;}.el-button--small i.iconfont{font-size: 12px !important;margin-right: 5px;}}; .table-th{word-break: break-all;white-space: pre-wrap;}.table-center{text-align: center;}`,
|
});
|
};
|
|
// 拖拽设置
|
const onSetTable = () => {
|
nextTick(() => {
|
const sortable = Sortable.create(toolSetRef.value, {
|
handle: '.handle',
|
dataIdAttr: 'data-key',
|
animation: 150,
|
onEnd: () => {
|
const headerList: EmptyObjectType[] = [];
|
sortable.toArray().forEach((val: any) => {
|
state.columns.forEach((v) => {
|
if (v.prop === val) headerList.push({ ...v });
|
});
|
});
|
emit('sortHeader', headerList);
|
},
|
});
|
});
|
};
|
|
const onRefreshTable = async () => {
|
state.loading = true;
|
let param = Object.assign({}, props.param, { ...state.page });
|
Object.keys(param).forEach((key) => param[key] === undefined && delete param[key]);
|
const res = await props.getData(param);
|
state.loading = false;
|
if (res && res.result && res.result.items) {
|
state.showPagination = true;
|
state.data = res.result?.items ?? [];
|
state.total = res.result?.total ?? 0;
|
} else {
|
state.showPagination = false;
|
state.data = res && res.result ? res.result : [];
|
}
|
};
|
|
const toggleSelection = (row: any, statu?: boolean) => {
|
tableRef.value!.toggleRowSelection(row, statu);
|
};
|
|
const getTableData = () => {
|
return state.data;
|
};
|
|
const setTableData = (data: Array<EmptyObjectType>, add: boolean = false) => {
|
if (add) {
|
// 追加, 去重
|
var repeat = false;
|
for (let newItem of data) {
|
repeat = false;
|
for (let item of state.data) {
|
if (newItem.id === item.id) {
|
repeat = true;
|
break;
|
}
|
}
|
if (!repeat) {
|
state.data.push(newItem);
|
}
|
}
|
} else {
|
state.data = data;
|
}
|
};
|
|
const clearFixed = () => {
|
for (let item of state.columns) delete item['fixed'];
|
};
|
|
const switchFixed = () => {
|
state.currentFixed = !state.currentFixed;
|
state.switchFixedContent = state.currentFixed ? '取消固定列' : '启用固定列';
|
if (state.currentFixed) {
|
state.fixedIconColor = themeConfig.value.primary;
|
state.columns = JSON.parse(JSON.stringify(state.oldColumns));
|
} else {
|
state.fixedIconColor = '';
|
clearFixed();
|
}
|
};
|
|
const refreshColumns = () => {
|
state.oldColumns = JSON.parse(JSON.stringify(props.columns));
|
state.columns = props.columns;
|
for (let item of state.columns) {
|
if (item.fixed !== undefined) {
|
state.haveFixed = true;
|
state.currentFixed = true;
|
if (item.fixed == 'left') {
|
state.serialNoFixed = true;
|
break;
|
}
|
}
|
}
|
};
|
|
onMounted(() => {
|
if (props.defaultSort) {
|
state.page.field = props.defaultSort.prop;
|
state.page.order = props.defaultSort.order;
|
}
|
state.page.pageSize = props.config.pageSize ?? 10;
|
refreshColumns();
|
onRefreshTable();
|
});
|
|
const handleList = onRefreshTable;
|
|
// 暴露变量
|
defineExpose({
|
pageReset,
|
handleList,
|
toggleSelection,
|
getTableData,
|
setTableData,
|
refreshColumns,
|
});
|
</script>
|
|
<style scoped lang="scss">
|
.table-container {
|
display: flex !important;
|
flex-direction: column;
|
height: 100%;
|
|
.el-table {
|
flex: 1;
|
}
|
|
.table-footer {
|
display: flex;
|
justify-content: flex-end;
|
}
|
|
.table-header {
|
display: flex;
|
|
.table-footer-tool {
|
flex: 1;
|
display: flex;
|
align-items: center;
|
justify-content: flex-end;
|
|
i {
|
margin-right: 5px !important;
|
cursor: pointer;
|
color: var(--el-text-color-regular);
|
|
&:last-of-type {
|
margin-right: 0;
|
}
|
}
|
|
.el-dropdown {
|
i {
|
margin-right: 10px;
|
color: var(--el-text-color-regular);
|
}
|
}
|
|
.tool-icon {
|
border: 1px solid #a7a7a7;
|
border-radius: 20%;
|
padding: 1px;
|
display: inline-flex;
|
align-items: center;
|
justify-content: center;
|
}
|
|
.el-icon.tool-icon {
|
font-size: 25px;
|
border: 1px solid #a7a7a7;
|
padding: 4px;
|
}
|
}
|
}
|
}
|
</style>
|