Files
wms-ui/src/views/system/user/index.vue

273 lines
8.3 KiB
Vue
Raw Normal View History

2026-02-26 17:31:18 +08:00
<template>
2026-03-03 11:37:26 +08:00
<div class="gi_table_page">
<GiTable
title="用户管理"
row-key="id"
:data="dataList"
:columns="columns"
:loading="loading"
:scroll="{ x: '100%', y: '100%', minWidth: 1500 }"
:pagination="pagination"
:disabled-tools="['size']"
:disabled-column-keys="['nickname']"
@refresh="search"
>
<template #top>
<GiForm
v-model="queryForm"
search :columns="queryFormColumns"
size="medium"
@search="search"
@reset="reset"
></GiForm>
2026-02-26 17:31:18 +08:00
</template>
2026-03-03 11:37:26 +08:00
<template #toolbar-left>
<a-button v-permission="['system:user:add']" type="primary" style="margin-right: 8px;" @click="onAdd">
<template #icon><icon-plus /></template>
<template #default>新增</template>
</a-button>
<a-button v-permission="['system:user:import']" style="margin-right: 8px;" @click="onImport">
<template #icon><icon-upload /></template>
<template #default>导入</template>
</a-button>
<a-button v-permission="['system:user:export']" @click="onExport">
<template #icon><icon-download /></template>
<template #default>导出</template>
</a-button>
</template>
<template #toolbar-right>
</template>
<!-- 表格插槽内容 -->
<template #nickname="{ record }">
<GiCellAvatar :avatar="record.avatar" :name="record.nickname" />
</template>
<template #gender="{ record }">
<GiCellGender :gender="record.gender" />
</template>
<template #roleNames="{ record }">
<GiCellTags :data="record.roleNames" />
</template>
<template #status="{ record }">
<GiCellStatus :status="record.status" />
</template>
<template #isSystem="{ record }">
<a-tag v-if="record.isSystem" color="red" size="small"></a-tag>
<a-tag v-else color="arcoblue" size="small"></a-tag>
</template>
<template #action="{ record }">
<a-space>
<a-link v-permission="['system:user:update']" title="修改" @click="onUpdate(record)">修改</a-link>
<a-link
v-permission="['system:user:delete']"
status="danger"
:disabled="record.isSystem"
:title="record.isSystem ? '系统内置数据不能删除' : '删除'"
@click="onDelete(record)"
>
删除
</a-link>
<a-dropdown>
<a-button v-if="has.hasPermOr(['system:user:resetPwd', 'system:user:updateRole'])" type="text" size="mini" title="更多">
<template #icon>
<icon-more :size="16" />
</template>
2026-02-26 17:31:18 +08:00
</a-button>
2026-03-03 11:37:26 +08:00
<template #content>
<a-doption v-permission="['system:user:resetPwd']" title="重置密码" @click="onResetPwd(record)">重置密码</a-doption>
<a-doption v-permission="['system:user:updateRole']" title="分配角色" @click="onUpdateRole(record)">分配角色</a-doption>
</template>
</a-dropdown>
</a-space>
2026-02-26 17:31:18 +08:00
</template>
2026-03-03 11:37:26 +08:00
</GiTable>
2026-02-26 17:31:18 +08:00
2026-03-03 11:37:26 +08:00
<!-- 弹窗/抽屉组件 -->
2026-02-26 17:31:18 +08:00
<UserAddDrawer ref="UserAddDrawerRef" @save-success="search" />
<UserImportDrawer ref="UserImportDrawerRef" @save-success="search" />
<UserResetPwdModal ref="UserResetPwdModalRef" />
<UserUpdateRoleModal ref="UserUpdateRoleModalRef" @save-success="search" />
</div>
</template>
<script setup lang="ts">
2026-03-03 11:37:26 +08:00
import { onMounted } from 'vue'
2026-02-26 17:31:18 +08:00
import UserAddDrawer from './UserAddDrawer.vue'
import UserImportDrawer from './UserImportDrawer.vue'
import UserResetPwdModal from './UserResetPwdModal.vue'
import UserUpdateRoleModal from './UserUpdateRoleModal.vue'
import { type UserResp, deleteUser, exportUser, listUser } from '@/apis/system/user'
import type { TableInstanceColumns } from '@/components/GiTable/type'
import { DisEnableStatusList } from '@/constant/common'
import { useDownload, useResetReactive, useTable } from '@/hooks'
import { isMobile } from '@/utils'
import has from '@/utils/has'
2026-03-04 16:46:52 +08:00
import type {ColumnItem} from "@/components/GiForm";
2026-02-26 17:31:18 +08:00
defineOptions({ name: 'SystemUser' })
2026-03-03 11:37:26 +08:00
// 查询表单
2026-02-26 17:31:18 +08:00
const [queryForm, resetForm] = useResetReactive({
2026-03-03 11:37:26 +08:00
description: undefined,
status: undefined,
createTime: undefined,
2026-02-26 17:31:18 +08:00
sort: ['t1.id,desc'],
})
2026-03-03 11:37:26 +08:00
2026-02-26 17:31:18 +08:00
const queryFormColumns: ColumnItem[] = reactive([
{
type: 'input',
field: 'description',
2026-03-03 11:37:26 +08:00
span: { xs: 24, sm: 4, xxl: 3 },
2026-02-26 17:31:18 +08:00
formItemProps: {
hideLabel: true,
},
props: {
2026-03-03 15:05:20 +08:00
placeholder: '搜索用户名',
2026-02-26 17:31:18 +08:00
showWordLimit: false,
},
},
{
type: 'select',
field: 'status',
2026-03-03 11:37:26 +08:00
span: { xs: 24, sm: 4, xxl: 4 },
2026-02-26 17:31:18 +08:00
formItemProps: {
hideLabel: true,
},
props: {
options: DisEnableStatusList,
placeholder: '请选择状态',
},
},
{
type: 'range-picker',
field: 'createTime',
2026-03-03 11:37:26 +08:00
span: { xs: 24, sm: 8, xxl: 5 },
2026-02-26 17:31:18 +08:00
formItemProps: {
hideLabel: true,
},
},
])
2026-03-03 11:37:26 +08:00
// 表格数据
2026-02-26 17:31:18 +08:00
const {
tableData: dataList,
loading,
pagination,
search,
handleDelete,
} = useTable((page) => listUser({ ...queryForm, ...page }), { immediate: false })
2026-03-03 11:37:26 +08:00
// 表格列配置
2026-02-26 17:31:18 +08:00
const columns: TableInstanceColumns[] = [
{
title: '序号',
width: 66,
align: 'center',
render: ({ rowIndex }) => h('span', {}, rowIndex + 1 + (pagination.current - 1) * pagination.pageSize),
fixed: !isMobile() ? 'left' : undefined,
},
2026-03-03 11:37:26 +08:00
{ title: '用户名', dataIndex: 'username', slotName: 'username', minWidth: 100, ellipsis: true, tooltip: true },
{ title:'卡号', dataIndex: 'cardNo', slotName: 'cardNo' },
2026-02-26 17:31:18 +08:00
{ title: '状态', dataIndex: 'status', slotName: 'status', align: 'center' },
{ title: '性别', dataIndex: 'gender', slotName: 'gender', align: 'center' },
{ title: '角色', dataIndex: 'roleNames', slotName: 'roleNames', minWidth: 165 },
{ title: '手机号', dataIndex: 'phone', minWidth: 170, ellipsis: true, tooltip: true },
{ title: '系统内置', dataIndex: 'isSystem', slotName: 'isSystem', width: 100, align: 'center', show: false },
{ title: '描述', dataIndex: 'description', minWidth: 130, ellipsis: true, tooltip: true },
{ title: '创建人', dataIndex: 'createUserString', width: 140, ellipsis: true, tooltip: true, show: false },
{ title: '创建时间', dataIndex: 'createTime', width: 180 },
{ title: '修改人', dataIndex: 'updateUserString', width: 140, ellipsis: true, tooltip: true, show: false },
{ title: '修改时间', dataIndex: 'updateTime', width: 180, show: false },
{
title: '操作',
dataIndex: 'action',
slotName: 'action',
width: 190,
align: 'center',
fixed: !isMobile() ? 'right' : undefined,
show: has.hasPermOr([
'system:user:detail',
'system:user:update',
'system:user:delete',
'system:user:resetPwd',
'system:user:updateRole',
]),
},
]
2026-03-03 11:37:26 +08:00
// 方法定义
2026-02-26 17:31:18 +08:00
const reset = () => {
resetForm()
search()
}
const onDelete = (record: UserResp) => {
return handleDelete(() => deleteUser(record.id), {
content: `是否确定删除用户「${record.nickname}(${record.username})」?`,
showModal: true,
})
}
const onExport = () => {
useDownload(() => exportUser(queryForm))
}
const UserImportDrawerRef = ref<InstanceType<typeof UserImportDrawer>>()
const onImport = () => {
UserImportDrawerRef.value?.onOpen()
}
const UserAddDrawerRef = ref<InstanceType<typeof UserAddDrawer>>()
const onAdd = () => {
UserAddDrawerRef.value?.onAdd()
}
const onUpdate = (record: UserResp) => {
UserAddDrawerRef.value?.onUpdate(record.id)
}
const UserResetPwdModalRef = ref<InstanceType<typeof UserResetPwdModal>>()
const onResetPwd = (record: UserResp) => {
UserResetPwdModalRef.value?.onOpen(record.id)
}
const UserUpdateRoleModalRef = ref<InstanceType<typeof UserUpdateRoleModal>>()
const onUpdateRole = (record: UserResp) => {
UserUpdateRoleModalRef.value?.onOpen(record.id)
}
2026-03-03 11:37:26 +08:00
onMounted(() => {
search()
})
2026-02-26 17:31:18 +08:00
</script>
<style scoped lang="scss">
2026-03-03 11:37:26 +08:00
.gi_table_page {
padding: 16px;
height: 100%;
box-sizing: border-box;
}
:deep(.gi-table) {
background: #ffffff;
}
:deep(.arco-table-toolbar-left) {
display: flex;
align-items: center;
flex-wrap: wrap; // 允许在小屏幕换行
gap: 8px; // 统一间距
.arco-input-search,
.arco-select,
.arco-range-picker {
}
2026-02-26 17:31:18 +08:00
}
2026-03-03 11:37:26 +08:00
:deep(.arco-table-toolbar-right) {
display: flex;
align-items: center;
gap: 8px;
2026-02-26 17:31:18 +08:00
}
</style>