checkitemLibrary.vue 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754
  1. <template>
  2. <div class="page-channel p-6">
  3. <div class="h-full w-full flex">
  4. <div class="mr-2">
  5. <el-card shadow="never" class="h-full">
  6. <template #header>
  7. <div class="flex items-center justify-between">
  8. <div class="flex items-center">
  9. <el-input
  10. v-model.trim="query.name"
  11. clearable
  12. placeholder="输入机构名称进行搜索"
  13. />
  14. <el-button
  15. type="primary"
  16. class="ml-2"
  17. @click="onSearch"
  18. :icon="Search"
  19. >搜索</el-button
  20. >
  21. </div>
  22. <el-button
  23. type="primary"
  24. :icon="Plus"
  25. class="ml-2"
  26. @click="
  27. editVisible = true;
  28. editForm = {
  29. id: undefined,
  30. name: '',
  31. comment: ''
  32. };
  33. "
  34. >新增</el-button
  35. >
  36. </div>
  37. </template>
  38. <div>
  39. <el-table
  40. :data="state.list"
  41. class="w-full"
  42. :show-header="false"
  43. max-height="85vh"
  44. @row-click="editRow"
  45. >
  46. <el-table-column prop="name" label="姓名">
  47. <template #default="{ row }">
  48. <div
  49. class="p-2 hover:bg-gray-50 cursor-pointer rounded"
  50. :class="{
  51. 'bg-gray-10 border-r-4 border-blue-500':
  52. currentValue?.id == row.id
  53. }"
  54. >
  55. <h3>{{ row.name }}</h3>
  56. <el-text>备注:{{ row.comment }}</el-text> <br />
  57. <el-text>最后更新时间:{{ row.updatedAt }}</el-text>
  58. </div>
  59. </template>
  60. </el-table-column>
  61. </el-table>
  62. <el-pagination
  63. background
  64. class="justify-end mt-4"
  65. layout="total, prev, pager, next"
  66. :page-size="10"
  67. :total="state.total"
  68. @change="handleCurrentChange"
  69. />
  70. </div>
  71. </el-card>
  72. </div>
  73. <div class="flex-1 h-full">
  74. <div v-if="currentValue?.id" class="h-full flex flex-col bg-white">
  75. <div class="p-4 border-b border-solid border-gray-200">
  76. <div class="flex items-center justify-between">
  77. <div>
  78. <h4>{{ currentValue.name }}</h4>
  79. <el-text>使用范围:{{ currentValue.comment }}</el-text>
  80. </div>
  81. <div class="ml-8">
  82. <el-button type="primary" :icon="Edit" link @click="handleEdit"
  83. >编辑</el-button
  84. >
  85. <el-popconfirm
  86. title="确定要删除这个机构吗?"
  87. width="300"
  88. @confirm="handleDelete"
  89. >
  90. <template #reference>
  91. <el-button type="danger" :icon="Delete" link
  92. >删除</el-button
  93. >
  94. </template>
  95. </el-popconfirm>
  96. </div>
  97. </div>
  98. </div>
  99. <div class="flex-1 p-4 overflow-y-auto">
  100. <div class="flex items-center">
  101. <el-button type="primary" @click="showEditCheckItem(null)"
  102. >新增检查项目</el-button
  103. >
  104. <div class="flex flex-1 justify-end">
  105. <el-input
  106. v-model="checkItem.query.name"
  107. class="w-64"
  108. placeholder="请输入检查项目名称"
  109. clearable
  110. @keyup.enter="
  111. checkItem.query.page = 1;
  112. getCheckItemList();
  113. "
  114. />
  115. <el-button
  116. type="primary"
  117. class="ml-2"
  118. @click="
  119. checkItem.query.page = 1;
  120. getCheckItemList();
  121. "
  122. >搜索</el-button
  123. >
  124. </div>
  125. </div>
  126. <el-table
  127. :data="checkItem.list"
  128. border
  129. class="mt-4"
  130. v-loading="checkItem.loading"
  131. element-loading-text="加载中..."
  132. >
  133. <el-table-column label="检查项目名称" prop="name" />
  134. <el-table-column label="检查项目类型">
  135. <template #default="{ row }">
  136. {{ ["检验", "检查"][row.type - 1] }}
  137. </template>
  138. </el-table-column>
  139. <el-table-column label="检查内容">
  140. <template #default="{ row }">
  141. <div>
  142. {{ row.checkItems.map(v => v.name).join() }}
  143. </div>
  144. </template>
  145. </el-table-column>
  146. <el-table-column label="操作" width="200">
  147. <template #default="{ row }">
  148. <el-button
  149. type="primary"
  150. :icon="Edit"
  151. link
  152. @click="showEditCheckItem(row)"
  153. >编辑</el-button
  154. >
  155. <el-popconfirm
  156. title="确定要删除这个检查项目吗?"
  157. width="300"
  158. @confirm="handleDeleteCheckItem(row)"
  159. >
  160. <template #reference>
  161. <el-button type="danger" :icon="Delete" link
  162. >删除</el-button
  163. >
  164. </template>
  165. </el-popconfirm>
  166. </template>
  167. </el-table-column></el-table
  168. >
  169. <el-pagination
  170. background
  171. class="justify-end mt-4"
  172. layout="total, prev, pager, next"
  173. v-model:current-page="checkItem.query.page"
  174. :total="checkItem.total"
  175. @current-change="
  176. p => {
  177. checkItem.query.page = p;
  178. getCheckItemList();
  179. }
  180. "
  181. />
  182. </div>
  183. </div>
  184. <el-card shadow="never" class="h-full" v-else>
  185. <div>
  186. <el-empty />
  187. </div>
  188. </el-card>
  189. </div>
  190. </div>
  191. <el-dialog title="检查机构管理" v-model="editVisible">
  192. <div>
  193. <el-form ref="formRef" :model="editForm" label-width="120px">
  194. <el-form-item
  195. label="检查机构名称:"
  196. prop="name"
  197. :rules="[
  198. {
  199. required: true,
  200. message: '请输入机构名称',
  201. trigger: 'blur'
  202. }
  203. ]"
  204. >
  205. <el-input
  206. v-model="editForm.name"
  207. placeholder="请输入机构名称"
  208. clearable
  209. />
  210. </el-form-item>
  211. <el-form-item label="备注:" prop="comment">
  212. <el-input
  213. v-model="editForm.comment"
  214. placeholder="请输入备注"
  215. clearable
  216. />
  217. </el-form-item>
  218. </el-form>
  219. </div>
  220. <template #footer>
  221. <div>
  222. <el-button @click="editVisible = false">取 消</el-button>
  223. <el-button type="primary" @click="saveEditTemplate">保 存</el-button>
  224. </div>
  225. </template>
  226. </el-dialog>
  227. <el-dialog
  228. v-model="editCheckItem.visible"
  229. title="检查项目编辑"
  230. :close-on-press-escape="false"
  231. :close-on-click-modal="false"
  232. destroy-on-close
  233. width="80%"
  234. >
  235. <el-form>
  236. <el-form-item label="检查项目名称">
  237. <el-input v-model="editCheckItem.row.name" />
  238. </el-form-item>
  239. <el-form-item label="系统体检项目">
  240. <el-select
  241. v-model="editCheckItem.row.systemCheck"
  242. placeholder="请选择"
  243. filterable
  244. >
  245. <el-option
  246. v-for="item in editCheckItem.systemCheckList"
  247. :key="item.id"
  248. :label="item.label"
  249. :value="item.id"
  250. />
  251. </el-select>
  252. </el-form-item>
  253. <el-form-item label="检查项目类型">
  254. <el-select v-model="editCheckItem.row.type" placeholder="请选择">
  255. <el-option label="检验" :value="1" />
  256. <el-option label="检查" :value="2" />
  257. </el-select>
  258. </el-form-item>
  259. <el-form-item label="检查内容">
  260. <el-button
  261. type="primary"
  262. plain
  263. :icon="Plus"
  264. @click="
  265. state.massAddvisible = true;
  266. state.content = '';
  267. "
  268. >批量添加</el-button
  269. >
  270. </el-form-item>
  271. </el-form>
  272. <el-table :data="editCheckItem.row.checkItems">
  273. <el-table-column
  274. :label="editCheckItem.row.type == 1 ? '检验项目名称' : '检验项目名称'"
  275. >
  276. <template #default="{ row }">
  277. <el-input v-model="row.name" placeholder="请输入" />
  278. </template>
  279. </el-table-column>
  280. <template v-if="editCheckItem.row.type == 1">
  281. <el-table-column label="单位">
  282. <template #default="{ row }">
  283. <el-input v-model="row.unit" placeholder="请输入" />
  284. </template>
  285. </el-table-column>
  286. <el-table-column label="参考范围">
  287. <template #default="{ row }">
  288. <el-input v-model="row.scope" placeholder="请输入" /> </template
  289. ></el-table-column>
  290. </template>
  291. <el-table-column label="排序">
  292. <template #default="{ row }">
  293. <el-input v-model.number="row.sort" placeholder="请输入" />
  294. </template>
  295. </el-table-column>
  296. <el-table-column
  297. v-for="(col, colIdx) in editCheckItem.row.type == 1
  298. ? ['系统身体物质/部位', '系统检查项目']
  299. : ['系统检查项目', '系统身体物质/部位']"
  300. :label="col"
  301. >
  302. <template #default="{ row, $index }">
  303. <template v-if="col == '系统身体物质/部位'">
  304. <el-select
  305. v-model="row.node"
  306. :disabled="editCheckItem.row.type == 2 && !row.matchProject"
  307. placeholder="请输入搜索后选择"
  308. filterable
  309. clearable
  310. collapse-tags
  311. remote
  312. :class="{
  313. 'no-match': !row.node
  314. }"
  315. :multiple="editCheckItem.row.type == 2"
  316. :remote-method="
  317. query =>
  318. editCheckItem.row.type == 1 && searchBodyNodes(query, row)
  319. "
  320. @change="matchNodeChange(row)"
  321. @clear="matchNodeClear(row)"
  322. >
  323. <el-option
  324. v-for="(project, projectIdx) in row.nodes"
  325. :key="project.id"
  326. :label="project.label"
  327. :value="project.id"
  328. class="h-auto"
  329. >
  330. <div class="">
  331. <div>{{ project.label }}</div>
  332. <div v-if="project.matched" class="text-gray-500 text-xs">
  333. 匹配名称:{{ project.matched }}
  334. </div>
  335. </div>
  336. </el-option>
  337. </el-select>
  338. </template>
  339. <template v-else>
  340. <el-select
  341. v-model="row.matchProject"
  342. :disabled="editCheckItem.row.type == 1 && !row.nodes.length"
  343. placeholder="请输入搜索后选择"
  344. filterable
  345. clearable
  346. remote
  347. :class="{
  348. 'no-match': !row.matchProject
  349. }"
  350. :remote-method="query => searchOcrIndicator(query, row)"
  351. @change="matchProjectChange(row)"
  352. @clear="matchProjectClear(row)"
  353. >
  354. <el-option
  355. v-for="(project, projectIdx) in row.indicators"
  356. :key="project.id"
  357. :label="project.label"
  358. :value="project.id"
  359. class="h-auto"
  360. >
  361. <div class="">
  362. <div>{{ project.label }}</div>
  363. <div v-if="project.matched" class="text-gray-500 text-xs">
  364. 匹配名称:{{ project.matched }}
  365. </div>
  366. </div>
  367. </el-option>
  368. </el-select>
  369. </template>
  370. </template></el-table-column
  371. >
  372. <el-table-column label="操作">
  373. <template #default="{ row, $index }">
  374. <el-button
  375. type="danger"
  376. link
  377. @click="editCheckItem.row.checkItems.splice($index, 1)"
  378. >
  379. 删除
  380. </el-button>
  381. </template>
  382. </el-table-column>
  383. </el-table>
  384. <div>
  385. <el-button
  386. type="primary"
  387. link
  388. class="mt-2"
  389. @click="addEditCheckItemLine"
  390. >
  391. + 新增一行</el-button
  392. >
  393. </div>
  394. <template #footer>
  395. <el-button type="primary" @click="saveEditCheckItem">保存</el-button>
  396. </template>
  397. </el-dialog>
  398. <el-dialog title="" v-model="state.massAddvisible" width="50%">
  399. <div>
  400. <el-input
  401. type="textarea"
  402. v-model="state.content"
  403. :rows="18"
  404. placeholder="输入检查内容,多个检查内容之间用中文输入法下的分号或顿号区隔"
  405. />
  406. </div>
  407. <template #footer>
  408. <el-button @click="state.massAddvisible = false">取 消</el-button>
  409. <el-button type="primary" @click="handleMassAddContent"
  410. >确 定</el-button
  411. >
  412. </template>
  413. </el-dialog>
  414. </div>
  415. </template>
  416. <script setup>
  417. import { ref, reactive } from "vue";
  418. import { Search, Plus, Edit, Delete } from "@element-plus/icons-vue";
  419. import { request, unique } from "@/utils";
  420. import { ElMessage } from "element-plus";
  421. const state = reactive({
  422. visible: false,
  423. list: [],
  424. total: 0,
  425. rowData: {},
  426. content: "",
  427. massAddvisible: false
  428. });
  429. const query = reactive({
  430. title: "",
  431. page: 1
  432. });
  433. const onSearch = () => {
  434. query.page = 1;
  435. getList();
  436. };
  437. const getList = async () => {
  438. const { data } = await request.get(
  439. "dataService/schemeManage/medicalCheck/hospital/paginate",
  440. { params: query }
  441. );
  442. state.list = data.list;
  443. state.total = data.total;
  444. if (query.page == 1 && data.list[0]) {
  445. editRow(data.list[0]);
  446. }
  447. };
  448. const handleCurrentChange = page => {
  449. query.page = page;
  450. getList();
  451. };
  452. const handleDelete = async () => {
  453. const { message } = await request.post(
  454. `dataService/schemeManage/medicalCheck/hospital/delete`,
  455. {
  456. id: currentValue.value.id
  457. }
  458. );
  459. ElMessage.success(message);
  460. getList();
  461. };
  462. getList();
  463. const editRow = row => {
  464. currentValue.value = row;
  465. checkItem.query.hospitalId = row.id;
  466. getCheckItemList();
  467. };
  468. const editForm = ref({
  469. id: "",
  470. name: "",
  471. comment: ""
  472. });
  473. const editVisible = ref(false);
  474. const currentValue = ref({
  475. id: ""
  476. });
  477. const handleEdit = () => {
  478. editForm.value = currentValue.value;
  479. editVisible.value = true;
  480. };
  481. const saveEditTemplate = async () => {
  482. const { message } = await request.post(
  483. `dataService/schemeManage/medicalCheck/hospital${
  484. editForm.value.id ? "/update" : ""
  485. }`,
  486. editForm.value
  487. );
  488. ElMessage.success(message);
  489. editVisible.value = false;
  490. if (!editForm.value.id) {
  491. onSearch();
  492. }
  493. getList();
  494. };
  495. const checkItem = reactive({
  496. query: {
  497. hospitalId: "",
  498. name: "",
  499. page: 1
  500. },
  501. list: [],
  502. total: 0,
  503. rowData: {}
  504. });
  505. const editCheckItem = reactive({
  506. visible: false,
  507. systemCheckList: [],
  508. row: {
  509. id: undefined,
  510. name: "",
  511. systemCheck: "",
  512. type: 1,
  513. checkItems: []
  514. }
  515. });
  516. const relatedRelationship = ["异常属于", "可检测"];
  517. const getCheckItemList = async () => {
  518. const { data } = await request.get(
  519. "dataService/schemeManage/medicalCheck/item/paginate",
  520. { params: checkItem.query }
  521. );
  522. checkItem.list = data.list || [];
  523. checkItem.total = data.total;
  524. };
  525. const showEditCheckItem = async row => {
  526. if (row) {
  527. editCheckItem.row = row;
  528. } else {
  529. editCheckItem.row = {
  530. id: undefined,
  531. name: "",
  532. systemCheck: "",
  533. type: 1,
  534. checkItems: []
  535. };
  536. }
  537. editCheckItem.visible = true;
  538. editCheckItem.systemCheckList = await getNodes({
  539. tag: "体检项目"
  540. });
  541. };
  542. const getNodes = async (query = {}) => {
  543. const { data } = await request.get(`/graphService/open/node/paginate`, {
  544. params: {
  545. page: 1,
  546. pageSize: 9999,
  547. ...query
  548. }
  549. });
  550. return (data.list || [])
  551. .map(v => {
  552. return {
  553. id: v.id,
  554. label: v.properties.name
  555. };
  556. })
  557. .sort((a, b) => a.label?.length - b.label?.length);
  558. };
  559. const addEditCheckItemLine = () => {
  560. editCheckItem.row.checkItems.push({
  561. name: "",
  562. unit: "",
  563. scope: "",
  564. sort: "",
  565. node: "",
  566. nodes: [],
  567. matchProject: [],
  568. indicators: []
  569. });
  570. };
  571. const deleteEditCheckItemLine = index => {
  572. editCheckItem.row.checkItems.splice(index, 1);
  573. };
  574. const saveEditCheckItem = async () => {
  575. console.log(editCheckItem);
  576. // return
  577. const { message } = await request.post(
  578. `dataService/schemeManage/medicalCheck/item${
  579. editCheckItem.row.id ? "/update" : ""
  580. }`,
  581. {
  582. ...editCheckItem.row,
  583. hospitalId: checkItem.query.hospitalId
  584. }
  585. );
  586. ElMessage.success(message);
  587. editCheckItem.visible = false;
  588. getCheckItemList();
  589. };
  590. const handleDeleteCheckItem = async row => {
  591. const { message } = await request.post(
  592. `dataService/schemeManage/medicalCheck/item/delete`,
  593. {
  594. id: row.id,
  595. hospitalId: checkItem.query.hospitalId
  596. }
  597. );
  598. ElMessage.success(message);
  599. getCheckItemList();
  600. };
  601. const searchRelateNodes = async (id, relationship, query = {}) => {
  602. const { data } = await request.get(`/graphService/open/node/related`, {
  603. params: { id, relationship: relatedRelationship[relationship], ...query }
  604. });
  605. return (data || [])
  606. .map(v => {
  607. return {
  608. id: v.id,
  609. label: v.properties.name
  610. };
  611. })
  612. .sort((a, b) => a.label?.length - b.label?.length);
  613. };
  614. const searchBodyNodes = async (key, row) => {
  615. if (!key) return;
  616. const nodes =
  617. (await getNodes({
  618. name: key,
  619. tag: "身体物质/部位"
  620. })) || [];
  621. const dataIds = nodes.map(v => v.id);
  622. console.log(row);
  623. const findItem = row.nodes.find(v => v.id == row.node);
  624. row.nodes =
  625. findItem && !dataIds.includes(findItem.id) ? [findItem, ...nodes] : nodes;
  626. };
  627. const matchNodeChange = async row => {
  628. console.log(row);
  629. // console.log(item, row, item.nodes[rowIdx]);
  630. const appendChildren = [];
  631. if (editCheckItem.row.type == 1) {
  632. const curNode = row.nodes.find(v => row.node == v.id);
  633. const projects = await searchRelateNodes(row.node, 1);
  634. row.matchProject = "";
  635. row.indicators = projects;
  636. }
  637. if (editCheckItem.row.type == 2) {
  638. const curNodes = row.node.filter(v => {
  639. const curNode = row.nodes.find(v2 => v == v2.id);
  640. return (
  641. curNode.properties["类型编号"] &&
  642. Number(curNode.properties["类型编号"][0]) == 0
  643. );
  644. });
  645. console.log(curNodes);
  646. if (curNodes?.length) {
  647. const data = await searchRelateNodes(curNodes.join(","), 0);
  648. data.length &&
  649. appendChildren.push({
  650. id: row.node,
  651. abnormal: data
  652. });
  653. }
  654. }
  655. };
  656. const matchNodeClear = row => {
  657. if (editCheckItem.row.type == 1) {
  658. row.nodes = [];
  659. row.matchProject = "";
  660. row.indicators = [];
  661. }
  662. };
  663. const searchOcrIndicator = async (key, row) => {
  664. if (!key) return;
  665. const nodes =
  666. (await getNodes({
  667. name: key,
  668. tag: "医学检查项目"
  669. })) || [];
  670. const dataIds = nodes.map(v => v.id);
  671. const findItem = row.indicators.find(v => v.id == row.matchProject);
  672. row.indicators =
  673. findItem && !dataIds.includes(findItem.id) ? [findItem, ...nodes] : nodes;
  674. };
  675. const matchProjectChange = async row => {
  676. if (editCheckItem.row.type == 2) {
  677. console.log(row);
  678. const nodes = await searchRelateNodes(row.matchProject, 1);
  679. row.node = [];
  680. row.nodes = nodes;
  681. }
  682. };
  683. const matchProjectClear = (item, rowIdx) => {
  684. if (editCheckItem.row.type == 2) {
  685. row.node = [];
  686. row.nodes = [];
  687. row.matchProject = "";
  688. row.indicators = [];
  689. }
  690. };
  691. const handleMassAddContent = () => {
  692. try {
  693. const contentArray = state.content
  694. .replaceAll(";", "、")
  695. .split("、")
  696. .filter(v => v);
  697. const array = [];
  698. contentArray.map(v => {
  699. array.push({
  700. name: v,
  701. unit: "",
  702. scope: "",
  703. sort: "",
  704. node: "",
  705. nodes: [],
  706. matchProject: [],
  707. indicators: []
  708. });
  709. });
  710. // 去重
  711. editCheckItem.row.checkItems = unique(
  712. [...editCheckItem.row.checkItems, ...array],
  713. "name"
  714. );
  715. } catch (error) {
  716. ElMessage.error(error);
  717. } finally {
  718. state.massAddvisible = false;
  719. }
  720. };
  721. </script>
  722. <style lang="scss">
  723. .el-scrollbar__view {
  724. height: 100%;
  725. .main-content.page-channel {
  726. height: 100%;
  727. margin: 0;
  728. }
  729. }
  730. ::v-deep {
  731. .el-card__header {
  732. height: 74px;
  733. }
  734. .el-table .el-table__body tr:hover > td {
  735. background-color: transparent;
  736. }
  737. }
  738. </style>