cabinetDetail.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. <template>
  2. <view class="main-view">
  3. <Navigation :scroll='scrollTop'></Navigation>
  4. <view class="content">
  5. <swiper v-if="shop_image.length!=0" class="swiper" :indicator-dots="true" :autoplay="true" :interval="2000"
  6. indicator-color="rgba(0, 0, 0, 0.3)" indicator-active-color="#000000" :duration="1000" circular>
  7. <swiper-item class="swiper-item" v-for="(item,index) in shop_image" :key="index">
  8. <image class="swiper-item-img" :src="item" mode="aspectFill"></image>
  9. </swiper-item>
  10. </swiper>
  11. <img v-else class="bg-img" src="https://qiniu.bms16.com/FhRnr7rADHHsOFfpWO4duD15SgIt" alt="">
  12. </view>
  13. <view class="cabinet-detail">
  14. <view class="cabinet-name-view flex-row flex-between">
  15. <view class="cabinet-name">{{cabinetInfo.cabinet_name}}</view>
  16. <!-- <view class="distance">{{cabinetInfo.distance}}km</view> -->
  17. </view>
  18. <view v-if="tagList.length==0" class="no_battery">
  19. 无可用电池
  20. </view>
  21. <view class="flex-row" style="justify-content: center;">
  22. <view class="icon-box flex-row">
  23. <view v-for="(item, index) in tagList" class="item-tag flex-row">
  24. <text class="tag_name">
  25. {{item.main_tag_name ? (item.main_tag_name) : ''}}{{item.child_tag_name ? '/' + item.child_tag_name : ''}}·
  26. </text>
  27. <text class="tag_num">{{'x' + item.num}}</text>
  28. </view>
  29. </view>
  30. </view>
  31. <view class="flex-row flex-between">
  32. <view :class="isWorkTimer ? 'left_grid_2' : 'left_grid_1'">
  33. <view class="flex-row">
  34. <img class="icon_grid"
  35. :src="!isWorkTimer ? 'https://zxappfile.bms16.com/zx_admin/cab_timer.png' : 'https://zxappfile.bms16.com/zx_admin/cab_timer_work.png'">
  36. <view :class="isWorkTimer ? 'grid_text_1' : 'grid_text_rest' ">{{isWorkTimer ? '营业中' : '已休息'}}
  37. </view>
  38. </view>
  39. <view class="grid_text_2">
  40. {{cabinetInfo.work_begin_time!=null?cabinetInfo.work_begin_time:'00:00'}}-{{cabinetInfo.work_end_time!=null?cabinetInfo.work_end_time:'23:59'}}
  41. </view>
  42. </view>
  43. <view @tap="navToCabinet" class="right_grid flex-between">
  44. <view>
  45. <view class="cab_distance">
  46. 直线距您{{cabinetInfo.distance>1?(cabinetInfo.distance+'千米'):(cabinetInfo.distance*1000+'米')}}
  47. </view>
  48. <view class="cab_address">{{cabinetInfo.address}}</view>
  49. </view>
  50. <view>
  51. <img class="icon_grid_1" src="https://zxappfile.bms16.com/zx_admin/cab_nav.png">
  52. <view class="grid_nav">导航</view>
  53. </view>
  54. </view>
  55. </view>
  56. </view>
  57. <view class="battery-list-view">
  58. <view class="battery-title flex-row flex-between">
  59. <text class="blod-text">格口详情</text>
  60. <text class="cabinet-ref">电柜编号:<text class="cabinet-ref-text"> {{cabinetInfo.dev_id}}</text></text>
  61. </view>
  62. <view class="battery-list-main flex-row flex-start">
  63. <view class="battery-list" v-for="(item, index) in batteryList" :key="item.unique">
  64. <!-- <block v-if="item.isReservation">
  65. <view class="battery-list-g">
  66. <view class="namber-view flex-row">
  67. <view class="namber namber-g">
  68. {{item.box_sn}}
  69. </view>
  70. </view>
  71. <view class="status-img-view flex-row">
  72. <img class="status-img" src="https://zxappfile.bms16.com/zx_admin/reservation.png"
  73. alt="">
  74. </view>
  75. <view class="tip-text tip-text-g1">已预约</view>
  76. </view>
  77. </block> -->
  78. <!-- 没有预约-->
  79. <!-- <block v-else> -->
  80. <!-- 空仓或者仓门不可用-->
  81. <view v-if="!item.cabinet_battery_sn||(item.fault_reason&&item.fault_reason.length>0)"
  82. :class="(item.fault_reason&&item.fault_reason.length>0)?'battery-list-s':'battery-list-g'">
  83. <!-- <view class="namber-view flex-row">
  84. <view
  85. :class="(item.fault_reason&&item.fault_reason.length>0)?'namber namber-s':'namber namber-g'">
  86. {{item.box_sn}}
  87. </view>
  88. </view> -->
  89. <view class="namber-view flex-row flex-between">
  90. <view
  91. :class="(item.fault_reason&&item.fault_reason.length>0)?'namber namber-s':'namber namber-g'">
  92. {{item.box_sn}}
  93. </view>
  94. <view v-if="item.tag_info" class="tag_type_n">
  95. {{item.tag_info.main_tag_name ? (item.tag_info.main_tag_name) : ''}}{{item.tag_info.child_tag_name ? '/' + item.tag_info.child_tag_name : ''}}
  96. </view>
  97. </view>
  98. <view class="status-img-view flex-row">
  99. <img v-if="!item.cabinet_battery_sn&&!(item.fault_reason&&item.fault_reason.length>0)"
  100. class="status-img" src="https://qiniu.bms16.com/FtOgmvwtoUCVzEyxIau6-6i0hjLt" alt="">
  101. <img v-if="item.fault_reason&&item.fault_reason.length>0" class="status-img"
  102. src="https://qiniu.bms16.com/FmMGYfe7eRSQvM8zeKEVeSmzbArd" alt="">
  103. </view>
  104. <!-- <view class="battery-ref">124513215</view> -->
  105. <view v-if="!item.cabinet_battery_sn&&!(item.fault_reason&&item.fault_reason.length>0)"
  106. class="tip-text tip-text-g">空仓</view>
  107. <view v-if="item.fault_reason&&item.fault_reason.length>0" class="tip-text tip-text-u">
  108. {{item.fault_reason[0]}}
  109. </view>
  110. </view>
  111. <!-- 满电或者电量未满 -->
  112. <view v-else :class="(item.is_full_soc==1)?'battery-list-b':'battery-list-o'">
  113. <view class="namber-view flex-row flex-between">
  114. <view :class="(item.is_full_soc==1)?'namber namber-b':'namber namber-o'">{{item.box_sn}}
  115. </view>
  116. <view v-if="item.tag_info" class="tag_type">
  117. {{item.tag_info.main_tag_name ? (item.tag_info.main_tag_name) : ''}}{{item.tag_info.child_tag_name ? '/' + item.tag_info.child_tag_name : ''}}
  118. </view>
  119. </view>
  120. <view class="status-img-view flex-row">
  121. <view v-if="item.cabinet_battery_sn" class="progress-bar-view">
  122. <progressView :soc="item.soc" :status="item.is_full_soc"></progressView>
  123. </view>
  124. <view v-if="item.cabinet_battery_sn" class="battery-number">
  125. {{item.battery_sn}}
  126. <!-- 0168 8256 9608 -->
  127. </view>
  128. </view>
  129. <view v-if="item.is_full_soc==1" class="tip-text tip-text-b">电池可用</view>
  130. <view v-else class="tip-text tip-text-o">待充满</view>
  131. </view>
  132. <!-- </block> -->
  133. </view>
  134. </view>
  135. <scanBtn :dev_id="dev_id" :listData='listData' :cab_info="cabinetInfo" @refreshCabinet="refreshCabinet"
  136. @popPackageModel="clickPopPackageModel" ref="scanRef"></scanBtn>
  137. </view>
  138. </view>
  139. </template>
  140. <script>
  141. import {
  142. getLocation,
  143. msg,
  144. strJoin
  145. } from '../../utils/util.js';
  146. import progressView from '@/component/progressView/progressView';
  147. const http = require('../../common/request.js');
  148. const config = require('../../common/config_gyq.js');
  149. const common = require('../../common/common.js');
  150. var user = require('../../common/user.js');
  151. var bluetooth = require('../../common/bluetooth.js');
  152. var cabinetDetailImpl = require('./cabinetDetailImpl.js');
  153. // const IndexBind = require('../index/model/indexBind.js');
  154. const DF_CAB_INFO_DONE = 10000; //机柜信息传输完成
  155. import scanBtn from '@/component/scanCabBtn/scanBtn';
  156. export default {
  157. components: {
  158. progressView,
  159. scanBtn
  160. },
  161. onPageScroll(e) {
  162. this.scrollTop = e.scrollTop
  163. },
  164. data() {
  165. return {
  166. dev_id: '',
  167. listData: {},
  168. isWorkTimer: true,
  169. tagList: [],
  170. batteryList: [], //电池列表
  171. isOpenBluetooth: false,
  172. isBluetooth: false, //存储蓝牙状态
  173. isShowToBuy: false,
  174. payType: 0,
  175. myLocation: {},
  176. reservation_info: [],
  177. cabinetInfo: {}, //机柜详情
  178. num: 0,
  179. is_scan: 0,
  180. fre_price: 0,
  181. online_status: 0, //0离线 1 在线
  182. wallet_money: 0, //钱包余额
  183. scan_dev_id: '', //扫码机柜id
  184. license_plate_number: '', //车牌号
  185. can_battery_num: 0, //可用电池数
  186. blueInfo: {}, //蓝牙信息
  187. cabBatterySn: '', //蓝牙换电需要的数据 cabinet_battery_sn
  188. shop_image: [],
  189. isShowAppoint: false,
  190. packType: 0,
  191. notShow: 0,
  192. car_info: {}
  193. };
  194. },
  195. /**
  196. * 生命周期函数--监听页面加载
  197. */
  198. onLoad: function(options) {
  199. this.notShow = options.notShow ? options.notShow : 1
  200. this.dev_id = options.dev_id
  201. const storedLocation = uni.getStorageSync('user_current_location');
  202. this.car_info = uni.getStorageSync('car_info') || {};
  203. const car_list = uni.getStorageSync('user_car_list') || null
  204. this.license_plate_number = car_list ? car_list.plate_number : ''
  205. const me = this
  206. if (storedLocation && storedLocation.longitude && storedLocation.latitude) {
  207. // 如果本地有存储的定位信息,则直接使用
  208. this.setData({
  209. myLocation: storedLocation
  210. });
  211. if (options && options.dev_id != '' && options.is_scan == 1) {
  212. this.is_scan = 1
  213. this.dev_id = options.dev_id
  214. }
  215. if (options && options.dev_id != '') {
  216. this.dev_id = options.dev_id
  217. this.loadCabinetDetail();
  218. }
  219. } else {
  220. // this.getLocationAndSave();
  221. }
  222. // bluetooth.initBluetooth()
  223. },
  224. /**
  225. * 生命周期函数--监听页面显示
  226. */
  227. onShow: function() {
  228. },
  229. methods: {
  230. clickClosePackage() {
  231. this.setData({
  232. isModelCenter: false
  233. })
  234. },
  235. clickPopPackageModel(e, notShow) {
  236. // console.log(e.packType,notShow,'sadasd');
  237. // this.notShow = notShow?notShow:this.notShow
  238. // if ((e.packType === 0&&this.notShow==1) || e.packType === 2) {
  239. // this.setData({
  240. // isModelCenter: true,
  241. // packType: e.packType
  242. // })
  243. // }
  244. // if (e.packType === 1) {
  245. // if (this.is_scan === 1) {
  246. // this.$refs.scanRef.loadNowCabinetDetail(this.dev_id)
  247. // }
  248. // }
  249. // if (e.packType === 3) {
  250. // if (this.is_scan === 1) {
  251. // this.getBindBattery(e)
  252. // }
  253. // }
  254. },
  255. getBindBattery(packInfo) {
  256. this.loadNowCabinetDetail(this.dev_id, this.myLocation, packInfo.order_sn, (bindSuccess) => {
  257. uni.showModal({
  258. title: '提示',
  259. content: '领取电池成功',
  260. confirmText: '确定',
  261. showCancel: false,
  262. success: function(res) {
  263. if (res.confirm) {
  264. setTimeout(() => {
  265. uni.switchTab({
  266. url: '/pages/index/index'
  267. });
  268. }, 1500)
  269. }
  270. },
  271. fail: function(res) {},
  272. complete: function(res) {},
  273. })
  274. })
  275. },
  276. //计算电池数量
  277. deduplicateAndCountByMainAndChild(data) {
  278. // 1. 统计次数:根据 main_tag_code 和 child_tag_code(如果存在)
  279. const countMap = {};
  280. data.forEach(item => {
  281. // 如果 child_tag_code 为空,则只根据 main_tag_code 统计
  282. const key = item.child_tag_code ?
  283. `${item.main_tag_code}_${item.child_tag_code}` :
  284. item.main_tag_code;
  285. countMap[key] = (countMap[key] || 0) + 1;
  286. });
  287. // 2. 去重并添加 num 字段
  288. const seen = new Set();
  289. return data.filter(item => {
  290. // 生成去重键(与统计时逻辑一致)
  291. const key = item.child_tag_code ?
  292. `${item.main_tag_code}_${item.child_tag_code}` :
  293. item.main_tag_code;
  294. // 如果是第一次出现,添加 num 并保留
  295. if (!seen.has(key)) {
  296. seen.add(key);
  297. item.num = countMap[key]; // 直接修改原对象(如需要深拷贝可改用 { ...item, num })
  298. return true;
  299. }
  300. // 否则过滤掉
  301. return false;
  302. });
  303. },
  304. async loadCabinetDetail() {
  305. //获取机柜信息
  306. const that = this
  307. var _can_num = 0
  308. let resp = await http.postApi(config.API_CABINET_INFO, {
  309. car_sn: this.car_info.car_sn,
  310. dev_id: this.dev_id,
  311. longitude: this.myLocation.longitude,
  312. latitude: this.myLocation.latitude,
  313. })
  314. if (resp.data.code === 200) {
  315. // const _batteryList = resp.data.data.boxList
  316. // // 计算可用电池数
  317. // for (let i = 0; i < _batteryList.length; i++) {
  318. // if (_batteryList[i].cabinet_battery_sn && (!(_batteryList[i].fault_reason &&
  319. // _batteryList[i].fault_reason.length > 0))) {
  320. // _can_num = _can_num + 1
  321. // }
  322. // }
  323. var cabinetInfo = resp.data.data.cabinetInfo
  324. cabinetInfo.work_begin_time = cabinetInfo.work_begin_time == null ? '00:00' : cabinetInfo
  325. .work_begin_time
  326. cabinetInfo.work_end_time = cabinetInfo.work_end_time == null ? '00:00' : cabinetInfo
  327. .work_end_time
  328. const isWorkTimer = cabinetDetailImpl.isWithinTimeRange(cabinetInfo.work_begin_time,
  329. cabinetInfo.work_end_time)
  330. var reservation_List = resp.data.data.cabinetInfo.reservation_info
  331. var box_list = resp.data.data.boxList
  332. // box_list = cabinetDetailImpl.getBoxReservation(box_list, reservation_List)
  333. console.log(box_list, "box_list")
  334. let tagList = []
  335. box_list.map(item => {
  336. if (item.tag_info) {
  337. tagList.push(item.tag_info)
  338. }
  339. })
  340. tagList = this.deduplicateAndCountByMainAndChild(tagList)
  341. that.setData({
  342. reservation_info: reservation_List,
  343. isWorkTimer: isWorkTimer,
  344. tagList,
  345. cabinetInfo: cabinetInfo,
  346. batteryList: box_list,
  347. listData: resp.data.data,
  348. online_status: cabinetInfo.online_status,
  349. can_battery_num: resp.data.data.can_exchange_num,
  350. shop_image: JSON.parse(cabinetInfo.imgs.split(',')) || []
  351. })
  352. } else {
  353. common.simpleToast(resp.data.msg)
  354. // setTimeout(() => {
  355. // uni.switchTab({
  356. // url: '/pages/index/index'
  357. // });
  358. // }, 1500)
  359. }
  360. },
  361. refreshCabinet() {
  362. this.loadCabinetDetail()
  363. },
  364. navToCabinet() {
  365. //导航去机柜
  366. const {
  367. latitude,
  368. longitude,
  369. cityname,
  370. address,
  371. cabinet_name
  372. } = this.cabinetInfo
  373. uni.openLocation({
  374. latitude: latitude - 0,
  375. longitude: longitude - 0,
  376. scale: 15,
  377. name: cabinet_name,
  378. address: address,
  379. success: function(res) {}
  380. });
  381. },
  382. getLocationAndSave: function() {
  383. const me = this;
  384. uni.getLocation({
  385. type: 'gcj02',
  386. success: function(res) {
  387. uni.hideLoading();
  388. var myLocation = {
  389. longitude: res.longitude,
  390. latitude: res.latitude
  391. }
  392. me.setData({
  393. myLocation: myLocation
  394. })
  395. uni.setStorageSync('user_current_location', myLocation);
  396. },
  397. fail: function(res) {
  398. uni.hideLoading();
  399. },
  400. complete: function(res) {
  401. uni.hideLoading();
  402. }
  403. });
  404. },
  405. loadNowCabinetDetail(dev_id, myLocation, order_sn, bindSuccess, bluetoothCallBack) {
  406. const timeNow = Date.now()
  407. const me = this
  408. const pData = {
  409. longitude: myLocation.longitude,
  410. latitude: myLocation.latitude,
  411. dev_id: dev_id || '',
  412. time: timeNow
  413. }
  414. http.postApi(config.API_DAYHIRE_CABINRT_CABINRT_INFO, pData, (resp) => {
  415. if (resp.data.code === 200) {
  416. online_status = resp.data.data.cabinetInfo.online_status
  417. const device = {
  418. device_type: "LSCabinet",
  419. bt_type: "",
  420. mac_id: resp.data.data.cabinetInfo.bt_mac,
  421. btid: resp.data.data.cabinetInfo.bt_mac,
  422. dev_id: resp.data.data.cabinetInfo.dev_id,
  423. key: decodeKey(resp.data.data.cabinetInfo.bt_sec),
  424. btkey: resp.data.data.cabinetInfo.bt_mac,
  425. bt_sec: decodeKey(resp.data.data.cabinetInfo.bt_sec),
  426. bt_mac: resp.data.data.cabinetInfo.bt_mac
  427. }
  428. console.log(online_status)
  429. if (online_status == 1) {
  430. console.log("bb设备数据", device)
  431. toBindBattery(DEVICE_TYPE_CABINET, '', dev_id, order_sn, bindSuccess)
  432. } else {
  433. console.log("aa设备数据", device)
  434. bluetoothCallBack(device)
  435. }
  436. } else {
  437. common.simpleToast(resp.data.msg)
  438. }
  439. })
  440. }
  441. }
  442. };
  443. </script>
  444. <style>
  445. @import './cabinetDetail.css';
  446. </style>