index.vue 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338
  1. <template>
  2. <div class="dashboard_container">
  3. <div class="refresh">
  4. <!-- <el-date-picker
  5. style="width: 130px"
  6. v-model="value1"
  7. format="yyyy"
  8. size="mini"
  9. type="year"
  10. placeholder="选择日期"
  11. >
  12. </el-date-picker> -->
  13. <span style="color: #1d82ff; margin-left: 20px" @click="refreshData">刷新数据</span>
  14. <i class="el-icon-refresh-left" style="color: #1d82ff" @click="refreshData"></i>
  15. </div>
  16. <div class="container_top">
  17. <div class="tab" v-for="(item, index) in tabs" :key="index" @click="openOrder(item.type, item.title)">
  18. <el-image class="img" :src="item.url" fit="fit"></el-image>
  19. <div class="text">
  20. <div class="title">{{ item.title }}</div>
  21. <div class="num">{{ item.num }}</div>
  22. </div>
  23. </div>
  24. </div>
  25. <div class="container_center">
  26. <div class="chart">
  27. <div class="head">
  28. <div class="title">近7天订单趋势</div>
  29. <div class="more" @click="openOrder()">查看更多</div>
  30. </div>
  31. <div ref="chart1" style="width: 100%; height: 280px"></div>
  32. </div>
  33. <div class="chart">
  34. <div class="head">
  35. <div class="title">未来7天需上门</div>
  36. </div>
  37. <div ref="chart2" style="width: 100%; height: 280px"></div>
  38. </div>
  39. </div>
  40. <div class="container_bottom">
  41. <div class="chart">
  42. <div class="head">
  43. <div class="title">近30天完工类型分布</div>
  44. </div>
  45. <div ref="chart3" style="width: 100%; height: 330px"></div>
  46. </div>
  47. </div>
  48. </div>
  49. </template>
  50. <script>
  51. import { getCount } from '@/api/workOrder/orderHome'
  52. export default {
  53. data() {
  54. return {
  55. tabs: [
  56. { title: '待派工', num: 0, url: require('@/assets/orderHome/dpg.png'), type: 'DWDPG' },
  57. { title: '待预约', num: 0, url: require('@/assets/orderHome/dpg.png'), type: 'DYY' },
  58. { title: '待抢单', num: 0, url: require('@/assets/orderHome/djs.png'), type: 'DQD' },
  59. { title: '待接收', num: 0, url: require('@/assets/orderHome/djs.png'), type: 'DJD' },
  60. { title: '服务中', num: 0, url: require('@/assets/orderHome/ffz.png'), type: 'FWZ' },
  61. { title: '异常待处理', num: 0, url: require('@/assets/orderHome/yc.png'), type: 'YCD' },
  62. { title: '差评统计', num: 0, url: require('@/assets/orderHome/pj.png'), type: 'appraise' }
  63. ],
  64. qtsm: [],
  65. qtddqs: {},
  66. wglxfb: {},
  67. value1: localStorage.getItem('orderYearVal') || new Date().getFullYear()
  68. }
  69. },
  70. watch: {
  71. value1(newVal) {
  72. this.getCount()
  73. }
  74. },
  75. computed: {
  76. getDate() {
  77. return function (d, t) {
  78. var date = new Date()
  79. var base = Date.parse(date) // 转换为时间戳
  80. var year = date.getFullYear() //获取当前年份
  81. var mon = date.getMonth() + 1 //获取当前月份
  82. var day = date.getDate() //获取当前日
  83. var oneDay = 24 * 3600 * 1000
  84. var daytimeArr = []
  85. for (var i = 0; i < d; i++) {
  86. //前七天的时间
  87. var now = new Date(t == 'before' ? base - oneDay * i : base + oneDay * i)
  88. var myear = now.getFullYear()
  89. var month = now.getMonth() + 1
  90. var mday = now.getDate()
  91. daytimeArr.push([month >= 10 ? month : '0' + month, mday >= 10 ? mday : '0' + mday].join('-'))
  92. }
  93. return t == 'before' ? daytimeArr.reverse() : daytimeArr
  94. }
  95. },
  96. filterOrderType() {
  97. return function (obj, type) {
  98. let names = []
  99. let series = []
  100. for (let key in this[obj]) {
  101. names.push(key)
  102. series.push({
  103. name: key,
  104. data: this[obj][key],
  105. type: 'line',
  106. smooth: true
  107. })
  108. }
  109. return type == 'name' ? names : series
  110. }
  111. }
  112. },
  113. created() {
  114. this.getCount()
  115. },
  116. mounted() {
  117. // this.drawChat1()
  118. },
  119. methods: {
  120. refreshData() {
  121. this.$echarts.init(this.$refs.chart1).dispose()
  122. this.$echarts.init(this.$refs.chart2).dispose()
  123. this.$echarts.init(this.$refs.chart3).dispose()
  124. this.getCount()
  125. },
  126. openOrder(type, name) {
  127. if (!type) {
  128. this.$router.push({
  129. name: window.isWorkOrderPoolPath,
  130. query: {}
  131. })
  132. } else if (type == 'appraise') {
  133. this.$router.push({
  134. name: 'appraise',
  135. query: {
  136. type: 1
  137. }
  138. })
  139. } else {
  140. this.$router.push({
  141. name: window.isWorkOrderPoolPath,
  142. params: {
  143. pageName: name || '-',
  144. pageType: 'orderStatus',
  145. pageCode: type
  146. }
  147. })
  148. }
  149. },
  150. getCount() {
  151. getCount(
  152. {}
  153. // this.value1
  154. // ? {
  155. // startTime: `${new Date(this.value1).getFullYear()}-01-01 00:00:00`,
  156. // endTime: `${new Date(this.value1).getFullYear()}-12-31 23:59:59`
  157. // }
  158. // : {}
  159. ).then(res => {
  160. this.tabs[0].num = res.data.dpg || 0
  161. this.tabs[1].num = res.data.dyy || 0
  162. this.tabs[2].num = res.data.dqd || 0
  163. this.tabs[3].num = res.data.djs || 0
  164. this.tabs[4].num = res.data.fwz || 0
  165. this.tabs[5].num = res.data.ycdcl || 0
  166. this.tabs[6].num = res.data.cp || 0
  167. this.qtsm = res.data.qtsm
  168. this.qtddqs = res.data.qtddqs
  169. this.wglxfb = res.data.wglxfb
  170. this.drawChat1()
  171. })
  172. },
  173. drawChat1() {
  174. let chart1 = this.$echarts.init(this.$refs.chart1)
  175. let chart2 = this.$echarts.init(this.$refs.chart2)
  176. let chart3 = this.$echarts.init(this.$refs.chart3)
  177. let option1 = {
  178. tooltip: {
  179. trigger: 'axis'
  180. },
  181. legend: {
  182. data: this.filterOrderType('qtddqs', 'name')
  183. },
  184. xAxis: {
  185. type: 'category',
  186. boundaryGap: false,
  187. data: this.getDate(7, 'before')
  188. },
  189. yAxis: {
  190. type: 'value'
  191. },
  192. min: 0,
  193. max: value => {
  194. // 百位起最大值向上取整
  195. return Math.ceil(value.max / 100) * 100
  196. },
  197. series: this.filterOrderType('qtddqs', 'series')
  198. }
  199. let option2 = {
  200. tooltip: {
  201. trigger: 'axis'
  202. },
  203. xAxis: {
  204. type: 'category',
  205. boundaryGap: false,
  206. data: this.getDate(7, 'after')
  207. },
  208. yAxis: {
  209. type: 'value'
  210. },
  211. min: 0,
  212. max: value => {
  213. // 百位起最大值向上取整
  214. return Math.ceil(value.max / 100) * 100
  215. },
  216. series: [
  217. {
  218. name: '工单数',
  219. data: this.qtsm,
  220. type: 'line',
  221. smooth: true
  222. }
  223. ]
  224. }
  225. let option3 = {
  226. tooltip: {
  227. trigger: 'axis'
  228. },
  229. legend: {
  230. data: this.filterOrderType('wglxfb', 'name')
  231. },
  232. xAxis: {
  233. type: 'category',
  234. boundaryGap: false,
  235. data: this.getDate(30, 'before')
  236. },
  237. min: 0,
  238. max: value => {
  239. // 百位起最大值向上取整
  240. return Math.ceil(value.max / 100) * 100
  241. },
  242. yAxis: {
  243. type: 'value'
  244. },
  245. series: this.filterOrderType('wglxfb', 'series')
  246. }
  247. chart1.setOption(option1)
  248. chart2.setOption(option2)
  249. chart3.setOption(option3)
  250. }
  251. }
  252. }
  253. </script>
  254. <style scoped lang="scss">
  255. .dashboard_container {
  256. background: #f5f5f5;
  257. padding: 20px;
  258. min-height: calc(100vh - 86px);
  259. box-sizing: border-box;
  260. .refresh {
  261. text-align: right;
  262. margin-bottom: 10px;
  263. span {
  264. cursor: pointer;
  265. }
  266. i {
  267. font-weight: bold;
  268. margin-left: 4px;
  269. cursor: pointer;
  270. }
  271. }
  272. .container_top {
  273. display: flex;
  274. .tab {
  275. display: flex;
  276. align-items: center;
  277. flex: 1;
  278. padding: 20px;
  279. margin-right: 20px;
  280. background-color: #ffffff;
  281. border-radius: 10px;
  282. cursor: pointer;
  283. .text {
  284. margin-left: 30px;
  285. .title {
  286. color: #666666;
  287. font-size: 16px;
  288. margin-bottom: 8px;
  289. }
  290. .num {
  291. font-size: 26px;
  292. font-weight: bold;
  293. }
  294. }
  295. .img {
  296. width: 36px;
  297. height: 36px;
  298. }
  299. }
  300. .tab:last-child {
  301. margin-right: 0;
  302. }
  303. }
  304. .container_center {
  305. display: flex;
  306. margin: 10px 0;
  307. margin-top: 20px;
  308. .chart {
  309. flex: 1;
  310. height: 300px;
  311. background-color: #ffffff;
  312. margin-right: 20px;
  313. }
  314. .chart:last-child {
  315. margin-right: 0;
  316. }
  317. }
  318. .container_bottom {
  319. width: 100%;
  320. height: 350px;
  321. background-color: #ffffff;
  322. }
  323. .head {
  324. width: 100%;
  325. display: flex;
  326. justify-content: space-between;
  327. padding: 16px 20px;
  328. font-weight: bold;
  329. .more {
  330. color: #1d82ff;
  331. cursor: pointer;
  332. }
  333. }
  334. }
  335. </style>