| <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="sampleCode"> | 
|                   <el-input | 
|                     v-model="form.sampleCode" | 
|                     placeholder="请输入检测样编号" | 
|                   /> | 
|                 </el-form-item> | 
|               </el-col> | 
|               <el-col :span="12"> | 
|                 <el-form-item label="检测数据" prop="testData"> | 
|                   <el-input | 
|                     v-model="form.testData" | 
|                     placeholder="请输入检测数据" | 
|                   /> | 
|                 </el-form-item> | 
|               </el-col> | 
|             </el-row> | 
|   | 
|             <el-row :gutter="20"> | 
|               <el-col :span="24"> | 
|                 <el-form-item label="检测类型" prop="testTypes"> | 
|                   <el-checkbox-group v-model="form.testTypes"> | 
|                     <el-checkbox label="photo">拍照</el-checkbox> | 
|                     <el-checkbox label="spectrum">图谱</el-checkbox> | 
|                   </el-checkbox-group> | 
|                 </el-form-item> | 
|               </el-col> | 
|             </el-row> | 
|   | 
|             <el-row v-if="showPhotoUpload" :gutter="20"> | 
|               <el-col :span="24"> | 
|                 <el-form-item label="检测拍照" prop="photos"> | 
|                   <div v-if="isIPad"> | 
|                     <el-button type="primary" @click="handleIPadPhoto">拍照</el-button> | 
|                     <el-button type="primary" @click="handleIPadUpload">上传</el-button> | 
|                   </div> | 
|                   <el-upload | 
|                     v-else | 
|                     class="upload-demo" | 
|                     action="#" | 
|                     :file-list="photoList" | 
|                     :auto-upload="false" | 
|                     list-type="picture-card" | 
|                     :on-change="handlePhotoChange" | 
|                   > | 
|                     <i class="el-icon-plus"></i> | 
|                     <!-- <div class="el-upload__tip" slot="tip">支持 jpg、png 格式文件</div> --> | 
|                   </el-upload> | 
|                 </el-form-item> | 
|               </el-col> | 
|             </el-row> | 
|   | 
|             <el-row v-if="showSpectrumUpload" :gutter="20"> | 
|               <el-col :span="24"> | 
|                 <el-form-item label="检测图谱" prop="spectrums"> | 
|                   <div v-if="isIPad"> | 
|                     <el-button type="primary" @click="handleIPadSpectrum">选择图谱</el-button> | 
|                   </div> | 
|                   <el-upload | 
|                     v-else | 
|                     class="upload-file" | 
|                     action="#" | 
|                     :file-list="spectrumList" | 
|                     :auto-upload="false" | 
|                     :on-change="handleSpectrumChange" | 
|                     :on-remove="handleSpectrumRemove" | 
|                   > | 
|                     <el-button type="primary"> | 
|                       <i class="el-icon-upload"></i> 选择文件 | 
|                     </el-button> | 
|                     <div class="el-upload__tip" slot="tip">支持 jpg、png、pdf 格式文件</div> | 
|                   </el-upload> | 
|                 </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> | 
| export default { | 
|   name: "AddDialog", | 
|   props: { | 
|     visible: { | 
|       type: Boolean, | 
|       default: false | 
|     } | 
|   }, | 
|   data() { | 
|     return { | 
|       isIPad: /iPad/i.test(navigator.userAgent), | 
|       form: { | 
|         sampleCode: "", | 
|         testData: "", | 
|         testTypes: ["photo"], // 默认选中拍照 | 
|         photos: [], | 
|         spectrums: [] | 
|       }, | 
|       rules: { | 
|         sampleCode: [ | 
|           { required: true, message: "请输入检测样编号", trigger: "blur" } | 
|         ], | 
|         testData: [ | 
|           { required: true, message: "请输入检测数据", trigger: "blur" } | 
|         ], | 
|         testTypes: [ | 
|           {  | 
|             type: 'array', | 
|             required: true,  | 
|             message: "请至少选择一种检测类型",  | 
|             trigger: "change"  | 
|           } | 
|         ], | 
|         photos: [ | 
|           {  | 
|             required: true,  | 
|             message: "请上传检测照片",  | 
|             trigger: "change", | 
|             validator: (rule, value, callback) => { | 
|               if (this.form.testTypes.includes('photo') && (!this.photoList || !this.photoList.length)) { | 
|                 callback(new Error('请上传检测照片')); | 
|               } else { | 
|                 callback(); | 
|               } | 
|             } | 
|           } | 
|         ], | 
|         spectrums: [ | 
|           {  | 
|             required: true,  | 
|             message: "请上传检测图谱",  | 
|             trigger: "change", | 
|             validator: (rule, value, callback) => { | 
|               if (this.form.testTypes.includes('spectrum') && (!this.spectrumList || !this.spectrumList.length)) { | 
|                 callback(new Error('请上传检测图谱')); | 
|               } else { | 
|                 callback(); | 
|               } | 
|             } | 
|           } | 
|         ] | 
|       }, | 
|       photoList: [], | 
|       spectrumList: [] | 
|     }; | 
|   }, | 
|   computed: { | 
|     dialogVisible: { | 
|       get() { | 
|         return this.visible; | 
|       }, | 
|       set(val) { | 
|         this.$emit("update:visible", val); | 
|       } | 
|     }, | 
|     showPhotoUpload() { | 
|       return this.form.testTypes.includes('photo'); | 
|     }, | 
|     showSpectrumUpload() { | 
|       return this.form.testTypes.includes('spectrum'); | 
|     } | 
|   }, | 
|   methods: { | 
|     setFormData(data) { | 
|       // 设置基础表单数据 | 
|       this.form.sampleCode = data.sampleCode; | 
|       this.form.testData = data.testData; | 
|       this.form.testTypes = data.testTypes; | 
|   | 
|       // 设置照片列表 | 
|       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(); | 
|       }); | 
|     }, | 
|     handleClose() { | 
|       this.dialogVisible = false; | 
|       this.$refs.form?.resetFields(); | 
|       this.photoList = []; | 
|       this.spectrumList = []; | 
|       this.form = { | 
|         sampleCode: "", | 
|         testData: "", | 
|         testTypes: ["photo"], | 
|         photos: [], | 
|         spectrums: [] | 
|       }; | 
|     }, | 
|     handleSubmit() { | 
|       this.$refs.form.validate((valid) => { | 
|         if (valid) { | 
|           const submitData = { | 
|             ...this.form, | 
|             photos: this.photoList, | 
|             spectrums: this.spectrumList | 
|           }; | 
|           this.$emit("success", submitData); | 
|         } | 
|       }); | 
|     }, | 
|     handlePhotoChange(file, fileList) { | 
|       this.photoList = fileList; | 
|       this.$refs.form.validateField('photos'); | 
|     }, | 
|     handleSpectrumChange(file, fileList) { | 
|       this.spectrumList = fileList; | 
|       this.$refs.form.validateField('spectrums'); | 
|     }, | 
|     handleSpectrumRemove(file) { | 
|       // 处理文件移除逻辑 | 
|     }, | 
|     // iPad 相关方法 | 
|     handleIPadPhoto() { | 
|       // TODO: 调用 iPad 拍照功能 | 
|       console.log('调用 iPad 拍照功能'); | 
|     }, | 
|     handleIPadUpload() { | 
|       // TODO: 调用 iPad 上传功能 | 
|       console.log('调用 iPad 上传功能'); | 
|     }, | 
|     handleIPadSpectrum() { | 
|       // TODO: 调用 iPad 选择图谱功能 | 
|       console.log('调用 iPad 选择图谱功能'); | 
|     } | 
|   } | 
| }; | 
| </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: 0 10px; | 
|       max-height: calc(90vh - 250px);  // 设置内容区域最大高度 | 
|        | 
|       &::-webkit-scrollbar { | 
|         width: 6px; | 
|       } | 
|        | 
|       &::-webkit-scrollbar-thumb { | 
|         background: #c0c4cc; | 
|         border-radius: 3px; | 
|       } | 
|        | 
|       &::-webkit-scrollbar-track { | 
|         background: #f5f7fa; | 
|       } | 
|     } | 
|   } | 
| } | 
|   | 
| .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 .3s; | 
|       &:hover { | 
|         background-color: #f5f7fa; | 
|       } | 
|     } | 
|     .el-upload__tip { | 
|       color: #909399; | 
|       font-size: 12px; | 
|       margin-top: 8px; | 
|     } | 
|   } | 
| } | 
| </style>  |