Elizabeth's proactive approach involves introducing urinal toilet attachment , an ingenious concept that optimizes space and functionality.

inventory.vue 21KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615
  1. <template>
  2. <div>
  3. <div v-show="showSearch">
  4. <label class="title"><span class="name">仓库</span> :</label>
  5. <el-select size="small" v-model="storehouse_id" filterable placeholder="请选择仓库" style="width:200px" @change="changeHouseList">
  6. <el-option
  7. v-for="(option, index) in houseList"
  8. :key="index"
  9. :label="option.storehouse_name"
  10. :value="option.id">
  11. </el-option>
  12. </el-select>
  13. <el-autocomplete
  14. class="checkSearch"
  15. popper-class="my-autocomplete"
  16. v-model="form.drug_name"
  17. :fetch-suggestions="querySearchAsync"
  18. :trigger-on-focus="true"
  19. placeholder="请输入耗材名称"
  20. @select="handleSelect"
  21. style="width:300px;"
  22. >
  23. <i class="el-icon-search el-input__icon" slot="suffix"></i>
  24. <template slot-scope="{ item }">
  25. <div class="name">{{ item.good_name +" " +item.specification_name + " "+item.manufacturer }}</div>
  26. </template>
  27. </el-autocomplete>
  28. </div>
  29. <div v-show="showTable">
  30. <div>日期:{{nowTime}} 盘点人:{{user_name}}</div>
  31. <div style="float:right;margin-bottom:10px"><el-button type="primary" @click="saveInentoryList">保存</el-button></div>
  32. <el-collapse v-model="activeNames">
  33. <el-collapse-item :name="index" v-for="(item,index) in tableData" :key="index">
  34. <template slot="title">
  35. <div style="font-size:16px">{{item.good_name}}</div> &nbsp;&nbsp;{{item.specification_name}} <el-button type="warning" icon="el-icon-remove" style="margin-left:20px;" @click="toMove(item,index)">移除</el-button>
  36. </template>
  37. <el-table :data="item.child" border :header-cell-style="{ backgroundColor: 'rgb(245, 247, 250)' }" style="width:100%">
  38. <el-table-column prop="date" label="耗材名称" width="150" align="center">
  39. <template slot-scope="scope">
  40. {{scope.row.good_name}}
  41. </template>
  42. </el-table-column>
  43. <el-table-column prop="name" label="规格" width="100" align="center">
  44. <template slot-scope="scope">
  45. {{scope.row.specification_name}}
  46. </template>
  47. </el-table-column>
  48. <el-table-column prop="name" label="入库单号" width="100" align="center">
  49. <template slot-scope="scope">
  50. {{scope.row.warehousing_order}}
  51. </template>
  52. </el-table-column>
  53. <el-table-column prop="name" label="有效期" width="100" align="center">
  54. <template slot-scope="scope">
  55. {{getTime(scope.row.expiry_date)}}
  56. </template>
  57. </el-table-column>
  58. <el-table-column prop="name" label="批号" width="100" align="center">
  59. <template slot-scope="scope">
  60. {{scope.row.number}}
  61. </template>
  62. </el-table-column>
  63. <el-table-column prop="name" label="进货价" width="100" align="center">
  64. <template slot-scope="scope">
  65. {{scope.row.price}}
  66. </template>
  67. </el-table-column>
  68. <el-table-column prop="name" label="生产厂商" width="100" align="center">
  69. <template slot-scope="scope">
  70. {{scope.row.manufacturer_name}}
  71. </template>
  72. </el-table-column>
  73. <el-table-column prop="name" label="仓库名称" width="100" align="center">
  74. <template slot-scope="scope">
  75. {{getStorehouseName(scope.row.storehouse_id)}}
  76. </template>
  77. </el-table-column>
  78. <el-table-column prop="name" label="盘点前数量" width="120" align="center">
  79. <template slot-scope="scope">
  80. <span><el-input style="width:80px" v-model="scope.row.stock_count" :disabled="true"></el-input>{{scope.row.max_unit}}</span>
  81. </template>
  82. </el-table-column>
  83. <el-table-column prop="name" label="盘点后数量" width="120" align="center">
  84. <template slot-scope="scope">
  85. <el-input style="width:80px" v-model="scope.row.last_stock_count"></el-input>
  86. </template>
  87. </el-table-column>
  88. <el-table-column prop="name" label="盘点原因" width="180" align="center">
  89. <template slot-scope="scope">
  90. <el-select v-model="scope.row.type" placeholder="请选择">
  91. <el-option
  92. v-for="item in reasonList"
  93. :key="item.id"
  94. :label="item.name"
  95. :value="item.id">
  96. </el-option>
  97. </el-select>
  98. </template>
  99. </el-table-column>
  100. <el-table-column prop="name" width="200" align="center">
  101. <template slot-scope="scope">
  102. <span><el-button type="danger" @click="toDelete(scope.$index,scope.row.drug_id)">不盘点此批次</el-button></span>
  103. </template>
  104. </el-table-column>
  105. </el-table>
  106. </el-collapse-item>
  107. <label class="title"><span class="name">仓库</span>:</label>
  108. <el-select size="small" v-model="storehouse_id" filterable placeholder="请选择仓库" style="width:200px" @change="changeHouseList">
  109. <el-option
  110. v-for="(option, index) in houseList"
  111. :key="index"
  112. :label="option.storehouse_name"
  113. :value="option.id">
  114. </el-option>
  115. </el-select>
  116. <el-autocomplete
  117. class="checkSearch"
  118. popper-class="my-autocomplete"
  119. v-model="form.drug_name"
  120. :fetch-suggestions="querySearchAsync"
  121. :trigger-on-focus="true"
  122. placeholder="请输入耗材名称"
  123. @select="handleSelect"
  124. style="width:300px;"
  125. v-show="showTableOne"
  126. >
  127. <i class="el-icon-search el-input__icon" slot="suffix"></i>
  128. <template slot-scope="{ item }">
  129. <div class="name">{{ item.good_name +" " +item.specification_name + " "+item.manufacturer }}</div>
  130. </template>
  131. </el-autocomplete>
  132. </el-collapse>
  133. </div>
  134. </div>
  135. </template>
  136. <script>
  137. const moment = require("moment");
  138. import { uParseTime } from '@/utils/tools'
  139. import { postSearchGoodWarehouseList,getInventoryList,SaveCheckedInventory,getInventoryDetail,deleteInventory,getWarehouseTotal,getGoodInventoryWarehouseList,saveStockInentoryList } from "@/api/stock"
  140. export default {
  141. name: "inventory",
  142. data() {
  143. return{
  144. user_name:this.$store.getters.xt_user.user.user_name,
  145. nowTime: moment(new Date()).format("YYYY-MM-DD HH:MM:SS"),
  146. searchKey:'',
  147. value1:'',
  148. value2:'',
  149. tableData: [],
  150. dialogVisible:false,
  151. total: 0,
  152. inventoryList:[
  153. {id:0,name:"全部"},
  154. {id:1,name:"盘点完成"},
  155. {id:2,name:"正在盘点"},
  156. ],
  157. start_time:"",
  158. end_time:"",
  159. form: {
  160. id:"",
  161. good_name: '',
  162. packing_price:'',
  163. new_price:"",
  164. count:"",
  165. remark:"",
  166. warehousing_order:"",
  167. manufacturer:"",
  168. dealer:"",
  169. license_number:"",
  170. warehousing_unit:"",
  171. specification_name:"",
  172. buy_price:"",
  173. good_id:"",
  174. packing_unit:"",
  175. good_origin_place:"",
  176. report_count:"",
  177. total:"",
  178. expiry_date:"",
  179. product_date:"",
  180. number:"",
  181. proof_count:"",
  182. storehouse_id:"",
  183. },
  184. manufacturerList:[],
  185. dealerList:[],
  186. goodList:[],
  187. currentIndex: 0,
  188. inventory_status:0,
  189. limit:10,
  190. page:1,
  191. doctorList:[],
  192. total:0,
  193. tableList:[],
  194. ids:"",
  195. checker:this.$store.getters.xt_user.user.id,
  196. check_time:new Date(),
  197. checkDialogVisible:false,
  198. editPriceDialogVisible:false,
  199. modefiyePriceDialogVisible:false,
  200. proofPriceDialogVisible:false,
  201. id:0,
  202. numberList:[],
  203. reasonList:[
  204. {id:6,name:"默认"},
  205. {id:1,name:"到期退货"},
  206. {id:2,name:"异常退货"},
  207. {id:3,name:"退货"},
  208. {id:4,name:"损坏"},
  209. {id:5,name:"不计入报损分析"},
  210. {id:7,name:"有效期到期"},
  211. ],
  212. activeNames: [0,1,2,3,4,5,6,7,8,9,10],
  213. showTable:false,
  214. showSearch:true,
  215. inventory_total:0,
  216. WarehouseList:[],
  217. showTableOne:false,
  218. houseList:[],
  219. storehouse_id:"",
  220. org_id:0,
  221. }
  222. },
  223. methods:{
  224. querySearchAsync(keyword, cb) {
  225. let key = '';
  226. if (keyword != undefined) {
  227. key = keyword
  228. }
  229. var params = {
  230. keyword:key,
  231. storehouse_id:this.storehouse_id,
  232. }
  233. postSearchGoodWarehouseList(params).then(response => {
  234. if (response.data.state == 1) {
  235. var list = response.data.data.list
  236. this.goodList = list
  237. var manufacturerList = response.data.data.manufacturerList
  238. this.manufacturerList = manufacturerList
  239. var dealer = response.data.data.dealerList
  240. this.dealerList = dealer
  241. for(let i=0;i<this.goodList.length;i++){
  242. for(let j=0;j<this.manufacturerList.length;j++){
  243. if(this.goodList[i].manufacturer == this.manufacturerList[j].id){
  244. this.goodList[i].manufacturer = this.manufacturerList[j].manufacturer_name
  245. }
  246. }
  247. }
  248. for(let i=0;i<this.goodList.length;i++){
  249. for(let j=0;j<this.dealerList.length;j++){
  250. if(this.goodList[i].dealer == this.dealerList[j].id){
  251. this.goodList[i].dealer = this.dealerList[j].dealer_name
  252. }
  253. }
  254. }
  255. cb(this.goodList)
  256. } else {
  257. cb([])
  258. }
  259. })
  260. },
  261. changeGoodName(val){
  262. this.currentIndex = val
  263. },
  264. handleSelect(val){
  265. this.getGoodInventoryWarehouseList(val.good_id)
  266. },
  267. search(){
  268. this.getlist()
  269. },
  270. print(){
  271. if(this.ids == ""){
  272. this.$message.error("请勾选打印数据")
  273. return
  274. }else{
  275. this.$router.push({path:'/stock/inventoryPrint?ids='+this.ids})
  276. }
  277. },
  278. getTime(val) {
  279. if(val < 0){
  280. return ""
  281. }
  282. if(val == ""){
  283. return ""
  284. }else {
  285. return uParseTime(val, '{y}-{m}-{d}')
  286. }
  287. },
  288. handleSizeChange(val){
  289. this.limit = val
  290. this.getlist()
  291. },
  292. handleCurrentChange(val){
  293. this.page = val
  294. this.getlist()
  295. },
  296. changeHouseList(){
  297. this.querySearchAsync()
  298. },
  299. getlist(){
  300. var params = {
  301. keyword:this.searchKey,
  302. start_time:this.start_time,
  303. end_time:this.end_time,
  304. inventory_status:this.inventory_status,
  305. page:this.page,
  306. limit:this.limit,
  307. }
  308. getInventoryList(params).then(response=>{
  309. if(response.data.state ==1 ){
  310. var list = response.data.data.list
  311. this.tableList = list
  312. var total = response.data.data.total
  313. this.total = total
  314. this.doctorList = response.data.data.doctorlist
  315. this.houseList = response.data.data.houseList
  316. console.log("仓库行吗32323232232332",this.houseList)
  317. var houseConfig = response.data.data.houseConfig
  318. this.storehouse_id = houseConfig.storehouse_out_info
  319. }
  320. })
  321. },
  322. getChecker(id){
  323. var name = ""
  324. for(let i=0;i<this.doctorList.length;i++){
  325. if(id == this.doctorList[i].admin_user_id){
  326. name = this.doctorList[i].user_name
  327. }
  328. }
  329. return name
  330. },
  331. toCheck(){
  332. if(this.ids.length <=0){
  333. this.$message.error("请勾选核对数据")
  334. return
  335. }else{
  336. this.checkDialogVisible = true
  337. }
  338. },
  339. changePrice(val){
  340. var arr = []
  341. for(let i=0;i<val.length;i++){
  342. arr.push(val[i].id)
  343. }
  344. var str = arr.join(",")
  345. this.ids = str
  346. },
  347. SaveCheckedInventory(){
  348. var params = {
  349. ids:this.ids,
  350. check_time:this.getTime(this.check_time),
  351. checker:this.checker,
  352. }
  353. SaveCheckedInventory(params).then(response=>{
  354. if(response.data.state == 1){
  355. this.$message.success("保存成功")
  356. var inventory = response.data.data.inventory
  357. this.checkDialogVisible = false
  358. this.getlist()
  359. }
  360. })
  361. },
  362. toDelete(index){
  363. this.tableData.splice(index,1)
  364. },
  365. getInventory(id){
  366. getInventoryDetail(id).then(response=>{
  367. if(response.data.state == 1){
  368. var detail = response.data.data.detail
  369. this.form.good_name = detail.good_name
  370. this.form.specification_name = detail.specification_name
  371. this.form.count = detail.count
  372. this.form.remark = detail.remark
  373. this.form.total = detail.total
  374. this.form.warehousing_info_id = detail.warehousing_info_id
  375. this.form.good_id = detail.good_id
  376. this.id = detail.id
  377. this.proofPriceDialogVisible = true
  378. }
  379. })
  380. },
  381. deleteInventory(id,check_status,index){
  382. if(check_status == 1){
  383. this.$message.error("已核对的不能编辑")
  384. return
  385. }
  386. this.$confirm('是否删除所选内容?', '提示', {
  387. confirmButtonText: '确定',
  388. cancelButtonText: '取消',
  389. type: 'warning'
  390. }).then(() => {
  391. deleteInventory(id).then(response => {
  392. if (response.data.state == 1) {
  393. var msg = response.data.data.msg
  394. this.$message.success("保存成功")
  395. this.tableList.splice(index,1)
  396. }
  397. })
  398. }).catch(() => {
  399. this.loading = false
  400. });
  401. },
  402. exportList(){
  403. import('@/vendor/Export2Excel').then(excel => {
  404. const tHeader = ['盘点编码', '盘点日期', '盘点数量','盘点人','审核日期','核对人','状态']
  405. const filterVal = ['warehousing_order', 'start_time', 'count','user_name','check_time','checker','checker_status']
  406. for(let i=0;i<this.tableList.length;i++){
  407. this.tableList[i].start_time = this.getTime(this.tableList[i].start_time)
  408. this.tableList[i].check_time = this.getTime(this.tableList[i].check_time)
  409. this.tableList[i].checker = this.getChecker(this.tableList[i].checker)
  410. if(this.tableList[i].checker_status == 1){
  411. this.tableList[i].checker_status = "已核对"
  412. }
  413. if(this.tableList[i].checker_status == 2){
  414. this.tableList[i].checker_status = "未核对"
  415. }
  416. }
  417. const data = this.formatJson(filterVal, this.tableList)
  418. excel.export_json_to_excel({
  419. header: tHeader,
  420. data,
  421. filename: '耗材盘点'
  422. })
  423. this.downloadLoading = false
  424. })
  425. },
  426. formatJson(filterVal, jsonData) {
  427. return jsonData.map(v => filterVal.map(j => v[j]));
  428. },
  429. getGoodInventoryWarehouseList(id){
  430. var params = {
  431. id:id,
  432. storehouse_id:this.storehouse_id,
  433. }
  434. getGoodInventoryWarehouseList(params).then(response=>{
  435. if(response.data.state == 1){
  436. var list = response.data.data.list
  437. for(let i=0;i<list.length;i++){
  438. if(this.org_id == 10489){
  439. list[i].type = 4
  440. }else{
  441. list[i].type = 6
  442. }
  443. list[i].good_name = list[i].GoodInfo.good_name
  444. list[i].specification_name = list[i].GoodInfo.specification_name
  445. list[i].manufacturer_name = ""
  446. for(let j=0;j<this.manufacturerList.length;j++){
  447. if(list[i].manufacturer == this.manufacturerList[j].id){
  448. list[i].manufacturer_name = this.manufacturerList[j].manufacturer_name
  449. }
  450. }
  451. this.WarehouseList.push(list[i])
  452. }
  453. let objInfo = {}
  454. this.WarehouseList.forEach((item,index)=>{
  455. let { good_id } = item
  456. if(!objInfo[good_id]){
  457. objInfo[good_id] = {
  458. good_id,
  459. child:[],
  460. good_name:item.good_name,
  461. specification_name:item.specification_name,
  462. warehouseing_unit:item.max_unit,
  463. price:item.price,
  464. number:item.number,
  465. expiry_date:item.expiry_date,
  466. type:0,
  467. last_stock_count:0,
  468. }
  469. }
  470. })
  471. let newList = Object.values(objInfo);
  472. for(let i=0;i<this.WarehouseList.length;i++){
  473. for(let j=0;j<newList.length;j++){
  474. if(this.WarehouseList[i].good_id == newList[j].good_id){
  475. newList[j].child.push(this.WarehouseList[i])
  476. }
  477. }
  478. }
  479. this.tableData = newList
  480. this.showSearch = false
  481. this.showTable = true
  482. if(this.tableData.length >=1){
  483. this.showTableOne = true
  484. }
  485. this.inventory_total = this.tableData.length
  486. }
  487. })
  488. },
  489. changeInventory(){
  490. this.getlist()
  491. },
  492. changeStartTime(val){
  493. this.start_time = this.getTime(val)
  494. this.getlist()
  495. },
  496. changeEndTime(val){
  497. this.end_time = this.getTime(val)
  498. this.getlist()
  499. },
  500. changeNumber(val){
  501. this.getWarehouseTotal(val)
  502. this.form.warehousing_info_id = val
  503. },
  504. getWarehouseTotal(id){
  505. var params = {
  506. id:id,
  507. }
  508. getWarehouseTotal(params).then(response=>{
  509. if(response.data.state == 1){
  510. var list = response.data.data.list
  511. this.form.total = list.stock_count
  512. }
  513. })
  514. },
  515. toMove(item,index){
  516. this.$confirm('此操作将移除耗材整个批次, 是否继续?', '提示', {
  517. confirmButtonText: '确定',
  518. cancelButtonText: '取消',
  519. type: 'warning'
  520. }).then(() => {
  521. this.tableData.splice(index,1)
  522. this.WarehouseList = []
  523. this.activeNames = [0,1,2,3,4,5,6,7,8,9,10]
  524. }).catch(() => {
  525. this.loading = false
  526. });
  527. },
  528. toDelete(index,drug_id){
  529. for(let i=0;i<this.tableData.length;i++){
  530. if(drug_id == this.tableData[i].drug_id){
  531. this.tableData[i].child.splice(index,1)
  532. if(this.tableData[i].child.length == 0){
  533. this.tableData.splice(index,1)
  534. }
  535. }
  536. }
  537. this.WarehouseList = []
  538. this.activeNames = [0,1,2,3,4,5,6,7,8,9,10]
  539. },
  540. saveInentoryList(){
  541. var arr = []
  542. for(let i=0;i<this.tableData.length;i++){
  543. for(let j=0;j<this.tableData[i].child.length;j++){
  544. arr.push(this.tableData[i].child[j])
  545. }
  546. }
  547. for(let i=0;i<arr.length;i++){
  548. if(arr[i].last_stock_count == undefined){
  549. arr[i].last_stock_count = 0
  550. }else{
  551. arr[i].last_stock_count = parseInt(arr[i].last_stock_count)
  552. }
  553. arr[i].storehouse_id = this.storehouse_id
  554. }
  555. var params = {
  556. tableData:arr,
  557. storehouse_id:this.storehouse_id,
  558. }
  559. // console.log("param23323232323232323",params)
  560. saveStockInentoryList(params).then(response=>{
  561. if(response.data.state == 1){
  562. var inventory = response.data.data.inventory
  563. this.$message.success("保存成功")
  564. this.WarehouseList = []
  565. this.tableData = []
  566. }
  567. })
  568. },
  569. getStorehouseName(id){
  570. var storehouse_name = ""
  571. for(let i=0;i<this.houseList.length;i++){
  572. if(id == this.houseList[i].id){
  573. storehouse_name = this.houseList[i].storehouse_name
  574. }
  575. }
  576. return storehouse_name
  577. }
  578. },
  579. created(){
  580. this.org_id = this.$store.getters.xt_user.org.id
  581. this.getlist()
  582. },
  583. mounted() {
  584. },
  585. };
  586. </script>