| <template> | 
|   <el-dialog | 
|     title="添加表数据" | 
|     :visible.sync="dialogVisible" | 
|     width="60%" | 
|     :close-on-click-modal="false" | 
|     @close="handleClose" | 
|   > | 
|     <div class="sample-dialog"> | 
|       <div class="sample-content"> | 
|         <div class="form-content"> | 
|           <el-form ref="form" :model="form" :rules="rules" label-position="top"> | 
|             <el-row :gutter="20"> | 
|               <el-col :span="12"> | 
|                 <el-form-item label="更新时间" prop="updateTime"> | 
|                   <el-input | 
|                     v-model="form.updateTime" | 
|                     placeholder="更新时间" | 
|                     disabled | 
|                   /> | 
|                 </el-form-item> | 
|               </el-col> | 
|               <el-col | 
|                 :span="12" | 
|                 v-for="(header, index) in headerList" | 
|                 :key="index" | 
|               > | 
|                 <el-form-item | 
|                   :label="header.name" | 
|                   :prop="header.name" | 
|                   :rules="{ | 
|                     required: header.required === true || header.required === 'true', | 
|                     message: header.message || `请输入${header.name}`, | 
|                     trigger: ['blur', 'change'] | 
|                   }" | 
|                   v-if="header.type == 'text'" | 
|                 > | 
|                   <el-input | 
|                     v-model="form[header.name]" | 
|                     :placeholder="'请输入' + header.name" | 
|                     :disabled="!checkEditPermission(header)" | 
|                   /> | 
|                 </el-form-item> | 
|                 <el-form-item | 
|                   :label="header.name" | 
|                   :prop="header.name" | 
|                   :rules="{ | 
|                     required: header.required === true || header.required === 'true', | 
|                     message: header.message || `请输入${header.name}`, | 
|                     trigger: ['blur', 'change'] | 
|                   }" | 
|                   v-if="header.type == 'image'" | 
|                 > | 
|                   <el-upload | 
|                     class="upload-demo" | 
|                     action="#" | 
|                     :file-list="spectrumList" | 
|                     :auto-upload="false" | 
|                     list-type="picture-card" | 
|                     :on-change="handleSpectrumChange" | 
|                     :on-remove="handleSpectrumRemove" | 
|                     :disabled="!checkEditPermission(header)" | 
|                   > | 
|                     <i class="el-icon-plus"></i> | 
|                     <!-- <div slot="tip" class="el-upload__tip">暂未连接服务器,使用默认图片</div> --> | 
|                   </el-upload> | 
|                 </el-form-item> | 
|                 <el-form-item | 
|                   :label="header.name" | 
|                   :prop="header.name" | 
|                   :rules="{ | 
|                     required: header.required === true || header.required === 'true', | 
|                     message: header.message || `请输入${header.name}`, | 
|                     trigger: ['blur', 'change'] | 
|                   }" | 
|                   v-if="header.type == 'date'" | 
|                 > | 
|                   <el-date-picker | 
|                     v-model="form[header.name]" | 
|                     type="datetime" | 
|                     placeholder="选择日期时间" | 
|                     value-format="yyyy-MM-dd HH:mm:ss" | 
|                     :disabled="!checkEditPermission(header)" | 
|                     :picker-options="{ | 
|                       shortcuts: [{ | 
|                         text: '今天', | 
|                         onClick(picker) { | 
|                           picker.$emit('pick', new Date()); | 
|                         } | 
|                       }, { | 
|                         text: '昨天', | 
|                         onClick(picker) { | 
|                           const date = new Date(); | 
|                           date.setTime(date.getTime() - 3600 * 1000 * 24); | 
|                           picker.$emit('pick', date); | 
|                         } | 
|                       }, { | 
|                         text: '一周前', | 
|                         onClick(picker) { | 
|                           const date = new Date(); | 
|                           date.setTime(date.getTime() - 3600 * 1000 * 24 * 7); | 
|                           picker.$emit('pick', date); | 
|                         } | 
|                       }] | 
|                     }" | 
|                   /> | 
|                 </el-form-item> | 
|                 <el-form-item | 
|                   :label="header.name" | 
|                   :prop="header.name" | 
|                   :rules="{ | 
|                     required: header.required === true || header.required === 'true', | 
|                     message: header.message || `请输入${header.name}`, | 
|                     trigger: ['blur', 'change'] | 
|                   }" | 
|                   v-if="header.type == 'user'" | 
|                 > | 
|                   <el-select | 
|                     v-model="form[header.name]" | 
|                     multiple | 
|                     filterable | 
|                     placeholder="请选择用户" | 
|                     :disabled="!checkEditPermission(header)" | 
|                   > | 
|                     <el-option | 
|                       v-for="item in userOptions" | 
|                       :key="item.value" | 
|                       :label="item.label" | 
|                       :value="item.value" | 
|                     /> | 
|                   </el-select> | 
|                 </el-form-item> | 
|               </el-col> | 
|             </el-row> | 
|           </el-form> | 
|         </div> | 
|       </div> | 
|     </div> | 
|     <div slot="footer" class="dialog-footer select-member-footer"> | 
|       <el-button @click="handleClose">取 消</el-button> | 
|       <el-button type="primary" @click="handleSubmit">确 定</el-button> | 
|     </div> | 
|   </el-dialog> | 
| </template> | 
|    | 
| <script> | 
| import { listByRole } from './service'; | 
|   | 
| export default { | 
|   name: "AddDialog", | 
|   props: { | 
|     visible: { | 
|       type: Boolean, | 
|       default: false, | 
|     }, | 
|     headerList: { | 
|       type: Array, | 
|       default: () => [], | 
|     }, | 
|     editData: { | 
|       type: Object, | 
|       default: () => ({}), | 
|     }, | 
|     isEdit: { | 
|       type: Boolean, | 
|       default: false, | 
|     } | 
|   }, | 
|   data() { | 
|     return { | 
|       isIPad: /iPad/i.test(navigator.userAgent), | 
|       showHeaderList: [{"name":"sd","type":"text","required":true,"message":"请输入表头名称","role":[]}], | 
|       form: {}, | 
|       rules: {}, | 
|       photoList: [], | 
|       spectrumList: [], | 
|       defaultImageUrl: 'https://fuss10.elemecdn.com/e/5d/4a731a90594a4af544c0c25941171jpeg.jpeg', // 默认图片地址 | 
|       userOptions: [] | 
|     }; | 
|   }, | 
|   computed: { | 
|     dialogVisible: { | 
|       get() { | 
|         return this.visible; | 
|       }, | 
|       set(val) { | 
|         this.$emit("update:visible", val); | 
|       }, | 
|     }, | 
|     currentUserId() { | 
|       const userInfo = JSON.parse(sessionStorage.getItem('userInfo') || '{}'); | 
|       return userInfo.userId; | 
|     } | 
|   }, | 
|   watch: { | 
|     visible: { | 
|       handler(newVal) { | 
|         if (newVal) { | 
|           this.showHeaderList = JSON.parse(JSON.stringify(this.headerList)); | 
|           this.$forceUpdate(); | 
|           if (this.isEdit && this.editData) { | 
|             // 编辑模式:设置回显数据 | 
|             this.setFormData(this.editData); | 
|           } else { | 
|             // 新增模式:初始化空表单 | 
|             this.initFormData(); | 
|           } | 
|           this.initRules(); | 
|         } | 
|       }, | 
|     }, | 
|     headerList: { | 
|       immediate: true, | 
|       handler(newVal) { | 
|         if (newVal && newVal.length) { | 
|           if (this.isEdit && this.editData) { | 
|             this.setFormData(this.editData); | 
|           } else { | 
|             this.initFormData(); | 
|           } | 
|           this.initRules(); | 
|         } | 
|       }, | 
|     }, | 
|   }, | 
|   methods: { | 
|     getUserOptions() { | 
|       listByRole().then(res => { | 
|         if (res) { | 
|           this.userOptions = res.map(user => ({ | 
|             value: user.userId, | 
|             label: user.nickName || user.userName | 
|           })); | 
|         } else { | 
|           this.$message.error('获取用户列表失败'); | 
|         } | 
|       }).catch(err => { | 
|         console.error('获取用户列表失败', err); | 
|       }); | 
|     }, | 
|     checkEditPermission(header) { | 
|       if (!header.role || !Array.isArray(header.role)) { | 
|         return true; | 
|       } | 
|       return header.role.includes(this.currentUserId); | 
|     }, | 
|     initRules() { | 
|       // 初始化校验规则 | 
|       const rules = {}; | 
|       if (this.headerList && this.headerList.length) { | 
|         this.headerList.forEach(header => { | 
|           // 处理required可能是字符串的情况 | 
|           const isRequired = header.required === true || header.required === 'true'; | 
|           if (isRequired) { | 
|             rules[header.name] = [{ | 
|               required: true, | 
|               message: header.message || `请输入${header.name}`, | 
|               trigger: ['blur', 'change'] | 
|             }]; | 
|           } | 
|         }); | 
|       } | 
|       this.rules = rules; | 
|     }, | 
|     initFormData() { | 
|       // 初始化基础表单数据 | 
|       const formData = { | 
|         updateTime: this.formatDateTime(new Date()), | 
|       }; | 
|        | 
|       // 根据headerList初始化表单数据 | 
|       if (this.headerList && this.headerList.length) { | 
|         this.headerList.forEach(header => { | 
|           if (header.type === 'user') { | 
|             formData[header.name] = []; | 
|           } else { | 
|             formData[header.name] = ''; | 
|           } | 
|         }); | 
|       } | 
|        | 
|       // 使用Vue.set确保响应式 | 
|       Object.keys(formData).forEach(key => { | 
|         this.$set(this.form, key, formData[key]); | 
|       }); | 
|     }, | 
|     setFormData(data) { | 
|       // 设置基础表单数据 | 
|       const formData = { | 
|         updateTime: this.formatDateTime(new Date()), | 
|       }; | 
|   | 
|       // 根据headerList设置表单数据 | 
|       if (this.headerList && this.headerList.length) { | 
|         this.headerList.forEach(header => { | 
|           if (header.type === 'user') { | 
|             formData[header.name] = data[header.name] || []; | 
|           } else { | 
|             formData[header.name] = data[header.name] || ''; | 
|           } | 
|         }); | 
|       } | 
|   | 
|       // 使用Vue.set确保响应式 | 
|       Object.keys(formData).forEach(key => { | 
|         this.$set(this.form, key, formData[key]); | 
|       }); | 
|   | 
|       // 设置照片列表 | 
|       if (data.photos && data.photos.length) { | 
|         this.photoList = data.photos.map((photo) => ({ | 
|           name: photo.name, | 
|           url: photo.url, | 
|           status: "success", | 
|         })); | 
|       } else { | 
|         this.photoList = []; | 
|       } | 
|   | 
|       // 设置图谱列表 | 
|       if (data.spectrums && data.spectrums.length) { | 
|         this.spectrumList = data.spectrums.map((spectrum) => ({ | 
|           name: spectrum.name, | 
|           url: spectrum.url, | 
|           status: "success", | 
|         })); | 
|       } else { | 
|         this.spectrumList = []; | 
|       } | 
|   | 
|       // 重置表单校验状态 | 
|       this.$nextTick(() => { | 
|         this.$refs.form?.clearValidate(); | 
|       }); | 
|     }, | 
|     formatDateTime(date) { | 
|       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'); | 
|       const seconds = String(date.getSeconds()).padStart(2, '0'); | 
|       return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`; | 
|     }, | 
|     handleClose() { | 
|       this.dialogVisible = false; | 
|       this.$refs.form?.resetFields(); | 
|       this.photoList = []; | 
|       this.spectrumList = []; | 
|       this.initFormData(); | 
|     }, | 
|     handleSubmit() { | 
|        | 
|       this.$refs.form.validate((valid) => { | 
|         if (valid) { | 
|           const submitData = { | 
|             ...this.form, | 
|             photos: this.photoList, | 
|             spectrums: this.spectrumList, | 
|           }; | 
|            | 
|           // 为用户类型字段添加用户完整信息 | 
|           if (this.headerList && this.headerList.length) { | 
|             this.headerList.forEach(header => { | 
|               if (header.type === 'user' && Array.isArray(submitData[header.name])) { | 
|                 // 为每个用户类型字段添加userInfo属性,包含用户完整信息 | 
|                 submitData[`${header.name}_userInfo`] = submitData[header.name].map(userId => { | 
|                   const userInfo = this.userOptions.find(user => user.value === userId); | 
|                   return userInfo ? userInfo : { value: userId, label: userId }; | 
|                 }); | 
|               } | 
|             }); | 
|           } | 
|            | 
|           console.log(submitData,'修改的数据') | 
|           this.$emit("success", submitData); | 
|           this.handleClose(); | 
|         } else { | 
|           this.$message.error('请填写必填项'); | 
|         } | 
|       }); | 
|     }, | 
|     handlePhotoChange(file, fileList) { | 
|       this.photoList = fileList; | 
|       this.$refs.form.validateField("photos"); | 
|     }, | 
|     handleSpectrumChange(file, fileList) { | 
|       // 使用默认图片替代实际上传 | 
|       this.spectrumList = [{ | 
|         name: '默认图片.jpg', | 
|         url: this.defaultImageUrl, | 
|         status: 'success' | 
|       }]; | 
|        | 
|       // 同时更新form中对应的字段值以通过表单验证 | 
|       const imageHeader = this.headerList.find(h => h.type === 'image'); | 
|       if (imageHeader && imageHeader.name) { | 
|         // 保存图片URL,这样在表格中可以直接使用 | 
|         this.$set(this.form, imageHeader.name, this.defaultImageUrl); | 
|         console.log('设置图片字段:', imageHeader.name, this.defaultImageUrl); | 
|       } | 
|        | 
|       this.$refs.form.validateField("spectrums"); | 
|     }, | 
|     handleSpectrumRemove(file) { | 
|       this.spectrumList = []; | 
|     }, | 
|   | 
|   }, | 
|   mounted() { | 
|     // 获取用户列表数据 | 
|     this.getUserOptions(); | 
|   }, | 
| }; | 
| </script> | 
|    | 
| <style scoped lang="less"> | 
| ::v-deep .el-dialog__body { | 
|   padding: 0; | 
|   max-height: calc(100vh - 200px); // 设置最大高度 | 
| } | 
|   | 
| ::v-deep .el-dialog { | 
|   margin-top: 5vh !important; // 调整弹窗位置 | 
|   max-height: 90vh; // 设置弹窗最大高度 | 
|   display: flex; | 
|   flex-direction: column; | 
|   | 
|   .el-dialog__body { | 
|     flex: 1; | 
|     overflow: hidden; // 防止出现双滚动条 | 
|   } | 
| } | 
|   | 
| .sample-dialog { | 
|   height: 100%; | 
|   | 
|   .sample-content { | 
|     background: #ffffff; | 
|     border-radius: 10px; | 
|     padding: 20px; | 
|     height: 100%; | 
|     display: flex; | 
|     flex-direction: column; | 
|   | 
|     .form-content { | 
|       flex: 1; | 
|       overflow-y: auto; | 
|       padding: 10px 10px; | 
|       max-height: calc(90vh - 250px); // 设置内容区域最大高度 | 
|   | 
|       &::-webkit-scrollbar { | 
|         width: 6px; | 
|       } | 
|   | 
|       &::-webkit-scrollbar-thumb { | 
|         background: #c0c4cc; | 
|         border-radius: 3px; | 
|       } | 
|   | 
|       &::-webkit-scrollbar-track { | 
|         background: #f5f7fa; | 
|       } | 
|     } | 
|     .el-form-item::after{ | 
|       height: 10px !important; | 
|     } | 
|   } | 
| } | 
|   | 
| .dialog-footer { | 
|   align-items: center; | 
|   display: flex; | 
|   justify-content: center; | 
|   padding: 15px 20px; | 
|   border-top: 1px solid #e4e7ed; | 
|   margin-top: auto; // 保持在底部 | 
|   | 
|   button { | 
|     width: 150px; | 
|   } | 
| } | 
|   | 
| .upload-demo { | 
|   ::v-deep { | 
|     .el-upload--picture-card { | 
|       width: 120px; | 
|       height: 120px; | 
|       line-height: 120px; | 
|     } | 
|      | 
|     .el-upload-list--picture-card { | 
|       display: flex; | 
|       flex-wrap: wrap; | 
|       gap: 8px; | 
|        | 
|       .el-upload-list__item { | 
|         width: 120px; | 
|         height: 120px; | 
|         margin: 0; | 
|       } | 
|     } | 
|      | 
|     // 让上传按钮始终显示在列表最后 | 
|     .el-upload--picture-card { | 
|       order: 9999; | 
|       margin: 0; | 
|     } | 
|   } | 
|    | 
|   // 包裹容器也使用flex布局 | 
|   display: flex; | 
|   flex-wrap: wrap; | 
|   gap: 8px; | 
| } | 
|   | 
| .el-row { | 
|   margin-bottom: 20px; | 
|   | 
|   &:last-child { | 
|     margin-bottom: 0; // 最后一行不要margin | 
|   } | 
| } | 
|   | 
| ::v-deep .el-form-item--small.el-form-item { | 
|   margin-bottom: 0; | 
| } | 
|   | 
| ::v-deep .el-form-item__label { | 
|   padding-bottom: 8px; | 
| } | 
|   | 
| // 优化表单布局 | 
| ::v-deep .el-form { | 
|   .el-form-item { | 
|     margin-bottom: 15px; // 减小表单项间距 | 
|   | 
|     &:last-child { | 
|       margin-bottom: 0; | 
|     } | 
|   } | 
| } | 
|   | 
| // 优化上传区域布局 | 
| ::v-deep .el-upload-list--picture-card { | 
|   display: flex; | 
|   flex-wrap: wrap; | 
|   gap: 8px; // 设置图片间距 | 
| } | 
|   | 
| .upload-file { | 
|   ::v-deep { | 
|     .el-upload-list { | 
|       margin-top: 10px; | 
|     } | 
|     .el-upload-list__item { | 
|       transition: all 0.3s; | 
|       &:hover { | 
|         background-color: #f5f7fa; | 
|       } | 
|     } | 
|     .el-upload__tip { | 
|       color: #909399; | 
|       font-size: 12px; | 
|       margin-top: 8px; | 
|     } | 
|   } | 
| } | 
| </style>  |