用户管理

This commit is contained in:
仲么了
2024-01-22 14:21:31 +08:00
parent 08e0e9e013
commit 0bf750e5ae
13 changed files with 432 additions and 451 deletions

View File

@@ -0,0 +1,221 @@
<template>
<div class='default-main'>
<TableHeader>
<template v-slot:select>
<el-form-item label='用户状态'>
<el-select v-model='tableStore.table.params.searchState' placeholder='选择用户状态'>
<el-option
v-for='(item, index) in userState'
:label='item.label'
:key='index'
:value='item.value'
></el-option>
</el-select>
</el-form-item>
<el-form-item label='用户类型:'>
<el-select v-model='tableStore.table.params.casualUser' placeholder='选择用户类型'>
<el-option
v-for='(item, index) in casualUser'
:label='item.label'
:key='index'
:value='item.value'
></el-option>
</el-select>
</el-form-item>
<el-form-item label='关键词:'>
<el-input
style='width: 240px'
v-model='tableStore.table.params.searchValue'
clearable
placeholder='仅根据用户名/登录名'
/>
</el-form-item>
</template>
<template v-slot:operation>
<el-button :icon='Plus' type='primary' @click='addUser'>添加</el-button>
</template>
</TableHeader>
<Table ref='tableRef' />
<PopupEdit ref='popupEditRef'></PopupEdit>
<PopupPwd ref='popupPwdRef'></PopupPwd>
</div>
</template>
<script setup lang='ts'>
import { Plus } from '@element-plus/icons-vue'
import { ref, onMounted, provide } from 'vue'
import { ElMessageBox, ElMessage } from 'element-plus'
import { activateUser, deluser, passwordConfirm } from '@/api/user-boot/user'
import TableStore from '@/utils/tableStore'
import Table from '@/components/table/index.vue'
import TableHeader from '@/components/table/header/index.vue'
import PopupEdit from './popupEdit.vue'
import PopupPwd from './popupPwd.vue'
defineOptions({
name: 'auth/userlist'
})
const popupEditRef = ref()
const popupPwdRef = ref()
const tableStore = new TableStore({
url: '/user-boot/user/list',
method: 'POST',
column: [
{ title: '用户名称', field: 'name', minWidth: '130' },
{ title: '登录名', field: 'loginName', minWidth: '130' },
{ title: '角色', field: 'roleName', minWidth: '130' },
{ title: '部门', field: 'deptName', minWidth: '200' },
{ title: '电话', field: 'phoneShow', minWidth: '100' },
{ title: '注册时间', field: 'registerTime', minWidth: '130' },
{ title: '登录时间', field: 'loginTime', minWidth: '130' },
{ title: '类型', field: 'casualUserName', minWidth: '80' },
{ title: '状态', field: 'stateName', minWidth: '80' },
{
title: '操作',
width: '180',
render: 'buttons',
fixed: 'right',
buttons: [
{
name: 'edit',
title: '编辑',
type: 'primary',
icon: 'el-icon-EditPen',
render: 'tipButton',
disabled: row => {
return row.state !== 1
},
click: row => {
popupEditRef.value.open('编辑用户', row)
}
},
{
name: 'edit',
title: '修改密码',
type: 'primary',
icon: 'el-icon-Lock',
render: 'tipButton',
disabled: row => {
return row.state !== 1
},
click: row => {
popupPwdRef.value.open(row.id)
}
},
{
name: 'edit',
title: '激活',
type: 'success',
icon: 'el-icon-Open',
render: 'tipButton',
disabled: row => {
return row.state !== 2 && row.state !== 5 && row.state !== 0 && row.state !== 4
},
click: row => {
activateUser({
id: row.id
}).then(() => {
ElMessage.success('激活成功')
tableStore.index()
})
}
},
{
name: 'edit',
title: '注销',
type: 'danger',
icon: 'el-icon-SwitchButton',
render: 'tipButton',
disabled: row => {
return row.state !== 1 && row.state !== 3
},
click: row => {
ElMessageBox.prompt('二次校验密码确认', '注销用户', {
confirmButtonText: '确认',
cancelButtonText: '取消',
inputType: 'password'
})
.then(({ value }) => {
passwordConfirm(value).then(res => {
deluser({
id: row.id
}).then(() => {
ElMessage.success('注销成功')
tableStore.index()
})
})
})
}
}
]
}
],
loadCallback: () => {
tableStore.table.data.forEach((item: any) => {
item.deptName = item.deptName || '/'
item.phoneShow = item.phone || '/'
item.roleName = item.role.length ? item.role : '/'
switch (item.casualUser) {
case 0:
item.casualUserName = '临时用户'
break
case 1:
item.casualUserName = '长期用户'
break
default:
item.casualUserName = '/'
break
}
switch (item.state) {
case 0:
item.stateName = '注销'
break
case 1:
item.stateName = '正常'
break
case 2:
item.stateName = '锁定'
break
case 3:
item.stateName = '待审核'
break
case 4:
item.stateName = '休眠'
break
case 5:
item.stateName = '密码过期'
break
default:
item.stateName = '/'
break
}
})
}
})
provide('tableStore', tableStore)
tableStore.table.params.searchState = 1
tableStore.table.params.casualUser = -1
tableStore.table.params.orderBy = ''
const userState = [
{ label: '全部', value: -1 },
{ label: '注销', value: 0 },
{ label: '正常', value: 1 },
{ label: '锁定', value: 2 },
{ label: '待审核', value: 3 },
{ label: '休眠', value: 4 },
{ label: '密码过期', value: 5 }
]
const casualUser = [
{ label: '全部', value: -1 },
{ label: '临时用户', value: 0 },
{ label: '长期用户', value: 1 }
]
onMounted(() => {
tableStore.index()
})
const addUser = () => {
popupEditRef.value.open('新增用户')
}
</script>

