GoodsRanking.vue 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. <template>
  2. <div ref="chartsWrapperRef" class="diagrams-chart-container"></div>
  3. </template>
  4. <script>
  5. import echarts from 'echarts'
  6. export default {
  7. props: {
  8. rankingData: { type: Array, default: () => [] },
  9. categoryAlias: { type: String, default: 'productName' },
  10. valueAlias: { type: String, default: 'productTransactionTotal' },
  11. animate: { type: Boolean, default: false }
  12. },
  13. data() {
  14. return {
  15. instance: null,
  16. chartData: {},
  17. animateTimer: null,
  18. startIndex: 0
  19. }
  20. },
  21. watch: {
  22. rankingData: {
  23. handler(newData) {
  24. this.chartData = newData
  25. this.updateChartData()
  26. },
  27. immediate: true,
  28. deep: true
  29. }
  30. },
  31. mounted() {
  32. this.init()
  33. },
  34. methods: {
  35. init() {
  36. this.instance = echarts.init(this.$refs.chartsWrapperRef)
  37. this.instance.setOption({
  38. grid: { left: '12%', right: '3%', top: '2%', bottom: '9%' },
  39. xAxis: { type: 'value', show: false },
  40. yAxis: {
  41. type: 'category',
  42. axisLine: { show: false },
  43. axisTick: { show: false }
  44. }
  45. })
  46. this.updateChartData()
  47. },
  48. updateChartData() {
  49. if (!this.instance || !this.instance.setOption) {
  50. return
  51. }
  52. this.instance.setOption({
  53. yAxis: {
  54. data: this.chartData.map((item) => item[this.categoryAlias]).slice(this.startIndex, this.startIndex + 5).reverse(),
  55. axisLabel: {
  56. color: '#A4A4A4',
  57. fontSize: 14,
  58. formatter(value) {
  59. const valueStr = value.length > 5 ? value.slice(0, 5) + '...' : value
  60. return '{lineHeight|' + valueStr + '}'
  61. },
  62. rich: { lineHeight: { lineHeight: 18, color: '#A4A4A4', fontSize: 14 } }
  63. }
  64. },
  65. series: [
  66. {
  67. zlevel: 100,
  68. name: '销量',
  69. type: 'bar',
  70. data: this.chartData.map((item) => item[this.valueAlias]).slice(this.startIndex, this.startIndex + 5).reverse(),
  71. barWidth: '16px',
  72. itemStyle: { borderRadius: 5, color: '#00CE7F' },
  73. label: { show: true, precision: 1, position: 'right', valueAnimation: true, fontFamily: 'monospace', color: '#3d3d3d' }
  74. }
  75. ]
  76. })
  77. this.animate && this.startAnimate()
  78. },
  79. startAnimate() {
  80. if (this.animateTimer) {
  81. this.stopAnimate()
  82. }
  83. this.animateTimer = setInterval(() => {
  84. if (this.startIndex >= 15) {
  85. this.startIndex = -1
  86. }
  87. this.startIndex += 1
  88. this.updateChartData()
  89. }, 1000)
  90. },
  91. stopAnimate() {
  92. clearInterval(this.animateTimer)
  93. this.animateTimer = null
  94. },
  95. resize() {
  96. this.instance.resize()
  97. }
  98. }
  99. }
  100. </script>
  101. <style lang="scss" scoped>
  102. .diagrams-chart-container {
  103. width: 729px;
  104. height: 265px;
  105. }
  106. </style>