164 lines
3.8 KiB
Vue
164 lines
3.8 KiB
Vue
<template>
|
|
<a-modal :title="titleMap[mode]" :open="visible" :width="450" :destroy-on-close="true" :mask-closable="false"
|
|
:footer="null" @cancel="handleCancel">
|
|
<a-form :model="form" :rules="rules" ref="dialogForm" :label-col="{ span: 5 }" :wrapper-col="{ span: 18 }">
|
|
<a-form-item label="所属字典" name="group_id">
|
|
<a-tree-select v-model:value="form.group_id" :tree-data="dicData"
|
|
:field-names="{ label: 'title', value: 'id', children: 'children' }" tree-default-expand-all
|
|
show-search placeholder="请选择所属字典" allow-clear tree-node-filter-prop="title" />
|
|
</a-form-item>
|
|
<a-form-item label="项名称" name="title">
|
|
<a-input v-model:value="form.title" placeholder="请输入项名称" allow-clear />
|
|
</a-form-item>
|
|
<a-form-item label="键值" name="values">
|
|
<a-input v-model:value="form.values" placeholder="请输入键值" allow-clear />
|
|
</a-form-item>
|
|
<a-form-item label="排序" name="sort">
|
|
<a-input-number v-model:value="form.sort" :min="0" style="width: 100%" placeholder="请输入排序" />
|
|
</a-form-item>
|
|
<a-form-item label="是否有效" name="status">
|
|
<a-radio-group v-model:value="form.status">
|
|
<a-radio :value="1">是</a-radio>
|
|
<a-radio :value="0">否</a-radio>
|
|
</a-radio-group>
|
|
</a-form-item>
|
|
</a-form>
|
|
<template #footer>
|
|
<a-button @click="handleCancel">取 消</a-button>
|
|
<a-button type="primary" :loading="isSaveing" @click="submit">保 存</a-button>
|
|
</template>
|
|
</a-modal>
|
|
</template>
|
|
|
|
<script setup>
|
|
import { ref, reactive } from 'vue'
|
|
import { message } from 'ant-design-vue'
|
|
import systemApi from '@/api/system'
|
|
|
|
const emit = defineEmits(['success', 'closed'])
|
|
|
|
const mode = ref('add')
|
|
const titleMap = {
|
|
add: '新增项',
|
|
edit: '编辑项'
|
|
}
|
|
const visible = ref(false)
|
|
const isSaveing = ref(false)
|
|
|
|
// 表单数据
|
|
const form = reactive({
|
|
id: '',
|
|
group_id: '',
|
|
title: '',
|
|
values: '',
|
|
sort: 0,
|
|
status: 1
|
|
})
|
|
|
|
// 表单引用
|
|
const dialogForm = ref()
|
|
|
|
// 验证规则
|
|
const rules = {
|
|
group_id: [
|
|
{ required: true, message: '请选择所属字典', trigger: 'change' }
|
|
],
|
|
title: [
|
|
{ required: true, message: '请输入项名称', trigger: 'blur' }
|
|
],
|
|
values: [
|
|
{ required: true, message: '请输入键值', trigger: 'blur' }
|
|
]
|
|
}
|
|
|
|
// 字典树数据
|
|
const dicData = ref([])
|
|
|
|
// 显示对话框
|
|
const open = (openMode = 'add') => {
|
|
mode.value = openMode
|
|
visible.value = true
|
|
loadDic()
|
|
return {
|
|
setData,
|
|
open,
|
|
close
|
|
}
|
|
}
|
|
|
|
// 关闭对话框
|
|
const close = () => {
|
|
visible.value = false
|
|
}
|
|
|
|
// 处理取消
|
|
const handleCancel = () => {
|
|
emit('closed')
|
|
visible.value = false
|
|
}
|
|
|
|
// 加载字典列表
|
|
const loadDic = async () => {
|
|
try {
|
|
const res = await systemApi.dictionary.category.get({ is_tree: 1 })
|
|
if (res.code === 1) {
|
|
dicData.value = res.data || []
|
|
}
|
|
} catch (error) {
|
|
console.error('加载字典列表失败:', error)
|
|
}
|
|
}
|
|
|
|
// 表单提交方法
|
|
const submit = async () => {
|
|
try {
|
|
await dialogForm.value.validate()
|
|
isSaveing.value = true
|
|
let res
|
|
|
|
if (mode.value === 'add') {
|
|
res = await systemApi.dictionary.add.post(form)
|
|
} else {
|
|
res = await systemApi.dictionary.edit.post(form)
|
|
}
|
|
|
|
isSaveing.value = false
|
|
if (res.code === 1) {
|
|
emit('success', form, mode.value)
|
|
visible.value = false
|
|
message.success('操作成功')
|
|
} else {
|
|
message.error(res.message || '操作失败')
|
|
}
|
|
} catch (error) {
|
|
isSaveing.value = false
|
|
console.error('表单验证失败', error)
|
|
}
|
|
}
|
|
|
|
// 表单注入数据
|
|
const setData = (data) => {
|
|
if (data.dic_type) {
|
|
form.dic_type = data.dic_type
|
|
}
|
|
form.id = data.id
|
|
form.title = data.title
|
|
form.values = data.values
|
|
form.sort = data.sort || 0
|
|
form.status = data.status ?? 1
|
|
form.group_id = data.group_id
|
|
return
|
|
}
|
|
|
|
// 暴露方法给父组件
|
|
defineExpose({
|
|
open,
|
|
setData,
|
|
close
|
|
})
|
|
</script>
|
|
|
|
<style scoped lang="scss">
|
|
// 弹窗样式可根据需要添加
|
|
</style>
|