| | |
| | | <template> |
| | | <div class="organ-assessment-form"> |
| | | <el-form :model="organData" label-width="120px"> |
| | | <el-form :model="assessmentData" label-width="120px"> |
| | | <el-form-item label="功能状态"> |
| | | <el-select v-model="organData.functionStatus" :disabled="readonly"> |
| | | <el-select v-model="assessmentData.functionStatus"> |
| | | <el-option label="正常" value="1" /> |
| | | <el-option label="轻度异常" value="2" /> |
| | | <el-option label="重度异常" value="3" /> |
| | |
| | | <el-form-item label="评估意见"> |
| | | <el-input |
| | | type="textarea" |
| | | v-model="organData.assessmentOpinion" |
| | | v-model="assessmentData.assessmentOpinion" |
| | | :rows="4" |
| | | :disabled="readonly" |
| | | placeholder="请输入评估意见" |
| | | /> |
| | | </el-form-item> |
| | | |
| | | <!-- 器官级别附件管理 --> |
| | | <!-- 附件上传区域 --> |
| | | <el-form-item label="相关附件"> |
| | | <div class="attachment-section"> |
| | | <el-upload |
| | | v-if="!readonly" |
| | | class="attachment-upload" |
| | | action="/api/attachment/upload" |
| | | :headers="uploadHeaders" |
| | | :on-success="handleUploadSuccess" |
| | | :on-error="handleUploadError" |
| | | :before-upload="beforeUpload" |
| | | :show-file-list="false" |
| | | :multiple="true" |
| | | > |
| | | <el-button size="small" type="primary" icon="el-icon-upload"> |
| | | 上传附件 |
| | | </el-button> |
| | | <div slot="tip" class="el-upload__tip"> |
| | | 支持图片、文档等格式,单个文件不超过10MB |
| | | </div> |
| | | </el-upload> |
| | | |
| | | <!-- 附件列表 --> |
| | | <div v-if="organData.attachments && organData.attachments.length > 0" class="attachment-list"> |
| | | <div v-for="file in organData.attachments" :key="file.id" class="attachment-item"> |
| | | <div class="file-info"> |
| | | <i :class="getFileIcon(file.fileType)" class="file-icon"></i> |
| | | <span class="file-name" :title="file.fileName">{{ file.fileName }}</span> |
| | | <span class="file-size">({{ formatFileSize(file.fileSize) }})</span> |
| | | </div> |
| | | <div class="file-actions"> |
| | | <el-button type="text" size="mini" @click="handlePreview(file)">预览</el-button> |
| | | <el-button type="text" size="mini" @click="handleDownload(file)">下载</el-button> |
| | | <el-button |
| | | v-if="!readonly" |
| | | type="text" |
| | | size="mini" |
| | | style="color: #F56C6C;" |
| | | @click="handleDeleteFile(file)" |
| | | >删除</el-button> |
| | | </div> |
| | | </div> |
| | | <div class="attachment-header"> |
| | | <i class="el-icon-paperclip"></i> |
| | | <span class="attachment-title">附件上传</span> |
| | | <span class="attachment-tip" |
| | | >支持上传检验报告单等文件 (最多{{ attachmentLimit }}个)</span |
| | | > |
| | | </div> |
| | | |
| | | <div v-else class="no-attachment"> |
| | | <el-empty description="暂无附件" :image-size="50"></el-empty> |
| | | <!-- 使用 UploadAttachment 组件 --> |
| | | <UploadAttachment |
| | | ref="uploadAttachment" |
| | | :file-list="attachmentFileList" |
| | | :limit="attachmentLimit" |
| | | :accept="attachmentAccept" |
| | | :multiple="true" |
| | | @change="handleAttachmentChange" |
| | | @upload-success="handleUploadSuccess" |
| | | @upload-error="handleUploadError" |
| | | @remove="handleAttachmentRemove" |
| | | /> |
| | | |
| | | <!-- 附件列表 --> |
| | | <div |
| | | class="attachment-list" |
| | | v-if=" |
| | | assessmentData.attachments && |
| | | assessmentData.attachments.length > 0 |
| | | " |
| | | > |
| | | <div class="list-title"> |
| | | 已上传附件 ({{ assessmentData.attachments.length }}) |
| | | </div> |
| | | <el-table |
| | | :data="assessmentData.attachments" |
| | | style="width: 100%" |
| | | size="small" |
| | | > |
| | | <el-table-column label="文件名" min-width="200"> |
| | | <template slot-scope="scope"> |
| | | <i |
| | | class="el-icon-document" |
| | | :style="{ color: getFileIconColor(scope.row.fileName) }" |
| | | ></i> |
| | | <span class="file-name">{{ scope.row.fileName }}</span> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="文件类型" width="100"> |
| | | <template slot-scope="scope"> |
| | | <el-tag |
| | | :type="getFileTagType(scope.row.fileName)" |
| | | size="small" |
| | | > |
| | | {{ getFileTypeText(scope.row.fileName) }} |
| | | </el-tag> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="上传时间" width="160"> |
| | | <template slot-scope="scope"> |
| | | <span>{{ formatDateTime(scope.row.uploadTime) }}</span> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="文件大小" width="100"> |
| | | <template slot-scope="scope"> |
| | | <span>{{ formatFileSize(scope.row.fileSize) }}</span> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="操作" width="266" fixed="right"> |
| | | <template slot-scope="scope"> |
| | | <el-button |
| | | size="mini" |
| | | type="primary" |
| | | @click="handlePreview(scope.row)" |
| | | :disabled="!isPreviewable(scope.row.fileName)" |
| | | > |
| | | 预览 |
| | | </el-button> |
| | | <el-button |
| | | size="mini" |
| | | type="success" |
| | | @click="handleDownload(scope.row)" |
| | | > |
| | | 下载 |
| | | </el-button> |
| | | <el-button |
| | | v-if="!readonly" |
| | | size="mini" |
| | | type="danger" |
| | | @click="handleRemoveAttachment(scope.$index)" |
| | | > |
| | | 删除 |
| | | </el-button> |
| | | </template> |
| | | </el-table-column> |
| | | </el-table> |
| | | </div> |
| | | </div> |
| | | </el-form-item> |
| | |
| | | <el-form-item v-if="!readonly"> |
| | | <el-button type="primary" @click="handleSave">保存评估</el-button> |
| | | <el-button @click="handleCancel">取消</el-button> |
| | | <el-button |
| | | type="success" |
| | | @click="handleAddAssessment" |
| | | icon="el-icon-plus" |
| | | > |
| | | 新增评估记录 |
| | | </el-button> |
| | | </el-form-item> |
| | | </el-form> |
| | | |
| | | <!-- 附件预览弹窗 --> |
| | | <el-dialog |
| | | :title="`附件预览 - ${currentFile.fileName}`" |
| | | :visible.sync="previewVisible" |
| | | width="60%" |
| | | > |
| | | <div v-if="isImage(currentFile.fileType)" class="image-preview"> |
| | | <img :src="currentFile.fileUrl" alt="预览图片" style="max-width: 100%;" /> |
| | | </div> |
| | | <div v-else class="file-preview"> |
| | | <el-alert |
| | | title="该文件格式不支持在线预览,请下载后查看" |
| | | type="info" |
| | | show-icon |
| | | /> |
| | | <div style="text-align: center; margin-top: 20px;"> |
| | | <el-button type="primary" @click="handleDownload(currentFile)"> |
| | | <i class="el-icon-download"></i> 下载文件 |
| | | </el-button> |
| | | </div> |
| | | </div> |
| | | </el-dialog> |
| | | <!-- 文件预览弹窗 --> |
| | | <FilePreviewDialog |
| | | :visible="previewVisible" |
| | | :file="currentPreviewFile" |
| | | @close="previewVisible = false" |
| | | @download="handleDownload" |
| | | /> |
| | | </div> |
| | | </template> |
| | | |
| | | <script> |
| | | import { getToken } from '@/utils/auth' |
| | | import UploadAttachment from "@/components/UploadAttachment"; |
| | | import FilePreviewDialog from "@/components/FilePreviewDialog"; |
| | | import dayjs from "dayjs"; |
| | | |
| | | export default { |
| | | name: "OrganAssessmentForm", |
| | | components: { |
| | | UploadAttachment, |
| | | FilePreviewDialog |
| | | }, |
| | | props: { |
| | | organData: { |
| | | type: Object, |
| | | default: () => ({}) |
| | | }, |
| | | assessmentData: { |
| | | type: Object, |
| | | default: () => ({ |
| | | attachments: [] // 确保有附件数组 |
| | | functionStatus: "", |
| | | assessmentOpinion: "", |
| | | attachments: [] |
| | | }) |
| | | }, |
| | | assessmentIndex: { |
| | | type: Number, |
| | | default: 0 |
| | | }, |
| | | readonly: { |
| | | type: Boolean, |
| | |
| | | }, |
| | | data() { |
| | | return { |
| | | // 预览相关 |
| | | previewVisible: false, |
| | | currentFile: {}, |
| | | uploadHeaders: { |
| | | Authorization: 'Bearer ' + getToken() |
| | | } |
| | | currentPreviewFile: null, |
| | | |
| | | // 附件相关配置 |
| | | attachmentLimit: 10, |
| | | attachmentAccept: |
| | | ".pdf,.jpg,.jpeg,.png,.doc,.docx,.xls,.xlsx,.ppt,.pptx,.txt", |
| | | attachmentFileList: [] |
| | | }; |
| | | }, |
| | | watch: { |
| | | organData: { |
| | | handler(newVal, oldVal) { |
| | | // 只有当附件数据真正发生变化时才初始化 |
| | | if ( |
| | | !oldVal || |
| | | !oldVal.attachments || |
| | | JSON.stringify(newVal.attachments) !== |
| | | JSON.stringify(oldVal.attachments) |
| | | ) { |
| | | this.initAttachmentList(); |
| | | } |
| | | }, |
| | | immediate: true, |
| | | deep: true |
| | | } |
| | | }, |
| | | methods: { |
| | | /** 初始化附件列表 */ |
| | | initAttachmentList() { |
| | | if (this.organData.attachments && this.organData.attachments.length > 0) { |
| | | this.assessmentData.attachments = [...this.organData.attachments]; |
| | | this.attachmentFileList = this.organData.attachments.map(item => ({ |
| | | uid: item.id || Math.random(), |
| | | name: item.fileName, |
| | | url: item.path || item.fileUrl, |
| | | status: "success" |
| | | })); |
| | | } else { |
| | | this.assessmentData.attachments = []; |
| | | this.attachmentFileList = []; |
| | | } |
| | | }, |
| | | handleSave() { |
| | | this.$emit('save', { |
| | | ...this.organData, |
| | | assessmentTime: new Date().toISOString(), |
| | | assessor: '当前用户' |
| | | }); |
| | | const saveData = { |
| | | organData: this.organData, |
| | | assessmentData: this.assessmentData, |
| | | assessmentIndex: this.assessmentIndex |
| | | }; |
| | | this.$emit("save", saveData); |
| | | }, |
| | | |
| | | handleCancel() { |
| | | this.$emit('cancel'); |
| | | this.$emit("cancel"); |
| | | }, |
| | | |
| | | // 文件上传处理 |
| | | handleUploadSuccess(response, file) { |
| | | if (response.code === 200) { |
| | | // 将新文件添加到附件列表 |
| | | const newFile = { |
| | | id: response.data.fileId, |
| | | handleAddAssessment() { |
| | | this.$emit("add-assessment", { |
| | | organData: this.organData, |
| | | currentIndex: this.assessmentIndex |
| | | }); |
| | | }, |
| | | /** 附件变化处理 */ |
| | | handleAttachmentChange(fileList) { |
| | | this.attachmentFileList = fileList; |
| | | }, |
| | | |
| | | /** 附件移除处理 */ |
| | | handleAttachmentRemove(file) { |
| | | if (file.url) { |
| | | const index = this.assessmentData.attachments.findIndex( |
| | | item => item.path === file.url || item.fileUrl === file.url |
| | | ); |
| | | if (index > -1) { |
| | | this.assessmentData.attachments.splice(index, 1); |
| | | } |
| | | } |
| | | }, |
| | | |
| | | /** 手动删除附件 */ |
| | | handleRemoveAttachment(index) { |
| | | this.assessmentData.attachments.splice(index, 1); |
| | | this.attachmentFileList.splice(index, 1); |
| | | this.$message.success("附件删除成功"); |
| | | }, |
| | | |
| | | /** 上传成功处理 */ |
| | | handleUploadSuccess({ file, fileList, response }) { |
| | | console.log(response); |
| | | |
| | | if (response.code == 200) { |
| | | console.log(1); |
| | | |
| | | const attachmentObj = { |
| | | // id: |
| | | // response.data.fileId || |
| | | // Math.random() |
| | | // .toString(36) |
| | | // .substr(2), |
| | | fileName: file.name, |
| | | path: response.data.fileUrl || file.url, |
| | | fileUrl: response.data.fileUrl || file.url, |
| | | fileType: this.getFileExtension(file.name), |
| | | fileSize: file.size, |
| | | fileUrl: response.data.fileUrl, |
| | | uploadTime: new Date().toISOString() |
| | | uploadTime: dayjs().format("YYYY-MM-DD HH:mm:ss"), |
| | | assessmentType: this.assessmentType, |
| | | organType: this.organData.organType |
| | | }; |
| | | console.log(2,attachmentObj); |
| | | |
| | | if (!this.organData.attachments) { |
| | | this.organData.attachments = []; |
| | | } |
| | | this.organData.attachments.push(newFile); |
| | | |
| | | this.$message.success('文件上传成功'); |
| | | this.assessmentData.attachments.push(attachmentObj); |
| | | this.$message.success("文件上传成功"); |
| | | } |
| | | }, |
| | | |
| | | handleUploadError(error) { |
| | | console.error('文件上传失败:', error); |
| | | this.$message.error('文件上传失败'); |
| | | /** 上传错误处理 */ |
| | | handleUploadError({ file, fileList, error }) { |
| | | console.error("附件上传失败:", error); |
| | | this.$message.error("文件上传失败,请重试"); |
| | | }, |
| | | |
| | | beforeUpload(file) { |
| | | const isLt10M = file.size / 1024 / 1024 < 10; |
| | | if (!isLt10M) { |
| | | this.$message.error('文件大小不能超过 10MB'); |
| | | return false; |
| | | } |
| | | return true; |
| | | }, |
| | | |
| | | // 文件操作 |
| | | /** 文件预览 */ |
| | | handlePreview(file) { |
| | | this.currentFile = file; |
| | | this.currentPreviewFile = { |
| | | fileName: file.fileName, |
| | | fileUrl: file.path || file.fileUrl, |
| | | fileType: this.getFileType(file.fileName) |
| | | }; |
| | | this.previewVisible = true; |
| | | }, |
| | | |
| | | /** 文件下载 */ |
| | | handleDownload(file) { |
| | | // 模拟文件下载 |
| | | const link = document.createElement('a'); |
| | | link.href = file.fileUrl; |
| | | link.download = file.fileName; |
| | | link.click(); |
| | | this.$message.success('开始下载文件'); |
| | | const fileUrl = file.path || file.fileUrl; |
| | | const fileName = file.fileName; |
| | | |
| | | if (fileUrl) { |
| | | const link = document.createElement("a"); |
| | | link.href = fileUrl; |
| | | link.download = fileName; |
| | | link.style.display = "none"; |
| | | document.body.appendChild(link); |
| | | link.click(); |
| | | document.body.removeChild(link); |
| | | this.$message.success("开始下载文件"); |
| | | } else { |
| | | this.$message.warning("文件路径不存在,无法下载"); |
| | | } |
| | | }, |
| | | |
| | | handleDeleteFile(file) { |
| | | this.$confirm('确定要删除这个附件吗?', '提示', { |
| | | confirmButtonText: '确定', |
| | | cancelButtonText: '取消', |
| | | type: 'warning' |
| | | }).then(() => { |
| | | const index = this.organData.attachments.findIndex(f => f.id === file.id); |
| | | if (index !== -1) { |
| | | this.organData.attachments.splice(index, 1); |
| | | this.$message.success('删除成功'); |
| | | } |
| | | }); |
| | | /** 获取文件类型 */ |
| | | getFileType(fileName) { |
| | | if (!fileName) return "other"; |
| | | |
| | | const extension = fileName |
| | | .split(".") |
| | | .pop() |
| | | .toLowerCase(); |
| | | const imageTypes = ["jpg", "jpeg", "png", "gif", "bmp", "webp"]; |
| | | const pdfTypes = ["pdf"]; |
| | | const officeTypes = ["doc", "docx", "xls", "xlsx", "ppt", "pptx"]; |
| | | |
| | | if (imageTypes.includes(extension)) return "image"; |
| | | if (pdfTypes.includes(extension)) return "pdf"; |
| | | if (officeTypes.includes(extension)) return "office"; |
| | | return "other"; |
| | | }, |
| | | |
| | | // 工具方法 |
| | | getFileIcon(fileType) { |
| | | const iconMap = { |
| | | 'pdf': 'el-icon-document', |
| | | 'doc': 'el-icon-document', |
| | | 'docx': 'el-icon-document', |
| | | 'xls': 'el-icon-document', |
| | | 'xlsx': 'el-icon-document', |
| | | 'jpg': 'el-icon-picture', |
| | | 'jpeg': 'el-icon-picture', |
| | | 'png': 'el-icon-picture', |
| | | 'gif': 'el-icon-picture' |
| | | /** 获取文件图标颜色 */ |
| | | getFileIconColor(fileName) { |
| | | const type = this.getFileType(fileName); |
| | | const colorMap = { |
| | | image: "#67C23A", |
| | | pdf: "#F56C6C", |
| | | office: "#409EFF", |
| | | other: "#909399" |
| | | }; |
| | | return iconMap[fileType] || 'el-icon-document'; |
| | | return colorMap[type] || "#909399"; |
| | | }, |
| | | |
| | | /** 获取文件标签类型 */ |
| | | getFileTagType(fileName) { |
| | | const type = this.getFileType(fileName); |
| | | const typeMap = { |
| | | image: "success", |
| | | pdf: "danger", |
| | | office: "primary", |
| | | other: "info" |
| | | }; |
| | | return typeMap[type] || "info"; |
| | | }, |
| | | |
| | | /** 获取文件类型文本 */ |
| | | getFileTypeText(fileName) { |
| | | const type = this.getFileType(fileName); |
| | | const textMap = { |
| | | image: "图片", |
| | | pdf: "PDF", |
| | | office: "文档", |
| | | other: "其他" |
| | | }; |
| | | return textMap[type] || "未知"; |
| | | }, |
| | | |
| | | /** 检查是否可预览 */ |
| | | isPreviewable(fileName) { |
| | | const type = this.getFileType(fileName); |
| | | return ["image", "pdf"].includes(type); |
| | | }, |
| | | |
| | | /** 获取文件扩展名 */ |
| | | getFileExtension(filename) { |
| | | return filename.split('.').pop().toLowerCase(); |
| | | return filename |
| | | .split(".") |
| | | .pop() |
| | | .toLowerCase(); |
| | | }, |
| | | |
| | | /** 格式化文件大小 */ |
| | | formatFileSize(bytes) { |
| | | if (bytes === 0) return '0 B'; |
| | | if (!bytes || bytes === 0) return "0 B"; |
| | | const k = 1024; |
| | | const sizes = ['B', 'KB', 'MB', 'GB']; |
| | | const sizes = ["B", "KB", "MB", "GB"]; |
| | | const i = Math.floor(Math.log(bytes) / Math.log(k)); |
| | | return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i]; |
| | | return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + " " + sizes[i]; |
| | | }, |
| | | |
| | | isImage(fileType) { |
| | | const imageTypes = ['jpg', 'jpeg', 'png', 'gif', 'bmp']; |
| | | return imageTypes.includes(fileType); |
| | | /** 日期时间格式化 */ |
| | | formatDateTime(dateTime) { |
| | | if (!dateTime) return ""; |
| | | |
| | | try { |
| | | const date = new Date(dateTime); |
| | | if (isNaN(date.getTime())) return dateTime; |
| | | |
| | | const year = date.getFullYear(); |
| | | const month = String(date.getMonth() + 1).padStart(2, "0"); |
| | | const day = String(date.getDate()).padStart(2, "0"); |
| | | const hours = String(date.getHours()).padStart(2, "0"); |
| | | const minutes = String(date.getMinutes()).padStart(2, "0"); |
| | | |
| | | return `${year}-${month}-${day} ${hours}:${minutes}`; |
| | | } catch (error) { |
| | | return dateTime; |
| | | } |
| | | } |
| | | } |
| | | } |
| | | }; |
| | | </script> |
| | | |
| | | <style scoped> |
| | |
| | | border: 1px solid #ebeef5; |
| | | border-radius: 4px; |
| | | padding: 15px; |
| | | background: #fafafa; |
| | | } |
| | | |
| | | .attachment-upload { |
| | | margin-bottom: 15px; |
| | | .attachment-header { |
| | | display: flex; |
| | | align-items: center; |
| | | margin-bottom: 16px; |
| | | padding-bottom: 8px; |
| | | border-bottom: 1px solid #ebeef5; |
| | | } |
| | | |
| | | .attachment-title { |
| | | font-weight: bold; |
| | | margin: 0 8px; |
| | | color: #303133; |
| | | } |
| | | |
| | | .attachment-tip { |
| | | font-size: 12px; |
| | | color: #909399; |
| | | margin-left: auto; |
| | | } |
| | | |
| | | .attachment-list { |
| | | max-height: 300px; |
| | | overflow-y: auto; |
| | | margin-top: 16px; |
| | | } |
| | | |
| | | .attachment-item { |
| | | display: flex; |
| | | justify-content: space-between; |
| | | align-items: center; |
| | | padding: 8px 12px; |
| | | border-bottom: 1px solid #f0f0f0; |
| | | } |
| | | |
| | | .attachment-item:last-child { |
| | | border-bottom: none; |
| | | } |
| | | |
| | | .file-info { |
| | | display: flex; |
| | | align-items: center; |
| | | flex: 1; |
| | | } |
| | | |
| | | .file-icon { |
| | | margin-right: 8px; |
| | | font-size: 16px; |
| | | color: #409EFF; |
| | | .list-title { |
| | | font-weight: bold; |
| | | margin-bottom: 12px; |
| | | color: #303133; |
| | | font-size: 14px; |
| | | } |
| | | |
| | | .file-name { |
| | | flex: 1; |
| | | overflow: hidden; |
| | | text-overflow: ellipsis; |
| | | white-space: nowrap; |
| | | margin-right: 8px; |
| | | font-size: 13px; |
| | | margin-left: 8px; |
| | | } |
| | | |
| | | .file-size { |
| | | color: #909399; |
| | | ::v-deep .el-upload__tip { |
| | | font-size: 12px; |
| | | } |
| | | |
| | | .file-actions { |
| | | flex-shrink: 0; |
| | | } |
| | | |
| | | .no-attachment { |
| | | text-align: center; |
| | | padding: 20px; |
| | | color: #909399; |
| | | margin-top: 8px; |
| | | } |
| | | |
| | | ::v-deep .el-table .cell { |
| | | padding: 8px 12px; |
| | | } |
| | | |
| | | ::v-deep .el-button--mini { |
| | | padding: 5px 10px; |
| | | font-size: 12px; |
| | | } |
| | | </style> |