| | |
| | | > |
| | | <template #default="scope"> |
| | | <div class="item-name-cell"> |
| | | <span :class="{'required-item': scope.row.required}"> |
| | | <span :class="{ 'required-item': scope.row.required }"> |
| | | {{ scope.row.itemName }} |
| | | </span> |
| | | <el-tooltip |
| | |
| | | /> |
| | | <div v-else class="value-display-container"> |
| | | <span class="value-text" :title="scope.row.values[index]"> |
| | | {{ scope.row.values[index] || '-' }} |
| | | {{ scope.row.values[index] || "-" }} |
| | | </span> |
| | | <span v-if="scope.row.values[index] && scope.row.unit" class="unit-text"> |
| | | <span |
| | | v-if="scope.row.values[index] && scope.row.unit" |
| | | class="unit-text" |
| | | > |
| | | {{ scope.row.unit }} |
| | | </span> |
| | | </div> |
| | | <div v-if="scope.row.reference && scope.row.values[index]" class="validation-indicator"> |
| | | <div |
| | | v-if="scope.row.reference && scope.row.values[index]" |
| | | class="validation-indicator" |
| | | > |
| | | <i |
| | | v-if="isValueValid(scope.row, scope.row.values[index])" |
| | | class="el-icon-success valid-icon" |
| | |
| | | </div> |
| | | </template> |
| | | </el-table-column> |
| | | |
| | | <!-- <el-table-column |
| | | v-if="isEditing" |
| | | label="操作" |
| | | width="120" |
| | | fixed="right" |
| | | class-name="leave-alone" |
| | | > |
| | | <template #default> |
| | | <el-button link type="primary" @click="addColumn" size="small"> |
| | | 新增列 |
| | | </el-button> |
| | | </template> |
| | | </el-table-column> --> |
| | | </el-table> |
| | | |
| | | <!-- 统计信息 --> |
| | |
| | | <el-card shadow="never"> |
| | | <div class="stats-content"> |
| | | <span class="stats-title">数据统计:</span> |
| | | <span class="stats-item">总记录数: {{ dynamicColumns.length }} 个时间点</span> |
| | | <span class="stats-item" |
| | | >总记录数: {{ dynamicColumns.length }} 个时间点</span |
| | | > |
| | | <span class="stats-item">已填写: {{ filledCount }} 项</span> |
| | | <span class="stats-item">完成度: {{ completionRate }}%</span> |
| | | </div> |
| | |
| | | <span class="attachment-title">附件上传</span> |
| | | <span class="attachment-tip">支持上传检验报告单等文件 (最多10个)</span> |
| | | </div> |
| | | <upload-attachment |
| | | :file-list="attachments" |
| | | |
| | | <!-- 使用 UploadAttachment 组件 --> |
| | | <UploadAttachment |
| | | ref="uploadAttachment" |
| | | :file-list="attachmentFileList" |
| | | :limit="attachmentLimit" |
| | | :accept="attachmentAccept" |
| | | :multiple="true" |
| | | @change="handleAttachmentChange" |
| | | :limit="10" |
| | | :accept="'.pdf,.jpg,.jpeg,.png,.doc,.docx'" |
| | | @upload-success="handleUploadSuccess" |
| | | @upload-error="handleUploadError" |
| | | @remove="handleAttachmentRemove" |
| | | /> |
| | | |
| | | <!-- 附件列表展示 --> |
| | | <div class="attachment-list" v-if="attachments && attachments.length > 0"> |
| | | <div class="list-title">已上传附件 ({{ attachments.length }})</div> |
| | | <el-table :data="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 |
| | | v-if="isEditing" |
| | | size="mini" |
| | | type="success" |
| | | @click="handleDownload(scope.row)" |
| | | > |
| | | 下载 |
| | | </el-button> |
| | | <el-button |
| | | v-if="isEditing" |
| | | size="mini" |
| | | type="danger" |
| | | @click="handleRemoveAttachment(scope.$index)" |
| | | > |
| | | 删除 |
| | | </el-button> |
| | | </template> |
| | | </el-table-column> |
| | | </el-table> |
| | | </div> |
| | | </div> |
| | | |
| | | <!-- 列编辑对话框 --> |
| | |
| | | <el-form |
| | | :model="columnForm" |
| | | label-width="80px" |
| | | ref="columnForm" |
| | | ref="columnFormB" |
| | | :rules="columnRules" |
| | | > |
| | | <el-form-item label="日期" prop="date"> |
| | |
| | | > |
| | | 删除 |
| | | </el-button> |
| | | <el-button type="primary" @click="confirmAddColumn" :loading="saveLoading"> |
| | | <el-button |
| | | type="primary" |
| | | @click="confirmAddColumn" |
| | | :loading="saveLoading" |
| | | > |
| | | 确定 |
| | | </el-button> |
| | | </span> |
| | | </el-dialog> |
| | | |
| | | <!-- 文件预览弹窗 --> |
| | | <FilePreviewDialog |
| | | :visible="previewVisible" |
| | | :file="currentPreviewFile" |
| | | @close="previewVisible = false" |
| | | @download="handleDownload" |
| | | /> |
| | | </div> |
| | | </template> |
| | | |
| | | <script> |
| | | import UploadAttachment from "@/components/UploadAttachment"; |
| | | import FilePreviewDialog from "@/components/FilePreviewDialog"; |
| | | import dayjs from "dayjs"; |
| | | |
| | | export default { |
| | | name: 'BloodRoutinePanel', |
| | | name: "BloodRoutinePanel", |
| | | components: { |
| | | UploadAttachment, |
| | | FilePreviewDialog |
| | | }, |
| | | props: { |
| | | isEditing: { |
| | |
| | | default: false |
| | | }, |
| | | initialData: { |
| | | type: Array, |
| | | default: () => [] |
| | | type: Object, |
| | | default: () => ({}) |
| | | }, |
| | | showStatistics: { |
| | | type: Boolean, |
| | |
| | | data() { |
| | | return { |
| | | tableData: [], |
| | | dynamicColumns: [ |
| | | { |
| | | label: '2024-12-27\n08:00', |
| | | key: 'time1', |
| | | date: '2024-12-27', |
| | | time: '08:00', |
| | | remark: '晨起检测' |
| | | } |
| | | ], |
| | | dynamicColumns: [], |
| | | attachments: [], |
| | | columnDialogVisible: false, |
| | | columnForm: { |
| | | date: '', |
| | | time: '', |
| | | remark: '' |
| | | date: "", |
| | | time: "", |
| | | remark: "" |
| | | }, |
| | | editingColumnIndex: null, |
| | | tableKey: 0, |
| | | tableLoading: false, |
| | | saveLoading: false, |
| | | internalData: {}, |
| | | columnRules: { |
| | | date: [ |
| | | { required: true, message: '请选择日期', trigger: 'change' } |
| | | ], |
| | | time: [ |
| | | { required: true, message: '请选择时间', trigger: 'change' } |
| | | ] |
| | | } |
| | | date: [{ required: true, message: "请选择日期", trigger: "change" }], |
| | | time: [{ required: true, message: "请选择时间", trigger: "change" }] |
| | | }, |
| | | |
| | | // 预览相关 |
| | | previewVisible: false, |
| | | currentPreviewFile: null, |
| | | |
| | | // 附件相关配置 |
| | | attachmentLimit: 10, |
| | | attachmentAccept: |
| | | ".pdf,.jpg,.jpeg,.png,.doc,.docx,.xls,.xlsx,.ppt,.pptx,.txt", |
| | | attachmentFileList: [] |
| | | }; |
| | | }, |
| | | computed: { |
| | |
| | | let count = 0; |
| | | this.tableData.forEach(row => { |
| | | row.values.forEach(value => { |
| | | if (value && value.toString().trim() !== '') { |
| | | if (value && value.toString().trim() !== "") { |
| | | count++; |
| | | } |
| | | }); |
| | |
| | | } |
| | | }, |
| | | watch: { |
| | | initialData: { |
| | | handler(newData) { |
| | | if (newData && Object.keys(newData).length > 0) { |
| | | this.internalData = { ...newData }; |
| | | this.initFromExternalData(); |
| | | } |
| | | }, |
| | | immediate: true, |
| | | deep: true |
| | | }, |
| | | isEditing(newVal) { |
| | | if (!newVal) { |
| | | this.$emit('data-change', { |
| | | type: 'blood_routine', |
| | | data: this.tableData, |
| | | columns: this.dynamicColumns, |
| | | attachments: this.attachments |
| | | }); |
| | | this.saveData(); |
| | | } |
| | | this.$nextTick(() => { |
| | | this.forceTableLayout(); |
| | |
| | | }, |
| | | deep: true, |
| | | immediate: true |
| | | }, |
| | | // 监听附件数据变化 |
| | | attachments: { |
| | | handler(newAttachments) { |
| | | this.attachmentFileList = newAttachments.map(item => ({ |
| | | uid: item.id || Math.random(), |
| | | name: item.fileName, |
| | | fileSize: item.fileSize, |
| | | url: item.path || item.fileUrl, |
| | | uploadTime: item.uploadTime, |
| | | status: "success" |
| | | })); |
| | | }, |
| | | deep: true |
| | | } |
| | | }, |
| | | methods: { |
| | | // 从外部数据初始化组件 |
| | | initFromExternalData() { |
| | | if (this.internalData.data && this.internalData.columns) { |
| | | this.tableData = this.internalData.data.map(item => ({ |
| | | ...item, |
| | | values: |
| | | item.values || new Array(this.internalData.columns.length).fill("") |
| | | })); |
| | | this.dynamicColumns = [...this.internalData.columns]; |
| | | } else { |
| | | this.initTableData(); |
| | | } |
| | | |
| | | // 初始化附件 |
| | | if (this.internalData.attachments) { |
| | | this.attachments = [...this.internalData.attachments]; |
| | | } |
| | | }, |
| | | |
| | | // 初始化默认表格数据 |
| | | initTableData() { |
| | | const medicalItems = [ |
| | | { |
| | | itemName: 'WBC', |
| | | unit: '×10⁹/L', |
| | | required: true, |
| | | reference: '3.5-9.5', |
| | | min: 3.5, |
| | | max: 9.5, |
| | | type: 'number' |
| | | }, |
| | | { |
| | | itemName: 'NEUT%', |
| | | unit: '%', |
| | | required: true, |
| | | reference: '40-75', |
| | | min: 40, |
| | | max: 75, |
| | | type: 'number' |
| | | }, |
| | | { |
| | | itemName: 'Hb', |
| | | unit: 'g/L', |
| | | required: true, |
| | | reference: '130-175', |
| | | min: 130, |
| | | max: 175, |
| | | type: 'number' |
| | | }, |
| | | { |
| | | itemName: '血小板', |
| | | unit: '×10⁹/L', |
| | | required: true, |
| | | reference: '125-350', |
| | | min: 125, |
| | | max: 350, |
| | | type: 'number' |
| | | } |
| | | ]; |
| | | const medicalItems = this.getMedicalItems(); |
| | | |
| | | if (this.dynamicColumns.length === 0) { |
| | | this.dynamicColumns = [ |
| | | { |
| | | label: `${new Date().toISOString().split("T")[0]}\n08:00`, |
| | | key: "time1", |
| | | date: new Date().toISOString().split("T")[0], |
| | | time: "08:00", |
| | | remark: "晨起检测" |
| | | } |
| | | ]; |
| | | } |
| | | |
| | | this.tableData = medicalItems.map(item => ({ |
| | | ...item, |
| | | values: new Array(this.dynamicColumns.length).fill('') |
| | | values: new Array(this.dynamicColumns.length).fill("") |
| | | })); |
| | | }, |
| | | |
| | | // 血常规检测项目定义 |
| | | getMedicalItems() { |
| | | return [ |
| | | { |
| | | itemName: "WBC", |
| | | unit: "×10⁹/L", |
| | | required: true, |
| | | reference: "3.5-9.5", |
| | | min: 3.5, |
| | | max: 9.5, |
| | | type: "number" |
| | | }, |
| | | { |
| | | itemName: "NEUT%", |
| | | unit: "%", |
| | | required: true, |
| | | reference: "40-75", |
| | | min: 40, |
| | | max: 75, |
| | | type: "number" |
| | | }, |
| | | { |
| | | itemName: "Hb", |
| | | unit: "g/L", |
| | | required: true, |
| | | reference: "130-175", |
| | | min: 130, |
| | | max: 175, |
| | | type: "number" |
| | | }, |
| | | { |
| | | itemName: "血小板", |
| | | unit: "×10⁹/L", |
| | | required: true, |
| | | reference: "125-350", |
| | | min: 125, |
| | | max: 350, |
| | | type: "number" |
| | | } |
| | | ]; |
| | | }, |
| | | |
| | | // 保存数据到父组件 |
| | | saveData() { |
| | | const dataToEmit = { |
| | | type: "blood_routine", |
| | | data: this.tableData, |
| | | columns: this.dynamicColumns, |
| | | attachments: this.attachments |
| | | }; |
| | | this.$emit("data-change", dataToEmit); |
| | | }, |
| | | |
| | | getPlaceholder(row) { |
| | | return row.reference ? `参考: ${row.reference}` : '请输入数值'; |
| | | return row.reference ? `参考: ${row.reference}` : "请输入数值"; |
| | | }, |
| | | |
| | | isValueValid(row, value) { |
| | |
| | | addColumn() { |
| | | this.editingColumnIndex = null; |
| | | this.columnForm = { |
| | | date: new Date().toISOString().split('T')[0], |
| | | time: '08:00', |
| | | remark: '' |
| | | date: new Date().toISOString().split("T")[0], |
| | | time: "08:00", |
| | | remark: "" |
| | | }; |
| | | this.columnDialogVisible = true; |
| | | this.$nextTick(() => { |
| | | this.$refs.columnForm && this.$refs.columnForm.clearValidate(); |
| | | this.$refs.columnFormB && this.$refs.columnFormB.clearValidate(); |
| | | }); |
| | | }, |
| | | |
| | |
| | | this.columnForm = { |
| | | date: column.date, |
| | | time: column.time, |
| | | remark: column.remark || '' |
| | | remark: column.remark || "" |
| | | }; |
| | | this.columnDialogVisible = true; |
| | | }, |
| | | |
| | | confirmAddColumn() { |
| | | this.$refs.columnForm.validate((valid) => { |
| | | this.$refs.columnFormB.validate(valid => { |
| | | if (!valid) { |
| | | this.$message.warning('请完善时间点信息'); |
| | | this.$message.warning("请完善时间点信息"); |
| | | return; |
| | | } |
| | | |
| | | this.saveLoading = true; |
| | | |
| | | if (this.editingColumnIndex !== null) { |
| | | // 编辑现有列 |
| | | const column = this.dynamicColumns[this.editingColumnIndex]; |
| | | column.label = `${this.columnForm.date}\n${this.columnForm.time}`; |
| | | column.date = this.columnForm.date; |
| | | column.time = this.columnForm.time; |
| | | column.remark = this.columnForm.remark; |
| | | this.$message.success('时间点修改成功'); |
| | | this.$message.success("时间点修改成功"); |
| | | } else { |
| | | // 新增列 |
| | | const newIndex = this.dynamicColumns.length + 1; |
| | | const newColumn = { |
| | | label: `${this.columnForm.date}\n${this.columnForm.time}`, |
| | | key: `time${Date.now()}`, |
| | |
| | | time: this.columnForm.time, |
| | | remark: this.columnForm.remark |
| | | }; |
| | | this.internalData.columns.push(newColumn); |
| | | |
| | | this.dynamicColumns.push(newColumn); |
| | | this.tableData.forEach(row => { |
| | | row.values.push(''); |
| | | if (!row.values) row.values = []; |
| | | row.values.push(""); |
| | | }); |
| | | this.$message.success('时间点添加成功'); |
| | | this.$message.success("时间点添加成功"); |
| | | } |
| | | |
| | | this.columnDialogVisible = false; |
| | | this.saveLoading = false; |
| | | this.tableKey += 1; |
| | | this.saveData(); |
| | | }); |
| | | }, |
| | | |
| | | handleDeleteColumn() { |
| | | if (this.editingColumnIndex !== null) { |
| | | this.$confirm('确定要删除这个时间点吗?', '提示', { |
| | | confirmButtonText: '确定', |
| | | cancelButtonText: '取消', |
| | | type: 'warning' |
| | | this.$confirm("确定要删除这个时间点吗?", "提示", { |
| | | confirmButtonText: "确定", |
| | | cancelButtonText: "取消", |
| | | type: "warning" |
| | | }).then(() => { |
| | | this.dynamicColumns.splice(this.editingColumnIndex, 1); |
| | | this.tableData.forEach(row => { |
| | |
| | | }); |
| | | this.columnDialogVisible = false; |
| | | this.tableKey += 1; |
| | | this.$message.success('时间点删除成功'); |
| | | this.saveData(); |
| | | this.$message.success("时间点删除成功"); |
| | | }); |
| | | } |
| | | }, |
| | |
| | | handleDialogClosed() { |
| | | this.editingColumnIndex = null; |
| | | this.columnForm = { |
| | | date: '', |
| | | time: '', |
| | | remark: '' |
| | | date: "", |
| | | time: "", |
| | | remark: "" |
| | | }; |
| | | this.$refs.columnForm && this.$refs.columnForm.clearValidate(); |
| | | this.$refs.columnFormB && this.$refs.columnFormB.clearValidate(); |
| | | }, |
| | | |
| | | disableFutureDates(time) { |
| | |
| | | }, |
| | | |
| | | handleValueChange(row, columnIndex) { |
| | | this.$emit('data-change', { |
| | | type: 'blood_routine', |
| | | data: this.tableData, |
| | | columns: this.dynamicColumns |
| | | }); |
| | | this.saveData(); |
| | | }, |
| | | |
| | | /** 附件变化处理 */ |
| | | handleAttachmentChange(fileList) { |
| | | this.attachments = fileList; |
| | | this.$emit('attachment-change', { |
| | | type: 'blood_routine', |
| | | attachments: fileList |
| | | }); |
| | | this.attachmentFileList = fileList; |
| | | }, |
| | | |
| | | /** 附件移除处理 */ |
| | | handleAttachmentRemove(file) { |
| | | if (file.url) { |
| | | const index = this.attachments.findIndex( |
| | | item => item.path === file.url || item.fileUrl === file.url |
| | | ); |
| | | if (index > -1) { |
| | | this.attachments.splice(index, 1); |
| | | this.saveData(); |
| | | this.$message.success("附件删除成功"); |
| | | } |
| | | } |
| | | }, |
| | | |
| | | /** 上传成功处理 */ |
| | | handleUploadSuccess({ file, fileList, response }) { |
| | | if (response.code === 200) { |
| | | const attachmentObj = { |
| | | fileName: file.name, |
| | | path: response.fileUrl || file.url, |
| | | fileUrl: response.fileUrl || file.url, |
| | | fileType: this.getFileExtension(file.name), |
| | | fileSize: file.size, |
| | | uploadTime: dayjs().format("YYYY-MM-DD HH:mm:ss") |
| | | }; |
| | | |
| | | if (!Array.isArray(this.attachments)) { |
| | | this.attachments = []; |
| | | } |
| | | |
| | | this.attachments.push(attachmentObj); |
| | | this.attachmentFileList = fileList; |
| | | this.saveData(); |
| | | this.$message.success("文件上传成功"); |
| | | } |
| | | }, |
| | | |
| | | /** 上传错误处理 */ |
| | | handleUploadError({ file, fileList, error }) { |
| | | console.error("附件上传失败:", error); |
| | | this.$message.error("文件上传失败,请重试"); |
| | | }, |
| | | |
| | | /** 手动删除附件 */ |
| | | handleRemoveAttachment(index) { |
| | | this.attachments.splice(index, 1); |
| | | this.attachmentFileList.splice(index, 1); |
| | | this.saveData(); |
| | | this.$message.success("附件删除成功"); |
| | | }, |
| | | |
| | | /** 文件预览 */ |
| | | handlePreview(file) { |
| | | this.currentPreviewFile = { |
| | | fileName: file.fileName, |
| | | fileUrl: file.path || file.fileUrl, |
| | | fileType: this.getFileType(file.fileName) |
| | | }; |
| | | this.previewVisible = true; |
| | | }, |
| | | |
| | | /** 文件下载 */ |
| | | handleDownload(file) { |
| | | 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("文件路径不存在,无法下载"); |
| | | } |
| | | }, |
| | | |
| | | /** 获取文件类型 */ |
| | | 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"; |
| | | }, |
| | | |
| | | /** 获取文件图标颜色 */ |
| | | getFileIconColor(fileName) { |
| | | const type = this.getFileType(fileName); |
| | | const colorMap = { |
| | | image: "#67C23A", |
| | | pdf: "#F56C6C", |
| | | office: "#409EFF", |
| | | other: "#909399" |
| | | }; |
| | | 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(); |
| | | }, |
| | | |
| | | /** 格式化文件大小 */ |
| | | formatFileSize(bytes) { |
| | | if (!bytes || bytes === 0) return "0 B"; |
| | | const k = 1024; |
| | | 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]; |
| | | }, |
| | | |
| | | /** 日期时间格式化 */ |
| | | 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; |
| | | } |
| | | }, |
| | | |
| | | forceTableLayout() { |
| | | this.$nextTick(() => { |
| | | const table = this.$el.querySelector('.el-table'); |
| | | const table = this.$el.querySelector(".el-table"); |
| | | if (table) { |
| | | window.dispatchEvent(new Event('resize')); |
| | | window.dispatchEvent(new Event("resize")); |
| | | } |
| | | }); |
| | | }, |
| | |
| | | } |
| | | }, |
| | | mounted() { |
| | | this.initTableData(); |
| | | this.$nextTick(() => { |
| | | if (Object.keys(this.internalData).length === 0) { |
| | | this.initTableData(); |
| | | } |
| | | this.forceTableLayout(); |
| | | }); |
| | | } |
| | | }; |
| | | </script> |
| | |
| | | font-size: 14px; |
| | | } |
| | | |
| | | /* 附件上传区域样式优化 */ |
| | | .attachment-section { |
| | | margin-top: 24px; |
| | | padding: 20px; |
| | |
| | | .attachment-tip { |
| | | font-size: 12px; |
| | | color: #909399; |
| | | } |
| | | |
| | | .attachment-list { |
| | | margin-top: 16px; |
| | | } |
| | | |
| | | .list-title { |
| | | font-weight: bold; |
| | | margin-bottom: 12px; |
| | | color: #303133; |
| | | font-size: 14px; |
| | | } |
| | | |
| | | .file-name { |
| | | font-size: 13px; |
| | | margin-left: 8px; |
| | | } |
| | | |
| | | /* 响应式设计 */ |
| | |
| | | flex-direction: column; |
| | | gap: 4px; |
| | | } |
| | | } |
| | | |
| | | /* 动画效果 */ |
| | | .fade-enter-active, .fade-leave-active { |
| | | transition: opacity 0.3s; |
| | | } |
| | | .fade-enter, .fade-leave-to { |
| | | opacity: 0; |
| | | } |
| | | </style> |