vue中使用echarts词云

2023-11-03

1. 安装

cnpm install echarts-wordcloud

2. 创建模板组件

  • WordCloudChart
<template>
    <div :id="id" :style="{ height:height,width:width }" />
</template>

<script>
    import echarts from "echarts/lib/echarts";
    import resize from "@/mixins/resize";
    import "echarts-wordcloud/dist/echarts-wordcloud";
    import "echarts-wordcloud/dist/echarts-wordcloud.min";

    export default {
        mixins: [resize],
        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: []
            },
            title: {
                type: String,
                default: ""
            }
        },
        data() {
            return {
                chart: null
            };
        },
        watch: {
            data(data) {
                this.initChart();
            },
        },
        mounted() {
            this.initChart();
        },
        beforeDestroy() {
            if (!this.chart) {
                return;
            }
            this.chart.dispose();
            this.chart = null;
        },
        methods: {
            initChart() {
                this.chart = echarts.init(document.getElementById(this.id));


                //let maskImage = new Image();   /*自定义形状关键点1*/
                //maskImage.src = "";

                //maskImage.src = require('@/assets/c.png')

                let option = {
                    title: {
                        text: this.title,
                        x: "center"
                    },
                    backgroundColor: "#fff",
                    tooltip: {
                      show: true,
                        formatter: function (data) {
                            let result = "<span>词条名称:</span><span>"+data.data.name+"</span><br/>";
                            result += "<span>词条数量:</span><span>"+data.data.value+"</span><br/>";
                            result += "<span>词条覆盖率:</span><span>"+data.data.rate+"%</span><br/>";
                            return result;
                        }
                    },
                    // tooltip: {
                    //   pointFormat: "{series.name}: <b>{point.percentage:.1f}%</b>"
                    // },
                    series: [
                        {
                            type: "wordCloud",
                            //用来调整词之间的距离
                            gridSize: 15,
                            //用来调整字的大小范围
                            // Text size range which the value in data will be mapped to.
                            // Default to have minimum 12px and maximum 60px size.
                            sizeRange: [15, 80],
                            // Text rotation range and step in degree. Text will be rotated randomly in range [-90,                                                                             90] by rotationStep 45
                            //用来调整词的旋转方向,,[0,0]--代表着没有角度,也就是词为水平方向,需要设置角度参考注释内容
                            // rotationRange: [-45, 0, 45, 90],
                            // rotationRange: [ 0,90],
                            rotationRange: [0, 0],
                            //随机生成字体颜色
                            //shape:'circle',
                            //maskImage: maskImage,
                            textStyle: {
                                normal: {
                                    color: function(data) {
                                        console.log("color-data")
                                        console.log(data)
                                        let colors = [
                                            '#092d52',
                                            '#2b4968',
                                            '#0f387a',
                                            '#336190',
                                            '#3d87d2',
                                            '#5690cc',
                                            '#68a3f5',
                                            '#7e9fc2',
                                            '#a7bdd4'
                                        ]
                                        /*return (
                                            "rgb(" +
                                            Math.round(Math.random() * 255) +
                                            ", " +
                                            Math.round(Math.random() * 255) +
                                            ", " +
                                            Math.round(Math.random() * 255) +
                                            ")"
                                        );*/
                                        return colors[parseInt(data.dataIndex / 4)];
                                    }
                                    /*color: [
                                        '#092d52', '#2b4968', '#0f387a', '#336190', '#3d87d2', '#5690cc',  '#68a3f5', '#7e9fc2', '#a7bdd4'
                                    ]*/
                                }
                            },
                            //位置相关设置
                            // 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
                        }
                    ]
                };

                /*let _self = this;
                maskImage.onload = function(){   /!* 自定义形状,关键点4*!/
                    _self.chart.setOption(option);
                };*/

                this.chart.setOption(option);
            }
        }
    };
</script>
<style lang='scss' scoped>
    .chartsClass {
        padding-left: 1.2rem;
    }
</style>

  • resize.js
