feat(meeting): 新增会议编辑全页面 MeetingEdit.vue,替代弹窗形式

- 新增 MeetingEdit.vue,容器改为 ContentWrap 全页面
- 增加组织单位(选填)字段
- 模式由路由参数决定:无 id=新建,有 id=编辑,query.mode=view=查看
- 底部操作区:保存草稿 + 返回按钮
- 在 remaining.ts 中新增 ReviewMeetingEdit / ReviewMeetingEditById 两条路由
- 修正 index.vue 中 goToEdit 路由名对应关系

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
pull/874/head
Codewoc 2026-03-23 13:57:14 +08:00
parent 83adecdcb8
commit bc8258a149
3 changed files with 434 additions and 1 deletions

View File

@ -747,6 +747,30 @@ const remainingRouter: AppRouteRecordRaw[] = [
activeMenu: '/review/meeting'
},
component: () => import('@/views/review/meeting/ProjectList.vue')
},
{
path: 'review-meeting/edit',
name: 'ReviewMeetingEdit',
meta: {
title: '新建会议',
noCache: true,
hidden: true,
canTo: true,
activeMenu: '/review/meeting'
},
component: () => import('@/views/review/meeting/MeetingEdit.vue')
},
{
path: 'review-meeting/edit/:id(\\d+)',
name: 'ReviewMeetingEditById',
meta: {
title: '编辑会议',
noCache: true,
hidden: true,
canTo: true,
activeMenu: '/review/meeting'
},
component: () => import('@/views/review/meeting/MeetingEdit.vue')
}
]
},

View File

