consumableList.vue 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. <template>
  2. <view class="consumableList">
  3. <view class="head" >
  4. <uni-search-bar v-model="dataInfo.keyWord" placeholder="请搜索耗材" bgColor="#F8F8F8" @input="search" cancelButton="none" focus :radius="18" />
  5. </view>
  6. <view class="body" v-if="dataInfo.list.length">
  7. <view class="body_item ellipsis" v-for="data in dataInfo.list" :key="data.id" @click="numberClick(data, 'addConsumable')">
  8. {{data.name}}
  9. <template v-if="data.model">({{data.model}})</template>
  10. <template v-if="data.unit">({{data.unit}})</template>
  11. <template v-if="data.price">({{data.price}}元)</template>
  12. </view>
  13. </view>
  14. <view class="zanwu" v-else>
  15. <text class="newicon newicon-zanwu"></text>
  16. </view>
  17. <view class="foot_common_btns">
  18. <button @click="goBack" type="default" class="primaryButton btn">返回</button>
  19. </view>
  20. <NumberModal v-if="dataInfo.isNumber" @cancelEmit="cancelNumber" @confirmEmit="conformNumber" :selectData="dataInfo.selectData" :selectType="dataInfo.selectType"></NumberModal>
  21. </view>
  22. </template>
  23. <script setup>
  24. import { debounce, uniqBy } from 'lodash-es'
  25. import { ref, reactive} from 'vue'
  26. import NumberModal from '@/components/NumberModal.vue';
  27. import { onLoad, onPullDownRefresh, onReachBottom } from '@dcloudio/uni-app'
  28. import { api_consumable, getFetchDataList, api_incidentDetail, api_addSummaryDoc, api_incidentCategoryConsumable } from "@/http/api.js"
  29. import { defaultColor } from '@/static/js/theme.js'
  30. import { useSetTitle } from '@/share/useSetTitle.js'
  31. import { useLoginUserStore } from '@/stores/loginUser'
  32. import { useGoBack } from '@/share/useGoBack.js'
  33. useSetTitle();
  34. const loginUserStore = useLoginUserStore();
  35. const { goBack } = useGoBack();
  36. // 主题颜色
  37. const primaryColor = ref(defaultColor)
  38. // 数据
  39. const dataInfo = reactive({
  40. list: [],//工单列表
  41. idx: 0,//页码
  42. hasMore: true,//是否有更多数据
  43. incidentId: undefined,//事件ID
  44. summaryId: undefined,//汇总单ID
  45. incidentData: {},//事件对象
  46. keyWord: '',//搜索的关键词
  47. isNumber: false,//修改数量弹窗
  48. evtNumber: 1,//弹窗返回的数量
  49. selectData: {},//选择的对象
  50. selectType: {},//选择的对象类型
  51. itsmZeroStock:null, //是否支持零库存
  52. itsmParentConsumable:null //是否支持父级科室耗材
  53. })
  54. // 点击修改数量
  55. function numberClick(data, type){
  56. console.log(data)
  57. dataInfo.isNumber = true;
  58. dataInfo.selectData = {
  59. consumableBrandModel: data.brandModel,
  60. consumableUnit: data.unit,
  61. consumableEndPrice: data.price,
  62. consumableId: data.id,
  63. consumableName: data.name,
  64. };
  65. dataInfo.selectType = type;
  66. }
  67. // 确认修改数量
  68. function conformNumber(evtNumber){
  69. dataInfo.evtNumber = evtNumber;
  70. dataInfo.isNumber = false;
  71. addSummaryDoc();
  72. }
  73. // 关闭修改数量
  74. function cancelNumber(){
  75. dataInfo.isNumber = false;
  76. }
  77. // 添加耗材
  78. function addSummaryDoc(){
  79. uni.showLoading({
  80. title: "加载中",
  81. mask: true,
  82. });
  83. let postData = {
  84. "consumableList": [
  85. {
  86. "consumablesId": dataInfo.selectData.consumableId,
  87. "consumablesNum": dataInfo.evtNumber
  88. }
  89. ],
  90. "summaryId": dataInfo.summaryId
  91. };
  92. api_addSummaryDoc(postData).then(res => {
  93. uni.hideLoading();
  94. if(res.status == 200){
  95. uni.showToast({
  96. icon: 'none',
  97. title: '添加耗材成功',
  98. mask: true,
  99. });
  100. setTimeout(() => {
  101. uni.navigateTo({
  102. url: `/pages/handler/handler?incidentId=${dataInfo.incidentId}`,
  103. })
  104. }, 1500)
  105. }else{
  106. uni.showToast({
  107. icon: 'none',
  108. title: res.msg || '请求数据失败!'
  109. });
  110. }
  111. })
  112. }
  113. // 获取事件详情
  114. function getIncidentDetail(){
  115. uni.showLoading({
  116. title: "加载中",
  117. mask: true,
  118. });
  119. api_incidentDetail(dataInfo.incidentId).then(res => {
  120. uni.hideLoading();
  121. if(res.status == 200){
  122. dataInfo.incidentData = res.data || {};
  123. getHosConfig()
  124. }else{
  125. uni.showToast({
  126. icon: 'none',
  127. title: res.msg || '请求数据失败!'
  128. });
  129. }
  130. })
  131. }
  132. // 搜索
  133. const search = debounce(getList.bind(null, 0), 500);
  134. // 是否没有故障耗材配置
  135. const isFlag = ref(false);
  136. // 获取故障耗材配置
  137. async function getIncidentCategoryConsumable(){
  138. uni.showLoading({
  139. title: "加载中",
  140. mask: true,
  141. });
  142. let postData = {
  143. idx: 0,
  144. sum: 20,
  145. incidentCategoryConsumable: {
  146. duty: dataInfo.incidentData.duty.id,
  147. category: dataInfo.incidentData.category,
  148. }
  149. }
  150. let res = await api_incidentCategoryConsumable(postData);
  151. uni.hideLoading();
  152. uni.stopPullDownRefresh();
  153. if(res.status == 200){
  154. let list = res.list || [];
  155. if(list.length){
  156. isFlag.value = false;
  157. let arr = list.map(v => v.consumableDTOS).flat();
  158. dataInfo.list = uniqBy(arr, 'id');
  159. }else{
  160. isFlag.value = true;
  161. }
  162. }else{
  163. isFlag.value = false;
  164. uni.showToast({
  165. icon: 'none',
  166. title: res.msg || '请求数据失败!'
  167. });
  168. }
  169. }
  170. // 获取列表信息
  171. function getList(idx){
  172. // 判断是否满足故障耗材配置
  173. // await getIncidentCategoryConsumable();
  174. // if(!isFlag.value){
  175. // return;
  176. // }
  177. if(dataInfo.keyWord.trim() === ''){
  178. dataInfo.list = [];
  179. uni.stopPullDownRefresh();
  180. return;
  181. }
  182. uni.showLoading({
  183. title: "加载中",
  184. mask: true,
  185. });
  186. dataInfo.idx = idx === undefined ? dataInfo.idx : idx;
  187. if(dataInfo.idx === 0){
  188. dataInfo.list = [];
  189. }
  190. let postData = {
  191. idx: dataInfo.idx,
  192. sum: 20,
  193. consumable: {
  194. name: dataInfo.keyWord,
  195. upHosId:loginUserStore.loginUser.user.currentHospital.id,
  196. hosId: loginUserStore.loginUser.user.currentHospital.id
  197. }
  198. }
  199. if(dataInfo.itsmParentConsumable.value==1){
  200. delete postData.consumable.hosId
  201. }else{
  202. delete postData.consumable.upHosId
  203. }
  204. getFetchDataList("simple/data", "consumable", postData)
  205. .then((res) => {
  206. uni.hideLoading();
  207. uni.stopPullDownRefresh();
  208. if(res.status == 200){
  209. let list = res.list || [];
  210. if(list.length){
  211. dataInfo.hasMore = true;
  212. if(dataInfo.itsmZeroStock.value==0){
  213. let arr = dataInfo.idx === 0 ? list : dataInfo.list.concat(list)
  214. dataInfo.list = arr.filter(i=>i.stock!=0)
  215. }else{
  216. dataInfo.list = dataInfo.idx === 0 ? list : dataInfo.list.concat(list);
  217. }
  218. }else{
  219. dataInfo.hasMore = false;
  220. }
  221. }else{
  222. uni.showToast({
  223. icon: 'none',
  224. title: res.msg || '请求数据失败!'
  225. });
  226. }
  227. });
  228. }
  229. // 获取院区配置是否支持零库存/是否支持父级科室耗材
  230. function getHosConfig(){
  231. let postData = {
  232. idx: 0,
  233. sum: 9999,
  234. hospitalConfig:{
  235. hosId: loginUserStore.loginUser.user.currentHospital.id,
  236. model: "itsm"
  237. }
  238. };
  239. getFetchDataList("simple/data", "hospitalConfig", postData)
  240. .then((res) => {
  241. dataInfo.itsmZeroStock = res.list.find(i=>i.key=='itsmZeroStock')
  242. dataInfo.itsmParentConsumable = res.list.find(i=>i.key=='itsmParentConsumable')
  243. getList(0);
  244. });
  245. }
  246. onLoad((option) => {
  247. dataInfo.incidentId = option.incidentId;
  248. dataInfo.summaryId = option.summaryId;
  249. getIncidentDetail();
  250. })
  251. onPullDownRefresh(() => {
  252. getList(0)
  253. })
  254. onReachBottom(() => {
  255. dataInfo.idx += 1;
  256. if (dataInfo.hasMore) {
  257. getList(); // 当触底时加载更多数据
  258. }
  259. })
  260. </script>
  261. <style lang="scss" scoped>
  262. .consumableList{
  263. display: flex;
  264. flex-direction: column;
  265. justify-content: space-between;
  266. .head{
  267. height: 88rpx;
  268. display: flex;
  269. align-items: center;
  270. justify-content: center;
  271. padding: 0 24rpx;
  272. position: fixed;
  273. z-index: 99;
  274. width: 100%;
  275. box-sizing: border-box;
  276. background: linear-gradient( 90deg, #58CF66 0%, #DDE9FC 100%);
  277. }
  278. .body{
  279. margin-bottom: 140rpx;
  280. margin-top: 88rpx;
  281. font-size: 26rpx;
  282. .body_item{
  283. border-bottom: 1rpx solid #DEDEDE;
  284. padding: 24rpx;
  285. }
  286. }
  287. .zanwu{
  288. margin-bottom: 140rpx;
  289. margin-top: 88rpx;
  290. display: flex;
  291. justify-content: center;
  292. .newicon-zanwu{
  293. font-size: 256rpx;
  294. color: #D6D6D6;
  295. margin-top: 140rpx;
  296. }
  297. }
  298. .foot_common_btns{
  299. position: fixed;
  300. left: 0;
  301. bottom: 0;
  302. background-color: #fff;
  303. }
  304. }
  305. </style>