statistics.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647
  1. <!--
  2. * @Description: create
  3. * @Version: 1.0
  4. * @Autor: XuTongZhang
  5. * @Date: 2020-08-03 15:00:31
  6. * @LastEditors: XuTongZhang
  7. * @LastEditTime: 2020-08-12 18:37:45
  8. -->
  9. <template>
  10. <div>
  11. <el-card class="box-card">
  12. <div class="flex">
  13. <div ref="mychart1" :style="{width: '75%', height: '500px'}"></div>
  14. <div class="flex1">
  15. <div>
  16. <div class="num">校招投递简历总数 {{schoolTotal||0}}条</div>
  17. <div class="num">社招投递简历总数 {{clubTotal||0}}条</div>
  18. </div>
  19. <div>
  20. <div class="title">本周</div>
  21. <div class="num">校招投递简历数 {{schoolWeek||0}}条</div>
  22. <div class="num">社招投递简历数 {{clubWeek||0}}条</div>
  23. </div>
  24. </div>
  25. </div>
  26. </el-card>
  27. <el-card class="box-card">
  28. <div class="flex">
  29. <div ref="mychart2" :style="{width: '100%', height: '600px', marginTop: '30px'}"></div>
  30. </div>
  31. </el-card>
  32. <el-table :data="resumeTable" border style="width: 95%;margin: 0 auto">
  33. <el-table-column prop="name" label="渠道名称">
  34. </el-table-column>
  35. <el-table-column prop="0" label="已收到简历">
  36. <template slot-scope="scope">
  37. <span>{{scope.row[0] || 0}}</span>
  38. </template>
  39. </el-table-column>
  40. <el-table-column prop="1" label="已通过简历">
  41. <template slot-scope="scope">
  42. <span>{{scope.row[1] || 0}}</span>
  43. </template>
  44. </el-table-column>
  45. <el-table-column prop="2" label="已淘汰简历">
  46. <template slot-scope="scope">
  47. <span>{{scope.row[2] || 0}}</span>
  48. </template>
  49. </el-table-column>
  50. <el-table-column prop="3" label="待定简历">
  51. <template slot-scope="scope">
  52. <span>{{scope.row[3] || 0}}</span>
  53. </template>
  54. </el-table-column>
  55. </el-table>
  56. <div class="title">简历状态统计</div>
  57. <el-table :data="tableData" border style="width: 95%;margin: 0 auto">
  58. <el-table-column prop="positionApplied" label="职位">
  59. </el-table-column>
  60. <el-table-column prop="job" label="51job">
  61. </el-table-column>
  62. <el-table-column prop="zhilian" label="智联">
  63. </el-table-column>
  64. <el-table-column prop="tongcheng" label="58同城">
  65. </el-table-column>
  66. <el-table-column prop="platform" label="平台">
  67. </el-table-column>
  68. </el-table>
  69. <div class="title">简历投递职位统计</div>
  70. <el-card class="box-card">
  71. <div ref="mychart3" :style="{width: '100%', height: '600px', marginTop: '30px'}"></div>
  72. </el-card>
  73. <el-table :data="interviewTable" border style="width: 95%;margin: 0 auto">
  74. <el-table-column prop="name" label="渠道名称">
  75. </el-table-column>
  76. <el-table-column prop="0" label="已收到简历">
  77. <template slot-scope="scope">
  78. <span>{{scope.row[0] || 0}}</span>
  79. </template>
  80. </el-table-column>
  81. <el-table-column prop="1" label="已通过简历">
  82. <template slot-scope="scope">
  83. <span>{{scope.row[1] || 0}}</span>
  84. </template>
  85. </el-table-column>
  86. <el-table-column prop="2" label="已淘汰简历">
  87. <template slot-scope="scope">
  88. <span>{{scope.row[2] || 0}}</span>
  89. </template>
  90. </el-table-column>
  91. <el-table-column prop="3" label="待定简历">
  92. <template slot-scope="scope">
  93. <span>{{scope.row[3] || 0}}</span>
  94. </template>
  95. </el-table-column>
  96. </el-table>
  97. <div class="title">面试状态统计</div>
  98. <el-table :data="tableList" border style="width: 95%;margin: 0 auto">
  99. <el-table-column prop="positionApplied" label="职位">
  100. </el-table-column>
  101. <el-table-column prop="job" label="51job">
  102. </el-table-column>
  103. <el-table-column prop="zhilian" label="智联">
  104. </el-table-column>
  105. <el-table-column prop="tongcheng" label="58同城">
  106. </el-table-column>
  107. <el-table-column prop="platform" label="平台">
  108. </el-table-column>
  109. </el-table>
  110. <div class="title">面试者意向职位统计</div>
  111. </div>
  112. </template>
  113. <script>
  114. import echarts from 'echarts'
  115. export default {
  116. data () {
  117. return {
  118. timeList: [],
  119. clubRecruit: [],
  120. schoolRecruit: [],
  121. clubTotal: 0,
  122. schoolTotal: 0,
  123. clubWeek: 0,
  124. schoolWeek: 0,
  125. platformList: ['51job', '智联', '58同城', '平台'],
  126. resumeList: [],
  127. resumeTable: [],
  128. interviewList: [],
  129. interviewTable: [],
  130. tableData: [],
  131. tableList: []
  132. }
  133. },
  134. mounted () {
  135. this.queryData()
  136. this.queryData2()
  137. this.queryData3()
  138. this.countResumeTotal()
  139. this.countInterviewTotal()
  140. },
  141. methods: {
  142. queryData () {
  143. this.$api
  144. .post('/chart/getResumeChart', {
  145. reqdata: {
  146. recruitType: 0
  147. }
  148. })
  149. .then((res) => {
  150. console.log(res)
  151. this.timeList = res.object.weekList || []
  152. this.clubRecruit = res.object.countList || []
  153. this.clubTotal = res.object.countTotal
  154. this.clubWeek = this.clubRecruit[this.clubRecruit.length - 1]
  155. this.$api
  156. .post('/chart/getResumeChart', {
  157. reqdata: {
  158. recruitType: 1
  159. }
  160. })
  161. .then((res) => {
  162. let arr = []
  163. let length = this.timeList.indexOf(res.object.weekList && res.object.weekList[0])
  164. arr.length = length === -1 ? 0 : length
  165. this.schoolRecruit = [...arr, ...(res.object.countList || [])]
  166. this.schoolTotal = res.object.countTotal
  167. // console.log(this.schoolTotal)
  168. this.schoolWeek = this.schoolRecruit[this.schoolRecruit.length - 1]
  169. this.initChart()
  170. })
  171. })
  172. },
  173. queryData2 () {
  174. this.$api
  175. .post('/chart/getResumeTotal', {
  176. reqdata: {}
  177. }).then(res => {
  178. this.resumeList = res.list
  179. this.resumeTable = res.list.map((item, index) => {
  180. item.name = this.platformList[index]
  181. return item
  182. })
  183. console.log((this.resumeList[0]))
  184. this.initChart2()
  185. })
  186. },
  187. queryData3 () {
  188. this.$api
  189. .post('/chart/getInterviewTotal', {
  190. reqdata: {}
  191. }).then(res => {
  192. this.interviewList = res.list
  193. this.interviewTable = res.list.map((item, index) => {
  194. item.name = this.platformList[index]
  195. return item
  196. })
  197. this.initChart3()
  198. })
  199. },
  200. countResumeTotal () {
  201. this.$api
  202. .post('/chart/countResumeTotal', {
  203. reqdata: {}
  204. }).then(res => {
  205. this.tableData = res.list
  206. })
  207. },
  208. countInterviewTotal () {
  209. this.$api
  210. .post('/chart/countInterviewTotal', {
  211. reqdata: {}
  212. }).then(res => {
  213. this.tableList = res.list
  214. })
  215. },
  216. initChart () {
  217. let myChart = echarts.init(this.$refs.mychart1)
  218. myChart.setOption({
  219. title: {
  220. text: '简历投递信息'
  221. },
  222. tooltip: {
  223. trigger: 'axis'
  224. },
  225. legend: {
  226. data: ['校招投递简历', '社招投递简历']
  227. },
  228. grid: {
  229. left: '3%',
  230. right: '4%',
  231. bottom: '3%',
  232. containLabel: true
  233. },
  234. // toolbox: {
  235. // feature: {
  236. // saveAsImage: {}
  237. // }
  238. // },
  239. xAxis: {
  240. type: 'category',
  241. boundaryGap: false,
  242. data: this.timeList.map(item => item.substring(0, 10)),
  243. axisLabel: {
  244. interval: 0,
  245. rotate: -80
  246. }
  247. },
  248. yAxis: {
  249. type: 'value'
  250. },
  251. series: [{
  252. name: '校招投递简历',
  253. type: 'line',
  254. data: this.schoolRecruit
  255. },
  256. {
  257. name: '社招投递简历',
  258. type: 'line',
  259. data: this.clubRecruit
  260. }
  261. ]
  262. })
  263. setTimeout(function () {
  264. window.onresize = function () {
  265. myChart.resize()
  266. }
  267. }, 200)
  268. },
  269. initChart2 () {
  270. let myChart = echarts.init(this.$refs.mychart2)
  271. let app = {}
  272. var posList = [
  273. 'left', 'right', 'top', 'bottom',
  274. 'inside',
  275. 'insideTop', 'insideLeft', 'insideRight', 'insideBottom',
  276. 'insideTopLeft', 'insideTopRight', 'insideBottomLeft', 'insideBottomRight'
  277. ]
  278. app.configParameters = {
  279. rotate: {
  280. min: -90,
  281. max: 90
  282. },
  283. align: {
  284. options: {
  285. left: 'left',
  286. center: 'center',
  287. right: 'right'
  288. }
  289. },
  290. verticalAlign: {
  291. options: {
  292. top: 'top',
  293. middle: 'middle',
  294. bottom: 'bottom'
  295. }
  296. },
  297. position: {
  298. options: echarts.util.reduce(posList, function (map, pos) {
  299. map[pos] = pos
  300. return map
  301. }, {})
  302. },
  303. distance: {
  304. min: 0,
  305. max: 100
  306. }
  307. }
  308. app.config = {
  309. rotate: 90,
  310. align: 'left',
  311. verticalAlign: 'middle',
  312. position: 'insideBottom',
  313. distance: 15,
  314. onChange: function () {
  315. let labelOption = {
  316. normal: {
  317. rotate: app.config.rotate,
  318. align: app.config.align,
  319. verticalAlign: app.config.verticalAlign,
  320. position: app.config.position,
  321. distance: app.config.distance
  322. }
  323. }
  324. myChart.setOption({
  325. series: [{
  326. label: labelOption
  327. }, {
  328. label: labelOption
  329. }, {
  330. label: labelOption
  331. }, {
  332. label: labelOption
  333. }]
  334. })
  335. }
  336. }
  337. // let labelOption = {
  338. // show: true,
  339. // position: app.config.position,
  340. // distance: app.config.distance,
  341. // align: app.config.align,
  342. // verticalAlign: app.config.verticalAlign,
  343. // rotate: app.config.rotate,
  344. // formatter: '{c} {name|{a}}',
  345. // fontSize: 16,
  346. // rich: {
  347. // name: {
  348. // textBorderColor: '#fff'
  349. // }
  350. // }
  351. // }
  352. let option = {
  353. title: {
  354. text: '招聘渠道分析:简历管理'
  355. },
  356. color: ['#003366', '#006699', '#4cabce', '#e5323e'],
  357. tooltip: {
  358. trigger: 'axis',
  359. axisPointer: {
  360. type: 'shadow'
  361. }
  362. },
  363. legend: {
  364. data: ['已收到', '已通过', '已淘汰', '待定']
  365. },
  366. toolbox: {
  367. show: true,
  368. orient: 'vertical',
  369. left: 'right',
  370. top: 'center',
  371. feature: {
  372. mark: {
  373. show: true
  374. },
  375. dataView: {
  376. show: true,
  377. readOnly: false
  378. },
  379. magicType: {
  380. show: true,
  381. type: ['line', 'bar', 'stack', 'tiled']
  382. },
  383. restore: {
  384. show: true
  385. },
  386. saveAsImage: {
  387. show: true
  388. }
  389. }
  390. },
  391. xAxis: [{
  392. type: 'category',
  393. axisTick: {
  394. show: false
  395. },
  396. data: this.platformList
  397. }],
  398. yAxis: [{
  399. type: 'value'
  400. }],
  401. series: [{
  402. name: '已收到',
  403. type: 'bar',
  404. barGap: 0,
  405. // label: labelOption,
  406. data: [this.resumeList[0]['0'] || 0, this.resumeList[1]['0'] || 0, this.resumeList[2]['0'] || 0, this.resumeList[3]['0'] || 0]
  407. },
  408. {
  409. name: '已通过',
  410. type: 'bar',
  411. // label: labelOption,
  412. data: [this.resumeList[0]['1'] || 0, this.resumeList[1]['1'] || 0, this.resumeList[2]['1'] || 0, this.resumeList[3]['1'] || 0]
  413. },
  414. {
  415. name: '已淘汰',
  416. type: 'bar',
  417. // label: labelOption,
  418. data: [this.resumeList[0]['2'] || 0, this.resumeList[1]['2'] || 0, this.resumeList[2]['2'] || 0, this.resumeList[3]['2'] || 0]
  419. },
  420. {
  421. name: '待定',
  422. type: 'bar',
  423. // label: labelOption,
  424. data: [this.resumeList[0]['3'] || 0, this.resumeList[1]['3'] || 0, this.resumeList[2]['3'] || 0, this.resumeList[3]['3'] || 0]
  425. }
  426. ]
  427. }
  428. myChart.setOption(option)
  429. setTimeout(function () {
  430. window.onresize = function () {
  431. myChart.resize()
  432. }
  433. }, 200)
  434. },
  435. initChart3 () {
  436. let myChart = echarts.init(this.$refs.mychart3)
  437. let app = {}
  438. var posList = [
  439. 'left', 'right', 'top', 'bottom',
  440. 'inside',
  441. 'insideTop', 'insideLeft', 'insideRight', 'insideBottom',
  442. 'insideTopLeft', 'insideTopRight', 'insideBottomLeft', 'insideBottomRight'
  443. ]
  444. app.configParameters = {
  445. rotate: {
  446. min: -90,
  447. max: 90
  448. },
  449. align: {
  450. options: {
  451. left: 'left',
  452. center: 'center',
  453. right: 'right'
  454. }
  455. },
  456. verticalAlign: {
  457. options: {
  458. top: 'top',
  459. middle: 'middle',
  460. bottom: 'bottom'
  461. }
  462. },
  463. position: {
  464. options: echarts.util.reduce(posList, function (map, pos) {
  465. map[pos] = pos
  466. return map
  467. }, {})
  468. },
  469. distance: {
  470. min: 0,
  471. max: 100
  472. }
  473. }
  474. app.config = {
  475. rotate: 90,
  476. align: 'left',
  477. verticalAlign: 'middle',
  478. position: 'insideBottom',
  479. distance: 15,
  480. onChange: function () {
  481. var labelOption = {
  482. normal: {
  483. rotate: app.config.rotate,
  484. align: app.config.align,
  485. verticalAlign: app.config.verticalAlign,
  486. position: app.config.position,
  487. distance: app.config.distance
  488. }
  489. }
  490. myChart.setOption({
  491. series: [{
  492. label: labelOption
  493. }, {
  494. label: labelOption
  495. }, {
  496. label: labelOption
  497. }, {
  498. label: labelOption
  499. }]
  500. })
  501. }
  502. }
  503. // let labelOption = {
  504. // show: true,
  505. // position: app.config.position,
  506. // distance: app.config.distance,
  507. // align: app.config.align,
  508. // verticalAlign: app.config.verticalAlign,
  509. // rotate: app.config.rotate,
  510. // formatter: '{c} {name|{a}}',
  511. // fontSize: 16,
  512. // rich: {
  513. // name: {
  514. // textBorderColor: '#fff'
  515. // }
  516. // }
  517. // }
  518. let {
  519. interviewList
  520. } = this
  521. let option = {
  522. title: {
  523. text: '招聘渠道分析:面试管理'
  524. },
  525. color: ['#003366', '#006699', '#4cabce', '#e5323e'],
  526. tooltip: {
  527. trigger: 'axis',
  528. axisPointer: {
  529. type: 'shadow'
  530. }
  531. },
  532. legend: {
  533. data: ['已通知', '已通过', '已淘汰', '待定']
  534. },
  535. toolbox: {
  536. show: true,
  537. orient: 'vertical',
  538. left: 'right',
  539. top: 'center',
  540. feature: {
  541. mark: {
  542. show: true
  543. },
  544. dataView: {
  545. show: true,
  546. readOnly: false
  547. },
  548. magicType: {
  549. show: true,
  550. type: ['line', 'bar', 'stack', 'tiled']
  551. },
  552. restore: {
  553. show: true
  554. },
  555. saveAsImage: {
  556. show: true
  557. }
  558. }
  559. },
  560. xAxis: [{
  561. type: 'category',
  562. axisTick: {
  563. show: false
  564. },
  565. data: this.platformList
  566. }],
  567. yAxis: [{
  568. type: 'value'
  569. }],
  570. series: [{
  571. name: '已通知',
  572. type: 'bar',
  573. barGap: 0,
  574. // label: labelOption,
  575. data: [interviewList[0]['0'] || 0, interviewList[1]['0'] || 0, interviewList[2]['0'] || 0, interviewList[3]['0'] || 0]
  576. },
  577. {
  578. name: '已通过',
  579. type: 'bar',
  580. // label: labelOption,
  581. data: [interviewList[0]['1'] || 0, interviewList[1]['1'] || 0, interviewList[2]['1'] || 0, interviewList[3]['1'] || 0]
  582. },
  583. {
  584. name: '已淘汰',
  585. type: 'bar',
  586. // label: labelOption,
  587. data: [interviewList[0]['2'] || 0, interviewList[1]['2'] || 0, interviewList[2]['2'] || 0, interviewList[3]['2'] || 0]
  588. },
  589. {
  590. name: '待定',
  591. type: 'bar',
  592. // label: labelOption,
  593. data: [interviewList[0]['3'] || 0, interviewList[1]['3'] || 0, interviewList[2]['3'] || 0, interviewList[3]['3'] || 0]
  594. }
  595. ]
  596. }
  597. myChart.setOption(option)
  598. setTimeout(function () {
  599. window.onresize = function () {
  600. myChart.resize()
  601. }
  602. }, 200)
  603. }
  604. }
  605. }
  606. </script>
  607. <style lang="scss" scoped>
  608. .flex {
  609. display: flex;
  610. }
  611. .flex1 {
  612. width: 200px;
  613. margin-top: 00px;
  614. margin-left: 80px;
  615. }
  616. .title {
  617. margin: 8px;
  618. text-align: center;
  619. font-size: 16px;
  620. }
  621. .num {
  622. margin: 20px;
  623. font-size: 14px;
  624. }
  625. .box-card {
  626. margin: 10px;
  627. }
  628. </style>