mirror of
https://github.com/chansee97/nova-admin.git
synced 2025-04-05 04:22:49 +08:00
feat: draggable list & useTableDrag hook (#38)
This commit is contained in:
parent
89d78b7ec7
commit
aee3e52f15
@ -89,6 +89,7 @@
|
||||
"route": {
|
||||
"appRoot": "Home",
|
||||
"cardList": "Card list",
|
||||
"draggableList": "Draggable list",
|
||||
"commonList": "Common list",
|
||||
"dashboard": "Dashboard",
|
||||
"demo": "Function example",
|
||||
|
@ -126,6 +126,7 @@
|
||||
"list": "列表页",
|
||||
"commonList": "常用列表",
|
||||
"cardList": "卡片列表",
|
||||
"draggableList": "拖拽列表",
|
||||
"demo": "功能示例",
|
||||
"fetch": "请求示例",
|
||||
"echarts": "Echarts示例",
|
||||
|
@ -60,6 +60,7 @@
|
||||
"quill": "^2.0.2",
|
||||
"radash": "^12.1.0",
|
||||
"vue": "^3.5.1",
|
||||
"vue-draggable-plus": "^0.5.3",
|
||||
"vue-i18n": "^9.14.0",
|
||||
"vue-router": "^4.4.3"
|
||||
},
|
||||
|
35
src/hooks/useTableDrag.ts
Normal file
35
src/hooks/useTableDrag.ts
Normal file
@ -0,0 +1,35 @@
|
||||
import type { NDataTable } from 'naive-ui'
|
||||
import { useDraggable } from 'vue-draggable-plus'
|
||||
|
||||
export function useTableDrag<T = unknown>(params: {
|
||||
tableRef: Ref<InstanceType<typeof NDataTable> | undefined>
|
||||
data: Ref<T[]>
|
||||
onRowDrag: (rows: T[]) => void
|
||||
}) {
|
||||
const tableEl = computed(() => params.tableRef?.value?.$el as HTMLElement)
|
||||
const tableBodyRef = ref<HTMLElement | undefined>(undefined)
|
||||
|
||||
const { start } = useDraggable(tableBodyRef, params.data, {
|
||||
immediate: false,
|
||||
animation: 150,
|
||||
handle: '.drag-handle',
|
||||
onEnd: (event) => {
|
||||
const { oldIndex, newIndex } = event
|
||||
const start = Math.min(oldIndex!, newIndex!)
|
||||
const end = Math.max(oldIndex!, newIndex!) - start + 1
|
||||
const changedRows = [...params.data.value].splice(start, end)
|
||||
params.onRowDrag(unref([...changedRows]))
|
||||
},
|
||||
})
|
||||
|
||||
onMounted(async () => {
|
||||
while (!tableBodyRef.value) {
|
||||
tableBodyRef.value = tableEl.value?.querySelector('tbody') || undefined
|
||||
await new Promise(resolve => setTimeout(resolve, 100))
|
||||
}
|
||||
})
|
||||
|
||||
watchOnce(() => tableBodyRef.value, (el) => {
|
||||
el && start()
|
||||
})
|
||||
}
|
@ -435,4 +435,14 @@ export const staticRoutes: AppRoute.RowRoute[] = [
|
||||
id: 43,
|
||||
pid: 13,
|
||||
},
|
||||
{
|
||||
name: 'draggableList',
|
||||
path: '/list/draggableList',
|
||||
title: '拖拽列表',
|
||||
requiresAuth: true,
|
||||
icon: 'icon-park-outline:menu-fold',
|
||||
componentPath: '/demo/list/draggableList/index.vue',
|
||||
id: 44,
|
||||
pid: 10,
|
||||
},
|
||||
]
|
||||
|
@ -14,3 +14,7 @@ body,
|
||||
.gray-mode {
|
||||
filter: grayscale(100%);
|
||||
}
|
||||
|
||||
.drag-handle {
|
||||
cursor: move;
|
||||
}
|
||||
|
170
src/views/demo/list/draggableList/index.vue
Normal file
170
src/views/demo/list/draggableList/index.vue
Normal file
@ -0,0 +1,170 @@
|
||||
<script setup lang="tsx">
|
||||
import type { DataTableColumns, FormInst, NDataTable } from 'naive-ui'
|
||||
import { Gender } from '@/constants'
|
||||
import { useBoolean } from '@/hooks'
|
||||
import { useTableDrag } from '@/hooks/useTableDrag'
|
||||
import { fetchUserPage } from '@/service'
|
||||
import { NButton, NPopconfirm, NSpace, NSwitch, NTag } from 'naive-ui'
|
||||
|
||||
const { bool: loading, setTrue: startLoading, setFalse: endLoading } = useBoolean(false)
|
||||
|
||||
const initialModel = {
|
||||
condition_1: '',
|
||||
condition_2: '',
|
||||
condition_3: '',
|
||||
condition_4: '',
|
||||
}
|
||||
const model = ref({ ...initialModel })
|
||||
|
||||
const formRef = ref<FormInst | null>()
|
||||
function sendMail(id?: number) {
|
||||
window.$message.success(`删除用户id:${id}`)
|
||||
}
|
||||
const columns: DataTableColumns<Entity.User> = [
|
||||
{
|
||||
title: '姓名',
|
||||
align: 'center',
|
||||
key: 'userName',
|
||||
},
|
||||
{
|
||||
title: '年龄',
|
||||
align: 'center',
|
||||
key: 'age',
|
||||
},
|
||||
{
|
||||
title: '性别',
|
||||
align: 'center',
|
||||
key: 'gender',
|
||||
render: (row) => {
|
||||
const tagType = {
|
||||
0: 'primary',
|
||||
1: 'success',
|
||||
} as const
|
||||
if (row.gender) {
|
||||
return (
|
||||
<NTag type={tagType[row.gender]}>
|
||||
{Gender[row.gender]}
|
||||
</NTag>
|
||||
)
|
||||
}
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '邮箱',
|
||||
align: 'center',
|
||||
key: 'email',
|
||||
},
|
||||
{
|
||||
title: '状态',
|
||||
align: 'center',
|
||||
key: 'status',
|
||||
render: (row) => {
|
||||
return (
|
||||
<NSwitch
|
||||
value={row.status}
|
||||
checked-value={1}
|
||||
unchecked-value={0}
|
||||
onUpdateValue={(value: 0 | 1) =>
|
||||
handleUpdateDisabled(value, row.id!)}
|
||||
>
|
||||
{{ checked: () => '启用', unchecked: () => '禁用' }}
|
||||
</NSwitch>
|
||||
)
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '操作',
|
||||
align: 'center',
|
||||
key: 'actions',
|
||||
render: (row) => {
|
||||
return (
|
||||
<NSpace justify="center">
|
||||
<NPopconfirm onPositiveClick={() => sendMail(row.id)}>
|
||||
{{
|
||||
default: () => '确认删除',
|
||||
trigger: () => <NButton size="small">删除</NButton>,
|
||||
}}
|
||||
</NPopconfirm>
|
||||
</NSpace>
|
||||
)
|
||||
},
|
||||
},
|
||||
]
|
||||
|
||||
const listData = ref<Entity.User[]>([])
|
||||
function handleUpdateDisabled(value: 0 | 1, id: number) {
|
||||
const index = listData.value.findIndex(item => item.id === id)
|
||||
if (index > -1)
|
||||
listData.value[index].status = value
|
||||
}
|
||||
|
||||
const tableRef = ref<InstanceType<typeof NDataTable>>()
|
||||
useTableDrag({
|
||||
tableRef,
|
||||
data: listData,
|
||||
onRowDrag(data) {
|
||||
const target = data[data.length - 1]
|
||||
window.$message.success(`拖拽数据 id: ${target.id} name: ${target.userName}`)
|
||||
},
|
||||
})
|
||||
|
||||
onMounted(() => {
|
||||
getUserList()
|
||||
})
|
||||
async function getUserList() {
|
||||
startLoading()
|
||||
await fetchUserPage().then((res: any) => {
|
||||
listData.value = res.data.list
|
||||
endLoading()
|
||||
})
|
||||
}
|
||||
function changePage(page: number, size: number) {
|
||||
window.$message.success(`分页器:${page},${size}`)
|
||||
}
|
||||
function handleResetSearch() {
|
||||
model.value = { ...initialModel }
|
||||
}
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<NSpace vertical size="large">
|
||||
<n-card>
|
||||
<n-form ref="formRef" :model="model" label-placement="left" inline :show-feedback="false">
|
||||
<n-flex>
|
||||
<n-form-item label="姓名" path="condition_1">
|
||||
<n-input v-model:value="model.condition_1" placeholder="请输入" />
|
||||
</n-form-item>
|
||||
<n-form-item label="年龄" path="condition_2">
|
||||
<n-input v-model:value="model.condition_2" placeholder="请输入" />
|
||||
</n-form-item>
|
||||
<n-form-item label="性别" path="condition_3">
|
||||
<n-input v-model:value="model.condition_3" placeholder="请输入" />
|
||||
</n-form-item>
|
||||
<n-form-item label="地址" path="condition_4">
|
||||
<n-input v-model:value="model.condition_4" placeholder="请输入" />
|
||||
</n-form-item>
|
||||
<n-flex class="ml-auto">
|
||||
<NButton type="primary" @click="getUserList">
|
||||
<template #icon>
|
||||
<icon-park-outline-search />
|
||||
</template>
|
||||
搜索
|
||||
</NButton>
|
||||
<NButton strong secondary @click="handleResetSearch">
|
||||
<template #icon>
|
||||
<icon-park-outline-redo />
|
||||
</template>
|
||||
重置
|
||||
</NButton>
|
||||
</n-flex>
|
||||
</n-flex>
|
||||
</n-form>
|
||||
</n-card>
|
||||
<n-card>
|
||||
<NSpace vertical size="large">
|
||||
<n-data-table ref="tableRef" row-class-name="drag-handle" :columns="columns" :data="listData" :loading="loading" />
|
||||
<Pagination :count="100" @change="changePage" />
|
||||
</NSpace>
|
||||
</n-card>
|
||||
</NSpace>
|
||||
</template>
|
Loading…
x
Reference in New Issue
Block a user