export default {
    data() {
        return {
            $_sidebarElm: null
        }
    },
    mounted() {
        this.__resizeHandler = this.debounce(() => {
            if (this.chart) {
                this.chart.resize()
            }
        }, 100)
        window.addEventListener('resize', this.__resizeHandler)

        this.$_sidebarElm = document.getElementsByClassName('sidebar-container')[0]
        this.$_sidebarElm && this.$_sidebarElm.addEventListener('transitionend', this.$_sidebarResizeHandler)
    },
    beforeDestroy() {
        window.removeEventListener('resize', this.__resizeHandler)

        this.$_sidebarElm && this.$_sidebarElm.removeEventListener('transitionend', this.$_sidebarResizeHandler)
    },
    methods: {
        // use $_ for mixins properties
        // https://vuejs.org/v2/style-guide/index.html#Private-property-names-essential
        $_sidebarResizeHandler(e) {
            if (e.propertyName === 'width') {
                this.__resizeHandler()
            }
        },
        debounce(func, wait, immediate) {
            let timeout, args, context, timestamp, result

            const later = function () {
                // 据上一次触发时间间隔
                const last = +new Date() - timestamp

                // 上次被包装函数被调用时间间隔 last 小于设定时间间隔 wait
                if (last < wait && last > 0) {
                    timeout = setTimeout(later, wait - last)
                } else {
                    timeout = null
                    // 如果设定为immediate===true,因为开始边界已经调用过了此处无需调用
                    if (!immediate) {
                        result = func.apply(context, args)
                        if (!timeout) context = args = null
                    }
                }
            }
        }
    }
}

3. 使用时引用

<!-- 图表案例 -->
<template>
  <div>
    <el-row>
      <!--<el-radio-group size="mini" v-model="pType" style="margin-bottom: 30px;" @change="typeChange">
				<el-radio-button  label="left">全部</el-radio-button>
				<el-radio-button  label="center">坐席</el-radio-button>
				<el-radio-button  label="right">客户</el-radio-button>
      </el-radio-group>-->
    </el-row>
    <el-row>
      <el-col :span="12">
        <div class="nodata" v-if="words.length == 0"></div>
        <!--<vue-word-cloud
          v-else="words.length == 0"
          style="width:500px;height:500px;"
          :words="words"
          :color="([, weight]) => weight % 9 == 0 ? colors[0] : weight % 8 == 0 ? colors[1] : weight % 7 == 0 ? colors[2] : weight % 6 == 0 ? colors[3] : weight % 5 == 0 ? colors[4] : weight % 4 == 0 ? colors[5] : weight % 3 == 0 ? colors[6] : weight % 2 == 0 ? colors[7] : colors[8]"
          font-family="Roboto"
          :font-size-ratio="fontSizeRatio"
          :spacing="spacing"
          :spiral="'rectangular'"
        >
          <template slot-scope="{text, weight, word}">
            <div :title="txtMove(word)" style="cursor: pointer;">{{ text }}</div>
          </template>
        </vue-word-cloud>-->
        <word-cloud-chart
                id="chartId"
                :title="''"
                :data="echarts05Data"
                :width="'500px'"
                :height="'500px'"
        >
        </word-cloud-chart>
      </el-col>
      <el-col :span="12">
        <el-table :data="tableData" height="500" border style="width: 100%">
          <!--<el-table-column type="expand">
						<template slot-scope="props">
							{{props.row.topicName}}
						</template>
          </el-table-column>-->
          <el-table-column type="index" label="序号" min-width="9%" align="center"></el-table-column>
          <!--:sort-method="sortDevName"-->
          <el-table-column prop="topicName" label="词条名称" sortable min-width="27%"></el-table-column>
          <el-table-column prop="topicNum" sortable label="词条数量" min-width="18%"></el-table-column>
          <el-table-column prop="topicCoverRate" sortable label="词条覆盖率" min-width="19%">
            <template slot-scope="scope">
              <span>{{scope.row.topicCoverRate}}%</span>
            </template>
          </el-table-column>
          <el-table-column
            v-if="!isShowOperation"
            min-width="14%"
            align="center"
            prop="address"
            label="操作"
          >
            <template slot-scope="scope">
              <el-button @click="handle(scope.row)" type="text" size="small">查看明细</el-button>
            </template>
          </el-table-column>
        </el-table>
      </el-col>
    </el-row>
  </div>
