orderList.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563
  1. <template>
  2. <view class="orderList">
  3. <!-- 头部筛选 -->
  4. <view class="orderList_header">
  5. <view class="orderList_header_item" @click="openPicker('gdState')">
  6. <text class="orderList_header_itemText">{{selectedGdState}}</text>
  7. <text class="pda pda-xiala"></text>
  8. </view>
  9. <view class="orderList_header_item" @click="openPicker('associationType')">
  10. <text class="orderList_header_itemText">{{selectedAssociationType}}</text>
  11. <text class="pda pda-xiala"></text>
  12. </view>
  13. </view>
  14. <!-- 列表 -->
  15. <view class="orderList_list">
  16. <view class="orderList_listItem" v-for="newOrder in newOrderList" :key="newOrder.id"
  17. @click="toDetail(newOrder.id)">
  18. <image class="ji" src="../../static/imgs/icon_ji.png" mode="widthFix"
  19. v-if="newOrder.emergencyType && newOrder.emergencyType.value == 2"></image>
  20. <image class="jiaji" src="../../static/imgs/icon_jiaji.png" mode="widthFix"
  21. v-if="newOrder.emergencyType && newOrder.emergencyType.value == 3"></image>
  22. <view class="orderList_listItem_header">
  23. <view class="orderList_listItem_header_title">
  24. <view class="associationType_icon" v-if="newOrder.isHalfInspect === 1"> 半 </view>
  25. <view class="associationType_icon" v-else-if="newOrder.taskType.associationType.value == 'specimen'">标
  26. </view>
  27. <view class="associationType_icon" v-else-if="newOrder.taskType.associationType.value == 'drugsBag'"> 药
  28. </view>
  29. <view class="associationType_icon" v-else-if="newOrder.taskType.associationType.value == 'specimenPlan'"> 巡
  30. </view>
  31. <view class="associationType_icon" v-else-if="newOrder.taskType.associationType.value == 'jPBag'"> 静 </view>
  32. <view class="associationType_icon" v-else-if="newOrder.taskType.associationType.value == 'inspect'"> 病
  33. </view>
  34. <view class="associationType_icon"
  35. v-else-if="newOrder.taskType.associationType.value == 'patientTransport'"> 病
  36. </view>
  37. <view class="associationType_icon" v-else-if="newOrder.taskType.associationType.value == 'other'"> 其
  38. </view>
  39. <view class="taskNameAndWorkerName">
  40. <text class="taskName">{{newOrder.isHalfInspect === 1? "半程陪检": newOrder.taskType.taskName}}</text>
  41. <text class="workerName"
  42. v-if="newOrder.patient &&(newOrder.taskType.associationType.value =='patientTransport' ||newOrder.taskType.associationType.value =='inspect')">-{{newOrder.patient.patientName}}</text>
  43. </view>
  44. </view>
  45. <text class="orderList_listItem_header_more"
  46. :class="newOrder.stateTextClass">{{newOrder.gdState.name }}</text>
  47. </view>
  48. <view class="orderList_listItem_item">
  49. <view class="orderList_listItem_item_content">
  50. <text class="orderList_listItem_item_name"
  51. v-if="newOrder.worker">{{newOrder.worker.name}}{{newOrder.worker.phone? "-" + newOrder.worker.phone: ""}}</text>
  52. <text class="orderList_listItem_item_name" v-else>暂未接单</text>
  53. <text class="orderList_listItem_item_time" v-if="newOrder.showCreateTime">{{newOrder.showCreateTime}}</text>
  54. <text class="orderList_listItem_item_time"
  55. v-else-if="newOrder.yyjdTime && newOrder.gdState.value == GDSTATE['定时预约']">{{newOrder.yyjdTime | formatDate('MM-dd HH:mm')}}</text>
  56. </view>
  57. <view class="orderList_listItem_item_btns" v-if="
  58. newOrder.gdState.value == GDSTATE['待抢单'] ||
  59. newOrder.gdState.value == GDSTATE['待接单'] ||
  60. newOrder.gdState.value == GDSTATE['待到达'] ||
  61. newOrder.gdState.value == GDSTATE['定时预约'] ||
  62. newOrder.gdState.value == GDSTATE['待评价']
  63. ">
  64. <button type="primary" class="btn" v-if="newOrder.gdState.value == GDSTATE['待评价']"
  65. @click.stop="showAppraise(newOrder.id)">评价</button>
  66. <button type="primary" class="btn" v-if="
  67. newOrder.gdState.value == GDSTATE['待抢单'] ||
  68. newOrder.gdState.value == GDSTATE['待接单'] ||
  69. newOrder.gdState.value == GDSTATE['待到达'] ||
  70. newOrder.gdState.value == GDSTATE['定时预约']
  71. " @click.stop="openRecallModal(newOrder.id)">撤销</button>
  72. <button type="primary" class="btn" v-if="newOrder.gdState.value == GDSTATE['定时预约']"
  73. @click.stop="openExecModal(newOrder.id)">立即执行</button>
  74. <button type="primary" class="btn" v-if="
  75. (newOrder.gdState.value == GDSTATE['待抢单'] ||
  76. newOrder.gdState.value == GDSTATE['待到达']) &&
  77. newOrder.taskType.allowUrgent == 1 &&
  78. !newOrder.urgentDetails
  79. " @click.stop="showJiaji(newOrder.id)">加急</button>
  80. </view>
  81. </view>
  82. </view>
  83. </view>
  84. <seiminFooterNav></seiminFooterNav>
  85. <seiminModel ref="seiminModel"></seiminModel>
  86. <seiminPicker ref="sPicker" titleColor="#808080" titleFontSize="28rpx" confirmColor="#333" confirmFontSize="38rpx"
  87. confirmFontWeight="500" itemFontSize="28rpx" @onClose="closePicker" @onConfirm="confirmPicker"
  88. :pickerList="pickerList">
  89. </seiminPicker>
  90. </view>
  91. </template>
  92. <script>
  93. import {
  94. mapState,
  95. mapActions
  96. } from "vuex";
  97. import {
  98. reqFetchDataList,
  99. } from "../../request/api.js";
  100. import {
  101. GDSTATE
  102. } from "../../utils/enum.gdstate.js";
  103. import {
  104. WORKORDER_EVALUATION
  105. } from "../../utils/enum.workorderEvaluation.js";
  106. import {
  107. showAppraise,
  108. openRecallModal,
  109. openExecModal,
  110. showJiaji,
  111. } from "../../utils/index.js";
  112. export default {
  113. name: 'orderList',
  114. data() {
  115. return {
  116. // 工单状态
  117. GDSTATE: GDSTATE,
  118. // 当前筛选的工单状态
  119. selectedGdState: '执行中',
  120. // 当前筛选的关联类型
  121. selectedAssociationType: '全部',
  122. //工单状态筛选列表
  123. gdStates: [{
  124. value: -1,
  125. label: '全部'
  126. },
  127. {
  128. value: 1,
  129. label: '执行中'
  130. },
  131. {
  132. value: 2,
  133. label: '待评价'
  134. },
  135. ],
  136. //关联类型筛选列表
  137. associationTypes: [{
  138. value: -1,
  139. label: '全部'
  140. }, ],
  141. newOrderList: [], //工单列表
  142. totalNum: 0, //工单总数量
  143. idx: 0, //页码
  144. pageNum: 20, //每页的工单数量
  145. pickerList: [], //picker列表
  146. type: '', //打开picker的类型,工单状态||关联类型
  147. checkStatus: {}, //审核状态列表
  148. };
  149. },
  150. computed: {
  151. ...mapState("login", ["loginInfo"]),
  152. },
  153. methods: {
  154. ...mapActions("dictionary", ["vxDictionary"]),
  155. // 前往工单详情
  156. toDetail(id) {
  157. uni.navigateTo({
  158. url: `/pages/orderDetail/orderDetail?id=${id}`
  159. })
  160. },
  161. //评价
  162. showAppraise(id) {
  163. showAppraise.call(this, id, () => {
  164. this.$refs.seiminModel.close();
  165. this.init();
  166. });
  167. },
  168. //撤销
  169. openRecallModal(id) {
  170. openRecallModal.call(this, id, () => {
  171. this.$refs.seiminModel.close();
  172. this.init();
  173. });
  174. },
  175. //立即执行
  176. openExecModal(id) {
  177. openExecModal.call(this, id, () => {
  178. this.$refs.seiminModel.close();
  179. this.init();
  180. });
  181. },
  182. //加急
  183. showJiaji(id) {
  184. showJiaji.call(this, id, () => {
  185. this.$refs.seiminModel.close();
  186. this.init();
  187. });
  188. },
  189. //关闭
  190. closePicker() {
  191. this.$refs.sPicker._close();
  192. },
  193. //打开
  194. openPicker(type) {
  195. this.type = type;
  196. this.$refs.sPicker._open();
  197. if (type === 'gdState') {
  198. //工单状态
  199. this.pickerList = this.gdStates;
  200. let index = this.pickerList.findIndex(v => v.label === this.selectedGdState);
  201. let obj = this.pickerList.find(v => v.label === this.selectedGdState);
  202. this.$refs.sPicker._changeValue(index);
  203. this.selectedGdState = obj && obj.label;
  204. } else if (type === 'associationType') {
  205. //关联类型
  206. this.pickerList = this.associationTypes;
  207. let index = this.pickerList.findIndex(v => v.label === this.selectedAssociationType);
  208. let obj = this.pickerList.find(v => v.label === this.selectedAssociationType);
  209. this.$refs.sPicker._changeValue(index);
  210. this.selectedAssociationType = obj && obj.label;
  211. }
  212. },
  213. //确定:接收子组件传来的参数
  214. confirmPicker(checkedObj) {
  215. if (this.type === 'gdState') {
  216. //工单状态
  217. this.selectedGdState = checkedObj.label;
  218. } else if (this.type === 'associationType') {
  219. //关联类型
  220. this.selectedAssociationType = checkedObj.label;
  221. }
  222. let index = this.pickerList.findIndex(v => v.label === checkedObj.label);
  223. this.$refs.sPicker._changeValue(index);
  224. this.init();
  225. },
  226. // 获取页面数据
  227. init() {
  228. uni.showLoading({
  229. title: "加载中",
  230. mask: true,
  231. });
  232. Promise.all([
  233. this.queryWorkOrdersRequest(), //查询最新工单列表
  234. this.queryDictionary('association_types'), //获取数据字典-关联类型
  235. this.queryDictionary('check_status'), //获取数据字典-审核状态
  236. ]).then((values) => {
  237. uni.hideLoading();
  238. uni.stopPullDownRefresh();
  239. this.queryWorkOrdersResponse(values[0]);
  240. this.queryDictionaryResponse(values[1], 'association_types');
  241. this.queryDictionaryResponse(values[2], 'check_status');
  242. });
  243. },
  244. // 查询最新工单列表(上拉)
  245. reachBottom() {
  246. //没有更多
  247. if (this.newOrderList.length == this.totalNum) {
  248. uni.showToast({
  249. icon: 'none',
  250. title: '没有更多工单了'
  251. })
  252. return;
  253. }
  254. uni.showLoading({
  255. title: "加载中",
  256. mask: true,
  257. });
  258. Promise.all([
  259. this.queryWorkOrdersRequest(true), //查询最新工单列表
  260. ]).then((values) => {
  261. uni.hideLoading();
  262. this.queryWorkOrdersResponse(values[0], true);
  263. });
  264. },
  265. // 查询最新工单列表
  266. queryWorkOrdersRequest(idxPlus = false) {
  267. console.log(this.selectedGdState, this.selectedAssociationType);
  268. console.log(this.gdStates, this.associationTypes);
  269. if (idxPlus) {
  270. //累加
  271. ++this.idx;
  272. } else {
  273. this.idx = 0;
  274. }
  275. let postData = {
  276. workOrder: {
  277. createDept: this.loginInfo.user.dept.id,
  278. platform: 2,
  279. searchDays: 2,
  280. },
  281. idx: this.idx,
  282. sum: this.pageNum,
  283. };
  284. let selectedGdStateValue = this.gdStates.find(v => v.label == this.selectedGdState).value;
  285. let associationTypesValue = this.associationTypes.find(v => v.label == this.selectedAssociationType).value;
  286. // 工单状态
  287. // 执行中包含状态:待抢单、待接单、待到达、待送达、执行中、定时预约
  288. if (selectedGdStateValue == 1) {
  289. // 执行中
  290. postData.workOrder.nurseState = 1;
  291. } else if (selectedGdStateValue == 2) {
  292. // 待评价
  293. postData.workOrder.gdState = {
  294. id: "73"
  295. };
  296. }
  297. //关联类型
  298. if (associationTypesValue != -1) {
  299. postData.workOrder.taskType = {
  300. associationType: {
  301. id: associationTypesValue,
  302. }
  303. }
  304. }
  305. return reqFetchDataList("nurse", "workOrder", postData);
  306. },
  307. // 查询最新工单列表
  308. queryWorkOrdersResponse(res, idxPlus = false) {
  309. if (res.status == 200) {
  310. res.list = res.list || [];
  311. this.totalNum = res.totalNum || 0;
  312. let newOrderList = res.list.map((v) => {
  313. if (v.gdState) {
  314. if (
  315. v.gdState.value == GDSTATE["待接单"] ||
  316. v.gdState.value == GDSTATE["待抢单"]
  317. ) {
  318. v.stateTextClass = "red";
  319. } else if (
  320. v.gdState.value == GDSTATE["待评价"] ||
  321. v.gdState.value == GDSTATE["已完成"]
  322. ) {
  323. v.stateTextClass = "green";
  324. } else {
  325. v.stateTextClass = "yellow";
  326. }
  327. }
  328. return v;
  329. });
  330. if (idxPlus) {
  331. //累加
  332. this.newOrderList = this.newOrderList.concat(newOrderList);
  333. } else {
  334. this.newOrderList = newOrderList;
  335. }
  336. } else {
  337. this.$refs.seiminModel.show({
  338. skin: "toast",
  339. icon: "error",
  340. content: res.msg || "获取数据失败",
  341. });
  342. throw new Error(res.msg || '获取数据失败');
  343. }
  344. },
  345. // 获取数据字典
  346. queryDictionary(key) {
  347. let postData = {
  348. "type": "list",
  349. key
  350. };
  351. return this.vxDictionary(postData);
  352. },
  353. // 获取数据字典
  354. queryDictionaryResponse(res, type) {
  355. let obj = {};
  356. let arr = [];
  357. switch (type) {
  358. case 'association_types':
  359. arr = res.map(v => ({
  360. value: v.id,
  361. label: v.name
  362. }));
  363. this.associationTypes = [{
  364. label: '全部',
  365. value: -1
  366. }, ...arr];
  367. break;
  368. case 'check_status':
  369. obj = {};
  370. res.forEach(v => {
  371. obj[v.value] = v.id;
  372. })
  373. this.checkStatus = obj;
  374. break;
  375. }
  376. },
  377. },
  378. mounted() {
  379. this.init();
  380. },
  381. onPullDownRefresh() {
  382. this.init();
  383. },
  384. onReachBottom() {
  385. this.reachBottom();
  386. }
  387. }
  388. </script>
  389. <style lang="scss" scoped>
  390. .orderList {
  391. padding-bottom: 108rpx;
  392. // 头部筛选
  393. .orderList_header {
  394. height: 88rpx;
  395. background-color: #fff;
  396. position: fixed;
  397. left: 0;
  398. z-index: 99;
  399. width: 100%;
  400. @include border(bottom);
  401. @include flex(center, center);
  402. .orderList_header_item {
  403. flex: 1;
  404. height: 100%;
  405. padding: 0 50rpx;
  406. @include border(right);
  407. @include flex(space-between, center);
  408. &:last-of-type {
  409. border-right: none;
  410. }
  411. .orderList_header_itemText {
  412. color: #333;
  413. font-size: 38rpx;
  414. }
  415. .pda-xiala {
  416. color: #DDE1E5;
  417. }
  418. }
  419. }
  420. // 列表
  421. .orderList_list {
  422. padding: 88rpx 24rpx 0;
  423. .orderList_listItem {
  424. width: 702rpx;
  425. min-height: 320rpx;
  426. background-color: #fff;
  427. margin-top: 8rpx;
  428. border-radius: 8rpx;
  429. position: relative;
  430. padding: 0 24rpx;
  431. font-size: 32rpx;
  432. @include border;
  433. @include semicircle(#f9fafb, 82rpx);
  434. @include flex(flex-start, stretch, column);
  435. .ji,
  436. .jiaji {
  437. width: 60rpx;
  438. position: absolute;
  439. right: 0;
  440. top: 0;
  441. }
  442. .orderList_listItem_header {
  443. height: 86rpx;
  444. @include border($directive:bottom, $style:dashed);
  445. @include flex(space-between, center);
  446. .orderList_listItem_header_title {
  447. color: #333;
  448. flex: 1;
  449. @include flex(flex-start, center);
  450. .associationType_icon {
  451. width: 48rpx;
  452. height: 48rpx;
  453. border-radius: 50%;
  454. background-color: #F0F6ED;
  455. font-size: 24rpx;
  456. color: #39b199;
  457. margin-right: 8rpx;
  458. @include border($color:#39b199);
  459. @include flex(center, center);
  460. }
  461. .taskNameAndWorkerName {
  462. flex: 1;
  463. @include flex;
  464. .taskName {
  465. max-width: 10em;
  466. @include clamp;
  467. }
  468. .workerName {
  469. flex: 1;
  470. @include clamp;
  471. }
  472. }
  473. }
  474. .orderList_listItem_header_more {
  475. color: #666;
  476. @include clamp;
  477. }
  478. }
  479. .orderList_listItem_item {
  480. height: 88rpx;
  481. color: #333;
  482. font-size: 30rpx;
  483. flex: 1;
  484. @include border(bottom);
  485. @include flex(flex-start, stretch, column);
  486. &:last-of-type {
  487. border-bottom: none;
  488. }
  489. .orderList_listItem_item_content {
  490. min-height: 143rpx;
  491. flex: 1;
  492. @include flex(space-between, center);
  493. .orderList_listItem_item_name {
  494. font-size: 38rpx;
  495. font-weight: bold;
  496. }
  497. .orderList_listItem_item_time {
  498. color: #666;
  499. font-size: 28rpx;
  500. }
  501. }
  502. .orderList_listItem_item_btns {
  503. position: relative;
  504. left: -24rpx;
  505. width: 698rpx;
  506. height: 88rpx;
  507. @include btn_background;
  508. @include flex;
  509. .btn {
  510. flex: 1;
  511. background-color: transparent;
  512. position: relative;
  513. @include flex(center, center);
  514. &::before {
  515. content: '';
  516. position: absolute;
  517. right: 0;
  518. top: 0;
  519. width: 1px;
  520. height: 100%;
  521. @include border(right, #fff);
  522. }
  523. &:last-of-type::before {
  524. border-right: none;
  525. }
  526. &::after {
  527. border: none;
  528. }
  529. }
  530. }
  531. }
  532. }
  533. }
  534. }
  535. </style>