View File

@@ -0,0 +1,270 @@
<template>
<el-dialog class='cn-operate-dialog' v-model='dialogVisible' :title='title'>
<el-scrollbar>
<el-form :inline='true' :model='form' label-width='120px' :rules='rules'>
<el-form-item label='用户名' prop='name'>
<el-input v-model='form.name' placeholder='请输入昵称' />
</el-form-item>
<el-form-item label='登录名' prop='loginName'>
<el-input v-model='form.loginName' placeholder='请输入登录名' />
</el-form-item>
<el-form-item label='默认密码' prop='password' v-if="title === '新增用户'">
<el-input v-model='form.password' placeholder='请输入密码' disabled />
</el-form-item>
<el-form-item label='权限类型' prop='type'>
<el-select
style='width: 100%'
v-model='form.type'
@change='changeValue'
disabled
placeholder='请选择权限类型'
>
<el-option
v-for='(item, index) in UserTypeOption'
:label='item.label'
:value='item.value'
:key='index'
></el-option>
</el-select>
</el-form-item>
<el-form-item label='用户类型' prop='casualUser'>
<el-select style='width: 100%' v-model='form.casualUser' placeholder='请选择权限类型'>
<el-option
v-for='(item, index) in TypeOptions'
:label='item.label'
:value='item.value'
:key='index'
></el-option>
</el-select>
</el-form-item>
<el-form-item label='所属部门' prop='deptId'>
<Area style='width: 100%' v-model='form.deptId' />
</el-form-item>
<el-form-item label='角色' prop='role'>
<el-select style='width: 100%' v-model='form.role' placeholder='请选择角色' multiple collapse-tags>
<el-option
v-for='(item, index) in roleOptions'
:label='item.label'
:value='item.value'
:key='index'
></el-option>
</el-select>
</el-form-item>
<el-form-item label='手机号' prop='phone'>
<el-input v-model='form.phone' placeholder='请输入手机号' />
</el-form-item>
<el-form-item label='邮箱' prop='email'>
<el-input v-model='form.email' placeholder='请输入描述' />
</el-form-item>
<el-form-item label='起始IP' prop='limitIpStart'>
<el-input v-model='form.limitIpStart' placeholder='请输入描述' />
</el-form-item>
<el-form-item label='结束IP' prop='limitIpEnd'>
<el-input v-model='form.limitIpEnd' placeholder='请输入描述' />
</el-form-item>
<el-form-item label='时间段' prop='limitTime'>
<el-slider v-model='form.limitTime' range show-stops :max='24' />
</el-form-item>
<el-form-item label='短信通知' prop='smsNotice'>
<el-radio-group v-model='form.smsNotice'>
<el-radio-button :label='0'></el-radio-button>
<el-radio-button :label='1'></el-radio-button>
</el-radio-group>
</el-form-item>
<el-form-item label='邮件通知' prop='emailNotice'>
<el-radio-group v-model='form.emailNotice'>
<el-radio-button :label='0'></el-radio-button>
<el-radio-button :label='1'></el-radio-button>
</el-radio-group>
</el-form-item>
<el-form-item label='用户ID'>
<div style='display: flex; width: 100%'>
<el-radio-group v-model='useId'>
<el-radio-button :label='1'></el-radio-button>
<el-radio-button :label='0'></el-radio-button>
</el-radio-group>
<el-input
:disabled="title !== '新增用户'"
v-model='form.id'
placeholder='请输入用户id'
v-if='useId'
style='flex: 1'
class='ml10'
></el-input>
</div>
</el-form-item>
</el-form>
</el-scrollbar>
<template #footer>
<span class='dialog-footer'>
<el-button @click='dialogVisible = false'>取消</el-button>
<el-button type='primary' @click='submit'>确认</el-button>
</span>
</template>
</el-dialog>
</template>
<script lang='ts' setup>
import { ref, inject } from 'vue'
import { reactive } from 'vue'
import TableStore from '@/utils/tableStore'
import { ElMessage } from 'element-plus'
import { roleList } from '@/api/user-boot/role'
import { add, edit } from '@/api/user-boot/user'
import { useAdminInfo } from '@/stores/adminInfo'
import Area from '@/components/form/area/index.vue'
import { de } from 'element-plus/es/locale'
const adminInfo = useAdminInfo()
const tableStore = inject('tableStore') as TableStore
// do not use same name with ref
const form = reactive({
id: '',
name: '',
password: '123456',
email: '',
limitIpStart: '',
deptId: '',
deptName: '',
casualUser: 1,
loginName: '',
phone: '',
limitIpEnd: '',
limitTime: [1, 2],
role: [],
smsNotice: 0,
emailNotice: 0,
type: 0
})
const rules = {
name: [{ required: true, message: '用户名不能为空', trigger: 'blur' }],
role: [{ required: true, message: '角色不能为空', trigger: 'blur' }],
password: [{ required: true, message: '用户密码不能为空', trigger: 'blur' }],
loginName: [{ required: true, message: '登录名不能为空', trigger: 'blur' }],
casualUser: [{ required: true, message: '用户类型不能为空', trigger: 'blur' }],
smsNotice: [{ required: true, message: '短信通知不能为空', trigger: 'blur' }],
emailNotice: [{ required: true, message: '邮件通知不能为空', trigger: 'blur' }],
email: [
{ required: false, message: '邮箱不能为空', trigger: 'blur' },
{
type: 'email',
message: '\'请输入正确的邮箱地址',
trigger: ['blur', 'change']
}
],
phone: [
{ required: false, message: '手机号不能为空', trigger: 'blur' },
{
pattern: /^1[3|4|5|6|7|8|9][0-9]\d{8}$/,
message: '请输入正确的手机号码',
trigger: 'blur'
}
],
limitTime: [{ required: true, message: '时间段不能为空', trigger: 'blur' }],
limitIpStart: [
{ required: true, message: '起始IP不能为空', trigger: 'blur' },
{
required: true,
validator: (rule: any, value: string, callback: any) => {
let regexp = /^((2(5[0-5]|[0-4]\d))|[0-1]?\d{1,2})(\.((2(5[0-5]|[0-4]\d))|[0-1]?\d{1,2})){3}$/
let isCorrect = regexp.test(value)
if (value == '') {
return callback(new Error('请输入IP地址'))
} else if (!isCorrect) {
callback(new Error('请输入正确的IP地址'))
} else {
callback()
}
},
trigger: 'blur'
}
],
limitIpEnd: [
{ required: true, message: '结束IP不能为空', trigger: 'blur' },
{
required: true,
validator: (rule: any, value: string, callback: any) => {
let regexp = /^((2(5[0-5]|[0-4]\d))|[0-1]?\d{1,2})(\.((2(5[0-5]|[0-4]\d))|[0-1]?\d{1,2})){3}$/
let isCorrect = regexp.test(value)
if (value == '') {
return callback(new Error('请输入IP地址'))
} else if (!isCorrect) {
callback(new Error('请输入正确的IP地址'))
} else {
callback()
}
},
trigger: 'blur'
}
]
}
const UserTypeOption = [
{ label: '管理员', value: 1 },
{ label: '普通用户', value: 2 }
]
const TypeOptions = [
{ label: '临时用户', value: 0 },
{ label: '长期用户', value: 1 }
]
const useId = ref(1)
const roleOptions = ref<treeData>()
const queryRole = () => {
roleList(adminInfo.$state.userType).then((res: any) => {
roleOptions.value = res.data.map((item: any) => {
return {
label: item.name,
value: item.id
}
})
})
}
queryRole()
const dialogVisible = ref(false)
const title = ref('新增菜单')
const open = (text: string, data?: anyObj) => {
title.value = text
dialogVisible.value = true
if (data) {
for (let key in form) {
form[key] = data[key]
}
form.limitTime = data.limitTime.split('-')
form.role = data.roleList
} else {
for (let key in form) {
form[key] = ''
}
form.casualUser = 1
form.limitTime = [0, 23]
form.role = []
form.smsNotice = 0
form.emailNotice = 0
useId.value = 1
form.id = ''
form.limitIpStart = '0.0.0.0'
form.limitIpEnd = '255.255.255.255'
form.password = '123456'
}
form.type = adminInfo.$state.userType + 1
}
const submit = async () => {
let obj = JSON.parse(JSON.stringify(form))
obj.limitTime = obj.limitTime.join('-')
delete obj.password
if (form.id) {
await edit(obj)
ElMessage.success('修改成功')
} else {
form.type = adminInfo.$state.userType + 1
await add(obj)
ElMessage.success('新增成功')
}
tableStore.index()
dialogVisible.value = false
}
const changeValue = () => {
}
defineExpose({ open })
</script>