@ -0,0 +1,405 @@
<template>
<ContentWrap>
<div class="meeting-edit-header">
<span class="page-title">{{ pageTitle }}</span>
</div>
<el-form
ref="formRef"
:model="formData"
:rules="isView ? {} : rules"
label-width="100px"
v-loading="formLoading"
>
<!-- 基本信息 -->
<el-divider content-position="left">基本信息</el-divider>
<el-row :gutter="16">
<el-col :span="12">
<el-form-item label="会议名称" prop="name">
<el-input v-model="formData.name" placeholder="请输入会议名称" :disabled="isView" />
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="组织单位" prop="organizationUnit">
<el-input v-model="formData.organizationUnit" placeholder="请输入组织单位" :disabled="isView" />
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="16">
<el-col :span="12">
<el-form-item label="会议时间" prop="meetingTimeRange">
<el-date-picker
v-model="formData.meetingTimeRange"
type="datetimerange"
range-separator="至"
start-placeholder="开始时间"
end-placeholder="结束时间"
format="YYYY-MM-DD HH:mm"
value-format="x"
:disabled="isView"
style="width: 100%"
/>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="会议地点" prop="location">
<el-input v-model="formData.location" placeholder="请输入会议地点" :disabled="isView" />
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="16">
<el-col :span="12">
<el-form-item label="资料查看时限" prop="materialViewTimeRange">
<el-date-picker
v-model="formData.materialViewTimeRange"
type="datetimerange"
range-separator="至"
start-placeholder="开始时间"
end-placeholder="结束时间"
format="YYYY-MM-DD HH:mm"
value-format="x"
:disabled="isView"
style="width: 100%"
/>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="议程附件">
<div class="agenda-attachment-wrap">
<el-upload
v-if="!isView"
:auto-upload="false"
:show-file-list="false"
accept=".pdf,.png,.jpg,.jpeg,.gif,.bmp,.webp"
:on-change="handleAgendaAttachmentChange"
>
<el-button type="primary" plain size="small">上传议程附件</el-button>
</el-upload>
<el-text type="info" size="small">仅支持图片或 PDF固定单附件</el-text>
<div v-if="formData.agendaAttachmentUrl" class="agenda-file-line">
<el-link type="primary" :underline="false" @click="previewAgendaAttachment">
{{ formData.agendaAttachmentName }}
</el-link>
<el-tag size="small">{{ (formData.agendaAttachmentType || '').toUpperCase() }}</el-tag>
<el-text type="info" size="small">{{ formatFileSize(formData.agendaAttachmentSize) }}</el-text>
<el-button v-if="!isView" type="danger" link @click="clearAgendaAttachment"></el-button>
</div>
</div>
</el-form-item>
</el-col>
</el-row>
<!-- 评审专家 -->
<el-divider content-position="left">评审专家</el-divider>
<el-row :gutter="16">
<el-col :span="24">
<el-form-item label="参会专家" prop="expertIds">
<el-select
v-model="formData.expertIds"
multiple
filterable
placeholder="请选择参会专家"
style="width: 100%"
:disabled="isView"
>
<el-option
v-for="expert in expertOptions"
:key="expert.id"
:label="expertLabel(expert)"
:value="expert.id"
/>
</el-select>
</el-form-item>
</el-col>
</el-row>
<!-- 评审项目 -->
<el-divider content-position="left">评审项目</el-divider>
<div v-if="!isView" class="import-section">
<el-upload
:auto-upload="false"
:on-change="handleExcelChange"
:show-file-list="false"
accept=".xls,.xlsx"
>
<el-button type="primary" plain>导入验收申请 Excel</el-button>
</el-upload>
<el-button type="success" plain @click="handleDownloadTemplate"></el-button>
<el-text type="info" size="small">格式序号开始时间结束时间议程分类项目标题汇报人报告人单位</el-text>
</div>
<div v-if="formData.projects && formData.projects.length > 0" class="mt-10">
<el-table :data="formData.projects" border size="small" max-height="360">
<el-table-column label="序号" prop="seqNo" width="60" align="center" />
<el-table-column label="开始时间" prop="startTime" width="80" align="center" />
<el-table-column label="结束时间" prop="endTime" width="80" align="center" />
<el-table-column label="议程分类" prop="agendaCategory" width="110" />
<el-table-column label="评审项目标题" prop="projectTitle" show-overflow-tooltip />
<el-table-column label="汇报人" prop="reporter" width="80" />
<el-table-column label="报告单位" prop="reporterUnit" show-overflow-tooltip width="130" />
</el-table>
</div>
<el-empty v-else-if="isView" description="暂无评审项目" :image-size="60" />
</el-form>
<!-- 底部操作区 -->
<div class="form-footer">
<el-button v-if="!isView" type="primary" :loading="formLoading" @click="submitForm">稿</el-button>
<el-button @click="handleBack"></el-button>
</div>
</ContentWrap>
</template>
<script setup lang="ts">
import { ref, reactive, computed, onMounted } from 'vue'
import { useRouter, useRoute } from 'vue-router'
import { ElMessage, ElMessageBox } from 'element-plus'
import type { FormRules } from 'element-plus'
import type { UploadFile } from 'element-plus'
import {
createReviewMeeting,
updateReviewMeeting,
getReviewMeeting,
importProjectsFromExcel,
getImportTemplate,
uploadAgendaAttachment,
type ReviewMeetingSaveReqVO,
type ReviewProjectItemVO
} from '@/api/review/meeting'
import { getExpertUserList } from '@/api/system/user/index'
import download from '@/utils/download'
defineOptions({ name: 'ReviewMeetingEdit' })
const router = useRouter()
const route = useRoute()
const formLoading = ref(false)
const expertOptions = ref<any[]>([])
const isProjectsModified = ref(false)
//
const meetingId = computed(() => {
const id = route.params.id
return id ? Number(id) : undefined
})
const isView = computed(() => route.query.mode === 'view')
const isEdit = computed(() => !!meetingId.value && !isView.value)
const pageTitle = computed(() => {
if (isView.value) return '查看会议'
return isEdit.value ? '编辑会议' : '新建会议'
})
type FormData = ReviewMeetingSaveReqVO & {
organizationUnit?: string
meetingTimeRange?: any[]
materialViewTimeRange?: any[]
}
const formData = reactive<FormData>({
id: undefined,
name: '',
organizationUnit: undefined,
startTime: undefined,
endTime: undefined,
location: '',
agendaAttachmentName: undefined,
agendaAttachmentUrl: undefined,
agendaAttachmentType: undefined,
agendaAttachmentSize: undefined,
materialViewStartTime: undefined,
materialViewEndTime: undefined,
materialViewRemark: undefined,
expertIds: [],
meetingTimeRange: undefined,
materialViewTimeRange: undefined,
projects: []
})
const rules: FormRules = {
name: [{ required: true, message: '会议名称不能为空', trigger: 'blur' }],
meetingTimeRange: [{ required: true, message: '会议时间不能为空', trigger: 'change' }],
location: [{ required: true, message: '会议地点不能为空', trigger: 'blur' }],
expertIds: [{ required: true, type: 'array', min: 1, message: '至少选择一位专家', trigger: 'change' }]
}
const formRef = ref()
const expertLabel = (e: any) =>
`${e.nickname}${e.title ? `${e.title}` : ''}${e.deptName ? ` ${e.deptName}` : ''}`
const loadDetail = async (id: number) => {
formLoading.value = true
try {
const detail = await getReviewMeeting(id)
Object.assign(formData, detail)
if (detail.startTime && detail.endTime) {
formData.meetingTimeRange = [
new Date(detail.startTime.replace(' ', 'T')).getTime(),
new Date(detail.endTime.replace(' ', 'T')).getTime()
]
}
if (detail.materialViewStartTime && detail.materialViewEndTime) {
formData.materialViewTimeRange = [
new Date(detail.materialViewStartTime.replace(' ', 'T')).getTime(),
new Date(detail.materialViewEndTime.replace(' ', 'T')).getTime()
]
}
} finally {
formLoading.value = false
}
}
onMounted(async () => {
expertOptions.value = await getExpertUserList().catch(() => [])
if (meetingId.value) {
await loadDetail(meetingId.value)
}
})
const handleExcelChange = async (uploadFile: UploadFile) => {
if (!uploadFile.raw) return
if (formData.projects && formData.projects.length > 0) {
await ElMessageBox.confirm('重新导入将覆盖已有评审项目列表,是否继续?', '提示', { type: 'warning' })
}
formLoading.value = true
try {
const result = await importProjectsFromExcel(uploadFile.raw)
formData.projects = result as ReviewProjectItemVO[]
isProjectsModified.value = true
ElMessage.success(`成功解析 ${formData.projects.length} 个评审项目`)
} catch {
ElMessage.error('Excel 解析失败,请检查文件格式')
} finally {
formLoading.value = false
}
}
const handleDownloadTemplate = async () => {
try {
const data = await getImportTemplate()
download.excel(data, '评审项目导入模板.xls')
} catch {
ElMessage.error('下载模板失败')
}
}
const handleAgendaAttachmentChange = async (uploadFile: UploadFile) => {
if (!uploadFile.raw) return
const ext = uploadFile.raw.name.includes('.')
? uploadFile.raw.name.substring(uploadFile.raw.name.lastIndexOf('.') + 1).toLowerCase()
: ''
const isPdf = ext === 'pdf' || uploadFile.raw.type === 'application/pdf'
const isImage = uploadFile.raw.type?.startsWith('image/')
if (!isPdf && !isImage) {
ElMessage.error('议程附件仅支持图片或 PDF')
return
}
formLoading.value = true
try {
const attachment = await uploadAgendaAttachment(uploadFile.raw)
formData.agendaAttachmentName = attachment.name
formData.agendaAttachmentUrl = attachment.url
formData.agendaAttachmentType = attachment.type
formData.agendaAttachmentSize = attachment.size
ElMessage.success('议程附件上传成功')
} finally {
formLoading.value = false
}
}
const clearAgendaAttachment = () => {
formData.agendaAttachmentName = undefined
formData.agendaAttachmentUrl = undefined
formData.agendaAttachmentType = undefined
formData.agendaAttachmentSize = undefined
}
const previewAgendaAttachment = () => {
if (formData.agendaAttachmentUrl) {
window.open(formData.agendaAttachmentUrl, '_blank')
}
}
const formatFileSize = (bytes?: number): string => {
if (!bytes) return '-'
if (bytes < 1024) return `${bytes} B`
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`
return `${(bytes / (1024 * 1024)).toFixed(1)} MB`
}
const submitForm = async () => {
const valid = await formRef.value?.validate().catch(() => false)
if (!valid) return
if (formData.meetingTimeRange?.length === 2) {
formData.startTime = formData.meetingTimeRange[0]
formData.endTime = formData.meetingTimeRange[1]
}
if (formData.materialViewTimeRange?.length === 2) {
formData.materialViewStartTime = formData.materialViewTimeRange[0]
formData.materialViewEndTime = formData.materialViewTimeRange[1]
} else if (formData.meetingTimeRange?.length === 2) {
formData.materialViewStartTime = formData.meetingTimeRange[0]
formData.materialViewEndTime = formData.meetingTimeRange[1]
} else {
formData.materialViewStartTime = undefined
formData.materialViewEndTime = undefined
}
formLoading.value = true
try {
const submitData = { ...formData }
if (isEdit.value && !isProjectsModified.value) {
delete submitData.projects
}
if (isEdit.value) {
await updateReviewMeeting(submitData)
ElMessage.success('更新成功')
} else {
await createReviewMeeting(submitData)
ElMessage.success('创建成功,会议已保存为草稿')
}
router.push({ name: 'ReviewMeeting' })
} finally {
formLoading.value = false
}
}
const handleBack = () => {
router.back()
}
</script>
<style scoped>
.meeting-edit-header {
margin-bottom: 16px;
}
.page-title {
font-size: 16px;
font-weight: 600;
color: var(--el-text-color-primary);
}
.import-section {
margin-bottom: 10px;
display: flex;
align-items: center;
gap: 8px;
flex-wrap: wrap;
}
.mt-10 {
margin-top: 10px;
}
.agenda-attachment-wrap {
display: flex;
flex-direction: column;
gap: 6px;
}
.agenda-file-line {
display: flex;
align-items: center;
gap: 8px;
}
.form-footer {
margin-top: 24px;
display: flex;
gap: 10px;
}
</style>

View File

@ -177,7 +177,11 @@ const handleQuery = () => { queryParams.pageNo = 1; getList() }
const resetQuery = () => { queryFormRef.value?.resetFields(); handleQuery() }
const goToEdit = (id?: number, mode?: string) => {
router.push({ name: 'ReviewMeetingEdit', params: id ? { id } : {}, query: mode ? { mode } : {} })
if (id) {
router.push({ name: 'ReviewMeetingEditById', params: { id }, query: mode ? { mode } : {} })
} else {
router.push({ name: 'ReviewMeetingEdit' })
}
}
const goToProjectList = (row: ReviewMeetingRespVO) => {