feat: 表格排序

This commit is contained in:
April 2023-06-14 23:37:06 +08:00
parent 3e4a21a9e7
commit 07383021b7
3 changed files with 106 additions and 16 deletions

View File

@ -41,7 +41,6 @@
] ]
}, },
"dependencies": { "dependencies": {
"@types/lodash": "^4.14.195",
"@vitejs/plugin-vue": "^4.2.3", "@vitejs/plugin-vue": "^4.2.3",
"@vueuse/core": "^10.1.2", "@vueuse/core": "^10.1.2",
"@wangeditor/editor": "^5.1.23", "@wangeditor/editor": "^5.1.23",
@ -65,6 +64,7 @@
"@commitlint/config-conventional": "^17.6.3", "@commitlint/config-conventional": "^17.6.3",
"@iconify-json/ep": "^1.1.10", "@iconify-json/ep": "^1.1.10",
"@types/codemirror": "^5.60.7", "@types/codemirror": "^5.60.7",
"@types/lodash": "^4.14.195",
"@types/nprogress": "^0.2.0", "@types/nprogress": "^0.2.0",
"@types/path-browserify": "^1.0.0", "@types/path-browserify": "^1.0.0",
"@typescript-eslint/eslint-plugin": "^5.59.6", "@typescript-eslint/eslint-plugin": "^5.59.6",

6
src/types/env.d.ts vendored
View File

@ -1,7 +1,7 @@
/// <reference types="vite/client" /> /// <reference types="vite/client" />
declare module '*.vue' { declare module "*.vue" {
import { DefineComponent } from 'vue'; import { DefineComponent } from "vue";
// eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/ban-types // eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/ban-types
const component: DefineComponent<{}, {}, any>; const component: DefineComponent<{}, {}, any>;
export default component; export default component;
@ -17,3 +17,5 @@ interface ImportMetaEnv {
interface ImportMeta { interface ImportMeta {
readonly env: ImportMetaEnv; readonly env: ImportMetaEnv;
} }
declare module "lodash-es";

View File

@ -12,7 +12,14 @@
> >
<template #header> <template #header>
<div class="table-head"> <div class="table-head">
<div>{{ oneCol.label }}</div> <div class="label-wrap" @click="handleClickHead(oneCol)">
<div>{{ oneCol.label }}</div>
<template v-if="oneCol.order">
<i-ep-sort-up v-if="oneCol.order === 'asc'"></i-ep-sort-up>
<i-ep-sort-down v-else></i-ep-sort-down>
</template>
</div>
<el-dropdown <el-dropdown
trigger="click" trigger="click"
@command="(command: string) => handleCommand(command, oneCol)" @command="(command: string) => handleCommand(command, oneCol)"
@ -20,10 +27,16 @@
<i-ep-arrow-down class="action-more"></i-ep-arrow-down> <i-ep-arrow-down class="action-more"></i-ep-arrow-down>
<template #dropdown> <template #dropdown>
<el-dropdown-menu> <el-dropdown-menu>
<!-- <el-dropdown-item>升序</el-dropdown-item> <template v-if="oneCol.prop === 'id'">
<el-dropdown-item>降序</el-dropdown-item> --> <el-dropdown-item :command="TableCommand.ASC">
升序
</el-dropdown-item>
<el-dropdown-item :command="TableCommand.DESC">
降序
</el-dropdown-item>
</template>
<el-dropdown-item :command="TableCommand.FIXED"> <el-dropdown-item :command="TableCommand.FIXED">
冻结 {{ oneCol.fixed ? "解冻" : "冻结" }}
</el-dropdown-item> </el-dropdown-item>
</el-dropdown-menu> </el-dropdown-menu>
</template> </template>
@ -36,7 +49,7 @@
</div> </div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { TableColumn } from "element-plus/es/components/table/src/table-column/defaults"; import { cloneDeep } from "lodash-es";
enum TableCommand { enum TableCommand {
// //
@ -46,7 +59,9 @@ enum TableCommand {
// //
DESC = "desc", DESC = "desc",
} }
const tableData = [
let tableData = ref<any[]>([]);
const orginData = [
{ {
date: "2016-05-03", date: "2016-05-03",
name: "admin", name: "admin",
@ -61,7 +76,7 @@ const tableData = [
date: "2016-05-02", date: "2016-05-02",
name: "admin", name: "admin",
address: "江西", address: "江西",
id: 2, id: 3,
gender: "男", gender: "男",
height: 167, height: 167,
school: "清华大学", school: "清华大学",
@ -71,7 +86,7 @@ const tableData = [
date: "2016-05-04", date: "2016-05-04",
name: "admin", name: "admin",
address: "湖南", address: "湖南",
id: 3, id: 4,
gender: "男", gender: "男",
height: 173, height: 173,
school: "清华大学", school: "清华大学",
@ -81,18 +96,20 @@ const tableData = [
date: "2016-05-01", date: "2016-05-01",
name: "admin", name: "admin",
address: "重庆", address: "重庆",
id: 4, id: 2,
gender: "女", gender: "女",
height: 178, height: 178,
school: "清华大学", school: "清华大学",
hobby: "游泳", hobby: "游泳",
}, },
]; ];
onMounted(() => {
tableData.value = orginData.map((item) => item);
});
interface ColumnProp { interface ColumnProp {
label: string; label: string;
prop: string; prop: string;
order: null | boolean; order: null | string;
fixed: null | string; fixed: null | string;
} }
const tableColList = ref<ColumnProp[]>([ const tableColList = ref<ColumnProp[]>([
@ -149,10 +166,65 @@ const handleCommand = (command: string, tableCol: ColumnProp) => {
if (command === TableCommand.FIXED) { if (command === TableCommand.FIXED) {
tableColList.value.forEach((item) => { tableColList.value.forEach((item) => {
if (item.prop === tableCol.prop) { if (item.prop === tableCol.prop) {
item.fixed = "left"; // /
item.fixed = item.fixed ? null : "left";
} }
}); });
} }
//
if (command === TableCommand.ASC) {
tableColList.value.forEach((item) => {
if (item.prop === tableCol.prop) {
item.order = "asc";
sortTableData(item);
}
});
}
//
if (command === TableCommand.DESC) {
tableColList.value.forEach((item) => {
if (item.prop === tableCol.prop) {
item.order = "desc";
sortTableData(item);
}
});
}
};
//
const handleClickHead = (oneCol: ColumnProp) => {
//
const sortArr = [null, "asc", "desc"];
const { order } = oneCol;
const findIndex = sortArr.findIndex((el) => el === order);
let currentIndex = findIndex + 1;
if (currentIndex > 2) {
currentIndex = 0;
}
//
tableColList.value.forEach((item) => {
if (item.prop === oneCol.prop) {
item.order = sortArr[currentIndex];
} else {
item.order = null;
}
});
sortTableData(oneCol);
};
const sortTableData = (sortCol: ColumnProp) => {
//
if (sortCol.order === TableCommand.ASC) {
const compare = (p: string) => (m: any, n: any) => m[p] - n[p];
tableData.value = cloneDeep(orginData).sort(compare(sortCol.prop));
}
//
if (sortCol.order === TableCommand.DESC) {
const compare = (p: string) => (m: any, n: any) => n[p] - m[p];
tableData.value = cloneDeep(orginData).sort(compare(sortCol.prop));
}
//
if (!sortCol.order) {
tableData.value = cloneDeep(orginData).map((item: any) => item);
}
}; };
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@ -160,18 +232,34 @@ const handleCommand = (command: string, tableCol: ColumnProp) => {
width: 100%; width: 100%;
height: calc(100vh - 84px); height: calc(100vh - 84px);
padding: 10px; padding: 10px;
.table-wrap { .table-wrap {
background-color: #fff;
height: 100%; height: 100%;
padding: 20px; padding: 20px;
background-color: #fff;
.el-table { .el-table {
width: 800px; width: 800px;
.table-head { .table-head {
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: space-between; justify-content: space-between;
.label-wrap {
display: flex;
flex: 1;
align-items: center;
cursor: pointer;
svg {
margin-left: 5px;
}
}
.action-more { .action-more {
cursor: pointer; cursor: pointer;
&:hover { &:hover {
color: #409eff; color: #409eff;
} }