</template>

<style>
#chart {
  display: inline-block;
  width: 100%;
  height: 300px;
  /*pointer-events : none;*/
}
</style>

<script>
import VueWordCloud from "vuewordcloud";
import { getFaq } from "@/api/enterprise/phoneBusinessTheme/complainCause"
import qs from "qs";
import { detailedRouter } from "@/utils/detailedRouter";

import WordCloudChart from "@/components/WordCloudChart"

export default {
  components: {
    VueWordCloud,
    WordCloudChart
  },
  props: {
    repeatComplain: {
      type: Number
    },
    types: {
      type: Object
    },
    tabIndex: {
      type: Number
    },
    type: {
      type: String
    },
    formInline: {
      type: Object
    },
    levelName: {
      type: String
    },
    clientIssue: {
      type: String
    },
    isShowOperation: {
      type: Boolean,
      defalult: false
    },
    firstGradeHotWords: {
      type: String
    }
  },
  mounted() {},
  watch: {
    firstGradeHotWords: "changeFirstGradeHotWords"
  },
  data() {
    let options = qs.parse(location.search, { ignoreQueryPrefix: true });

    return {
      options,
      //面板
      pType: "left",
      spacing: 0.3,
      fontSizeRatio: 5,
      colors: [
					'#092d52', '#2b4968', '#0f387a', '#336190', '#3d87d2', '#5690cc',  '#68a3f5', '#7e9fc2', '#a7bdd4'
				],
      words: [
        //['暂无数据', 1499]
      ],
      tableData: [],
      firstGradeWordsCompare: "", // 对比时用的一级
      clientIssueCompare: "", // 对比时用的文本检索

      echarts05Data: [

      ]


    };
  },
  created() {
    let params = {};
    this.initParams(params);
    this.initData(params);
  },
  methods: {
    sortDevName(str1, str2) {
      let res = 0;
      for (let i = 0; ; i++) {
        if (!str1[i] || !str2[i]) {
          res = str1.length - str2.length;
          break;
        }
        const char1 = str1[i];
        const char1Type = this.getChartType(char1);
        const char2 = str2[i];
        const char2Type = this.getChartType(char2);
        // 类型相同的逐个比较字符
        if (char1Type[0] === char2Type[0]) {
          if (char1 === char2) {
            continue;
          } else {
            if (char1Type[0] === "zh") {
              res = char1.localeCompare(char2);
            } else if (char1Type[0] === "en") {
              res = char1.charCodeAt(0) - char2.charCodeAt(0);
            } else {
              res = char1 - char2;
            }
            break;
          }
        } else {
          // 类型不同的,直接用返回的数字相减
          res = char1Type[1] - char2Type[1];
          break;
        }
      }
      return res;
    },
    getChartType(char) {
      // 数字可按照排序的要求进行自定义,我这边产品的要求是
      // 数字(0->9)->大写字母(A->Z)->小写字母(a->z)->中文拼音(a->z)
      if (/^[\u4e00-\u9fa5]$/.test(char)) {
        return ["zh", 300];
      }
      if (/^[a-zA-Z]$/.test(char)) {
        return ["en", 200];
      }
      if (/^[0-9]$/.test(char)) {
        return ["number", 100];
      }
      return ["others", 999];
    },
    changeFirstGradeHotWords() {
      console.log(this.firstGradeHotWords);
      let params = {};
      this.initParams(params);
      //筛选条件如果是全部,传过来" ",不用处理,还是取formInline里的值
      if (this.firstGradeHotWords !== "") {
        params.firstGradeComplain = this.firstGradeHotWords;
      }
      this.initData(params);
    },

    txtMove(_item) {
      let result =
        /*"词条名称:" +
        _item[0] +
        "\n词条数量:" +
        _item[1] +
        "\n词条覆盖率:" +
        _item[2] +
        "%\n";*/
        "aaa";
      return result;
    },
    handle(row) {
		/*row.jumpFrom = "howWords";
		row.firstGrade = this.firstGradeHotWords;
      this.formInlineCopy = this.formInline;
      let json = {
        formInlineCopy: this.formInlineCopy,
        data: row
      };
      this.$router.push({
        path: "/npsTheme/npsDataDetail",
        query: { json: json }
      });*/
      this.formInlineCopy = this.formInline;
      if (this.repeatComplain && this.repeatComplain === 1) {
        this.formInlineCopy.isComplain = "是"
        this.formInlineCopy.numOfComplain = 2;  //后台处理大于等于
      }
      this.formInlineCopy.topicName = row.topicName;
      //一级类别
      if (this.firstGradeHotWords !== "") {
        this.formInlineCopy.classifyA = this.firstGradeHotWords;
      }
      let json = {
          "formInlineCopy": this.formInlineCopy,
      };
      let themeName = "complainCause"
      detailedRouter(this, json, themeName);
    },
    typeChange(value) {
      console.log(value);
      this.initData();
    },
    initParams(params1) {
      if (this.firstGradeHotWords !== "") {
        params1.firstGradeComplain = this.firstGradeHotWords;
      } else {
        params1.firstGradeComplain = "";
      }

      params1.startTime = this.formInline.time[0];
      params1.endTime = this.formInline.time[1];
      /*let a1 = new Array();
      a1.push(this.formInline.business);
      params1.typeOfJob = this.formInline.business == '' ? '' : a1;
      let a2 = new Array();
      a2.push(this.formInline.institution);
      params1.areaOfJob = this.formInline.institution == '' ? '' : a2;*/
      params1.typeOfJob = this.formInline.business;
      params1.areaOfJob = this.formInline.institution;

      params1.product = this.formInline.product;
      params1.repGroup = this.formInline.repGroup;
      params1.repNo = this.formInline.repNo;
      params1.custName = this.formInline.custName;
      params1.callNumber = this.formInline.callNumber;
      params1.numOfComplain = this.formInline.numOfComplain;
      params1.isComplain = "是"
      if (this.repeatComplain && this.repeatComplain === 1) {
        params1.isComplain = "是"
        params1.numOfComplain = 2;  //后台处理大于等于
      }

      if (this.clientIssue) {
        params1.matchEvalWordTriplet = this.clientIssue;
      }
      //发请求时把一级存住,对比中使用
                if (this.firstGradeHotWords !== undefined ) {
                    this.formInline.firstGradeWordsCompare = JSON.parse(JSON.stringify(this.firstGradeHotWords));
                }
                if (this.formInline.firstGradeWordsCompare!== undefined) {
                    params1.firstGradeComplain = this.formInline.firstGradeWordsCompare
                }
                //发请求时把文本框存住,对比中使用
                if (this.clientIssue !== undefined ) {
                    this.formInline.clientIssueCompare = JSON.parse(JSON.stringify(this.clientIssue));
                }
                if (this.formInline.clientIssueCompare!== undefined) {
                    params1.matchEvalWordTriplet = this.formInline.clientIssueCompare
                }
      /*if(this.levelName){
					if(this.levelName.indexOf(",") != -1){
						let _callResult = this.levelName.split(",")[0];
						let _firstLevel = this.levelName.split(",")[1];
						params1.callResult = _callResult;
						params1.firstLevel = [_firstLevel];
					}else{
						params1.firstLevel = [this.levelName];
					}

				}*/
    },
    initData(params) {
      getFaq(params).then(res => {
          if (!res.words || res.words.length == 0) {
            this.tableData = [];
            this.words = [
              //['暂无数据', 1499]
            ];
            this.echarts05Data = [];

          }
          let tables = res.list;
          //if(tables.length > 20)tables.length = 20;

          let wordsData = res.words;
          if (wordsData.length > 30) wordsData.length = 30;


        let echarts05Data = []
        for (let i = 0; i < wordsData.length; i++) {
          let json = {
            name: wordsData[i][0],
            value: wordsData[i][1],
            rate: wordsData[i][2]
          }
          echarts05Data.push(json);
        }
        this.echarts05Data = echarts05Data;

          this.tableData = tables;
          this.words = wordsData;
        })
        .catch(err => {
          console.log(err);
        });
    },
    // 数据转换
    formatJson(filterVal, jsonData) {
      return jsonData.map(v =>
        filterVal.map(j => {
          if (j == "topicCoverRate") {
            return v[j] + "%";
          }
          return v[j];
        })
      );
    },
    downloadFAQ() {
      for (let i = 0; i < this.tableData.length; i++) {
        this.tableData[i].index = i + 1;
      }
      if (!this.tableData || this.tableData.length === 0) {
        this.$message({
          message: "暂无数据",
          type: "warning",
          duration: 2500
        });
      } else if (this.tableData.length > 0) {
        let _self = this;
        //用来刷新按钮小图标的
        import("@/vendor/Export2Excel").then(excel => {
          const tHeader = ["序号", "词条名称", "词条数量", "词条覆盖率"];
          const filterVal = ["index", "topicName", "topicNum", "topicCoverRate"];
          const data = this.formatJson(filterVal, _self.tableData);
          excel.export_json_to_excel({
            header: tHeader,
            data,
            filename: "热词分析"
          });
        });
      }
    }
  }
};
</script>

