checkData.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502
  1. <template>
  2. <div class="page-check-data h-full">
  3. <div class="flex items-center h-12 py-2 px-4 bg-white">
  4. <div class="flex-1 space-x-4">
  5. <el-tag class="cursor-copy" size="large" v-copy:click="route.query.sn">
  6. <span class="align-middle select-none"
  7. >数据批次号:{{ route.query.sn }}</span
  8. >
  9. <CopyDocument class="inline-block align-middle ml-2 w-4 h-4" />
  10. </el-tag>
  11. <el-tag
  12. v-if="detail.archivesId"
  13. class="cursor-copy"
  14. size="large"
  15. v-copy:click="detail.archivesId || '-'"
  16. >
  17. <span class="align-middle select-none"
  18. >关联档案号:{{ detail.archivesId || "-" }}</span
  19. >
  20. <CopyDocument class="inline-block align-middle ml-2 w-4 h-4" />
  21. </el-tag>
  22. <span class="align-middle">左屏:</span>
  23. <el-select v-model="leftView" class="w-32">
  24. <el-option
  25. v-for="(item, index) in viewOptions"
  26. :key="index"
  27. :value="item.val"
  28. :label="item.label"
  29. :disabled="rightView == item.val"
  30. ></el-option>
  31. </el-select>
  32. <span class="align-middle">右屏:</span>
  33. <el-select v-model="rightView" class="w-32">
  34. <el-option
  35. v-for="(item, index) in viewOptions"
  36. :key="index"
  37. :value="item.val"
  38. :label="item.label"
  39. :disabled="leftView == item.val"
  40. ></el-option>
  41. </el-select>
  42. <template v-if="rightView == 4 && pdfResultList.length > 1">
  43. <el-select v-model="curPdfTemplate" filterable>
  44. <el-option
  45. v-for="(item, index) in pdfResultList"
  46. :key="index"
  47. :label="item.pdfTemplate?.template.name"
  48. :value="index"
  49. ></el-option>
  50. </el-select>
  51. </template>
  52. </div>
  53. <div>
  54. <el-button type="primary" link @click="router.back()">
  55. <Fold class="h-4 w-4 mr-2" />
  56. <span>返回</span>
  57. </el-button>
  58. </div>
  59. </div>
  60. <div class="data-container">
  61. <el-row class="h-full">
  62. <el-col
  63. v-show="containerView.includes(1)"
  64. :style="{
  65. order: containerView.findIndex(v => v == 1)
  66. }"
  67. :span="containerView.length > 1 ? 12 : 24"
  68. class="h-full overflow-y-auto"
  69. >
  70. <div
  71. v-if="detail.files.length > 1 && curFileType != 'img'"
  72. class="h-8 flex justify-center items-center space-x-4"
  73. >
  74. <div
  75. class="flex items-center text-blue-500 hover:text-blue-700 cursor-pointer"
  76. @click="curFileIdx--"
  77. >
  78. <ArrowLeft class="h-4 w-4" /><span>上一份</span>
  79. </div>
  80. <span>{{ curFileIdx + 1 }}/{{ detail.files.length }}</span>
  81. <div
  82. class="flex items-center text-blue-500 hover:text-blue-700 cursor-pointer"
  83. @click="curFileIdx++"
  84. >
  85. <span>下一份</span><ArrowRight class="h-4 w-4" />
  86. </div>
  87. </div>
  88. <template v-if="curFileType == 'pdf'">
  89. <Webview
  90. :url="formatFileUrlProtocol(detail.files[curFileIdx].fileUrl)"
  91. class="w-full h-full"
  92. />
  93. </template>
  94. <template v-else-if="curFileType == 'img'">
  95. <div v-for="(item, index) in detail.files" :key="index">
  96. <img
  97. :src="formatFileUrlProtocol(item.fileUrl)"
  98. alt=""
  99. class="w-full h-fit select-none"
  100. />
  101. </div>
  102. </template>
  103. <template v-else-if="curFileType == 'json'">
  104. <json-viewer
  105. v-if="detail.files[curFileIdx].jsonData"
  106. :value="detail.files[curFileIdx].jsonData"
  107. :show-array-index="false"
  108. boxed
  109. expanded
  110. preview-mode
  111. >
  112. </json-viewer>
  113. </template>
  114. <el-empty
  115. v-else
  116. description="未上传原始数据或者不支持的文件类型"
  117. ></el-empty>
  118. </el-col>
  119. <el-col
  120. v-show="containerView.includes(2)"
  121. :style="{
  122. order: containerView.findIndex(v => v == 2)
  123. }"
  124. :span="containerView.length > 1 ? 12 : 24"
  125. class="h-full overflow-y-auto"
  126. >
  127. <el-card
  128. v-for="(item, index) in transferRawJson"
  129. :key="index"
  130. class="mb-4"
  131. shadow="never"
  132. >
  133. <div class="p-2 text-blue-600">检查数据</div>
  134. <el-table :data="item.bodyData">
  135. <el-table-column
  136. label="身体物质/部位"
  137. prop="name"
  138. ></el-table-column>
  139. <el-table-column
  140. label="检查项目"
  141. prop="itemName"
  142. ></el-table-column>
  143. <el-table-column label="结果" prop="value"></el-table-column>
  144. <el-table-column
  145. label="参考范围"
  146. prop="reference"
  147. ></el-table-column>
  148. <el-table-column label="单位" prop="unit"></el-table-column>
  149. </el-table>
  150. <div class="mt-2 p-2 text-yellow-600">异常项目</div>
  151. <el-table :data="item.abnormalData">
  152. <el-table-column
  153. label="异常项目"
  154. prop="name"
  155. width="200"
  156. ></el-table-column>
  157. <el-table-column label="结果">
  158. <template #default="{ row }">
  159. <div v-if="row.value" class="space-x-2 space-y-2">
  160. <el-tag
  161. v-for="(val, valKey) in row.value"
  162. :key="valKey"
  163. type="warning"
  164. >
  165. <span>{{ valKey }}</span
  166. >:
  167. <span>{{ val }}</span>
  168. </el-tag>
  169. </div>
  170. </template>
  171. </el-table-column>
  172. </el-table>
  173. </el-card>
  174. </el-col>
  175. <el-col
  176. v-show="containerView.includes(3)"
  177. :style="{
  178. order: containerView.findIndex(v => v == 3)
  179. }"
  180. :span="containerView.length > 1 ? 12 : 24"
  181. class="h-full overflow-y-auto"
  182. >
  183. <el-scrollbar>
  184. <json-viewer
  185. v-if="detail.resultRawJson"
  186. :value="detail.resultRawJson"
  187. :show-array-index="false"
  188. boxed
  189. expanded
  190. preview-mode
  191. >
  192. </json-viewer>
  193. </el-scrollbar>
  194. </el-col>
  195. <el-col
  196. v-show="containerView.includes(4)"
  197. :style="{
  198. order: containerView.findIndex(v => v == 4)
  199. }"
  200. :span="containerView.length > 1 ? 12 : 24"
  201. class="h-full"
  202. >
  203. <Webview
  204. v-if="pdfResultList[curPdfTemplate]?.pdfTransferResultUrl"
  205. :url="
  206. formatFileUrlProtocol(
  207. pdfResultList[curPdfTemplate]?.pdfTransferResultUrl
  208. )
  209. "
  210. class="w-full h-full"
  211. />
  212. <el-empty v-else description="暂无分析报告"></el-empty>
  213. </el-col>
  214. <el-col
  215. v-show="containerView.includes(8)"
  216. :style="{
  217. order: containerView.findIndex(v => v == 8)
  218. }"
  219. :span="containerView.length > 1 ? 12 : 24"
  220. class="h-full overflow-y-auto"
  221. >
  222. <component
  223. v-if="[4].includes(detail.type) && CompCheckData"
  224. :is="CompCheckData"
  225. ></component>
  226. </el-col>
  227. <el-col
  228. v-show="containerView.includes(5)"
  229. :style="{
  230. order: containerView.findIndex(v => v == 5)
  231. }"
  232. :span="containerView.length > 1 ? 12 : 24"
  233. class="h-full overflow-y-auto"
  234. >
  235. <component
  236. v-if="[6, 7].includes(detail.type) && VisitsFillData"
  237. :is="VisitsFillData"
  238. ></component>
  239. </el-col>
  240. <el-col
  241. v-show="containerView.includes(6)"
  242. :style="{
  243. order: containerView.findIndex(v => v == 6)
  244. }"
  245. :span="containerView.length > 1 ? 12 : 24"
  246. class="h-full overflow-y-auto"
  247. >
  248. <component
  249. v-if="[2].includes(detail.type) && MonitoringData"
  250. :is="MonitoringData"
  251. ></component>
  252. </el-col>
  253. <el-col
  254. v-show="containerView.includes(7)"
  255. :style="{
  256. order: containerView.findIndex(v => v == 7)
  257. }"
  258. :span="containerView.length > 1 ? 12 : 24"
  259. class="h-full overflow-y-auto"
  260. >
  261. <component
  262. v-if="[3].includes(detail.type) && DiseaseVisitData"
  263. :is="DiseaseVisitData"
  264. :form-show="true"
  265. ></component>
  266. </el-col>
  267. </el-row>
  268. </div>
  269. </div>
  270. </template>
  271. <script setup>
  272. import {
  273. computed,
  274. watch,
  275. ref,
  276. reactive,
  277. onMounted,
  278. shallowRef,
  279. provide,
  280. defineAsyncComponent
  281. } from "vue";
  282. import { useRoute, useRouter } from "vue-router";
  283. import {
  284. CopyDocument,
  285. Fold,
  286. ArrowLeft,
  287. ArrowRight
  288. } from "@element-plus/icons-vue";
  289. import { ElMessage } from "element-plus";
  290. import { request, parseJsonData, findScope } from "@/utils";
  291. import JsonViewer from "vue-json-viewer";
  292. import Webview from "@/components/WebView.vue";
  293. import dayjs from "dayjs";
  294. // import VisitsFillData from './components/VisitsFillData.vue';
  295. const VisitsFillData = defineAsyncComponent(() =>
  296. import("./components/VisitsFillData.vue")
  297. );
  298. const DiseaseVisitData = defineAsyncComponent(() =>
  299. import("./components/DiseaseVisitData.vue")
  300. );
  301. const MonitoringData = defineAsyncComponent(() =>
  302. import("@/views/dataCenter/components/MonitoringData.vue")
  303. );
  304. const CompCheckData = defineAsyncComponent(() =>
  305. import("@/views/dataCenter/components/CompCheckData.vue")
  306. );
  307. const [route, router] = [useRoute(), useRouter()];
  308. const sn = provide("sn", route.query.sn);
  309. const leftView = ref(1);
  310. const rightView = ref(2);
  311. const containerView = computed(() => {
  312. return [leftView.value, rightView.value].filter(v => v > 0);
  313. });
  314. const viewOptions = ref([]);
  315. const originViewOptions = [
  316. {
  317. label: "隐藏视图",
  318. val: -1
  319. },
  320. {
  321. label: "原始数据",
  322. val: 1
  323. }
  324. ];
  325. const viewOptionsMap = {
  326. 1: [
  327. {
  328. label: "处理后的数据",
  329. val: 2
  330. },
  331. // {
  332. // label: "分析结果",
  333. // val: 3
  334. // },
  335. // {
  336. // label: "分析报告",
  337. // val: 4
  338. // }
  339. ],
  340. 2: [
  341. {
  342. label: "处理后的数据",
  343. val: 6
  344. }
  345. ],
  346. 3: [
  347. {
  348. label: "处理后的数据",
  349. val: 7
  350. }
  351. ],
  352. 4: [
  353. {
  354. label: "处理后的数据",
  355. val: 8
  356. }
  357. ],
  358. 6: [
  359. {
  360. label: "填写数据",
  361. val: 5
  362. }
  363. ],
  364. 7: [
  365. {
  366. label: "填写数据",
  367. val: 5
  368. }
  369. ]
  370. };
  371. const defaultView = {
  372. 1: 2,
  373. 2: 6,
  374. 3: 7,
  375. 4: 8,
  376. 6: 5,
  377. 7: 5,
  378. };
  379. const curFileIdx = ref(0);
  380. watch(
  381. () => curFileIdx.value,
  382. idx => {
  383. if (idx < 0) {
  384. curFileIdx.value = detail.value.files.length - 1;
  385. } else if (idx > detail.value.files.length - 1) {
  386. curFileIdx.value = 0;
  387. }
  388. }
  389. );
  390. const detail = ref({
  391. sn: "",
  392. archivesId: "",
  393. transferRawJson: [],
  394. files: []
  395. });
  396. const curFileType = computed(() => {
  397. const curFile = detail.value.files[curFileIdx.value];
  398. if (curFile) {
  399. const afterFix = curFile.fileUrl.split(".").at(-1).toLocaleLowerCase();
  400. if (
  401. [
  402. "gif",
  403. "png",
  404. "jpg",
  405. "jpeg",
  406. "webp",
  407. "svg",
  408. "psd",
  409. "bmp",
  410. "tif"
  411. ].includes(afterFix)
  412. )
  413. return "img";
  414. return afterFix;
  415. }
  416. return "";
  417. });
  418. const archivesId = ref()
  419. const transferRawJson = ref([]);
  420. const transferRawJsonFormId = ref();
  421. const transferRawJsonDiseaseId = ref();
  422. const formatFileUrlProtocol = url => url.replace(/^https?:/, location.protocol);
  423. const getDetail = async () => {
  424. const { data } = await request.get(`/medicalData/detail`, {
  425. params: {
  426. sn: route.query.sn
  427. }
  428. });
  429. for (const file of data.detail.files) {
  430. if (file.fileUrl.split(".").at(-1) == "json") {
  431. const jsonData = await request.get(file.fileUrl);
  432. console.log(jsonData);
  433. file.jsonData = jsonData;
  434. }
  435. }
  436. try {
  437. data.detail.resultRawJson = parseJsonData(data.detail.resultRawJson, []);
  438. const temp = parseJsonData(data.detail.transferRawJson);
  439. transferRawJson.value = temp?.answers || temp || [];
  440. transferRawJsonFormId.value = temp?.formId;
  441. transferRawJsonDiseaseId.value = temp?.diseaseId;
  442. } catch (error) {
  443. console.log(error);
  444. }
  445. viewOptions.value = [
  446. ...originViewOptions,
  447. ...(viewOptionsMap[data.detail.type] || [])
  448. ];
  449. rightView.value = defaultView[data.detail.type];
  450. detail.value = data.detail;
  451. archivesId.value = data.detail.archivesId;
  452. if ([2].includes(data.detail.type)) {
  453. leftView.value = -1;
  454. transferRawJson.value[0].bodyData.forEach(v => {
  455. v.date = dayjs(v.date * 1000).format("YYYY-MM-DD HH:mm:ss");
  456. });
  457. }
  458. getPdfResultList(detail.value);
  459. };
  460. provide("transferRawJson", transferRawJson);
  461. provide("formId", transferRawJsonFormId);
  462. provide("archivesId", archivesId);
  463. provide("transferRawJsonDiseaseId", transferRawJsonDiseaseId);
  464. provide("getDetail", getDetail);
  465. const curPdfTemplate = ref(0);
  466. const pdfResultList = ref([]);
  467. const getPdfResultList = async row => {
  468. const { data } = await request.get(
  469. `/idcService/mechanism/medicalData/pdfResult/list`,
  470. {
  471. params: { sn: row.sn }
  472. }
  473. );
  474. pdfResultList.value = data.list || [];
  475. };
  476. onMounted(() => {
  477. getDetail();
  478. });
  479. </script>
  480. <style lang="scss">
  481. .el-scrollbar__view {
  482. height: 100%;
  483. .main-content.page-check-data {
  484. margin: 0;
  485. }
  486. }
  487. .page-check-data {
  488. .data-container {
  489. height: calc(100% - 48px);
  490. width: 100%;
  491. }
  492. }
  493. </style>