View File

@@ -0,0 +1,122 @@
<template>
<el-dialog class='cn-operate-dialog' v-model='dialogVisible' title='修改密码'>
<el-scrollbar>
<el-form :inline='false' :model='form' label-width='120px' :rules='rules' ref='formRef'>
<el-form-item label='校验密码' prop='password'>
<el-input v-model='form.password' type='password' placeholder='请输入校验密码' show-password />
</el-form-item>
<el-form-item label='新密码' prop='newPwd'>
<el-input v-model='form.newPwd' type='password' placeholder='请输入新密码' show-password />
</el-form-item>
<el-form-item label='确认密码' prop='confirmPwd'>
<el-input v-model='form.confirmPwd' type='password' placeholder='请输入确认密码' show-password />
</el-form-item>
</el-form>
</el-scrollbar>
<template #footer>
<span class='dialog-footer'>
<el-button @click='dialogVisible = false'>取消</el-button>
<el-button type='primary' @click='submit'>确认</el-button>
</span>
</template>
</el-dialog>
</template>
<script lang='ts' setup>
import { ref, inject } from 'vue'
import { reactive } from 'vue'
import TableStore from '@/utils/tableStore'
import { ElMessage } from 'element-plus'
import { validatePwd } from '@/utils/common'
import { passwordConfirm, updatePassword } from '@/api/user-boot/user'
const formRef = ref()
const tableStore = inject('tableStore') as TableStore
const form = reactive({
id: '',
password: '',
newPwd: '',
confirmPwd: ''
})
const rules = {
password: [
{ required: true, message: '请输入校验密码', trigger: 'blur' },
{
min: 6,
max: 16,
message: '长度在 6 到 16 个字符',
trigger: 'blur'
},
{
validator: (rule: any, value: string, callback: any) => {
if (value === '') {
callback(new Error('请输入旧密码'))
} else {
passwordConfirm(value).then(res => {
callback()
}).catch(() => {
callback(new Error('旧密码不正确'))
})
}
}, trigger: 'blur'
}
],
newPwd: [
{ required: true, message: '请输入密码', trigger: 'blur' },
{
min: 6,
max: 16,
message: '长度在 6 到 16 个字符',
trigger: 'blur'
},
{ validator: validatePwd, trigger: 'blur' }
],
confirmPwd: [
{ required: true, message: '请确认密码', trigger: 'blur' },
{
min: 6,
max: 16,
message: '长度在 6 到 16 个字符',
trigger: 'blur'
},
{
validator: (rule: any, value: string, callback: any) => {
if (value === '') {
callback(new Error('请再次输入密码'))
} else if (value !== form.newPwd) {
callback(new Error('两次输入密码不一致!'))
} else {
callback()
}
},
trigger: 'blur',
required: true
}
]
}
const dialogVisible = ref(false)
const title = ref('新增菜单')
const open = (id: string) => {
form.id = id
form.password = ''
form.newPwd = ''
form.confirmPwd = ''
dialogVisible.value = true
}
const submit = async () => {
formRef.value.validate((valid) => {
if (valid) {
updatePassword({
id: form.id,
newPassword: form.newPwd
}).then((res: any) => {
ElMessage.success('密码修改成功')
dialogVisible.value = false
})
}
})
}
defineExpose({ open })
</script>