4.echartswordcloud和vuewordcloud对比

  1. echarts有自带的提示框,可自定义;vuewordcloud需要手写一个
  2. echarts初始化是一个一个词出,不会感觉特别慢;vuewordcloud是等所有词一起出现
  3. echarts中数量和颜色的关系需要自己定义;vuewordcloud直接传color就好
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

vue中使用echarts词云 的相关文章

随机推荐

  • Unity Shader入门精要第3 章 Unity Shader 基础

    Unity系列文章目录 文章目录 Unity系列文章目录 前言 一 Unity Shader 概述 二 使用步骤 1 3 1 2 Unity 中的材质 2 Unity 中的Shader 3 Unity Shader 的基础 ShaderLa
  • python 时间加8小时后的时间

    eta temp one arrival encode utf 8 fd datetime datetime strptime eta temp Y m dT H M SZ 加8后的时间eta fd datetime timedelta h
  • sudo提权漏洞cve-2023-22809

    1 影响版本 Sudo 1 8 0 1 9 12p1均受影响 2 sudo V查看当前sudo版本 3 exp usr bin env bash Exploit Title sudo 1 8 0 1 9 12p1 Privilege Esc
  • Python网络爬虫:爬取CSDN热搜数据 并保存到本地文件中

    hello 大家好 我是wangzirui32 今天我们来学习如何爬取CSDN热搜数据 并保存到Excel表格中 开始学习吧 学习目录 1 数据包抓取 2 编写代码 1 数据包抓取 打开CSDN首页 再打开检查 或为审查元素 各大浏览器不同
  • STDOUT_FILENO stdout

    STDOUT FILENO表示标准输出 STDERR FILENO表示标准出错 使用时需要加头文件
  • ViewPager(一屏多页、无限滑动、自动切换)

    一 简介 前段时间在腾讯视频中看到一个效果 是一个广告轮播 然后一屏还显示了多页 看着这个效果看着还不错 就自己实现了下 国际惯例先上效果图 如下 虽然界面比较简陋 但是功能是全的 分割线 二 原理 实现如上效果需要两个功能 一屏多页 无限
  • Spark编程基础期末复习

    选择题 1 spark 的四大组件下面哪个不是 D A Spark Streaming B Mlib C Graphx D Spark R 2 下面哪个端口不是 spark 自带服务的端口 C A 8080 B 4040 C 8090 D
  • 30是什么意思_农村俗语“30要想,40要戒,50要数,60要放”什么意思?有道理吗...

    请您在阅读本文前点击右上方的 关注 以后您就可以每天免费收到 农夫也疯狂 分享的关于农村大小事 文 农夫也疯狂 中国的文化博大精深 而农村文化一样影响深远 其也是传统文化中的重要部分 在过去绝大多数的农村人都没有念过什么书 但是却能明白很多
  • 【大数据存储技术】实验2:MongoDB数据库的部署和操作

    目录 1 实现MongoDB单实例的部署 1 1 安装MongoDB Ubuntu版本 22 04 LTS 1 1 1 查看Ubuntu版本 1 1 2 使用Ubuntu命令安装 1 2 启动MongoDB 验证状态 1 3 测试Mongo
  • 五. SpringCloud Alibaba Sentinel 自定义降级

    目录 一 简单解释 二 SentinelResource 注解详解 三 SentinelResource 设置异常降级方法 三 SentinelResource 降级方法与业务接口的解耦 一 简单解释 在前面配置限流 熔断降级时 可以针对u
  • linux网络编程(一)

    1 linux的网络模型 linux使用的网络模型是TCP UP四层网络模型 主要由应用程序 传输层 网络层 网络接口层组成 与OSI七层模型不同 但是又相互对应 它们之间关系如下图 OSI模型的应用层 表示层 会话层对应着TCP IP模型
  • Android中Application的onCreate多次调用问题

    http blog csdn net peidonghui article details 46043943 版权声明 本文为博主原创文章 未经博主允许不得转载 1 问题描述 一个Android应用需要为一个service单独开一个进程以完
  • 设置cin不忽略空格

    cin读取字符是会忽略空格和换行的 可以用noskipws设为不跳过空格或者换行 char step cin gt gt noskipws gt gt step
  • ERROR 1045(28000):Access denied for user ‘root‘@‘local‘(using password:yes)问题解决

    首先出现这种问题一般是密码错误 但是也有可能 输入密码正确也显示报错的情况 笔者刚在Xshell中安装mysql 第二天输入了正确密码 我保证我输入的是正确的 出现此类报错 解决方法一般是重置密码 1 跳过MYSQL的密码验证过程 在Xsh
  • 蓝桥杯单片机学习过程记录(二十八)第五届国赛串口通信相关代码补充

    蓝桥杯单片机学习过程记录 二十八 第五届国赛串口通信相关代码补充 UART串口通信 第五届国赛uart串口内容相关补充 设置数组存储输入输入字符 并与设定的密码相判断 include
  • springboot2+shiro+redis限制同一账号同时在线人数

    springboot2 shiro redis限制同一账号同时在线人数 我们在写系统的时候 需要注意账号安全问题 最好的处理方法就是同一个账号只能在一个地方登录 原理 大概的原理就是每次登录的时候将登录的sessionId存入缓存 然后登录
  • c# 二维码生成

    dll下载 https pan baidu com s 1MDQalDEoV4iDXRYsEzDXtw 生成图片 此dll适合Framework版本较多 Imports ThoughtWorks QRCode Codec Dim qrCod
  • 技术人修炼之道阅读笔记(七)系统性思维方法

    在工作中有两种高手 一种是他们有成体系的逻辑 术法清晰 另一种是他们悟性高 对大多数人来说 前一种可借鉴性更高 但前提是要足够努力和坚持 塑造系统性思维并进行验证和升级 一 什么是系统性思维 系统性思维 是把物质系统当作一个整体进行思考的思
  • matlab 中num2str函数的使用

    参考 https zhidao baidu com question 431413920 html 问题描述 先前使用num2str函数只是使用了该函数最常用的功能 将数字转换为字符串 但其实该函数还有额外格式上的功能 今天使用图像批处理的
  • vue中使用echarts词云

    1 安装 cnpm install echarts wordcloud 2 创建模板组件 WordCloudChart