feat(mrhomepage): 病案首页质量校验

- 新增 IMrHomepageQualityAppService + impl,实现 checkQuality/getQualityResults
- 新增 MrHomepageQualityController (POST /quality/check, GET /quality/results/{id})
- 增强 MrHomepageQualityCheckServiceImpl:必填项+逻辑校验+ICD编码+费用一致性
- 新增 MrHomepageQualityCheck.vue 校验结果展示页面
- 更新前端 API 文件添加 checkQuality/getQualityResults 接口
This commit is contained in:
2026-06-17 14:02:42 +08:00
parent 09e43e4b8c
commit 00604b2d01
8 changed files with 420 additions and 36 deletions

View File

@@ -1,3 +1,5 @@
import request from "@/utils/request"
export function executeQualityCheck(id) { return request({ url: "/api/v1/mr-homepage/quality-check/" + id, method: "post" }) }
export function submitHomepage(id) { return request({ url: "/api/v1/mr-homepage/submit/" + id, method: "put" }) }
export function checkQuality(homepageId) { return request({ url: "/api/v1/mr-homepage/quality/check", method: "post", params: { homepageId } }) }
export function getQualityResults(homepageId) { return request({ url: "/api/v1/mr-homepage/quality/results/" + homepageId, method: "get" }) }

View File

@@ -6,3 +6,5 @@ export function executeQualityCheck(id) { return request({ url: '/api/v1/mr-home
export function getQualityCheck(homepageId) { return request({ url: '/api/v1/mr-homepage/quality-check/' + homepageId, method: 'get' }) }
export function getStatistics(params) { return request({ url: '/api/v1/mr-homepage/statistics', method: 'get', params }) }
export function submitHomepage(id) { return request({ url: '/api/v1/mr-homepage/submit/' + id, method: 'put' }) }
export function checkQuality(homepageId) { return request({ url: '/api/v1/mr-homepage/quality/check', method: 'post', params: { homepageId } }) }
export function getQualityResults(homepageId) { return request({ url: '/api/v1/mr-homepage/quality/results/' + homepageId, method: 'get' }) }

View File

@@ -0,0 +1,171 @@
<template>
<div class="app-container">
<el-card shadow="never">
<template #header>
<div class="card-header">
<span>病案首页质量校验</span>
<div>
<el-input
v-model="homepageId"
placeholder="请输入首页ID"
style="width: 200px; margin-right: 12px"
@keyup.enter="handleCheck"
/>
<el-button
type="primary"
:loading="checking"
@click="handleCheck"
>
执行校验
</el-button>
</div>
</div>
</template>
<div v-if="results">
<el-row :gutter="20" style="margin-bottom: 20px">
<el-col :span="6">
<el-statistic title="校验总数" :value="results.totalChecks" />
</el-col>
<el-col :span="6">
<el-statistic title="通过数" :value="results.passedChecks">
<template #suffix>
<span style="color: #67c23a"> </span>
</template>
</el-statistic>
</el-col>
<el-col :span="6">
<el-statistic title="不合格数" :value="results.failedChecks">
<template #suffix>
<span style="color: #f56c6c"> </span>
</template>
</el-statistic>
</el-col>
<el-col :span="6">
<el-statistic title="质量评分">
<template #default>
<div :style="{ color: scoreColor, fontSize: '28px', fontWeight: 'bold' }">
{{ results.score }}
</div>
</template>
</el-statistic>
</el-col>
</el-row>
<el-tabs v-model="activeTab">
<el-tab-pane label="全部" name="all" />
<el-tab-pane label="不合格项" name="FAIL" />
<el-tab-pane label="合格项" name="PASS" />
</el-tabs>
<el-table
:data="filteredChecks"
stripe
border
style="width: 100%"
>
<el-table-column
prop="checkCategory"
label="校验类别"
width="120"
/>
<el-table-column
prop="checkItem"
label="校验项目"
width="180"
/>
<el-table-column
prop="checkResult"
label="结果"
width="80"
align="center"
>
<template #default="scope">
<el-tag :type="scope.row.checkResult === 'PASS' ? 'success' : 'danger'" size="small">
{{ scope.row.checkResult === 'PASS' ? '合格' : '不合格' }}
</el-tag>
</template>
</el-table-column>
<el-table-column
prop="checkDetail"
label="详情"
show-overflow-tooltip
/>
<el-table-column
prop="suggestion"
label="整改建议"
show-overflow-tooltip
/>
<el-table-column
prop="checkTime"
label="校验时间"
width="170"
/>
</el-table>
</div>
<el-empty v-if="!results && !loading" description="请输入首页ID并执行校验" />
</el-card>
</div>
</template>
<script setup>
import { ref, computed } from 'vue'
import { checkQuality, getQualityResults } from '@/api/mrhomepage'
import { ElMessage } from 'element-plus'
const homepageId = ref('')
const loading = ref(false)
const checking = ref(false)
const results = ref(null)
const activeTab = ref('all')
const scoreColor = computed(() => {
if (!results.value) return '#909399'
if (results.value.score >= 90) return '#67c23a'
if (results.value.score >= 70) return '#e6a23c'
return '#f56c6c'
})
const filteredChecks = computed(() => {
if (!results.value?.checks) return []
if (activeTab.value === 'all') return results.value.checks
return results.value.checks.filter(c => c.checkResult === activeTab.value)
})
const handleCheck = async () => {
if (!homepageId.value) {
ElMessage.warning('请输入首页ID')
return
}
checking.value = true
try {
await checkQuality(homepageId.value)
await loadResults()
ElMessage.success('质量校验完成')
} catch (e) {
ElMessage.error('校验失败: ' + (e.message || '未知错误'))
} finally {
checking.value = false
}
}
const loadResults = async () => {
loading.value = true
try {
results.value = await getQualityResults(homepageId.value)
} catch (e) {
ElMessage.error('获取结果失败')
} finally {
loading.value = false
}
}
</script>
<style scoped>
.card-header {
display: flex;
justify-content: space-between;
align-items: center;
}
</style>