index.vue 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690
  1. <template>
  2. <view class="container">
  3. <view class="main-container">
  4. <view class="nav-tab"></view>
  5. <view class="main-body">
  6. <view class="panel">
  7. <view class="stat-pannel flex-row flex-between">
  8. <view class="stat-item">
  9. <view class="stat-des">
  10. <image class="des-icon" src="/static/resource/images/device.png"></image>
  11. <text>设备</text>
  12. </view>
  13. <view class="stat-data">{{ tools.showMoney(device_count) }}</view>
  14. </view>
  15. <view class="stat-item">
  16. <view class="stat-des">
  17. <image class="des-icon" src="/static/resource/images/my-yes.png"></image>
  18. <text>客户</text>
  19. </view>
  20. <view class="stat-data">{{ tools.showMoney(user_count) }}</view>
  21. </view>
  22. <view class="stat-item" style="border: none">
  23. <view class="stat-des">
  24. <image class="des-icon" src="/static/resource/images/b-j.png"></image>
  25. <text>今日报警</text>
  26. </view>
  27. <view class="stat-data">{{ tools.showMoney(alarm_count) }}</view>
  28. </view>
  29. </view>
  30. <view class="flex-row flex-between">
  31. <view class="chart-pannel topbar-pannel">
  32. <view class="chart-desc">设备地区排行</view>
  33. <view class="chart-container">
  34. <ec-canvas id="mychart-dom-top-bar" canvas-id="mychart-top-bar" :ec="ecTopBar"></ec-canvas>
  35. </view>
  36. </view>
  37. <view class="chart-pannel percent-pannel">
  38. <view class="chart-desc">设备状态</view>
  39. <view class="chart-container">
  40. <!-- <ec-canvas id="mychart-dom-percent-pie" canvas-id="mychart-percent-pie" :ec="ecPercentPie"></ec-canvas> -->
  41. </view>
  42. </view>
  43. </view>
  44. <view class="chart-pannel">
  45. <view class="flex-row flex-between">
  46. <view class="chart-desc">每月设备增长数</view>
  47. </view>
  48. <view class="chart-container">
  49. <ec-canvas id="mychart-dom-dynamics" canvas-id="mychart-dynamics" :ec="ecDynamics"></ec-canvas>
  50. </view>
  51. </view>
  52. <view class="chart-pannel">
  53. <view class="flex-row flex-between">
  54. <view class="chart-desc">实时报警</view>
  55. <view class="chart-desc" @tap="loadAlarmAll">
  56. <image class="fresh-icon" src="/static/resource/images/fr-icon.png"></image>
  57. <text>刷新</text>
  58. </view>
  59. </view>
  60. <view>
  61. <van-cell-group>
  62. <van-cell :label="tools.formatTime(item.send_time)" clickable v-for="(item, index) in alarmlist" :key="index">
  63. <view slot="title">
  64. <view class="van-cell-text">{{ item.macid }}</view>
  65. </view>
  66. <view>
  67. <view class="van-cell-text">{{ item.alarm_desc }}</view>
  68. <view class="van-cell-text"></view>
  69. </view>
  70. </van-cell>
  71. </van-cell-group>
  72. <!-- i-load-more tip="{{orderLoading?'正在加载 ...':'没有更多信息啦'}}" loading="{{isLoading}}" / -->
  73. </view>
  74. </view>
  75. </view>
  76. </view>
  77. </view>
  78. </view>
  79. </template>
  80. <script module="tools" lang="wxs" src="@/pages/common/wxs/tools.wxs"></script>
  81. <script>
  82. // import ecCanvas from './ec-canvas';
  83. import iLoadMore from '@/component/iview/load-more/index';
  84. // pages/index/index.js
  85. var config = require('../../common/config.js');
  86. var http = require('../../common/http.js');
  87. var common = require('../../common/common.js');
  88. var storage = require('../../common/storage.js');
  89. // import * as echarts from 'ec-canvas/echarts';
  90. var areaList = require('../../common/area.js');
  91. function LocationCodeToName(code, tail = 3, suffix = true) {
  92. if (!code) {
  93. return '';
  94. }
  95. if (code < 100) {
  96. code *= 10000;
  97. }
  98. if (code < 10000) {
  99. code *= 100;
  100. }
  101. var area = [];
  102. var province = areaList.province_list[code.toString().substring(0, 2) + '0000'];
  103. if (province) {
  104. if (!suffix) {
  105. province = province.replace('省', '').replace('市', '').replace('特别行政区', '').replace('自治区', '').replace('壮族', '').replace('回族', '').replace('维吾尔', '');
  106. }
  107. area.push(province);
  108. }
  109. var city = areaList.city_list[code.toString().substring(0, 4) + '00'];
  110. if (city && city != area[area.length - 1]) {
  111. area.push(city);
  112. }
  113. var county = areaList.county_list[code];
  114. if (county && county != area[area.length - 1]) {
  115. area.push(county);
  116. }
  117. return area.slice(-tail).join(' ');
  118. }
  119. export default {
  120. components: {
  121. // ecCanvas,
  122. iLoadMore
  123. },
  124. data() {
  125. return {
  126. ecTopBar: {
  127. lazyLoad: true
  128. },
  129. ecPercentPie: {
  130. lazyLoad: true
  131. },
  132. ecDynamics: {
  133. lazyLoad: true
  134. },
  135. message: '0',
  136. statisList: {},
  137. selectShop: {
  138. id: 0,
  139. name: '',
  140. head: ''
  141. },
  142. timeUnit: 'date',
  143. selectTime: new Date().getTime(),
  144. formatSelectTime: common.DateFormat(new Date(), 'yyyy年MM月dd日'),
  145. totalDynamicsData: {},
  146. shopDynamicsData: {},
  147. topNum: 5,
  148. showNum: 20,
  149. typeTitle: '营收',
  150. typeName: 'income',
  151. orderType: true,
  152. orderList: [],
  153. orderLoading: false,
  154. start_page: 1,
  155. limit_page: 10,
  156. orderNum: 0,
  157. orderTotal: 0,
  158. device_count: 0,
  159. user_count: 0,
  160. alarm_count: 0,
  161. unread_alarm_count: 0,
  162. areaList: [],
  163. deviceStatis: {},
  164. deviceMonthly: {},
  165. alarmlist: []
  166. };
  167. },
  168. /**
  169. * 生命周期函数--监听页面加载
  170. */
  171. onLoad: function (options) {
  172. this.loadUserInfo();
  173. this.loadStatisTop();
  174. this.loadStatisArea();
  175. this.loadStatisDevice();
  176. this.loadDeviceMonthly();
  177. this.loadAlarmAll();
  178. //
  179. },
  180. onReady: function () {
  181. console.log('ready');
  182. //this.drawChart([{ value: '', year: '', country: '' }])
  183. },
  184. /**
  185. * 生命周期函数--监听页面显示
  186. */
  187. onShow: function () {},
  188. /**
  189. * 用户点击右上角分享
  190. */
  191. onShareAppMessage: function () {},
  192. /**
  193. * 页面相关事件处理函数--监听用户下拉动作
  194. */
  195. onPullDownRefresh: function () {
  196. console.log('pull');
  197. this.loadStatisTop();
  198. this.loadStatisArea();
  199. this.loadStatisDevice();
  200. this.loadDeviceMonthly();
  201. this.loadAlarmAll();
  202. },
  203. onReachBottom: function () {
  204. /*if (this.data.orderLoading) return
  205. this.setData({
  206. orderLoading: true
  207. })
  208. this.data.start_page++*/
  209. //common.loading()
  210. //this.loadOrderList()
  211. },
  212. methods: {
  213. loadStatisTop: function () {
  214. const that = this;
  215. http.postApi(config.API_STATIS_TOP, {}, function (resp) {
  216. uni.stopPullDownRefresh();
  217. if (resp.data.code === 200) {
  218. that.setData({
  219. device_count: resp.data.data.device_count,
  220. user_count: resp.data.data.user_count,
  221. alarm_count: resp.data.data.alarm_count,
  222. unread_alarm_count: resp.data.data.unread_alarm_count
  223. });
  224. } else {
  225. common.simpleToast(resp.data.msg);
  226. }
  227. });
  228. },
  229. loadStatisArea: function () {
  230. const that = this;
  231. http.postApi(
  232. config.API_AREA_STATIS,
  233. {
  234. scope: 0
  235. },
  236. function (resp) {
  237. uni.stopPullDownRefresh();
  238. if (resp.data.code === 200) {
  239. that.setData({
  240. areaList: resp.data.data.list
  241. });
  242. that.initTopBar();
  243. } else {
  244. common.simpleToast(resp.data.msg);
  245. }
  246. }
  247. );
  248. },
  249. loadStatisDevice: function () {
  250. const that = this;
  251. http.postApi(config.API_DEVICE_STATIS, {}, function (resp) {
  252. uni.stopPullDownRefresh();
  253. if (resp.data.code === 200) {
  254. that.setData({
  255. deviceStatis: resp.data.data
  256. });
  257. that.initPercentPie();
  258. } else {
  259. common.simpleToast(resp.data.msg);
  260. }
  261. });
  262. },
  263. loadDeviceMonthly: function () {
  264. const that = this;
  265. http.postApi(config.API_DEVICE_MONTH, {}, function (resp) {
  266. uni.stopPullDownRefresh();
  267. if (resp.data.code === 200) {
  268. that.setData({
  269. deviceMonthly: resp.data.data.list
  270. });
  271. that.initDynamicsChart();
  272. } else {
  273. common.simpleToast(resp.data.msg);
  274. }
  275. });
  276. },
  277. loadAlarmAll: function () {
  278. const that = this;
  279. http.postApi(
  280. config.API_ALARM_ALL,
  281. {
  282. status: 0,
  283. size: 20
  284. },
  285. function (resp) {
  286. uni.stopPullDownRefresh();
  287. if (resp.data.code === 200) {
  288. that.setData({
  289. alarmlist: resp.data.data.list
  290. });
  291. } else {
  292. common.simpleToast(resp.data.msg);
  293. }
  294. }
  295. );
  296. },
  297. initTopBar() {
  298. // this.topBarComponent = this.zpSelectComponent('#mychart-dom-top-bar');
  299. // this.topBarComponent.init((canvas, width, height) => {
  300. // const chart = echarts.init(canvas, null, {
  301. // width: width,
  302. // height: height
  303. // });
  304. // this.topBarChart = chart;
  305. // this.setTopBarOptions();
  306. // return chart;
  307. // });
  308. },
  309. setTopBarOptions() {
  310. // this.topBarChart.setOption({
  311. // color: ['#409EFF'],
  312. // xAxis: {
  313. // type: 'value',
  314. // axisLine: {
  315. // show: false
  316. // },
  317. // axisLabel: {
  318. // show: false
  319. // },
  320. // axisTick: {
  321. // show: false
  322. // },
  323. // splitLine: {
  324. // show: false
  325. // }
  326. // },
  327. // grid: {
  328. // top: 0,
  329. // left: '5%',
  330. // right: '2%',
  331. // bottom: '2%',
  332. // containLabel: true
  333. // },
  334. // yAxis: {
  335. // type: 'category',
  336. // axisLine: {
  337. // show: false
  338. // },
  339. // axisLabel: {
  340. // show: false
  341. // },
  342. // axisTick: {
  343. // show: false
  344. // },
  345. // splitLine: {
  346. // show: false
  347. // },
  348. // data: this.areaList
  349. // .filter((p) => p.area != 0)
  350. // .sort((a, b) => {
  351. // return b.count - a.count;
  352. // })
  353. // .slice(0, 8)
  354. // .map((p) => LocationCodeToName(p.area, 1, false))
  355. // .reverse()
  356. // },
  357. // series: [
  358. // {
  359. // name: '',
  360. // smooth: true,
  361. // type: 'bar',
  362. // label: {
  363. // normal: {
  364. // show: true,
  365. // fontSize: 10,
  366. // position: 'insideLeft',
  367. // formatter: (obj) => {
  368. // var name = obj.name;
  369. // if (name.length > 6) {
  370. // name = name.slice(0, 5) + '..';
  371. // }
  372. // var value = obj.value;
  373. // if (value >= 10000) {
  374. // value = (value / 10000).toFixed(1) + '万';
  375. // }
  376. // return name + ':' + value;
  377. // } //'{b}: {c}'
  378. // }
  379. // },
  380. // itemStyle: {
  381. // color: '#409EFF'
  382. // },
  383. // data: this.areaList
  384. // .filter((p) => p.area != 0)
  385. // .sort((a, b) => {
  386. // return b.count - a.count;
  387. // })
  388. // .slice(0, 8)
  389. // .map((p) => p.count)
  390. // .reverse(),
  391. // animationDuration: 2800,
  392. // animationEasing: 'cubicInOut'
  393. // }
  394. // ]
  395. // });
  396. },
  397. initPercentPie() {
  398. // this.percentPieComponent = this.zpSelectComponent('#mychart-dom-percent-pie');
  399. // this.percentPieComponent.init((canvas, width, height) => {
  400. // const chart = echarts.init(canvas, null, {
  401. // width: width,
  402. // height: height
  403. // });
  404. // this.percentPieChart = chart;
  405. // this.percentPieConvertData = {};
  406. // this.setPercentPieOptions();
  407. // return chart;
  408. // });
  409. },
  410. setPercentPieOptions() {
  411. this.percentPieChart.setOption({
  412. series: [
  413. {
  414. name: '设备',
  415. type: 'pie',
  416. radius: '80%',
  417. data: [
  418. {
  419. value: this.deviceStatis.immobility_count,
  420. name: '静止',
  421. itemStyle: {
  422. color: '#409EFF'
  423. }
  424. },
  425. {
  426. value: this.deviceStatis.driving_count,
  427. name: '行驶',
  428. itemStyle: {
  429. color: '#67C23A'
  430. }
  431. },
  432. {
  433. value: this.deviceStatis.offline_count,
  434. name: '离线',
  435. itemStyle: {
  436. color: '#E6A23C'
  437. }
  438. },
  439. {
  440. value: this.deviceStatis.expire_count,
  441. name: '到期',
  442. itemStyle: {
  443. color: '#909399'
  444. }
  445. },
  446. {
  447. value: this.deviceStatis.disabled_count,
  448. name: '未启用',
  449. itemStyle: {
  450. color: '#F56C6C'
  451. }
  452. }
  453. ]
  454. .filter((p) => p.value != 0)
  455. .sort((a, b) => {
  456. return a.value - b.value;
  457. }),
  458. label: {
  459. position: 'outer',
  460. alignTo: 'none',
  461. bleedMargin: 5
  462. },
  463. roseType: 'radius',
  464. labelLine: {
  465. smooth: 0.2,
  466. length: 0,
  467. length2: 5
  468. },
  469. animationType: 'scale',
  470. animationEasing: 'elasticOut',
  471. animationDelay: function (idx) {
  472. return Math.random() * 200;
  473. }
  474. }
  475. ]
  476. });
  477. },
  478. initDynamicsChart() {
  479. // this.dynamicsComponent = this.zpSelectComponent('#mychart-dom-dynamics');
  480. // this.dynamicsComponent.init((canvas, width, height) => {
  481. // const chart = echarts.init(canvas, null, {
  482. // width: width,
  483. // height: height
  484. // });
  485. // this.dynamicsChart = chart;
  486. // this.setDynamicsOptions();
  487. // return chart;
  488. // });
  489. },
  490. setDynamicsOptions() {
  491. this.dynamicsChart.setOption({
  492. color: ['#67C23A'],
  493. grid: {
  494. left: '3%',
  495. right: '3%',
  496. top: '5%',
  497. bottom: '3%',
  498. containLabel: true
  499. },
  500. xAxis: {
  501. type: 'category',
  502. data: this.deviceMonthly.map((p) => p.time).reverse(),
  503. axisTick: {
  504. alignWithLabel: true
  505. }
  506. },
  507. yAxis: {
  508. type: 'value'
  509. },
  510. series: [
  511. {
  512. name: '设备增长',
  513. type: 'line',
  514. data: this.deviceMonthly.map((p) => p.count).reverse(),
  515. smooth: true
  516. }
  517. ]
  518. });
  519. },
  520. loadUserInfo: function () {
  521. const userInfo = storage.getUserInfo();
  522. if (!userInfo) {
  523. http.postApi(config.API_USER_INFO, {}, (resp) => {
  524. if (resp.data.code === 200) {
  525. //this.setUserInfo(resp.data.data)
  526. storage.setUserInfo(resp.data.data);
  527. //this.loadDynamicsData()
  528. } else {
  529. common.simpleToast(resp.data.msg);
  530. }
  531. });
  532. } else {
  533. //this.setUserInfo(userInfo)
  534. //this.loadDynamicsData()
  535. }
  536. },
  537. setUserInfo(userInfo) {
  538. var selectShop = {};
  539. selectShop.id = userInfo.userInfo.user_id;
  540. selectShop.name = userInfo.userInfo.shop_name;
  541. selectShop.head = userInfo.userInfo.shop_head;
  542. this.setData({
  543. selectShop: selectShop
  544. });
  545. },
  546. formatTime(time) {
  547. switch (this.timeUnit) {
  548. case 'date':
  549. return common.DateFormat(new Date(time), 'yyyy年MM月dd日');
  550. case 'month':
  551. return common.DateFormat(new Date(time), 'yyyy年MM月');
  552. case 'year':
  553. return common.DateFormat(new Date(time), 'yyyy年');
  554. default:
  555. return '';
  556. }
  557. },
  558. bindTimeSelectDate() {
  559. this.setData({
  560. timeUnit: 'date'
  561. });
  562. this.setData({
  563. formatSelectTime: this.formatTime(this.selectTime)
  564. });
  565. this.totalDynamicsData = {};
  566. this.shopDynamicsData = {};
  567. this.loadDynamicsData();
  568. this.refreshLoadOrderList();
  569. },
  570. bindTimeSelectMonth() {
  571. this.setData({
  572. timeUnit: 'month'
  573. });
  574. this.setData({
  575. formatSelectTime: this.formatTime(this.selectTime)
  576. });
  577. this.totalDynamicsData = {};
  578. this.shopDynamicsData = {};
  579. this.loadDynamicsData();
  580. this.refreshLoadOrderList();
  581. },
  582. bindTimeSelectYear() {
  583. this.setData({
  584. timeUnit: 'year'
  585. });
  586. this.setData({
  587. formatSelectTime: this.formatTime(this.selectTime)
  588. });
  589. this.totalDynamicsData = {};
  590. this.shopDynamicsData = {};
  591. this.loadDynamicsData();
  592. this.refreshLoadOrderList();
  593. },
  594. bindTimeSelectChange(event) {
  595. this.setData({
  596. selectTime: new Date(event.detail.value).getTime(),
  597. formatSelectTime: this.formatTime(new Date(event.detail.value).getTime())
  598. });
  599. this.totalDynamicsData = {};
  600. this.shopDynamicsData = {};
  601. this.loadDynamicsData();
  602. this.refreshLoadOrderList();
  603. },
  604. bindOrderSelectTotal() {
  605. this.setData({
  606. orderType: true
  607. });
  608. this.refreshLoadOrderList();
  609. },
  610. bindOrderSelectShop() {
  611. this.setData({
  612. orderType: false
  613. });
  614. this.refreshLoadOrderList();
  615. },
  616. refreshLoadOrderList: function () {
  617. if (this.orderLoading) {
  618. return;
  619. }
  620. this.setData({
  621. orderLoading: true
  622. });
  623. this.orderList = [];
  624. this.limit_page = 10;
  625. this.start_page = 1;
  626. this.loadOrderList();
  627. },
  628. loadOrderList: function () {
  629. var time = this.timeSelect();
  630. http.postApi(
  631. config.API_INCOME_USER,
  632. {
  633. limit: this.limit_page,
  634. page: this.start_page,
  635. shop_id: this.orderType ? '' : this.selectShop.id,
  636. begin_date: common.formatTime(time.begin_date / 1000),
  637. end_date: common.formatTime(time.end_date / 1000)
  638. },
  639. (resp) => {
  640. uni.stopPullDownRefresh();
  641. if (resp.data.code === 200) {
  642. let orderList = this.orderList;
  643. orderList.push.apply(orderList, resp.data.data.statisList.list);
  644. this.setData({
  645. orderList: orderList,
  646. orderNum: resp.data.data.statisList.total,
  647. orderTotal: resp.data.data.statisList.sum,
  648. orderLoading: false
  649. });
  650. } else {
  651. common.simpleToast(resp.data.msg);
  652. }
  653. }
  654. );
  655. },
  656. bindOrderClick(event) {
  657. uni.navigateTo({
  658. url: '/pages/searchOrder/searchOrder?type=2&search=' + event.target.dataset.phone
  659. });
  660. }
  661. }
  662. };
  663. </script>
  664. <style>
  665. @import './index.css';
  666. </style>