UserDataAnalysis.vue 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. <template>
  2. <div v-if="resultRawJson" class="bg-white p-4">
  3. <el-descriptions
  4. v-if="resultRawJson?.result"
  5. title="用户信息"
  6. :column="2"
  7. border
  8. class="mb-8"
  9. >
  10. <el-descriptions-item label="姓名">{{
  11. resultRawJson.result.userInfo.name
  12. }}</el-descriptions-item>
  13. <el-descriptions-item label="性别">{{
  14. resultRawJson.result.userInfo.gender
  15. }}</el-descriptions-item>
  16. <el-descriptions-item label="体检年龄">{{
  17. resultRawJson.result.userInfo.examAge
  18. }}</el-descriptions-item>
  19. <el-descriptions-item label="体检日期">
  20. <!-- {{resultRawJson.result.assessmentDate}} -->
  21. {{ dayjs(resultRawJson.result.examDate * 1000).format("YYYY-MM-DD") }}
  22. </el-descriptions-item>
  23. </el-descriptions>
  24. <page-title>
  25. <template #icon>
  26. <img
  27. src="@/assets/riskAnalysis/icon-overflow.png"
  28. alt=""
  29. class="h-full"
  30. />
  31. </template>
  32. <span>{{ "重大疾病风险概览" }}</span>
  33. </page-title>
  34. <div v-for="(item, index) in overViewList" :key="index">
  35. <div class="break-inside-avoid mb-12">
  36. <sub-title :type="rankLabel[item.rank].label">
  37. <span>{{ item.title }}</span>
  38. </sub-title>
  39. <SectionText :type="rankLabel[item.rank].label">{{
  40. item.note
  41. }}</SectionText>
  42. </div>
  43. <table
  44. border="0"
  45. v-if="item.children.length"
  46. class="border-collapse w-full mb-4 text-sm"
  47. >
  48. <tr class="page-break-avoid-auto page-break-inside-avoid">
  49. <td
  50. v-for="(column, columnIndex) in riskTableHeader[item.rank] || [
  51. '疾病名称',
  52. '患病风险',
  53. '重要已知风险因素',
  54. '对应科室'
  55. ]"
  56. :key="columnIndex"
  57. class="border border-gray-200 px-2 h-[48px] whitespace-nowrap"
  58. :class="rankLabel[item.rank]?.thClass"
  59. >
  60. {{ column }}
  61. </td>
  62. </tr>
  63. <tr
  64. v-for="(child, childIdx) in item.children"
  65. :key="childIdx"
  66. class="page-break-inside-avoid"
  67. >
  68. <td class="border border-gray-200">
  69. <div class="px-2 py-3">
  70. {{ child.name }}
  71. </div>
  72. </td>
  73. <td class="border border-gray-200">
  74. <div class="px-2 py-3 whitespace-nowrap">
  75. <span :class="riskDegreeLabel[child.riskDegree]?.class">{{
  76. // riskDegreeLabel[child.riskDegree].label
  77. child.riskDesc
  78. }}</span>
  79. </div>
  80. </td>
  81. <template v-if="riskTableHeader[item.rank]">
  82. <td class="border border-gray-200">
  83. <div class="px-2 py-3">
  84. {{
  85. Array.isArray(child.recommendItems)
  86. ? child.recommendItems.join()
  87. : child.recommendItems
  88. }}
  89. </div>
  90. </td>
  91. </template>
  92. <td v-else class="border border-gray-200">
  93. <div class="px-2 py-3">
  94. <div
  95. v-for="(know, knowIdx) in child.knownFactors"
  96. :key="knowIdx"
  97. class=""
  98. >
  99. <span class="align-middle">{{ know.itemName }}:</span>
  100. <span class="align-middle">{{
  101. Array.isArray(know.value) ? know.value.join() : know.value
  102. }}</span>
  103. <el-icon
  104. v-if="know.mark > 0"
  105. class="inline-block align-middle w-4 h-4 text-red-600"
  106. ><Top />
  107. </el-icon>
  108. <el-icon
  109. v-if="know.mark < 0"
  110. class="inline-block align-middle w-4 h-4 text-green-600"
  111. ><Bottom />
  112. </el-icon>
  113. </div>
  114. </div>
  115. </td>
  116. <td class="border border-gray-200">
  117. <div class="px-2 py-3">
  118. {{
  119. Array.isArray(child.department)
  120. ? child.department.join("/")
  121. : child.department || "/"
  122. }}
  123. </div>
  124. </td>
  125. </tr>
  126. </table>
  127. <div
  128. v-else
  129. class="break-inside-avoid flex flex-col items-center justify-center py-6 mb-4 border border-solid border-gray-200 rounded-b"
  130. >
  131. <img src="@/assets/riskAnalysis/empty-img.png" class="w-40" />
  132. <div class="text-gray-500 text-xs text-center mt-4">暂无分析数据</div>
  133. </div>
  134. </div>
  135. <el-empty v-if="!overViewList?.length" description="暂无数据"></el-empty>
  136. <UserDataAnalysisEvaluate
  137. v-for="(item, index) in 3"
  138. :key="index"
  139. :classification="index + 1"
  140. :latest="true"
  141. />
  142. <template v-if="suggestion">
  143. <page-title :bg="['#1DB198', '#7DFCBC']">
  144. <template #icon>
  145. <img
  146. src="@/assets/riskAnalysis/icon-interpretation.png"
  147. alt=""
  148. class="h-full"
  149. />
  150. </template>
  151. <span>个性化健康建议</span>
  152. </page-title>
  153. <div
  154. v-if="suggestion"
  155. class="relative mb-4 rounded p-4 text-gray-800 leading-6"
  156. >
  157. <div v-html="suggestion"></div>
  158. </div>
  159. <div
  160. v-else
  161. class="break-inside-avoid flex flex-col items-center justify-center py-6 mb-4"
  162. >
  163. <img src="@/assets/riskAnalysis/empty-img.png" class="w-40" />
  164. <div class="text-gray-500 text-xs text-center mt-4">暂无分析数据</div>
  165. </div>
  166. </template>
  167. <component
  168. v-if="
  169. props.needHorizontal && route.query.archivesId && HorizontalContrast
  170. "
  171. :is="HorizontalContrast"
  172. :archivesId="route.query.archivesId"
  173. :hideCtrl="true"
  174. >
  175. <template #title1>
  176. <page-title>
  177. <template #icon>
  178. <img
  179. src="@/assets/riskAnalysis/icon-tumor.png"
  180. alt=""
  181. class="h-full"
  182. />
  183. </template>
  184. <span>疾病风险评估对比及趋势</span>
  185. </page-title>
  186. </template>
  187. <template #title2>
  188. <page-title class="page-break-avoid-auto">
  189. <template #icon>
  190. <img
  191. src="@/assets/riskAnalysis/icon-tumor.png"
  192. alt=""
  193. class="h-full"
  194. />
  195. </template>
  196. <span>异常项目横向对比</span>
  197. </page-title>
  198. </template>
  199. </component>
  200. <template v-if="props.needHorizontal && compareSuggestionStatus == 4">
  201. <page-title :bg="['#1DB198', '#7DFCBC']">
  202. <template #icon>
  203. <img
  204. src="@/assets/riskAnalysis/icon-interpretation.png"
  205. alt=""
  206. class="h-full"
  207. />
  208. </template>
  209. <span>综合个性化健康建议</span>
  210. </page-title>
  211. <div class="relative mb-4 rounded p-4 text-gray-800 leading-6">
  212. <div v-html="compareSuggestion"></div>
  213. </div>
  214. <!-- <div
  215. v-else
  216. class="break-inside-avoid flex flex-col items-center justify-center py-6 mb-4"
  217. >
  218. <img src="@/assets/riskAnalysis/empty-img.png" class="w-40" />
  219. <div class="text-gray-500 text-xs text-center mt-4">暂无分析数据</div>
  220. </div> -->
  221. </template>
  222. </div>
  223. <el-empty v-else></el-empty>
  224. </template>
  225. <script setup>
  226. import {
  227. computed,
  228. shallowRef,
  229. watch,
  230. ref,
  231. nextTick,
  232. inject,
  233. reactive,
  234. onMounted,
  235. provide,
  236. defineAsyncComponent
  237. } from "vue";
  238. import { useRoute, useRouter } from "vue-router";
  239. import { ElMessage, ElMessageBox } from "element-plus";
  240. import { deduplicateArrayByKeys } from "@/utils";
  241. import dayjs from "dayjs";
  242. import PageTitle from "@/components/PageTitle.vue";
  243. import SubTitle from "@/components/SubTitle.vue";
  244. import SectionText from "@/components/SectionText.vue";
  245. import { Top, Bottom } from "@element-plus/icons-vue";
  246. import UserDataAnalysisEvaluate from "./UserDataAnalysisEvaluate.vue";
  247. const HorizontalContrast = defineAsyncComponent(() =>
  248. import("./HorizontalContrast.vue")
  249. );
  250. const [route, router] = [useRoute(), useRouter()];
  251. const props = defineProps({
  252. needHorizontal: false
  253. });
  254. const riskDegreeToRank = {
  255. 1: [4, 3],
  256. 2: [2],
  257. 3: [1],
  258. 4: [0]
  259. };
  260. const riskTableHeader = {
  261. 3: [
  262. "疾病名称",
  263. "患病风险",
  264. // '建议检查项目',
  265. // '建议就医时间',
  266. "建议随访项目",
  267. // '建议随访时间',
  268. "对应科室"
  269. ],
  270. 4: [
  271. "疾病名称",
  272. "患病风险",
  273. "建议完善项目",
  274. // '建议完善时间',
  275. "对应科室"
  276. ]
  277. };
  278. const riskDegreeLabel = {
  279. 4: {
  280. label: "已确诊",
  281. class: "font-bold text-red-500"
  282. },
  283. 3: {
  284. label: "有明显风险",
  285. class: "font-bold text-red-600"
  286. },
  287. 2: {
  288. label: "有一定风险",
  289. class: "font-bold text-yellow-500"
  290. },
  291. 1: {
  292. label: "暂未发现风险",
  293. class: "text-green-500"
  294. },
  295. 0: {
  296. label: "暂未发现风险",
  297. class: "text-gray-500"
  298. }
  299. };
  300. const rankLabel = {
  301. 1: { label: "danger", thClass: "text-red-500 bg-red-50" },
  302. 2: { label: "warning", thClass: "text-yellow-500 bg-yellow-50" },
  303. 3: { label: "primary", thClass: "text-blue-500 bg-blue-50" },
  304. 4: { label: "warning", thClass: "text-yellow-500 bg-yellow-50" }
  305. };
  306. const resultRawJson = inject("resultRawJson")
  307. // const latestReport = inject("latestReport");
  308. const suggestion = inject("suggestion");
  309. const compareSuggestion = inject("compareSuggestion");
  310. const compareSuggestionStatus = inject("compareSuggestionStatus");
  311. const overViewList = ref();
  312. const formatOverView = () => {
  313. overViewList.value = (
  314. resultRawJson.value?.result?.reportConfig.reportPreviewCfg.sectionContent ||
  315. []
  316. )
  317. .map(v => {
  318. const tempItem = { ...v, children: [] };
  319. const children = riskDegreeToRank[v.rank]
  320. .map(riskDegree => {
  321. return resultRawJson.value.result.majorDisease
  322. .filter(v2 => v2.riskDegree == riskDegree)
  323. .sort((a, b) => b.riskDegree - a.riskDegree);
  324. })
  325. .flat();
  326. tempItem.children = deduplicateArrayByKeys(children, ["nodeId"]);
  327. return tempItem;
  328. })
  329. .filter(v => v.children.length);
  330. console.log("overViewList.value", overViewList.value, resultRawJson.value);
  331. };
  332. watch(
  333. () => resultRawJson.value,
  334. () => {
  335. console.log("resultRawJson.value", resultRawJson.value);
  336. formatOverView();
  337. },
  338. { immediate: true }
  339. );
  340. </script>
  341. <style lang="scss"></style>