feat: 新增 antd 主子表 erp 模式代码生成示例
							parent
							
								
									1bfcdcb708
								
							
						
					
					
						commit
						d746b8f572
					
				|  | @ -1,3 +1,5 @@ | |||
| import type { Dayjs } from 'dayjs'; | ||||
| 
 | ||||
| import type { PageParam, PageResult } from '@vben/request'; | ||||
| 
 | ||||
| import { requestClient } from '#/api/request'; | ||||
|  | @ -24,7 +26,7 @@ export namespace Demo03StudentApi { | |||
|     id: number; // 编号
 | ||||
|     name?: string; // 名字
 | ||||
|     sex?: number; // 性别
 | ||||
|     birthday?: Date; // 出生日期
 | ||||
|     birthday?: Dayjs | string; // 出生日期
 | ||||
|     description?: string; // 简介
 | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,324 @@ | |||
| <script lang="ts" setup> | ||||
| import type { VxeTableInstance } from 'vxe-table'; | ||||
| 
 | ||||
| import type { Demo03StudentApi } from '#/api/infra/demo/demo03/erp'; | ||||
| 
 | ||||
| import { h, nextTick, onMounted, reactive, ref } from 'vue'; | ||||
| 
 | ||||
| import { Page, useVbenModal } from '@vben/common-ui'; | ||||
| import { Download, Plus } from '@vben/icons'; | ||||
| import { cloneDeep, formatDateTime } from '@vben/utils'; | ||||
| 
 | ||||
| import { | ||||
|   Button, | ||||
|   DatePicker, | ||||
|   Form, | ||||
|   Input, | ||||
|   message, | ||||
|   Pagination, | ||||
|   RangePicker, | ||||
|   Select, | ||||
|   Tabs, | ||||
| } from 'ant-design-vue'; | ||||
| import { VxeColumn, VxeTable } from 'vxe-table'; | ||||
| 
 | ||||
| import { | ||||
|   deleteDemo03Student, | ||||
|   exportDemo03Student, | ||||
|   getDemo03StudentPage, | ||||
| } from '#/api/infra/demo/demo03/erp'; | ||||
| import { ContentWrap } from '#/components/content-wrap'; | ||||
| import { DictTag } from '#/components/dict-tag'; | ||||
| import { TableToolbar } from '#/components/table-toolbar'; | ||||
| import { $t } from '#/locales'; | ||||
| import { getRangePickerDefaultProps } from '#/utils/date'; | ||||
| import { DICT_TYPE, getDictOptions } from '#/utils/dict'; | ||||
| import { downloadByData } from '#/utils/download'; | ||||
| 
 | ||||
| import Demo03CourseList from './modules/demo03-course-list.vue'; | ||||
| import Demo03GradeList from './modules/demo03-grade-list.vue'; | ||||
| import Demo03StudentForm from './modules/form.vue'; | ||||
| 
 | ||||
| /** 子表的列表 */ | ||||
| const subTabsName = ref('demo03Course'); | ||||
| const selectDemo03Student = ref<Demo03StudentApi.Demo03Student>(); | ||||
| async function onCellClick({ row }: { row: Demo03StudentApi.Demo03Student }) { | ||||
|   selectDemo03Student.value = row; | ||||
| } | ||||
| 
 | ||||
| const loading = ref(true); // 列表的加载中 | ||||
| const list = ref<Demo03StudentApi.Demo03Student[]>([]); // 列表的数据 | ||||
| 
 | ||||
| const total = ref(0); // 列表的总页数 | ||||
| const queryParams = reactive({ | ||||
|   pageNo: 1, | ||||
|   pageSize: 10, | ||||
|   name: undefined, | ||||
|   sex: undefined, | ||||
|   birthday: undefined, | ||||
|   description: undefined, | ||||
|   createTime: undefined, | ||||
| }); | ||||
| const queryFormRef = ref(); // 搜索的表单 | ||||
| const exportLoading = ref(false); // 导出的加载中 | ||||
| 
 | ||||
| /** 查询列表 */ | ||||
| const getList = async () => { | ||||
|   loading.value = true; | ||||
|   try { | ||||
|     const params = cloneDeep(queryParams) as any; | ||||
|     if (params.birthday && Array.isArray(params.birthday)) { | ||||
|       params.birthday = (params.birthday as string[]).join(','); | ||||
|     } | ||||
|     if (params.createTime && Array.isArray(params.createTime)) { | ||||
|       params.createTime = (params.createTime as string[]).join(','); | ||||
|     } | ||||
|     const data = await getDemo03StudentPage(params); | ||||
|     list.value = data.list; | ||||
|     total.value = data.total; | ||||
|   } finally { | ||||
|     loading.value = false; | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| /** 搜索按钮操作 */ | ||||
| const handleQuery = () => { | ||||
|   queryParams.pageNo = 1; | ||||
|   getList(); | ||||
| }; | ||||
| 
 | ||||
| /** 重置按钮操作 */ | ||||
| const resetQuery = () => { | ||||
|   queryFormRef.value.resetFields(); | ||||
|   handleQuery(); | ||||
| }; | ||||
| 
 | ||||
| const [FormModal, formModalApi] = useVbenModal({ | ||||
|   connectedComponent: Demo03StudentForm, | ||||
|   destroyOnClose: true, | ||||
| }); | ||||
| 
 | ||||
| /** 创建学生 */ | ||||
| function onCreate() { | ||||
|   formModalApi.setData({}).open(); | ||||
| } | ||||
| 
 | ||||
| /** 编辑学生 */ | ||||
| function onEdit(row: Demo03StudentApi.Demo03Student) { | ||||
|   formModalApi.setData(row).open(); | ||||
| } | ||||
| 
 | ||||
| /** 删除学生 */ | ||||
| async function onDelete(row: Demo03StudentApi.Demo03Student) { | ||||
|   const hideLoading = message.loading({ | ||||
|     content: $t('ui.actionMessage.deleting', [row.id]), | ||||
|     duration: 0, | ||||
|     key: 'action_process_msg', | ||||
|   }); | ||||
|   try { | ||||
|     await deleteDemo03Student(row.id as number); | ||||
|     message.success({ | ||||
|       content: $t('ui.actionMessage.deleteSuccess', [row.id]), | ||||
|       key: 'action_process_msg', | ||||
|     }); | ||||
|     await getList(); | ||||
|   } catch { | ||||
|     hideLoading(); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** 导出表格 */ | ||||
| async function onExport() { | ||||
|   try { | ||||
|     exportLoading.value = true; | ||||
|     const data = await exportDemo03Student(queryParams); | ||||
|     downloadByData(data, '学生.xls'); | ||||
|   } finally { | ||||
|     exportLoading.value = false; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** 隐藏搜索栏 */ | ||||
| const hiddenSearchBar = ref(false); | ||||
| const tableToolbarRef = ref<InstanceType<typeof TableToolbar>>(); | ||||
| const tableRef = ref<VxeTableInstance>(); | ||||
| 
 | ||||
| /** 初始化 */ | ||||
| onMounted(async () => { | ||||
|   await getList(); | ||||
|   await nextTick(); | ||||
|   // 挂载 toolbar 工具栏 | ||||
|   const table = tableRef.value; | ||||
|   const tableToolbar = tableToolbarRef.value; | ||||
|   if (table && tableToolbar) { | ||||
|     await table.connect(tableToolbar.getToolbarRef()!); | ||||
|   } | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <Page auto-content-height> | ||||
|     <FormModal @success="getList" /> | ||||
| 
 | ||||
|     <ContentWrap v-if="!hiddenSearchBar"> | ||||
|       <!-- 搜索工作栏 --> | ||||
|       <Form :model="queryParams" ref="queryFormRef" layout="inline"> | ||||
|         <Form.Item label="名字" name="name"> | ||||
|           <Input | ||||
|             v-model:value="queryParams.name" | ||||
|             placeholder="请输入名字" | ||||
|             allow-clear | ||||
|             @press-enter="handleQuery" | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item label="性别" name="sex"> | ||||
|           <Select | ||||
|             v-model:value="queryParams.sex" | ||||
|             placeholder="请选择性别" | ||||
|             allow-clear | ||||
|             class="w-full" | ||||
|           > | ||||
|             <Select.Option | ||||
|               v-for="dict in getDictOptions( | ||||
|                 DICT_TYPE.SYSTEM_USER_SEX, | ||||
|                 'number', | ||||
|               )" | ||||
|               :key="dict.value" | ||||
|               :value="dict.value" | ||||
|             > | ||||
|               {{ dict.label }} | ||||
|             </Select.Option> | ||||
|           </Select> | ||||
|         </Form.Item> | ||||
|         <Form.Item label="出生日期" name="birthday"> | ||||
|           <DatePicker | ||||
|             v-model:value="queryParams.birthday" | ||||
|             value-format="YYYY-MM-DD" | ||||
|             placeholder="选择出生日期" | ||||
|             allow-clear | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item label="创建时间" name="createTime"> | ||||
|           <RangePicker | ||||
|             v-model:value="queryParams.createTime" | ||||
|             v-bind="getRangePickerDefaultProps()" | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item> | ||||
|           <Button class="ml-2" @click="resetQuery"> 重置 </Button> | ||||
|           <Button class="ml-2" @click="handleQuery" type="primary"> | ||||
|             搜索 | ||||
|           </Button> | ||||
|         </Form.Item> | ||||
|       </Form> | ||||
|     </ContentWrap> | ||||
| 
 | ||||
|     <!-- 列表 --> | ||||
|     <ContentWrap title="学生"> | ||||
|       <template #extra> | ||||
|         <TableToolbar | ||||
|           ref="tableToolbarRef" | ||||
|           v-model:hidden-search="hiddenSearchBar" | ||||
|         > | ||||
|           <Button | ||||
|             class="ml-2" | ||||
|             :icon="h(Plus)" | ||||
|             type="primary" | ||||
|             @click="onCreate" | ||||
|             v-access:code="['infra:demo03-student:create']" | ||||
|           > | ||||
|             {{ $t('ui.actionTitle.create', ['学生']) }} | ||||
|           </Button> | ||||
|           <Button | ||||
|             :icon="h(Download)" | ||||
|             type="primary" | ||||
|             class="ml-2" | ||||
|             :loading="exportLoading" | ||||
|             @click="onExport" | ||||
|             v-access:code="['infra:demo03-student:export']" | ||||
|           > | ||||
|             {{ $t('ui.actionTitle.export') }} | ||||
|           </Button> | ||||
|         </TableToolbar> | ||||
|       </template> | ||||
|       <VxeTable | ||||
|         ref="tableRef" | ||||
|         :data="list" | ||||
|         @cell-click="onCellClick" | ||||
|         :row-config="{ | ||||
|           keyField: 'id', | ||||
|           isHover: true, | ||||
|           isCurrent: true, | ||||
|         }" | ||||
|         show-overflow | ||||
|         :loading="loading" | ||||
|       > | ||||
|         <VxeColumn field="id" title="编号" align="center" /> | ||||
|         <VxeColumn field="name" title="名字" align="center" /> | ||||
|         <VxeColumn field="sex" title="性别" align="center"> | ||||
|           <template #default="{ row }"> | ||||
|             <DictTag :type="DICT_TYPE.SYSTEM_USER_SEX" :value="row.sex" /> | ||||
|           </template> | ||||
|         </VxeColumn> | ||||
|         <VxeColumn field="birthday" title="出生日期" align="center"> | ||||
|           <template #default="{ row }"> | ||||
|             {{ formatDateTime(row.birthday) }} | ||||
|           </template> | ||||
|         </VxeColumn> | ||||
|         <VxeColumn field="description" title="简介" align="center" /> | ||||
|         <VxeColumn field="createTime" title="创建时间" align="center"> | ||||
|           <template #default="{ row }"> | ||||
|             {{ formatDateTime(row.createTime) }} | ||||
|           </template> | ||||
|         </VxeColumn> | ||||
|         <VxeColumn field="operation" title="操作" align="center"> | ||||
|           <template #default="{ row }"> | ||||
|             <Button | ||||
|               size="small" | ||||
|               type="link" | ||||
|               @click="onEdit(row as any)" | ||||
|               v-access:code="['infra:demo03-student:update']" | ||||
|             > | ||||
|               {{ $t('ui.actionTitle.edit') }} | ||||
|             </Button> | ||||
|             <Button | ||||
|               size="small" | ||||
|               type="link" | ||||
|               danger | ||||
|               class="ml-2" | ||||
|               @click="onDelete(row as any)" | ||||
|               v-access:code="['infra:demo03-student:delete']" | ||||
|             > | ||||
|               {{ $t('ui.actionTitle.delete') }} | ||||
|             </Button> | ||||
|           </template> | ||||
|         </VxeColumn> | ||||
|       </VxeTable> | ||||
|       <!-- 分页 --> | ||||
|       <div class="mt-2 flex justify-end"> | ||||
|         <Pagination | ||||
|           :total="total" | ||||
|           v-model:current="queryParams.pageNo" | ||||
|           v-model:page-size="queryParams.pageSize" | ||||
|           show-size-changer | ||||
|           @change="getList" | ||||
|         /> | ||||
|       </div> | ||||
|     </ContentWrap> | ||||
| 
 | ||||
|     <ContentWrap> | ||||
|       <!-- 子表的表单 --> | ||||
|       <Tabs v-model:active-key="subTabsName"> | ||||
|         <Tabs.TabPane key="demo03Course" tab="学生课程" force-render> | ||||
|           <Demo03CourseList :student-id="selectDemo03Student?.id" /> | ||||
|         </Tabs.TabPane> | ||||
|         <Tabs.TabPane key="demo03Grade" tab="学生班级" force-render> | ||||
|           <Demo03GradeList :student-id="selectDemo03Student?.id" /> | ||||
|         </Tabs.TabPane> | ||||
|       </Tabs> | ||||
|     </ContentWrap> | ||||
|   </Page> | ||||
| </template> | ||||
|  | @ -0,0 +1,120 @@ | |||
| <script lang="ts" setup> | ||||
| import type { Rule } from 'ant-design-vue/es/form'; | ||||
| 
 | ||||
| import type { Demo03StudentApi } from '#/api/infra/demo/demo03/erp'; | ||||
| 
 | ||||
| import { computed, ref } from 'vue'; | ||||
| 
 | ||||
| import { useVbenModal } from '@vben/common-ui'; | ||||
| 
 | ||||
| import { Form, Input, message } from 'ant-design-vue'; | ||||
| 
 | ||||
| import { | ||||
|   createDemo03Course, | ||||
|   getDemo03Course, | ||||
|   updateDemo03Course, | ||||
| } from '#/api/infra/demo/demo03/erp'; | ||||
| import { $t } from '#/locales'; | ||||
| 
 | ||||
| const emit = defineEmits(['success']); | ||||
| const getTitle = computed(() => { | ||||
|   return formData.value?.id | ||||
|     ? $t('ui.actionTitle.edit', ['学生课程']) | ||||
|     : $t('ui.actionTitle.create', ['学生课程']); | ||||
| }); | ||||
| 
 | ||||
| const formRef = ref(); | ||||
| const formData = ref<Partial<Demo03StudentApi.Demo03Course>>({ | ||||
|   id: undefined, | ||||
|   studentId: undefined, | ||||
|   name: undefined, | ||||
|   score: undefined, | ||||
| }); | ||||
| const rules: Record<string, Rule[]> = { | ||||
|   studentId: [{ required: true, message: '学生编号不能为空', trigger: 'blur' }], | ||||
|   name: [{ required: true, message: '名字不能为空', trigger: 'blur' }], | ||||
|   score: [{ required: true, message: '分数不能为空', trigger: 'blur' }], | ||||
| }; | ||||
| 
 | ||||
| const [Modal, modalApi] = useVbenModal({ | ||||
|   async onConfirm() { | ||||
|     await formRef.value?.validate(); | ||||
| 
 | ||||
|     modalApi.lock(); | ||||
|     // 提交表单 | ||||
|     const data = formData.value as Demo03StudentApi.Demo03Course; | ||||
|     try { | ||||
|       await (formData.value?.id | ||||
|         ? updateDemo03Course(data) | ||||
|         : createDemo03Course(data)); | ||||
|       // 关闭并提示 | ||||
|       await modalApi.close(); | ||||
|       emit('success'); | ||||
|       message.success({ | ||||
|         content: $t('ui.actionMessage.operationSuccess'), | ||||
|         key: 'action_process_msg', | ||||
|       }); | ||||
|     } finally { | ||||
|       modalApi.unlock(); | ||||
|     } | ||||
|   }, | ||||
|   async onOpenChange(isOpen: boolean) { | ||||
|     if (!isOpen) { | ||||
|       resetForm(); | ||||
|       return; | ||||
|     } | ||||
| 
 | ||||
|     // 加载数据 | ||||
|     let data = modalApi.getData<Demo03StudentApi.Demo03Course>(); | ||||
|     if (!data) { | ||||
|       return; | ||||
|     } | ||||
|     if (data.id) { | ||||
|       modalApi.lock(); | ||||
|       try { | ||||
|         data = await getDemo03Course(data.id); | ||||
|       } finally { | ||||
|         modalApi.unlock(); | ||||
|       } | ||||
|     } | ||||
|     // 设置到 values | ||||
|     formData.value = data; | ||||
|   }, | ||||
| }); | ||||
| 
 | ||||
| /** 重置表单 */ | ||||
| const resetForm = () => { | ||||
|   formData.value = { | ||||
|     id: undefined, | ||||
|     studentId: undefined, | ||||
|     name: undefined, | ||||
|     score: undefined, | ||||
|   }; | ||||
|   formRef.value?.resetFields(); | ||||
| }; | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <Modal :title="getTitle"> | ||||
|     <Form | ||||
|       ref="formRef" | ||||
|       :model="formData" | ||||
|       :rules="rules" | ||||
|       :label-col="{ span: 5 }" | ||||
|       :wrapper-col="{ span: 18 }" | ||||
|     > | ||||
|       <Form.Item label="学生编号" name="studentId"> | ||||
|         <Input | ||||
|           v-model:value="formData.studentId" | ||||
|           placeholder="请输入学生编号" | ||||
|         /> | ||||
|       </Form.Item> | ||||
|       <Form.Item label="名字" name="name"> | ||||
|         <Input v-model:value="formData.name" placeholder="请输入名字" /> | ||||
|       </Form.Item> | ||||
|       <Form.Item label="分数" name="score"> | ||||
|         <Input v-model:value="formData.score" placeholder="请输入分数" /> | ||||
|       </Form.Item> | ||||
|     </Form> | ||||
|   </Modal> | ||||
| </template> | ||||
|  | @ -0,0 +1,265 @@ | |||
| <script lang="ts" setup> | ||||
| import type { VxeTableInstance } from 'vxe-table'; | ||||
| 
 | ||||
| import type { Demo03StudentApi } from '#/api/infra/demo/demo03/erp'; | ||||
| 
 | ||||
| import { h, nextTick, onMounted, reactive, ref, watch } from 'vue'; | ||||
| 
 | ||||
| import { useVbenModal } from '@vben/common-ui'; | ||||
| import { Plus } from '@vben/icons'; | ||||
| import { cloneDeep, formatDateTime } from '@vben/utils'; | ||||
| 
 | ||||
| import { | ||||
|   Button, | ||||
|   Form, | ||||
|   Input, | ||||
|   message, | ||||
|   Pagination, | ||||
|   RangePicker, | ||||
| } from 'ant-design-vue'; | ||||
| import { VxeColumn, VxeTable } from 'vxe-table'; | ||||
| 
 | ||||
| import { | ||||
|   deleteDemo03Course, | ||||
|   getDemo03CoursePage, | ||||
| } from '#/api/infra/demo/demo03/erp'; | ||||
| import { ContentWrap } from '#/components/content-wrap'; | ||||
| import { TableToolbar } from '#/components/table-toolbar'; | ||||
| import { $t } from '#/locales'; | ||||
| import { getRangePickerDefaultProps } from '#/utils/date'; | ||||
| 
 | ||||
| import Demo03CourseForm from './demo03-course-form.vue'; | ||||
| 
 | ||||
| const props = defineProps<{ | ||||
|   studentId?: number; // 学生编号(主表的关联字段) | ||||
| }>(); | ||||
| 
 | ||||
| const [FormModal, formModalApi] = useVbenModal({ | ||||
|   connectedComponent: Demo03CourseForm, | ||||
|   destroyOnClose: true, | ||||
| }); | ||||
| 
 | ||||
| /** 创建学生课程 */ | ||||
| function onCreate() { | ||||
|   if (!props.studentId) { | ||||
|     message.warning('请先选择一个学生!'); | ||||
|     return; | ||||
|   } | ||||
|   formModalApi.setData({ studentId: props.studentId }).open(); | ||||
| } | ||||
| 
 | ||||
| /** 编辑学生课程 */ | ||||
| function onEdit(row: Demo03StudentApi.Demo03Course) { | ||||
|   formModalApi.setData(row).open(); | ||||
| } | ||||
| 
 | ||||
| /** 删除学生课程 */ | ||||
| async function onDelete(row: Demo03StudentApi.Demo03Course) { | ||||
|   const hideLoading = message.loading({ | ||||
|     content: $t('ui.actionMessage.deleting', [row.id]), | ||||
|     duration: 0, | ||||
|     key: 'action_process_msg', | ||||
|   }); | ||||
|   try { | ||||
|     await deleteDemo03Course(row.id as number); | ||||
|     message.success({ | ||||
|       content: $t('ui.actionMessage.deleteSuccess', [row.id]), | ||||
|       key: 'action_process_msg', | ||||
|     }); | ||||
|     getList(); | ||||
|   } catch { | ||||
|     hideLoading(); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| const loading = ref(true); // 列表的加载中 | ||||
| const list = ref<Demo03StudentApi.Demo03Course[]>([]); // 列表的数据 | ||||
| const total = ref(0); // 列表的总页数 | ||||
| const queryFormRef = ref(); // 搜索的表单 | ||||
| const queryParams = reactive({ | ||||
|   pageNo: 1, | ||||
|   pageSize: 10, | ||||
|   studentId: undefined, | ||||
|   name: undefined, | ||||
|   score: undefined, | ||||
|   createTime: undefined, | ||||
| }); | ||||
| 
 | ||||
| /** 搜索按钮操作 */ | ||||
| const handleQuery = () => { | ||||
|   queryParams.pageNo = 1; | ||||
|   getList(); | ||||
| }; | ||||
| 
 | ||||
| /** 重置按钮操作 */ | ||||
| const resetQuery = () => { | ||||
|   queryFormRef.value.resetFields(); | ||||
|   handleQuery(); | ||||
| }; | ||||
| /** 查询列表 */ | ||||
| const getList = async () => { | ||||
|   loading.value = true; | ||||
|   try { | ||||
|     if (!props.studentId) { | ||||
|       return []; | ||||
|     } | ||||
|     const params = cloneDeep(queryParams) as any; | ||||
|     if (params.birthday && Array.isArray(params.birthday)) { | ||||
|       params.birthday = (params.birthday as string[]).join(','); | ||||
|     } | ||||
|     if (params.createTime && Array.isArray(params.createTime)) { | ||||
|       params.createTime = (params.createTime as string[]).join(','); | ||||
|     } | ||||
|     params.studentId = props.studentId; | ||||
|     const data = await getDemo03CoursePage(params); | ||||
|     list.value = data.list; | ||||
|     total.value = data.total; | ||||
|   } finally { | ||||
|     loading.value = false; | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| /** 监听主表的关联字段的变化,加载对应的子表数据 */ | ||||
| watch( | ||||
|   () => props.studentId, | ||||
|   async (val) => { | ||||
|     if (!val) { | ||||
|       return; | ||||
|     } | ||||
|     await nextTick(); | ||||
|     await getList(); | ||||
|   }, | ||||
|   { immediate: true }, | ||||
| ); | ||||
| 
 | ||||
| /** 隐藏搜索栏 */ | ||||
| const hiddenSearchBar = ref(false); | ||||
| const tableToolbarRef = ref<InstanceType<typeof TableToolbar>>(); | ||||
| const tableRef = ref<VxeTableInstance>(); | ||||
| 
 | ||||
| /** 初始化 */ | ||||
| onMounted(async () => { | ||||
|   await getList(); | ||||
|   await nextTick(); | ||||
|   // 挂载 toolbar 工具栏 | ||||
|   const table = tableRef.value; | ||||
|   const tableToolbar = tableToolbarRef.value; | ||||
|   if (table && tableToolbar) { | ||||
|     await table.connect(tableToolbar.getToolbarRef()!); | ||||
|   } | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <FormModal @success="getList" /> | ||||
|   <div class="h-[600px]"> | ||||
|     <ContentWrap v-if="!hiddenSearchBar"> | ||||
|       <!-- 搜索工作栏 --> | ||||
|       <Form :model="queryParams" ref="queryFormRef" layout="inline"> | ||||
|         <Form.Item label="学生编号" name="studentId"> | ||||
|           <Input | ||||
|             v-model:value="queryParams.studentId" | ||||
|             placeholder="请输入学生编号" | ||||
|             allow-clear | ||||
|             @press-enter="handleQuery" | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item label="名字" name="name"> | ||||
|           <Input | ||||
|             v-model:value="queryParams.name" | ||||
|             placeholder="请输入名字" | ||||
|             allow-clear | ||||
|             @press-enter="handleQuery" | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item label="分数" name="score"> | ||||
|           <Input | ||||
|             v-model:value="queryParams.score" | ||||
|             placeholder="请输入分数" | ||||
|             allow-clear | ||||
|             @press-enter="handleQuery" | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item label="创建时间" name="createTime"> | ||||
|           <RangePicker | ||||
|             v-model:value="queryParams.createTime" | ||||
|             v-bind="getRangePickerDefaultProps()" | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item> | ||||
|           <Button class="ml-2" @click="resetQuery"> 重置 </Button> | ||||
|           <Button class="ml-2" @click="handleQuery" type="primary"> | ||||
|             搜索 | ||||
|           </Button> | ||||
|         </Form.Item> | ||||
|       </Form> | ||||
|     </ContentWrap> | ||||
| 
 | ||||
|     <!-- 列表 --> | ||||
|     <ContentWrap title="学生"> | ||||
|       <template #extra> | ||||
|         <TableToolbar | ||||
|           ref="tableToolbarRef" | ||||
|           v-model:hidden-search="hiddenSearchBar" | ||||
|         > | ||||
|           <Button | ||||
|             class="ml-2" | ||||
|             :icon="h(Plus)" | ||||
|             type="primary" | ||||
|             @click="onCreate" | ||||
|             v-access:code="['infra:demo03-student:create']" | ||||
|           > | ||||
|             {{ $t('ui.actionTitle.create', ['学生']) }} | ||||
|           </Button> | ||||
|         </TableToolbar> | ||||
|       </template> | ||||
|       <VxeTable ref="tableRef" :data="list" show-overflow :loading="loading"> | ||||
|         <VxeColumn field="id" title="编号" align="center" /> | ||||
|         <VxeColumn field="studentId" title="学生编号" align="center" /> | ||||
|         <VxeColumn field="name" title="名字" align="center" /> | ||||
|         <VxeColumn field="score" title="分数" align="center" /> | ||||
|         <VxeColumn field="createTime" title="创建时间" align="center"> | ||||
|           <template #default="{ row }"> | ||||
|             {{ formatDateTime(row.createTime) }} | ||||
|           </template> | ||||
|         </VxeColumn> | ||||
|         <VxeColumn field="operation" title="操作" align="center"> | ||||
|           <template #default="{ row }"> | ||||
|             <Button | ||||
|               size="small" | ||||
|               type="link" | ||||
|               @click="onEdit(row as any)" | ||||
|               v-access:code="['infra:demo03-student:update']" | ||||
|             > | ||||
|               {{ $t('ui.actionTitle.edit') }} | ||||
|             </Button> | ||||
|             <Button | ||||
|               size="small" | ||||
|               type="link" | ||||
|               danger | ||||
|               class="ml-2" | ||||
|               @click="onDelete(row as any)" | ||||
|               v-access:code="['infra:demo03-student:delete']" | ||||
|             > | ||||
|               {{ $t('ui.actionTitle.delete') }} | ||||
|             </Button> | ||||
|           </template> | ||||
|         </VxeColumn> | ||||
|       </VxeTable> | ||||
|       <!-- 分页 --> | ||||
|       <div class="mt-2 flex justify-end"> | ||||
|         <Pagination | ||||
|           :total="total" | ||||
|           v-model:current="queryParams.pageNo" | ||||
|           v-model:page-size="queryParams.pageSize" | ||||
|           show-size-changer | ||||
|           @change="getList" | ||||
|         /> | ||||
|       </div> | ||||
|     </ContentWrap> | ||||
|   </div> | ||||
| </template> | ||||
|  | @ -0,0 +1,120 @@ | |||
| <script lang="ts" setup> | ||||
| import type { Rule } from 'ant-design-vue/es/form'; | ||||
| 
 | ||||
| import type { Demo03StudentApi } from '#/api/infra/demo/demo03/erp'; | ||||
| 
 | ||||
| import { computed, ref } from 'vue'; | ||||
| 
 | ||||
| import { useVbenModal } from '@vben/common-ui'; | ||||
| 
 | ||||
| import { Form, Input, message } from 'ant-design-vue'; | ||||
| 
 | ||||
| import { | ||||
|   createDemo03Grade, | ||||
|   getDemo03Grade, | ||||
|   updateDemo03Grade, | ||||
| } from '#/api/infra/demo/demo03/erp'; | ||||
| import { $t } from '#/locales'; | ||||
| 
 | ||||
| const emit = defineEmits(['success']); | ||||
| const getTitle = computed(() => { | ||||
|   return formData.value?.id | ||||
|     ? $t('ui.actionTitle.edit', ['学生班级']) | ||||
|     : $t('ui.actionTitle.create', ['学生班级']); | ||||
| }); | ||||
| 
 | ||||
| const formRef = ref(); | ||||
| const formData = ref<Partial<Demo03StudentApi.Demo03Grade>>({ | ||||
|   id: undefined, | ||||
|   studentId: undefined, | ||||
|   name: undefined, | ||||
|   teacher: undefined, | ||||
| }); | ||||
| const rules: Record<string, Rule[]> = { | ||||
|   studentId: [{ required: true, message: '学生编号不能为空', trigger: 'blur' }], | ||||
|   name: [{ required: true, message: '名字不能为空', trigger: 'blur' }], | ||||
|   teacher: [{ required: true, message: '班主任不能为空', trigger: 'blur' }], | ||||
| }; | ||||
| 
 | ||||
| const [Modal, modalApi] = useVbenModal({ | ||||
|   async onConfirm() { | ||||
|     await formRef.value?.validate(); | ||||
| 
 | ||||
|     modalApi.lock(); | ||||
|     // 提交表单 | ||||
|     const data = formData.value as Demo03StudentApi.Demo03Grade; | ||||
|     try { | ||||
|       await (formData.value?.id | ||||
|         ? updateDemo03Grade(data) | ||||
|         : createDemo03Grade(data)); | ||||
|       // 关闭并提示 | ||||
|       await modalApi.close(); | ||||
|       emit('success'); | ||||
|       message.success({ | ||||
|         content: $t('ui.actionMessage.operationSuccess'), | ||||
|         key: 'action_process_msg', | ||||
|       }); | ||||
|     } finally { | ||||
|       modalApi.unlock(); | ||||
|     } | ||||
|   }, | ||||
|   async onOpenChange(isOpen: boolean) { | ||||
|     if (!isOpen) { | ||||
|       resetForm(); | ||||
|       return; | ||||
|     } | ||||
| 
 | ||||
|     // 加载数据 | ||||
|     let data = modalApi.getData<Demo03StudentApi.Demo03Grade>(); | ||||
|     if (!data) { | ||||
|       return; | ||||
|     } | ||||
|     if (data.id) { | ||||
|       modalApi.lock(); | ||||
|       try { | ||||
|         data = await getDemo03Grade(data.id); | ||||
|       } finally { | ||||
|         modalApi.unlock(); | ||||
|       } | ||||
|     } | ||||
|     // 设置到 values | ||||
|     formData.value = data; | ||||
|   }, | ||||
| }); | ||||
| 
 | ||||
| /** 重置表单 */ | ||||
| const resetForm = () => { | ||||
|   formData.value = { | ||||
|     id: undefined, | ||||
|     studentId: undefined, | ||||
|     name: undefined, | ||||
|     teacher: undefined, | ||||
|   }; | ||||
|   formRef.value?.resetFields(); | ||||
| }; | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <Modal :title="getTitle"> | ||||
|     <Form | ||||
|       ref="formRef" | ||||
|       :model="formData" | ||||
|       :rules="rules" | ||||
|       :label-col="{ span: 5 }" | ||||
|       :wrapper-col="{ span: 18 }" | ||||
|     > | ||||
|       <Form.Item label="学生编号" name="studentId"> | ||||
|         <Input | ||||
|           v-model:value="formData.studentId" | ||||
|           placeholder="请输入学生编号" | ||||
|         /> | ||||
|       </Form.Item> | ||||
|       <Form.Item label="名字" name="name"> | ||||
|         <Input v-model:value="formData.name" placeholder="请输入名字" /> | ||||
|       </Form.Item> | ||||
|       <Form.Item label="班主任" name="teacher"> | ||||
|         <Input v-model:value="formData.teacher" placeholder="请输入班主任" /> | ||||
|       </Form.Item> | ||||
|     </Form> | ||||
|   </Modal> | ||||
| </template> | ||||
|  | @ -0,0 +1,265 @@ | |||
| <script lang="ts" setup> | ||||
| import type { VxeTableInstance } from 'vxe-table'; | ||||
| 
 | ||||
| import type { Demo03StudentApi } from '#/api/infra/demo/demo03/erp'; | ||||
| 
 | ||||
| import { h, nextTick, onMounted, reactive, ref, watch } from 'vue'; | ||||
| 
 | ||||
| import { useVbenModal } from '@vben/common-ui'; | ||||
| import { Plus } from '@vben/icons'; | ||||
| import { cloneDeep, formatDateTime } from '@vben/utils'; | ||||
| 
 | ||||
| import { | ||||
|   Button, | ||||
|   Form, | ||||
|   Input, | ||||
|   message, | ||||
|   Pagination, | ||||
|   RangePicker, | ||||
| } from 'ant-design-vue'; | ||||
| import { VxeColumn, VxeTable } from 'vxe-table'; | ||||
| 
 | ||||
| import { | ||||
|   deleteDemo03Grade, | ||||
|   getDemo03GradePage, | ||||
| } from '#/api/infra/demo/demo03/erp'; | ||||
| import { ContentWrap } from '#/components/content-wrap'; | ||||
| import { TableToolbar } from '#/components/table-toolbar'; | ||||
| import { $t } from '#/locales'; | ||||
| import { getRangePickerDefaultProps } from '#/utils/date'; | ||||
| 
 | ||||
| import Demo03GradeForm from './demo03-grade-form.vue'; | ||||
| 
 | ||||
| const props = defineProps<{ | ||||
|   studentId?: number; // 学生编号(主表的关联字段) | ||||
| }>(); | ||||
| 
 | ||||
| const [FormModal, formModalApi] = useVbenModal({ | ||||
|   connectedComponent: Demo03GradeForm, | ||||
|   destroyOnClose: true, | ||||
| }); | ||||
| 
 | ||||
| /** 创建学生班级 */ | ||||
| function onCreate() { | ||||
|   if (!props.studentId) { | ||||
|     message.warning('请先选择一个学生!'); | ||||
|     return; | ||||
|   } | ||||
|   formModalApi.setData({ studentId: props.studentId }).open(); | ||||
| } | ||||
| 
 | ||||
| /** 编辑学生班级 */ | ||||
| function onEdit(row: Demo03StudentApi.Demo03Grade) { | ||||
|   formModalApi.setData(row).open(); | ||||
| } | ||||
| 
 | ||||
| /** 删除学生班级 */ | ||||
| async function onDelete(row: Demo03StudentApi.Demo03Grade) { | ||||
|   const hideLoading = message.loading({ | ||||
|     content: $t('ui.actionMessage.deleting', [row.id]), | ||||
|     duration: 0, | ||||
|     key: 'action_process_msg', | ||||
|   }); | ||||
|   try { | ||||
|     await deleteDemo03Grade(row.id as number); | ||||
|     message.success({ | ||||
|       content: $t('ui.actionMessage.deleteSuccess', [row.id]), | ||||
|       key: 'action_process_msg', | ||||
|     }); | ||||
|     getList(); | ||||
|   } catch { | ||||
|     hideLoading(); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| const loading = ref(true); // 列表的加载中 | ||||
| const list = ref<Demo03StudentApi.Demo03Grade[]>([]); // 列表的数据 | ||||
| const total = ref(0); // 列表的总页数 | ||||
| const queryFormRef = ref(); // 搜索的表单 | ||||
| const queryParams = reactive({ | ||||
|   pageNo: 1, | ||||
|   pageSize: 10, | ||||
|   studentId: undefined, | ||||
|   name: undefined, | ||||
|   teacher: undefined, | ||||
|   createTime: undefined, | ||||
| }); | ||||
| 
 | ||||
| /** 搜索按钮操作 */ | ||||
| const handleQuery = () => { | ||||
|   queryParams.pageNo = 1; | ||||
|   getList(); | ||||
| }; | ||||
| 
 | ||||
| /** 重置按钮操作 */ | ||||
| const resetQuery = () => { | ||||
|   queryFormRef.value.resetFields(); | ||||
|   handleQuery(); | ||||
| }; | ||||
| /** 查询列表 */ | ||||
| const getList = async () => { | ||||
|   loading.value = true; | ||||
|   try { | ||||
|     if (!props.studentId) { | ||||
|       return []; | ||||
|     } | ||||
|     const params = cloneDeep(queryParams) as any; | ||||
|     if (params.birthday && Array.isArray(params.birthday)) { | ||||
|       params.birthday = (params.birthday as string[]).join(','); | ||||
|     } | ||||
|     if (params.createTime && Array.isArray(params.createTime)) { | ||||
|       params.createTime = (params.createTime as string[]).join(','); | ||||
|     } | ||||
|     params.studentId = props.studentId; | ||||
|     const data = await getDemo03GradePage(params); | ||||
|     list.value = data.list; | ||||
|     total.value = data.total; | ||||
|   } finally { | ||||
|     loading.value = false; | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| /** 监听主表的关联字段的变化,加载对应的子表数据 */ | ||||
| watch( | ||||
|   () => props.studentId, | ||||
|   async (val) => { | ||||
|     if (!val) { | ||||
|       return; | ||||
|     } | ||||
|     await nextTick(); | ||||
|     await getList(); | ||||
|   }, | ||||
|   { immediate: true }, | ||||
| ); | ||||
| 
 | ||||
| /** 隐藏搜索栏 */ | ||||
| const hiddenSearchBar = ref(false); | ||||
| const tableToolbarRef = ref<InstanceType<typeof TableToolbar>>(); | ||||
| const tableRef = ref<VxeTableInstance>(); | ||||
| 
 | ||||
| /** 初始化 */ | ||||
| onMounted(async () => { | ||||
|   await getList(); | ||||
|   await nextTick(); | ||||
|   // 挂载 toolbar 工具栏 | ||||
|   const table = tableRef.value; | ||||
|   const tableToolbar = tableToolbarRef.value; | ||||
|   if (table && tableToolbar) { | ||||
|     await table.connect(tableToolbar.getToolbarRef()!); | ||||
|   } | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <FormModal @success="getList" /> | ||||
|   <div class="h-[600px]"> | ||||
|     <ContentWrap v-if="!hiddenSearchBar"> | ||||
|       <!-- 搜索工作栏 --> | ||||
|       <Form :model="queryParams" ref="queryFormRef" layout="inline"> | ||||
|         <Form.Item label="学生编号" name="studentId"> | ||||
|           <Input | ||||
|             v-model:value="queryParams.studentId" | ||||
|             placeholder="请输入学生编号" | ||||
|             allow-clear | ||||
|             @press-enter="handleQuery" | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item label="名字" name="name"> | ||||
|           <Input | ||||
|             v-model:value="queryParams.name" | ||||
|             placeholder="请输入名字" | ||||
|             allow-clear | ||||
|             @press-enter="handleQuery" | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item label="班主任" name="teacher"> | ||||
|           <Input | ||||
|             v-model:value="queryParams.teacher" | ||||
|             placeholder="请输入班主任" | ||||
|             allow-clear | ||||
|             @press-enter="handleQuery" | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item label="创建时间" name="createTime"> | ||||
|           <RangePicker | ||||
|             v-model:value="queryParams.createTime" | ||||
|             v-bind="getRangePickerDefaultProps()" | ||||
|             class="w-full" | ||||
|           /> | ||||
|         </Form.Item> | ||||
|         <Form.Item> | ||||
|           <Button class="ml-2" @click="resetQuery"> 重置 </Button> | ||||
|           <Button class="ml-2" @click="handleQuery" type="primary"> | ||||
|             搜索 | ||||
|           </Button> | ||||
|         </Form.Item> | ||||
|       </Form> | ||||
|     </ContentWrap> | ||||
| 
 | ||||
|     <!-- 列表 --> | ||||
|     <ContentWrap title="学生"> | ||||
|       <template #extra> | ||||
|         <TableToolbar | ||||
|           ref="tableToolbarRef" | ||||
|           v-model:hidden-search="hiddenSearchBar" | ||||
|         > | ||||
|           <Button | ||||
|             class="ml-2" | ||||
|             :icon="h(Plus)" | ||||
|             type="primary" | ||||
|             @click="onCreate" | ||||
|             v-access:code="['infra:demo03-student:create']" | ||||
|           > | ||||
|             {{ $t('ui.actionTitle.create', ['学生']) }} | ||||
|           </Button> | ||||
|         </TableToolbar> | ||||
|       </template> | ||||
|       <VxeTable ref="tableRef" :data="list" show-overflow :loading="loading"> | ||||
|         <VxeColumn field="id" title="编号" align="center" /> | ||||
|         <VxeColumn field="studentId" title="学生编号" align="center" /> | ||||
|         <VxeColumn field="name" title="名字" align="center" /> | ||||
|         <VxeColumn field="teacher" title="班主任" align="center" /> | ||||
|         <VxeColumn field="createTime" title="创建时间" align="center"> | ||||
|           <template #default="{ row }"> | ||||
|             {{ formatDateTime(row.createTime) }} | ||||
|           </template> | ||||
|         </VxeColumn> | ||||
|         <VxeColumn field="operation" title="操作" align="center"> | ||||
|           <template #default="{ row }"> | ||||
|             <Button | ||||
|               size="small" | ||||
|               type="link" | ||||
|               @click="onEdit(row as any)" | ||||
|               v-access:code="['infra:demo03-student:update']" | ||||
|             > | ||||
|               {{ $t('ui.actionTitle.edit') }} | ||||
|             </Button> | ||||
|             <Button | ||||
|               size="small" | ||||
|               type="link" | ||||
|               danger | ||||
|               class="ml-2" | ||||
|               @click="onDelete(row as any)" | ||||
|               v-access:code="['infra:demo03-student:delete']" | ||||
|             > | ||||
|               {{ $t('ui.actionTitle.delete') }} | ||||
|             </Button> | ||||
|           </template> | ||||
|         </VxeColumn> | ||||
|       </VxeTable> | ||||
|       <!-- 分页 --> | ||||
|       <div class="mt-2 flex justify-end"> | ||||
|         <Pagination | ||||
|           :total="total" | ||||
|           v-model:current="queryParams.pageNo" | ||||
|           v-model:page-size="queryParams.pageSize" | ||||
|           show-size-changer | ||||
|           @change="getList" | ||||
|         /> | ||||
|       </div> | ||||
|     </ContentWrap> | ||||
|   </div> | ||||
| </template> | ||||
|  | @ -0,0 +1,141 @@ | |||
| <script lang="ts" setup> | ||||
| import type { Rule } from 'ant-design-vue/es/form'; | ||||
| 
 | ||||
| import type { Demo03StudentApi } from '#/api/infra/demo/demo03/erp'; | ||||
| 
 | ||||
| import { computed, ref } from 'vue'; | ||||
| 
 | ||||
| import { useVbenModal } from '@vben/common-ui'; | ||||
| 
 | ||||
| import { | ||||
|   DatePicker, | ||||
|   Form, | ||||
|   Input, | ||||
|   message, | ||||
|   Radio, | ||||
|   RadioGroup, | ||||
| } from 'ant-design-vue'; | ||||
| 
 | ||||
| import { | ||||
|   createDemo03Student, | ||||
|   getDemo03Student, | ||||
|   updateDemo03Student, | ||||
| } from '#/api/infra/demo/demo03/erp'; | ||||
| import { Tinymce as RichTextarea } from '#/components/tinymce'; | ||||
| import { $t } from '#/locales'; | ||||
| import { DICT_TYPE, getDictOptions } from '#/utils/dict'; | ||||
| 
 | ||||
| const emit = defineEmits(['success']); | ||||
| 
 | ||||
| const formRef = ref(); | ||||
| const formData = ref<Partial<Demo03StudentApi.Demo03Student>>({ | ||||
|   id: undefined, | ||||
|   name: undefined, | ||||
|   sex: undefined, | ||||
|   birthday: undefined, | ||||
|   description: undefined, | ||||
| }); | ||||
| const rules: Record<string, Rule[]> = { | ||||
|   name: [{ required: true, message: '名字不能为空', trigger: 'blur' }], | ||||
|   sex: [{ required: true, message: '性别不能为空', trigger: 'blur' }], | ||||
|   birthday: [{ required: true, message: '出生日期不能为空', trigger: 'blur' }], | ||||
|   description: [{ required: true, message: '简介不能为空', trigger: 'blur' }], | ||||
| }; | ||||
| const getTitle = computed(() => { | ||||
|   return formData.value?.id | ||||
|     ? $t('ui.actionTitle.edit', ['学生']) | ||||
|     : $t('ui.actionTitle.create', ['学生']); | ||||
| }); | ||||
| 
 | ||||
| /** 重置表单 */ | ||||
| const resetForm = () => { | ||||
|   formData.value = { | ||||
|     id: undefined, | ||||
|     name: undefined, | ||||
|     sex: undefined, | ||||
|     birthday: undefined, | ||||
|     description: undefined, | ||||
|   }; | ||||
|   formRef.value?.resetFields(); | ||||
| }; | ||||
| 
 | ||||
| const [Modal, modalApi] = useVbenModal({ | ||||
|   async onConfirm() { | ||||
|     await formRef.value?.validate(); | ||||
|     modalApi.lock(); | ||||
|     // 提交表单 | ||||
|     const data = formData.value as Demo03StudentApi.Demo03Student; | ||||
|     try { | ||||
|       await (formData.value?.id | ||||
|         ? updateDemo03Student(data) | ||||
|         : createDemo03Student(data)); | ||||
|       // 关闭并提示 | ||||
|       await modalApi.close(); | ||||
|       emit('success'); | ||||
|       message.success({ | ||||
|         content: $t('ui.actionMessage.operationSuccess'), | ||||
|         key: 'action_process_msg', | ||||
|       }); | ||||
|     } finally { | ||||
|       modalApi.unlock(); | ||||
|     } | ||||
|   }, | ||||
|   async onOpenChange(isOpen: boolean) { | ||||
|     if (!isOpen) { | ||||
|       resetForm(); | ||||
|       return; | ||||
|     } | ||||
|     // 加载数据 | ||||
|     let data = modalApi.getData<Demo03StudentApi.Demo03Student>(); | ||||
|     if (!data) { | ||||
|       return; | ||||
|     } | ||||
|     if (data.id) { | ||||
|       modalApi.lock(); | ||||
|       try { | ||||
|         data = await getDemo03Student(data.id); | ||||
|       } finally { | ||||
|         modalApi.unlock(); | ||||
|       } | ||||
|     } | ||||
|     formData.value = data; | ||||
|   }, | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <Modal :title="getTitle"> | ||||
|     <Form | ||||
|       ref="formRef" | ||||
|       :model="formData" | ||||
|       :rules="rules" | ||||
|       :label-col="{ span: 5 }" | ||||
|       :wrapper-col="{ span: 18 }" | ||||
|     > | ||||
|       <Form.Item label="名字" name="name"> | ||||
|         <Input v-model:value="formData.name" placeholder="请输入名字" /> | ||||
|       </Form.Item> | ||||
|       <Form.Item label="性别" name="sex"> | ||||
|         <RadioGroup v-model:value="formData.sex"> | ||||
|           <Radio | ||||
|             v-for="dict in getDictOptions(DICT_TYPE.SYSTEM_USER_SEX, 'number')" | ||||
|             :key="dict.value" | ||||
|             :value="dict.value" | ||||
|           > | ||||
|             {{ dict.label }} | ||||
|           </Radio> | ||||
|         </RadioGroup> | ||||
|       </Form.Item> | ||||
|       <Form.Item label="出生日期" name="birthday"> | ||||
|         <DatePicker | ||||
|           v-model:value="formData.birthday" | ||||
|           value-format="x" | ||||
|           placeholder="选择出生日期" | ||||
|         /> | ||||
|       </Form.Item> | ||||
|       <Form.Item label="简介" name="description"> | ||||
|         <RichTextarea v-model="formData.description" height="500px" /> | ||||
|       </Form.Item> | ||||
|     </Form> | ||||
|   </Modal> | ||||
| </template> | ||||
		Loading…
	
		Reference in New Issue
	
	 puhui999
						puhui999