healthPlan.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487
  1. <template>
  2. <div class="page-health-plan p-2 w-full h-full flex flex-col">
  3. <div class="flex items-center p-4 rounded bg-white shadow mb-2">
  4. <!-- <div class="">
  5. </div> -->
  6. <div class="h-full grid grid-cols-2 grid-rows-2 gap-1">
  7. <div
  8. class="col-start-1 col-end-2 row-start-1 row-end-3 rounded p-2 flex items-center cursor-pointer bg-blue-100 hover:bg-blue-200 text-blue-600"
  9. @click="openDiseaseManageDialog"
  10. >
  11. <img src="@/assets/icon-disease.png" alt="" class="w-5 mr-1" />
  12. <span>{{ route.query?.name }}管理</span>
  13. <ArrowRight class="w-4"></ArrowRight>
  14. </div>
  15. <div
  16. class="col-start-2 col-end-3 row-start-1 row-end-2 rounded p-2 flex items-center cursor-pointer bg-green-100 hover:bg-green-200 text-green-600"
  17. @click="historyVisible = true"
  18. >
  19. <Tickets class="w-4 mr-1" />
  20. <span>评估记录</span>
  21. <ArrowRight class="w-4"></ArrowRight>
  22. </div>
  23. <div
  24. class="col-start-2 col-end-3 row-start-2 row-end-3 rounded p-2 flex items-center cursor-pointer bg-red-100 hover:bg-red-200 text-red-600"
  25. @click="router.back()"
  26. >
  27. <SwitchButton class="w-4 mr-1" />
  28. <span>退出管理</span>
  29. <ArrowRight class="w-4"></ArrowRight>
  30. </div>
  31. </div>
  32. <div class="flex items-center justify-around px-4 pt-10 flex-1">
  33. <div
  34. v-for="(item, index) in paths"
  35. :key="index"
  36. class="relative health-path-item py-2 px-4 rounded text-sm select-none cursor-pointer"
  37. :class="item.class"
  38. :style="current == index ? '' : 'opacity: 0.7'"
  39. @click="changePath(item, index)"
  40. >
  41. <span>{{ item.label }}</span>
  42. <div v-if="current == index" class="absolute -right-1 -top-1">
  43. <span class="relative flex h-3 w-3">
  44. <span
  45. class="animate-ping absolute inline-flex h-full w-full rounded-full opacity-75"
  46. :class="item.dotBgClass"
  47. ></span>
  48. <span
  49. class="relative inline-flex rounded-full h-3 w-3"
  50. :class="item.dotClass"
  51. ></span>
  52. </span>
  53. </div>
  54. </div>
  55. </div>
  56. </div>
  57. <div
  58. class="flex-1 w-full rounded bg-white shadow"
  59. style="height: calc(100% - 120px)"
  60. >
  61. <component
  62. v-if="isInit && pathCaches.includes(current) && paths[current].component"
  63. :is="paths[current].component"
  64. ></component>
  65. </div>
  66. <el-dialog title="评估历史详情" v-model="historyVisible" width="85%">
  67. <div>
  68. <component
  69. :is="DialogHealthManageEvaluationHistory"
  70. :list="transformDataExpand(historyList, 'histories')"
  71. ref="historyRef"
  72. ></component>
  73. </div>
  74. </el-dialog>
  75. <el-dialog
  76. v-model="diseaseManageVisible"
  77. :title="route.query.name + '管理'"
  78. >
  79. <div v-if="diseaseManageList.length" class="grid grid-cols-3 gap-4">
  80. <div
  81. v-for="(item, index) in diseaseManageList"
  82. :key="index"
  83. class="bg-white shadow p-4 rounded hover:bg-gray-50 cursor-pointer flex items-center"
  84. @click="getDiseaseManageDetail(item)"
  85. >
  86. <img src="@/assets/icon-disease.png" alt="" class="w-5 mr-1" />
  87. <div class="max-[10em] whitespace-nowrap truncate">
  88. {{ item.label }}
  89. </div>
  90. <ArrowRight class="w-4 text-gray-500" />
  91. </div>
  92. </div>
  93. <el-empty v-else :image="emptyImg"></el-empty>
  94. </el-dialog>
  95. <el-dialog
  96. v-model="diseaseManageDetailVisible"
  97. :title="diseaseManageDetail?.title"
  98. >
  99. <div v-html="diseaseManageDetail?.content"></div>
  100. </el-dialog>
  101. </div>
  102. </template>
  103. <script setup>
  104. import LeaderLine from "@/plugins/leader-line.min.js";
  105. import {
  106. computed,
  107. watch,
  108. markRaw,
  109. ref,
  110. reactive,
  111. onMounted,
  112. provide,
  113. defineAsyncComponent,
  114. nextTick,
  115. onUnmounted
  116. } from "vue";
  117. import { onBeforeRouteLeave, useRoute, useRouter } from "vue-router";
  118. import {
  119. CopyDocument,
  120. Fold,
  121. TrendCharts,
  122. Tickets,
  123. SwitchButton,
  124. ArrowLeft,
  125. ArrowRight
  126. } from "@element-plus/icons-vue";
  127. import { request, sleep, transformDataExpand } from "@/utils";
  128. import emptyImg from "@/assets/icon-empty-left.png";
  129. const DialogHealthManageEvaluationHistory = defineAsyncComponent(() =>
  130. import("./components/DialogHealthManageEvaluationHistory.vue")
  131. );
  132. const DataCollection = defineAsyncComponent(() =>
  133. import("./healthPlan/DataCollection.vue")
  134. );
  135. const AnalysisOfEtiology = defineAsyncComponent(() =>
  136. import("./healthPlan/AnalysisOfEtiology.vue")
  137. );
  138. const HealthPortrait = defineAsyncComponent(() =>
  139. import("./healthPlan/HealthPortrait.vue")
  140. );
  141. const SituationAssessment = defineAsyncComponent(() =>
  142. import("./healthPlan/SituationAssessment.vue")
  143. );
  144. const IndividualInterventionProgram = defineAsyncComponent(() =>
  145. import("./healthPlan/IndividualInterventionProgram.vue")
  146. );
  147. const NextFeedbackSetting = defineAsyncComponent(() =>
  148. import("./healthPlan/NextFeedbackSetting.vue")
  149. );
  150. const [route, router] = [useRoute(), useRouter()];
  151. const current = ref(0);
  152. const pathCaches = ref([0]);
  153. const changePath = (item, index) => {
  154. !pathCaches.value.includes(index) && pathCaches.value.push(index);
  155. current.value = index;
  156. };
  157. const paths = [
  158. {
  159. label: "数据收集",
  160. component: DataCollection,
  161. key: "DataCollection",
  162. dotClass: "bg-green-500",
  163. dotBgClass: "bg-green-400",
  164. class: "text-green-800 bg-green-200" // 绿色代表数据和增长
  165. },
  166. {
  167. label: "病因分析",
  168. component: AnalysisOfEtiology,
  169. key: "AnalysisOfEtiology",
  170. dotClass: "bg-orange-500",
  171. dotBgClass: "bg-orange-400",
  172. class: "text-orange-800 bg-orange-300" // 橙色代表警示和关注
  173. },
  174. {
  175. label: "健康画像",
  176. component: HealthPortrait,
  177. key: "HealthPortrait",
  178. dotClass: "bg-blue-500",
  179. dotBgClass: "bg-blue-400",
  180. class: "text-blue-800 bg-blue-300" // 蓝色代表信任和可靠
  181. },
  182. {
  183. label: "管理效果评价",
  184. component: SituationAssessment,
  185. key: "SituationAssessment",
  186. dotClass: "bg-indigo-500",
  187. dotBgClass: "bg-indigo-400",
  188. class: "text-indigo-800 bg-indigo-300" // 靛青色代表分析和深度思考
  189. },
  190. {
  191. label: "个性干预方案",
  192. component: IndividualInterventionProgram,
  193. key: "IndividualInterventionProgram",
  194. dotClass: "bg-purple-500",
  195. dotBgClass: "bg-purple-400",
  196. class: "text-purple-800 bg-purple-300" // 紫色代表个性化和创新
  197. },
  198. {
  199. label: "下次反馈设定",
  200. component: NextFeedbackSetting,
  201. key: "NextFeedbackSetting",
  202. dotClass: "bg-teal-500",
  203. dotBgClass: "bg-teal-400",
  204. class: "text-teal-800 bg-teal-300" // 青色代表规划和未来
  205. }
  206. ];
  207. const pathLines = ref([]);
  208. const initPathLink = async () => {
  209. pathLines.value = [];
  210. const commonStyle = {
  211. path: "grid",
  212. color: "#0052D4",
  213. gradient: {
  214. startColor: "#65C7F7",
  215. endColor: "#0052D4"
  216. },
  217. dash: {
  218. animation: true
  219. },
  220. size: 2
  221. };
  222. await sleep(6e2);
  223. const els = document.querySelectorAll(".health-path-item");
  224. els.forEach((el, i) => {
  225. let next = els[i + 1];
  226. if (next) {
  227. pathLines.value.push(
  228. new LeaderLine(el, next, {
  229. ...commonStyle,
  230. startSocket: "right",
  231. endSocket: "left",
  232. path: "straight",
  233. endPlug: "arrow1"
  234. })
  235. );
  236. }
  237. });
  238. pathLines.value.push(
  239. new LeaderLine(els[els.length - 1], els[0], {
  240. ...commonStyle,
  241. startSocket: "top",
  242. endSocket: "top",
  243. startPlug: "behind",
  244. endPlug: "arrow1"
  245. })
  246. );
  247. for (let i = 0; i < 6; i++) {
  248. await sleep(16 * 4);
  249. pathLinkLinePosition();
  250. }
  251. };
  252. const pathLinkLinePosition = () => {
  253. pathLines.value.forEach(v => v && v.position && v.position());
  254. };
  255. onMounted(async () => {
  256. await getDetail();
  257. initPathLink();
  258. initDisease();
  259. });
  260. onUnmounted(() => {
  261. pathLines.value.forEach(v => v.remove());
  262. });
  263. const state = ref({
  264. archiveId: route.query.archivesId, // 档案号
  265. disease: route.query.name, // 疾病名称
  266. diseaseId: route.query.diseaseId, // 疾病id
  267. etiologies: [], // 病因数组
  268. effectId: "", // 目前管理效果评价id
  269. target: "", // 下阶段期望效果
  270. nextTime: "", // 下次评估时间
  271. source: Number(route.query.source) || 0,
  272. score: 100,
  273. personalizeInterventions: [
  274. // 个性化干预
  275. // {
  276. // id: "", // 图数据库id
  277. // title: "", // 标题
  278. // content: "", // content
  279. // source: 0 // 历史记录里的source,第一次创建时传0
  280. // }
  281. ],
  282. furtherInfo: [], // 进一步
  283. additionalInfo: "" // 其他补充信
  284. });
  285. const isInit = ref(false)
  286. provide("parentState", state);
  287. const updateStateValueForKey = (key, value, source = "") => {
  288. state.value[key] = value;
  289. console.log(
  290. "updateStateValueForKey",
  291. "修改字段:",
  292. key,
  293. "修改值:",
  294. value,
  295. "完整数据:",
  296. state.value,
  297. "来源:",
  298. source
  299. );
  300. };
  301. provide("updateStateValueForKey", updateStateValueForKey);
  302. const getNodes = async query => {
  303. const { data } = await request.get(`/graphService/open/node/paginate`, {
  304. params: {
  305. page: 1,
  306. pageSize: 9999,
  307. pagesize: 9999,
  308. ...query
  309. }
  310. });
  311. return data;
  312. };
  313. provide("getNodes", getNodes);
  314. const getNodeRelationship = async (id, tag, reverse = undefined) => {
  315. const { data } = await request.get(`/graphService/open/node/relationship`, {
  316. params: {
  317. id,
  318. rName: tag,
  319. reverse
  320. }
  321. });
  322. return data;
  323. };
  324. provide("getNodeRelationship", getNodeRelationship);
  325. const getBatchNodeRelationships = async (ids, tag, reverse = undefined) => {
  326. const { data } = await request.get(`/graphService/open/node/relationships`, {
  327. params: {
  328. ids,
  329. rName: tag,
  330. reverse
  331. }
  332. });
  333. return data;
  334. };
  335. provide("getBatchNodeRelationships", getBatchNodeRelationships);
  336. const getArticle = async id => {
  337. const { data } = await request.get("/articleService/open/article", {
  338. params: {
  339. id,
  340. mechanismId: "algor"
  341. }
  342. });
  343. return data;
  344. };
  345. provide("getArticle", getArticle);
  346. const diseaseMangeNode = ref();
  347. const initDisease = async () => {
  348. const nodes = await getNodeRelationship(
  349. route.query.diseaseId,
  350. "功能医学管理"
  351. );
  352. if (nodes[0]) {
  353. diseaseMangeNode.value = nodes[0].mId;
  354. }
  355. };
  356. provide("diseaseMangeNode", diseaseMangeNode);
  357. const openDiseaseManageDialog = () => {
  358. getDiseaseManageList();
  359. diseaseManageVisible.value = true;
  360. };
  361. const diseaseManageVisible = ref(false);
  362. const diseaseManageList = ref(false);
  363. const getDiseaseManageList = async () => {
  364. const data = await getNodeRelationship(route.query.diseaseId, "疾病相关知识");
  365. diseaseManageList.value = (data || []).map(v => {
  366. return {
  367. id: v.mId,
  368. label: v.mProperties.name,
  369. articleId: v.mProperties["文章ID"]
  370. };
  371. });
  372. };
  373. const diseaseManageDetailVisible = ref(false);
  374. const diseaseManageDetail = ref();
  375. const getDiseaseManageDetail = async item => {
  376. diseaseManageDetail.value = await getArticle(item.articleId);
  377. diseaseManageDetailVisible.value = true;
  378. };
  379. const historyVisible = ref(false);
  380. const historyList = ref([]);
  381. const getDetail = async () => {
  382. const { data } = await request.get(
  383. "/platformApi/dataService/healthManage/detail",
  384. {
  385. params: {
  386. archiveId: route.query.archivesId
  387. }
  388. }
  389. );
  390. historyList.value = data.filter(v => v.diseaseId == route.query.diseaseId);
  391. const last = historyList.value[0];
  392. console.log('last', last)
  393. if (last) {
  394. state.value["diseaseId"] = last.diseaseId;
  395. state.value["etiologies"] = last.etiologies || [];
  396. state.value["score"] = last.score;
  397. state.value["target"] = last.target;
  398. state.value["nextTime"] = last.nextTime;
  399. state.value["personalizeInterventions"] =
  400. last.personalizeInterventions || [];
  401. state.value["furtherInfo"] = last.furtherInfo || [];
  402. state.value["additionalInfo"] = last.additionalInfo;
  403. }
  404. isInit.value =true
  405. };
  406. provide("healthHistoryList", historyList);
  407. const getSurveyFormByCode = async customFields => {
  408. if (!customFields.length) return [];
  409. const { data } = await request.post(
  410. `/formService/mechanism/form/detailByCustomFields`,
  411. {
  412. customFields
  413. }
  414. );
  415. return data?.formFields || [];
  416. };
  417. provide("getSurveyFormByCode", getSurveyFormByCode);
  418. const getFormFieldsSurveyIds = fields => {
  419. let ids = [];
  420. fields.forEach(v => {
  421. if (v.fields?.length) {
  422. v.fields.forEach(filed => {
  423. if (filed.extra?.subjectIds) {
  424. ids.push(...filed.extra.subjectIds);
  425. } else {
  426. filed.extra?.sn && ids.push(filed.extra?.sn);
  427. }
  428. });
  429. }
  430. if (v.extra?.subjectIds) {
  431. ids.push(...v.extra.subjectIds);
  432. } else {
  433. v.extra?.sn && ids.push(v.extra.sn);
  434. }
  435. });
  436. ids = [...new Set(ids)];
  437. return ids;
  438. };
  439. provide("getFormFieldsSurveyIds", getFormFieldsSurveyIds);
  440. const getFormAnswer = async ids => {
  441. if (!ids.length) return;
  442. const { data } = await request.post(
  443. `/archivesService/mechanism/form/data/query`,
  444. {
  445. archivesId: route.query.archivesId,
  446. ids
  447. }
  448. );
  449. return (data.list || []).map(v => {
  450. return {
  451. ...v,
  452. answer: v.returnAnswer
  453. };
  454. });
  455. };
  456. provide("getFormAnswer", getFormAnswer);
  457. </script>
  458. <style lang="scss">
  459. .el-scrollbar__view {
  460. height: 100%;
  461. .main-content.page-health-plan {
  462. margin: 0;
  463. }
  464. }
  465. .page-health-plan {
  466. overflow: hidden;
  467. .data-container {
  468. height: calc(100% - 48px);
  469. width: 100%;
  470. }
  471. }
  472. </style>