Newer
Older
CloudBrainNew / src / components / text / wordCloud / wordCloud.vue
StephanieGitHub on 4 Feb 2021 23 KB first commit
<!--
 * @Description: 词云
 * @Author: 王晓颖
 * @Date: 2020-08-17
 -->
<template>
  <div :id="id" :class="className" :style="{ height:curHeight,width:curWidth }" />
</template>

<script>
// import resize from './mixins/resize'
import 'echarts-wordcloud/dist/echarts-wordcloud'
import 'echarts-wordcloud/dist/echarts-wordcloud.min'

export default {
  name: 'WordCloud',
  props: {
    className: {
      type: String,
      default: 'chart'
    },
    id: {
      type: String,
      default: 'chart'
    },
    width: {
      type: String,
      default: '100%'
    },
    height: {
      type: String,
      default: '400px'
    },
    data: {
      type: Array,
      default: () => { return [] }
    },
    title: {
      type: String,
      default: ''
    },
    shape: {
      type: String,
      default: 'circle'
    }
  },
  watch: {
    width (newVal, oldVal) {
      this.curWidth = newVal
      this.refreshEchart()
    },
    height (newVal, oldVal) {
      this.curHeight = newVal
      this.refreshEchart()
    },
    shape (newVal, oldVal) {
      this.refreshEchart()
    },
    data (newVal, oldVal) {
      this.option.series[0].data = newVal
      this.refreshEchart()
    }
  },
  data () {
    return {
      chart: null,
      curWidth: this.width, // 宽
      curHeight: this.height, // 高
      options: {
        title: {
          text: '', // 标题
          x: 'center',
          color: '#ffffff'
        },
        backgroundColor: 'transparent', // 背景颜色, 透明
        series: [
          {
            type: 'wordCloud',
            shape: 'circle', // circle, cardioid, diamond, triangle, triangle-forward
            gridSize: 20, // 用来调整词之间的距离
            sizeRange: [10, 30], // 用来调整字的大小范围
            rotationRange: [0, 0], // 字体旋转的角度,[-45, 0, 45, 90],[[ 0,90]]
            textStyle: {
              normal: {
                color: function () {
                  return 'rgb(' + [
                    // Math.round(Math.random() * 255),
                    // Math.round(Math.random() * 255),
                    // Math.round(Math.random() * 255)
                    255, 255, 255
                  ].join(',') + ')' // 字体的随机颜色
                }
              },
              emphasis: {// 鼠标移入的特效样式
                shadowBlur: 10,
                shadowColor: '#b8b8b8'
              }
            },
            // 位置相关设置
            // Folllowing left/top/width/height/right/bottom are used for positioning the word cloud
            // Default to be put in the center and has 75% x 80% size.
            left: 'center',
            top: 'center',
            right: null,
            bottom: null,
            width: '100%',
            height: '100%',
            data: this.data // 数据
          }
        ]
      },
      shapeList: ['circle', 'cardioid', 'diamond', 'triangle', 'triangle-forward'],
      images: {
        'circle': '',
        'heart': '',
        'bird': ''
      }
    }
  },
  mounted () {
    this.initChart()
  },
  beforeDestroy () {
    if (!this.chart) {
      return
    }
    this.chart.dispose()
    this.chart = null
  },
  methods: {
    initChart () {
      // 处理标题
      if (this.title) {
        this.options.title.text = this.title
      }
      // 有图片
      if (this.shape) {
        if (this.shapeList.indexOf(this.shape)) {
          this.options.shape = this.shape
          this.setEchart()
        } else {
          const image = this.images[this.shape]
          const maskImage = new Image()
          maskImage.src = image
          this.options.maskImage = maskImage
          maskImage.onload = () => {
            this.setEchart()
          }
        }
      } else {
        this.setEchart()
      }
    },
    setEchart () {
      setTimeout(() => {
        const _div = document.getElementById(this.id)
        this.chart = this.$echarts.init(_div)
        this.chart.setOption(this.options)
      }, 500)
    },
    refreshEchart () {
      if (this.curWidth && this.curHeight && this.data.length > 0) {
        this.initEchart()
      }
    }
  }
}
</script>
<style lang='scss' scoped>
  .chartsClass {
    padding-left: 1.2rem;
  }
</style>