checkAfterScanning.vue 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682
  1. <template>
  2. <view class="checkAfterScanning">
  3. <view class="checkAfterScanning_title">标本核对</view>
  4. <view class="page_item_cont">
  5. <view class="page_item_cont_T">
  6. <view class="page_item_cont_title">
  7. <view> 工单单号 </view>
  8. <view class="text_big">{{info.gdCode}}</view>
  9. </view>
  10. <view class="page_item_cont_title">
  11. <view class="text_title"> 科室名称 </view>
  12. <view class="text_big">{{info.LCDept}}</view>
  13. </view>
  14. <view class="page_item_cont_title">
  15. <view> 扫描标本数量 </view>
  16. <view class="text_big" @click="goToSpeDetail()"><text class="underline">{{info.total}}</text>只</view>
  17. </view>
  18. <view class="page_item_cont_title2">
  19. 检验科室标本数量:
  20. </view>
  21. <view class="page_item_cont_title" v-for="(item, i) in info.data" :key="i">
  22. <view>{{item[1]}}</view>
  23. <view class="text_big" @click="goToSpeDetail(item[1],item[2])"><text class="underline">{{item[3]}}</text>只</view>
  24. </view>
  25. </view>
  26. </view>
  27. <view class="foot_btn_spe" v-if="enterDynamicDigitalKey==0">
  28. <view class="btn1" @click="Scanning_complete('scan')">核对完成</view>
  29. <view class="btn3" @click="goBack">取消</view>
  30. </view>
  31. <view class="foot_btn_spe" v-if="enterDynamicDigitalKey==1">
  32. <view class="btn1" @click="Scanning_complete('scan')">扫一扫核对</view>
  33. <view class="btn1" @click="isShowKey()">数字核对</view>
  34. <view class="btn3" @click="goBack">返回</view>
  35. </view>
  36. <!-- 弹窗 -->
  37. <showModel :title="models.title" :icon="models.icon" :disjunctor="models.disjunctor" :content="models.content"
  38. @ok="ok" @cancel="cancel" :operate="models.operate"></showModel>
  39. <!-- 弹窗 -->
  40. <showModel :title="models2.title" :icon="models2.icon" :disjunctor="models2.disjunctor" :content="models2.content"
  41. @ok="ok2" @know="know2" @cancel="cancel2" :operate="models2.operate"></showModel>
  42. <!-- 动态密钥 -->
  43. <numberKeyModel v-if="showKey" :keyArr="keyArr" @cancel="showKey = false" @confirm="confirmKey($event)"></numberKeyModel>
  44. </view>
  45. </template>
  46. <script>
  47. import {
  48. get,
  49. post,
  50. SM,
  51. webHandle
  52. } from "../../http/http.js";
  53. import numberKeyModel from "../../components/numberKeyModel/numberKeyModel.vue";
  54. export default {
  55. data() {
  56. return {
  57. handoverId: undefined,
  58. handoverDeptId: undefined,
  59. info: {},
  60. wcFlag: false,
  61. queryObj: {}, //路由传递过来的数据
  62. // 弹窗model
  63. models: {
  64. disjunctor: false,
  65. },
  66. // 完成扫描弹窗model1
  67. models2: {
  68. disjunctor: false,
  69. },
  70. showKey:false,
  71. keyNum:4, //密钥位数
  72. isNumberKey:false, //是否开启动态密钥
  73. keyArr: [],
  74. enterDynamicDigitalKey:null,
  75. keyType:null,
  76. contentData:null,
  77. };
  78. },
  79. components: {
  80. numberKeyModel
  81. },
  82. methods: {
  83. confirmKey(data){
  84. this.Scanning_complete('key')
  85. this.contentData = data
  86. this.showKey = false
  87. },
  88. isShowKey(){
  89. this.showKey = true
  90. },
  91. // 获取配置
  92. getConfig() {
  93. // 判断开关
  94. const userData = uni.getStorageSync("userData");
  95. let postData = {
  96. idx: 0,
  97. sum: 9999,
  98. hospitalConfig:{
  99. hosId:userData.user.currentHospital.id,
  100. model:"all"
  101. }
  102. };
  103. post("/simple/data/fetchDataList/hospitalConfig",postData).then((result) => {
  104. if (result.status == 200) {
  105. this.keyArr = []
  106. for(let i of result.list){
  107. if(i.key=='digitalSecretKey'){
  108. if(i.value==1){
  109. this.isNumberKey = true
  110. }else{
  111. this.isNumberKey = false
  112. }
  113. }else if(i.key=='numberDigitalSecretKey'){
  114. this.keyNum = Number(i.value)
  115. }
  116. }
  117. for(let i = 0; i < this.keyNum; i++){
  118. this.keyArr.push({
  119. value:null
  120. })
  121. }
  122. }
  123. });
  124. uni.showLoading({
  125. title: "加载中",
  126. mask: true,
  127. });
  128. // 查询标本配送业务
  129. post("/simple/data/fetchDataList/taskType",{
  130. "idx": 0,
  131. "sum": 10,
  132. "taskType": {
  133. "simpleQuery": true,
  134. "hosId": {
  135. "id": userData.user.currentHospital.id
  136. },
  137. "associationType": {
  138. "key": "association_types",
  139. "value": "specimen"
  140. }
  141. }
  142. }).then((res) => {
  143. if (res.status == 200) {
  144. let taskTypeDTO = res.list[0];
  145. if(taskTypeDTO){
  146. // 查询业务页面控制-标本
  147. post("/simple/data/fetchDataList/taskTypeConfig",{
  148. "idx": 0,
  149. "sum": 10,
  150. "taskTypeConfig": {
  151. taskTypeDTO,
  152. }
  153. }).then((res) => {
  154. if (res.status == 200) {
  155. let data = res.list[0];
  156. uni.hideLoading();
  157. if(data){
  158. // 输入动态数字密钥
  159. this.enterDynamicDigitalKey = data.enterDynamicDigitalKey
  160. console.log(55555,this.enterDynamicDigitalKey)
  161. }
  162. } else {
  163. uni.hideLoading();
  164. uni.showToast({
  165. icon: "none",
  166. title: res.msg || "接口获取数据失败!",
  167. });
  168. }
  169. });
  170. }else{
  171. uni.hideLoading();
  172. }
  173. } else {
  174. uni.hideLoading();
  175. uni.showToast({
  176. icon: "none",
  177. title: res.msg || "接口获取数据失败!",
  178. });
  179. }
  180. });
  181. },
  182. // 获取核对信息
  183. getInfo(gdId){
  184. uni.showLoading({
  185. title: '加载中',
  186. mask: true
  187. });
  188. post("/api/specimensCheck", {gdIds:gdId}).then((res) => {
  189. uni.hideLoading();
  190. if (res.status == 200) {
  191. console.log(res)
  192. res.LCDept = res.LCDept.join(',');
  193. this.info = res;
  194. } else {
  195. uni.showToast({
  196. icon: "none",
  197. title: res.msg || "接口获取数据失败!",
  198. });
  199. }
  200. })
  201. },
  202. // 返回
  203. goBack() {
  204. uni.navigateBack();
  205. },
  206. // 业务页面控制-标本-收取时需扫描二维码
  207. beforeScanning(fn) {
  208. this.handoverId = undefined;//扫动态码或静态码,后端会返回交接人,核对完成接口传过去
  209. this.handoverDeptId = undefined;//扫动态码或静态码,后端会返回科室,核对完成接口传过去
  210. // 判断开关
  211. const userData = uni.getStorageSync("userData");
  212. console.log('userData', userData);
  213. uni.showLoading({
  214. title: "加载中",
  215. mask: true,
  216. });
  217. // 查询标本配送业务
  218. post("/simple/data/fetchDataList/taskType",{
  219. "idx": 0,
  220. "sum": 10,
  221. "taskType": {
  222. "simpleQuery": true,
  223. "hosId": {
  224. "id": userData.user.currentHospital.id
  225. },
  226. "associationType": {
  227. "key": "association_types",
  228. "value": "specimen"
  229. }
  230. }
  231. }).then((res) => {
  232. if (res.status == 200) {
  233. let taskTypeDTO = res.list[0];
  234. if(taskTypeDTO){
  235. // 查询业务页面控制-标本
  236. post("/simple/data/fetchDataList/taskTypeConfig",{
  237. "idx": 0,
  238. "sum": 10,
  239. "taskTypeConfig": {
  240. taskTypeDTO,
  241. }
  242. }).then((res) => {
  243. if (res.status == 200) {
  244. let data = res.list[0];
  245. if(data){
  246. // 收取时需扫描二维码
  247. if(data.arriveScanCode == 1){
  248. // 扫描科室二维码
  249. uni.hideLoading();
  250. if(this.keyType=='key'){
  251. let postData = {
  252. code: this.contentData,
  253. type: 'specimen',
  254. hosId: userData.user.currentHospital.id,
  255. orderId: +this.queryObj.id,
  256. };
  257. console.log(postData)
  258. uni.showLoading({
  259. title: "加载中",
  260. mask: true,
  261. });
  262. post("/workerOrder/arriveScanCheck", postData).then((res) => {
  263. uni.hideLoading();
  264. if (res.state == 200) {
  265. this.handoverId = res.handover;//扫动态码或静态码,后端会返回交接人,核对完成接口传过去
  266. this.handoverDeptId = res.handoverDeptId;//扫动态码或静态码,后端会返回科室,核对完成接口传过去
  267. fn();
  268. } else {
  269. uni.showToast({
  270. icon: "none",
  271. title: res.msg || "接口获取数据失败!",
  272. });
  273. }
  274. })
  275. }else{
  276. SM().then((content) => {
  277. let postData = {
  278. code: content,
  279. type: 'specimen',
  280. hosId: userData.user.currentHospital.id,
  281. orderId: +this.queryObj.id,
  282. };
  283. console.log(postData)
  284. uni.showLoading({
  285. title: "加载中",
  286. mask: true,
  287. });
  288. post("/workerOrder/arriveScanCheck", postData).then((res) => {
  289. uni.hideLoading();
  290. if (res.state == 200) {
  291. this.handoverId = res.handover;//扫动态码或静态码,后端会返回交接人,核对完成接口传过去
  292. this.handoverDeptId = res.handoverDeptId;//扫动态码或静态码,后端会返回科室,核对完成接口传过去
  293. fn();
  294. } else {
  295. uni.showToast({
  296. icon: "none",
  297. title: res.msg || "接口获取数据失败!",
  298. });
  299. }
  300. })
  301. }).catch(err => {
  302. });
  303. }
  304. }else{
  305. uni.hideLoading();
  306. fn();
  307. }
  308. }else{
  309. uni.hideLoading();
  310. fn();
  311. }
  312. } else {
  313. uni.hideLoading();
  314. uni.showToast({
  315. icon: "none",
  316. title: res.msg || "接口获取数据失败!",
  317. });
  318. }
  319. });
  320. }else{
  321. uni.hideLoading();
  322. uni.showToast({
  323. icon: "none",
  324. title: "未查询到标本配送业务!",
  325. });
  326. }
  327. } else {
  328. uni.hideLoading();
  329. uni.showToast({
  330. icon: "none",
  331. title: res.msg || "接口获取数据失败!",
  332. });
  333. }
  334. });
  335. },
  336. // 完成核对
  337. Scanning_complete(type) {
  338. this.keyType = type
  339. this.beforeScanning(() => {
  340. if (
  341. this.queryObj.type1 == "plan-spe-ddd-2" ||
  342. this.queryObj.type1 == "spe-ddd-2"
  343. ) {
  344. this.wcFlag = true;
  345. this.models = {
  346. disjunctor: true,
  347. title: "提示",
  348. content: "是否确定标本已核对完成?",
  349. icon: "warn",
  350. operate: {
  351. ok: "确定",
  352. cancel: "取消",
  353. },
  354. };
  355. } else if (
  356. this.queryObj.type1 == "plan-spe-dsd-2" ||
  357. this.queryObj.type1 == "plan-spe-dsd-3" ||
  358. this.queryObj.type1 == "spe-dsd-2" ||
  359. this.queryObj.type1 == "spe-dsd-3"
  360. ) {
  361. this.wcFlag = false;
  362. this.models = {
  363. disjunctor: true,
  364. title: "提示",
  365. content: "是否确定标本已核对完成?",
  366. icon: "warn",
  367. operate: {
  368. ok: "确定",
  369. cancel: "取消",
  370. },
  371. };
  372. } else {
  373. this.gotoOver();
  374. }
  375. });
  376. },
  377. // 跳转完成工单页面
  378. gotoOver() {
  379. uni.navigateTo({
  380. url: `../scanning/scanning?type=${this.queryObj.type}&type1=${
  381. this.queryObj.type1
  382. }&id=${encodeURIComponent(JSON.stringify([this.queryObj.id]))}&deptCode=${
  383. this.queryObj.deptCode
  384. }&dept=${this.queryObj.dept}&speNum=${this.queryObj.speNum}&content=${this.queryObj.content}`,
  385. });
  386. },
  387. // 确定
  388. ok() {
  389. this.models.disjunctor = false;
  390. let postData = {
  391. ids: [this.queryObj.id],
  392. sign: true,
  393. deptQrCode: this.queryObj.deptCode
  394. };
  395. uni.showLoading({
  396. title: '加载中',
  397. mask: true
  398. });
  399. post("/workerOrder/expectedAndActual", postData).then((ress) => {
  400. uni.hideLoading();
  401. if (ress.status == 200) {
  402. if (this.wcFlag) {
  403. //正常完成扫描
  404. this.overFinish()
  405. } else {
  406. this.gotoOver();
  407. }
  408. } else if (ress.status == 1000035) {
  409. let content = '';
  410. if (this.queryObj.type1 === 'spe-ddd-2' || this.queryObj.type1 === 'plan-spe-ddd-2') {
  411. //待到达
  412. content =
  413. `系统内预计标本<strong class="red">${ress.expectReceiveNum}</strong>只,您扫描收取标本<strong class="red">${ress.actualReceiveNum}</strong>只,其中<strong class="red">${ress.notReceiveNum}</strong>只未扫描;`;
  414. } else {
  415. //待送达
  416. this.content =
  417. `本工单已签到<strong class="red">${ress.scanSet?ress.scanSet.join('、'):''}</strong>,剩余需签到科室<strong class="red">${ress.notScanSet?ress.notScanSet.join('、'):''}</strong>,总签收<strong class="red">${ress.totalAcceptance}</strong>只,剩余<strong class="red">${ress.notAcceptance}</strong>只未签收,您确定完成工单吗?`;
  418. content =
  419. `您在<strong class="red">${ress.deptName}</strong>检验科扫描了<strong class="red">${ress.deptScanNum}</strong>个标本,还需扫描<strong class="red">${ress.deptNotScanNum}</strong>标本。`;
  420. }
  421. // 取消弹窗 2022年10月10日
  422. this.ok2();
  423. // this.models2 = {
  424. // disjunctor: true,
  425. // title: "提示",
  426. // content,
  427. // icon: "warn",
  428. // operate: {
  429. // ok: "确定",
  430. // cancel: "取消",
  431. // },
  432. // };
  433. } else {
  434. uni.showToast({
  435. icon: "none",
  436. title: ress.msg || "接口获取数据失败!",
  437. });
  438. }
  439. })
  440. },
  441. // 取消
  442. cancel() {
  443. this.models.disjunctor = false;
  444. },
  445. // 确定
  446. ok2() {
  447. this.models2.disjunctor = false;
  448. let postData = {
  449. ids: [this.queryObj.id],
  450. deptQrCode: this.queryObj.deptCode
  451. };
  452. uni.showLoading({
  453. title: '加载中',
  454. mask: true
  455. });
  456. post("/workerOrder/expectedAndActual", postData).then((ress) => {
  457. uni.hideLoading();
  458. if (ress.status == 200) {
  459. if (this.wcFlag) {
  460. this.overFinish()
  461. } else {
  462. this.gotoOver();
  463. }
  464. } else {
  465. uni.showToast({
  466. icon: "none",
  467. title: ress.msg || "接口获取数据失败!",
  468. });
  469. }
  470. })
  471. },
  472. // 取消
  473. cancel2() {
  474. this.models2.disjunctor = false;
  475. },
  476. know2() {
  477. this.models2.disjunctor = false;
  478. uni.navigateTo({
  479. url: "../receiptpage/receiptpage",
  480. });
  481. },
  482. // 正常完成扫描
  483. overFinish() {
  484. let data = {
  485. type: this.queryObj.type1,
  486. ids: [this.queryObj.id],
  487. };
  488. //只要是标本轮巡1对多或者多对多
  489. // plan-spe-dsd-2 1对多
  490. // plan-spe-dsd-3 多对多
  491. if (
  492. this.queryObj.type1 === "plan-spe-ddd-2" ||
  493. this.queryObj.type1 === "plan-spe-dsd-2" ||
  494. this.queryObj.type1 === "plan-spe-dsd-3"
  495. ) {
  496. post("/workerOrder/finishPlanSpes", data).then((res) => {
  497. // uni.hideLoading()
  498. if (res.status == 200) {
  499. this.models2 = {
  500. disjunctor: true,
  501. title: "提示",
  502. content: `完成扫描成功`,
  503. icon: "success",
  504. operate: {
  505. know: "知道了",
  506. },
  507. };
  508. } else {
  509. uni.showToast({
  510. icon: "none",
  511. title: res.msg || "接口获取数据失败!",
  512. });
  513. }
  514. });
  515. } else {
  516. data.handover = this.handoverId || undefined;
  517. data.handoverDeptId = this.handoverDeptId || undefined;
  518. post("/workerOrder/finishSpes", data).then((res) => {
  519. // uni.hideLoading()
  520. if (res.status == 200) {
  521. uni.navigateTo({
  522. url: "../receiptpage/receiptpage",
  523. });
  524. } else {
  525. uni.showToast({
  526. icon: "none",
  527. title: res.msg || "接口获取数据失败!",
  528. });
  529. }
  530. });
  531. }
  532. },
  533. // 跳转到已扫描标本详情页
  534. goToSpeDetail(deptName='', qrCode=''){
  535. uni.navigateTo({
  536. url: `../noScanSpecimen/noScanSpecimen?deptName=${deptName}&workOrderId=${encodeURIComponent(JSON.stringify([this.queryObj.id]))}&deptCode=${qrCode}&isScan=1&specimensCheck=1`,
  537. });
  538. }
  539. },
  540. onLoad(options) {
  541. console.log(options);
  542. this.queryObj = options;
  543. this.getInfo(this.queryObj.id);
  544. this.getConfig();
  545. // #ifdef APP-PLUS
  546. webHandle("no", "app");
  547. // #endif
  548. // #ifdef H5
  549. webHandle("no", "wx");
  550. // #endif
  551. },
  552. };
  553. </script>
  554. <style lang="less" scoped>
  555. .checkAfterScanning {
  556. .checkAfterScanning_title {
  557. padding: 50rpx 0;
  558. font-size: 46rpx;
  559. font-weight: 550;
  560. text-align: center;
  561. border-bottom: 1px solid #ccc;
  562. }
  563. .page_item_cont {
  564. min-height: 90rpx;
  565. padding: 0 20rpx;
  566. text-align: left;
  567. position: relative;
  568. .text_title{
  569. flex-shrink: 0;
  570. margin-right: 16rpx;
  571. }
  572. .text_big {
  573. font-size: 32rpx;
  574. font-weight: 700;
  575. margin-top: 10rpx;
  576. .underline{
  577. text-decoration: underline;
  578. }
  579. }
  580. .page_item_cont_T {
  581. padding-top: 28rpx;
  582. padding-bottom: 28rpx;
  583. font-size: 28rpx;
  584. .page_item_cont_title {
  585. height: 100%;
  586. font-size: 32rpx;
  587. display: flex;
  588. justify-content: space-between;
  589. align-items: center;
  590. }
  591. .page_item_cont_title2{
  592. margin-top: 36rpx;
  593. margin-bottom: 18rpx;
  594. height: 100%;
  595. font-size: 32rpx;
  596. font-weight: bold;
  597. display: flex;
  598. align-items: center;
  599. }
  600. }
  601. .page_item_cont_B {
  602. padding-top: 28rpx;
  603. margin-bottom: 28rpx;
  604. .page_item_cont_title {
  605. font-size: 32rpx;
  606. display: flex;
  607. justify-content: space-between;
  608. }
  609. .page_item_cont_title1 {
  610. height: 60rpx;
  611. line-height: 60rpx;
  612. font-size: 32rpx;
  613. padding-left: 64rpx;
  614. }
  615. }
  616. }
  617. .foot_btn {
  618. line-height: 88rpx;
  619. height: 100rpx;
  620. margin-top: 40rpx;
  621. display: flex;
  622. justify-content: center;
  623. .btn1,
  624. .btn2,
  625. .btn3 {
  626. height: 88rpx;
  627. flex: 1;
  628. margin: 0 1%;
  629. background-image: linear-gradient(to right, #72c172, #3bb197);
  630. color: #fff;
  631. border-radius: 8rpx;
  632. font-size: 32rpx;
  633. margin-top: 16rpx;
  634. text-align: center;
  635. }
  636. }
  637. .foot_btn_spe {
  638. line-height: 88rpx;
  639. height: 100rpx;
  640. margin-top: 40rpx;
  641. text-align: center;
  642. display: flex;
  643. justify-content: space-between;
  644. flex-wrap: wrap;
  645. &::after {
  646. content: '';
  647. flex: 1;
  648. }
  649. view {
  650. height: 88rpx;
  651. width: 48%;
  652. margin: 0 1%;
  653. background-image: linear-gradient(to right, #72c172, #3bb197);
  654. color: #fff;
  655. border-radius: 8rpx;
  656. font-size: 32rpx;
  657. margin-top: 16rpx;
  658. }
  659. }
  660. }
  661. </style>