| | |
| | | :value="option.value" |
| | | /> |
| | | </el-select> |
| | | <span v-else class="value-text" :title="getSelectLabel(scope.row, index)"> |
| | | {{ getSelectLabel(scope.row, index) || '-' }} |
| | | <span |
| | | v-else |
| | | class="value-text" |
| | | :title="getSelectLabel(scope.row, index)" |
| | | > |
| | | {{ getSelectLabel(scope.row, index) || "-" }} |
| | | </span> |
| | | </template> |
| | | |
| | |
| | | /> |
| | | <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> |
| | | </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> |
| | |
| | | <div class="attachment-header"> |
| | | <i class="el-icon-paperclip"></i> |
| | | <span class="attachment-title">附件上传</span> |
| | | <span class="attachment-tip">支持上传尿常规检验报告单等文件 (最多10个)</span> |
| | | <span class="attachment-tip" |
| | | >支持上传尿常规检验报告单等文件 (最多10个)</span |
| | | > |
| | | </div> |
| | | <upload-attachment |
| | | :file-list="attachments" |
| | |
| | | > |
| | | 删除 |
| | | </el-button> |
| | | <el-button type="primary" @click="confirmAddColumn" :loading="saveLoading"> |
| | | <el-button |
| | | type="primary" |
| | | @click="confirmAddColumn" |
| | | :loading="saveLoading" |
| | | > |
| | | 确定 |
| | | </el-button> |
| | | </span> |
| | |
| | | import UploadAttachment from "@/components/UploadAttachment"; |
| | | |
| | | export default { |
| | | name: 'UrineRoutinePanel', |
| | | name: "UrineRoutinePanel", |
| | | components: { |
| | | UploadAttachment, |
| | | UploadAttachment |
| | | }, |
| | | props: { |
| | | isEditing: { |
| | | type: Boolean, |
| | | default: false |
| | | }, |
| | | // 修改为 Object 类型,支持复杂数据结构 |
| | | 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" }] |
| | | } |
| | | }; |
| | | }, |
| | |
| | | let count = 0; |
| | | this.tableData.forEach(row => { |
| | | row.values.forEach(value => { |
| | | if (value && value.toString().trim() !== '') { |
| | | if (value && value.toString().trim() !== "") { |
| | | count++; |
| | | } |
| | | }); |
| | |
| | | } |
| | | }, |
| | | watch: { |
| | | // 监听 initialData 变化,确保数据正确接收 |
| | | initialData: { |
| | | handler(newData) { |
| | | console.log(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: 'urine_routine', |
| | | data: this.tableData, |
| | | columns: this.dynamicColumns, |
| | | attachments: this.attachments |
| | | }); |
| | | this.saveData(); |
| | | } |
| | | this.$nextTick(() => { |
| | | this.forceTableLayout(); |
| | |
| | | } |
| | | }, |
| | | methods: { |
| | | initTableData() { |
| | | const medicalItems = [ |
| | | // 从外部数据初始化组件 |
| | | initFromExternalData() { |
| | | console.log(this.internalData,'this.internalData'); |
| | | |
| | | 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 = this.getMedicalItems(); |
| | | |
| | | // 如果没有动态列,初始化默认列 |
| | | if (this.dynamicColumns.length === 0) { |
| | | this.dynamicColumns = [ |
| | | { |
| | | itemName: '尿量', |
| | | type: 'number', |
| | | 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("") |
| | | })); |
| | | }, |
| | | |
| | | // 尿常规检测项目定义 |
| | | getMedicalItems() { |
| | | return [ |
| | | { |
| | | itemName: "尿量", |
| | | type: "number", |
| | | required: true, |
| | | unit: 'ml/h', |
| | | reference: '正常范围视情况而定' |
| | | unit: "ml/h", |
| | | reference: "正常范围视情况而定" |
| | | }, |
| | | { |
| | | itemName: '颜色', |
| | | type: 'select', |
| | | itemName: "颜色", |
| | | type: "select", |
| | | required: true, |
| | | options: [ |
| | | { value: '淡黄色', label: '淡黄色' }, |
| | | { value: '黄色', label: '黄色' }, |
| | | { value: '深黄色', label: '深黄色' }, |
| | | { value: '红色', label: '红色' }, |
| | | { value: '白色', label: '白色' }, |
| | | { value: '其他', label: '其他' } |
| | | { value: "淡黄色", label: "淡黄色" }, |
| | | { value: "黄色", label: "黄色" }, |
| | | { value: "深黄色", label: "深黄色" }, |
| | | { value: "红色", label: "红色" }, |
| | | { value: "白色", label: "白色" }, |
| | | { value: "其他", label: "其他" } |
| | | ] |
| | | }, |
| | | { |
| | | itemName: '外观', |
| | | type: 'select', |
| | | itemName: "外观", |
| | | type: "select", |
| | | required: false, |
| | | options: [ |
| | | { value: '清亮', label: '清亮' }, |
| | | { value: '微浊', label: '微浊' }, |
| | | { value: '浑浊', label: '浑浊' }, |
| | | { value: '沉淀', label: '沉淀' }, |
| | | { value: '其他', label: '其他' } |
| | | { value: "清亮", label: "清亮" }, |
| | | { value: "微浊", label: "微浊" }, |
| | | { value: "浑浊", label: "浑浊" }, |
| | | { value: "沉淀", label: "沉淀" }, |
| | | { value: "其他", label: "其他" } |
| | | ] |
| | | }, |
| | | { |
| | | itemName: '尿蛋白', |
| | | type: 'select', |
| | | itemName: "尿蛋白", |
| | | type: "select", |
| | | required: true, |
| | | options: [ |
| | | { value: '-', label: '阴性(-)' }, |
| | | { value: '±', label: '微量(±)' }, |
| | | { value: '+', label: '阳性(+)' }, |
| | | { value: '++', label: '阳性(++)' }, |
| | | { value: '+++', label: '阳性(+++)' } |
| | | { value: "-", label: "阴性(-)" }, |
| | | { value: "±", label: "微量(±)" }, |
| | | { value: "+", label: "阳性(+)" }, |
| | | { value: "++", label: "阳性(++)" }, |
| | | { value: "+++", label: "阳性(+++)" } |
| | | ], |
| | | reference: '正常为阴性(-)' |
| | | reference: "正常为阴性(-)" |
| | | }, |
| | | { |
| | | itemName: 'pH值', |
| | | type: 'number', |
| | | itemName: "pH值", |
| | | type: "number", |
| | | required: true, |
| | | placeholder: '4.5-8.0', |
| | | unit: '', |
| | | reference: '4.5-8.0', |
| | | placeholder: "4.5-8.0", |
| | | unit: "", |
| | | reference: "4.5-8.0", |
| | | min: 4.5, |
| | | max: 8.0 |
| | | }, |
| | | { |
| | | itemName: '白细胞', |
| | | type: 'select', |
| | | itemName: "白细胞", |
| | | type: "select", |
| | | required: true, |
| | | options: [ |
| | | { value: '-', label: '阴性(-)' }, |
| | | { value: '+', label: '阳性(+)' }, |
| | | { value: '++', label: '阳性(++)' }, |
| | | { value: '+++', label: '阳性(+++)' } |
| | | { value: "-", label: "阴性(-)" }, |
| | | { value: "+", label: "阳性(+)" }, |
| | | { value: "++", label: "阳性(++)" }, |
| | | { value: "+++", label: "阳性(+++)" } |
| | | ], |
| | | reference: '正常为阴性(-)' |
| | | reference: "正常为阴性(-)" |
| | | }, |
| | | { |
| | | itemName: '红细胞', |
| | | type: 'number', |
| | | itemName: "红细胞", |
| | | type: "number", |
| | | required: true, |
| | | unit: '/μL', |
| | | reference: '0-9.2', |
| | | unit: "/μL", |
| | | reference: "0-9.2", |
| | | min: 0, |
| | | max: 9.2 |
| | | }, |
| | | { |
| | | itemName: '细菌', |
| | | type: 'number', |
| | | itemName: "细菌", |
| | | type: "number", |
| | | required: true, |
| | | unit: '/μL', |
| | | reference: '0-385', |
| | | unit: "/μL", |
| | | reference: "0-385", |
| | | min: 0, |
| | | max: 385 |
| | | } |
| | | ]; |
| | | }, |
| | | |
| | | this.tableData = medicalItems.map(item => ({ |
| | | ...item, |
| | | values: new Array(this.dynamicColumns.length).fill('') |
| | | })); |
| | | // 保存数据到父组件 |
| | | saveData() { |
| | | const dataToEmit = { |
| | | type: "urine_routine", |
| | | data: this.tableData, |
| | | columns: this.dynamicColumns, |
| | | attachments: this.attachments |
| | | }; |
| | | this.$emit("data-change", dataToEmit); |
| | | }, |
| | | |
| | | getSelectLabel(row, columnIndex) { |
| | | if (!row.options || !row.values[columnIndex]) return row.values[columnIndex]; |
| | | const option = row.options.find(opt => opt.value === row.values[columnIndex]); |
| | | if (!row.options || !row.values[columnIndex]) |
| | | return row.values[columnIndex]; |
| | | const option = row.options.find( |
| | | opt => opt.value === row.values[columnIndex] |
| | | ); |
| | | return option ? option.label : row.values[columnIndex]; |
| | | }, |
| | | |
| | | 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.columnForm = { |
| | | date: column.date, |
| | | time: column.time, |
| | | remark: column.remark || '' |
| | | remark: column.remark || "" |
| | | }; |
| | | this.columnDialogVisible = true; |
| | | }, |
| | | |
| | | confirmAddColumn() { |
| | | this.$refs.columnFormU.validate((valid) => { |
| | | this.$refs.columnFormU.validate(valid => { |
| | | if (!valid) { |
| | | this.$message.warning('请完善时间点信息'); |
| | | this.$message.warning("请完善时间点信息"); |
| | | return; |
| | | } |
| | | |
| | |
| | | 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; |
| | |
| | | |
| | | 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.$message.success("时间点删除成功"); |
| | | }); |
| | | } |
| | | }, |
| | |
| | | handleDialogClosed() { |
| | | this.editingColumnIndex = null; |
| | | this.columnForm = { |
| | | date: '', |
| | | time: '', |
| | | remark: '' |
| | | date: "", |
| | | time: "", |
| | | remark: "" |
| | | }; |
| | | this.$refs.columnFormU && this.$refs.columnFormU.clearValidate(); |
| | | }, |
| | |
| | | }, |
| | | |
| | | handleValueChange(row, columnIndex) { |
| | | this.$emit('data-change', { |
| | | type: 'urine_routine', |
| | | data: this.tableData, |
| | | columns: this.dynamicColumns |
| | | }); |
| | | this.saveData(); |
| | | }, |
| | | |
| | | handleAttachmentChange(fileList) { |
| | | this.attachments = fileList; |
| | | this.$emit('attachment-change', { |
| | | type: 'urine_routine', |
| | | this.$emit("attachment-change", { |
| | | type: "urine_routine", |
| | | attachments: fileList |
| | | }); |
| | | }, |
| | | |
| | | 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.$nextTick(() => { |
| | | if (Object.keys(this.internalData).length === 0) { |
| | | this.initTableData(); |
| | | } |
| | | this.forceTableLayout(); |
| | | }); |
| | | } |
| | | }; |
| | | </script> |
| | |
| | | flex-direction: column; |
| | | gap: 4px; |
| | | } |
| | | } |
| | | |
| | | /* 动画效果 */ |
| | | .fade-enter-active, .fade-leave-active { |
| | | transition: opacity 0.3s; |
| | | } |
| | | .fade-enter, .fade-leave-to { |
| | | opacity: 0; |
| | | } |
| | | </style> |