一、组件封装
<template>
<div class="headerTable">
<div>
<slot name="leftTopButton"/>
</div>
<div class="tableSortable">
<el-table
ref="table"
v-loading="loading"
element-loading-text="Loading"
:data="tableData"
:border="border"
tooltip-effect="dark"
style="width:100%"
header-row-class-name="el-table_color"
:fit="fit"
:stripe="stripe"
:height="height"
:size="small"
@row-click="rowChangeHandler"
@selection-change="selectionChangeHandler"
>
<el-table-column
type="selection"
width="55"
/>
<el-table-column
v-for="(item, index) in tableLabel"
:key="index"
:width="item.width ? item.width : ''"
:align="item.align"
:label="item.label"
:prop="item.param"
:sortable="item.sortable ? 'custom' : false"
>
<template slot-scope="scope">
<i
v-if="index === 0"
class="el-icon-s-operation"
/>
<span v-if="item.render">
{{ item.render(scope.row) }}
</span>
<span v-else>{{ scope.row[item.param] }}</span>
</template>
</el-table-column>
<el-table-column
v-if="tableOption.label"
:width="tableOption.width"
:label="tableOption.label"
>
<template slot-scope="scope">
<el-button
v-for="(item, index) in tableOption.options"
:key="index"
:type="item.type"
:icon="item.icon"
size="mini"
@click="handleButton(item.action, scope)"
>
{{ item.label }}
</el-button>
</template>
</el-table-column>
</el-table>
</div>
<div>
<slot name="pagination"/>
</div>
</div>
</template>
<script>
// import {parseTime} from "../utils";
import Sortable from 'sortablejs';
export default {
name: 'TableCmp',
props: {
// 表格数据
tableData: {
type: Array,
default: () => {
return [];
}
},
// 表格表头标题(相关的属性)
tableLabel: {
type: Array,
default: () => {
return [];
}
},
// 操作按钮
tableOption: {
type: Object,
default: () => {
return {};
}
},
loading: {
type: Boolean
},
border: {
type: Boolean,
default: true
},
fit: {
type: Boolean,
default: true
},
stripe: {
type: Boolean,
default: true
},
height: {
type: String,
default: 'auto'
},
small: {
type: String,
default: 'mini'
}
},
data() {
return {
// loadingLog: true,
showBox: false
};
},
computed: {
loadingLog() {
return this.loading;
}
},
watch: {
loadingLog() {
if (this.tableData) {
setTimeout(() => {
this.loadingLog = false;
}, 1500);
}
}
},
mounted() {
this.onDragEnd();
},
methods: {
handleButton(action, scope) {
this.$emit('handleFun', action, scope);
},
rowChangeHandler(row) {
this.$emit('handleRowClick', row);
},
selectionChangeHandler(selected) {
this.$emit('selectionChange', selected);
},
onDragEnd() {
debugger;
const that = this;
// eslint-disable-next-line no-undef
const tbody = document.querySelector('.tableSortable .el-table__body-wrapper tbody');
Sortable.create(tbody, {
draggable: '.tableSortable .el-table__row',
onEnd({newIndex, oldIndex}) {
const currRow = that.tableData.splice(oldIndex, 1)[0];
that.$nextTick(() => {
that.tableData.splice(newIndex, 0, currRow);
});
}
});
}
}
};
</script>
<style scoped>
.headerTable {
margin: 20px;
}
</style>
二、组件使用
<template>
<div>
<vuedraggable
:table-data="tableData"
:table-label="tableLabel"
:table-option="tableOption"
:loading="loading"
:border="true"
@handleFun="handleFun"
@handleRowClick="handleRowClick"
@selectionChange="selectionChange"
>
<template v-slot:leftTopButton>
<el-button>Add Row</el-button>
</template>
</vuedraggable>
</div>
</template>
<script>
import vuedraggable from '@/components/draggable/vuedraggable.vue';
export default {
components: {
vuedraggable
},
data() {
// Generate mock table data
const generateMockData = () => {
const data = [];
for (let i = 0; i < 10; i++) {
data.push({
id: i + 1,
name: `User ${i + 1}`,
age: Math.floor(Math.random() * 50) + 20,
role: i % 2 === 0 ? 'Admin' : 'User'
});
}
return data;
};
// Generate mock table labels
const tableLabel = [
{label: 'ID', param: 'id', align: 'center', sortable: true},
{label: 'Name', param: 'name', align: 'center', sortable: false},
{label: 'Age', param: 'age', align: 'center', sortable: true},
{label: 'Role', param: 'role', align: 'center', sortable: false}
];
// Define mock table options
const tableOption = {
label: 'Actions',
width: 300,
options: [
{label: 'Edit', type: 'primary', icon: 'el-icon-edit', action: 'edit'},
{label: 'Delete', type: 'danger', icon: 'el-icon-delete', action: 'delete'}
]
};
return {
tableData: generateMockData(),
tableLabel,
tableOption,
loading: false
};
},
methods: {
handleFun(action, scope) {
switch (action) {
case 'edit':
console.log('Edit row:', scope.row);
break;
case 'delete':
console.log('Delete row:', scope.row);
break;
default:
break;
}
},
handleRowClick(row) {
console.log('Clicked row:', row);
},
selectionChange(selected) {
console.log('Selected rows:', selected);
}
}
};
</script>
视频预览