【vue3 + echarts】legend 点击图例报错 Cannot read properties of undefined (reading ‘type‘)

【vue3 + echarts】legend 点击图例报错 Cannot read properties of undefined (reading ‘type‘)_第1张图片
解决方案
参考官方:你可以选择退出默认的深度响应式/只读转换模式,并原始的,未被代理的对象嵌入状态图中。它们可以根据情况灵活运用:有些值不应该是响应式的,例如复杂的第三方类实例或vue组件对象。当渲染具有不可变数据源的大列表时,跳过proxy转换可以提高性能。
所以在实列化echart时,将期转换为非响应式即可。

import { markRaw } from "vue";
//这里的el 是我用的ref 
 const el = state["chartsWrapper"];
  state.instance = markRaw(ECharts.init(el));
 // 标签用id 
 const idEl=document.getElementById('id名')
  state.instance = markRaw(ECharts.init(idEl));
  

以上是遇到vue3+echart遇到的坑 Cannot read properties of undefined (reading ‘type’)

echart.vue界面代码格式

<template>
  <div class="charts-wrapper">
    <div v-show="visible" class="charts-content" ref="chartsWrapper" />
    <div v-show="!visible" class="charts-content image-wrapper">
      <div class="no-list-img" />
      <div>暂无数据</div>
    </div>
  </div>
</template>

<script>
import * as ECharts from "echarts";
import { markRaw } from "vue";
export default defineComponent({
  name:'echarts',
  props: {
    visible: {
      type: Boolean,
      default: true,
    },
    option: {
      type: Object,
      default() {
        return {};
      },
    },
  },
  emits: ["click"],
  setup(props, { emit }) {
    const state = reactive({
      instance: null,
      finalOption: {
        // 个性化配色列表
        color: [
          "#54a1d5",
          "#5fc2e5",
          "#ade4bb",
          "#fadc71",
          "#f3a385",
          "#ec7b94",
          "#e0bef0",
          "#807ae3",
          /*'#2d9ef6',
            '#4ddeef',
            '#53d55c',
            '#fae94a',
            '#f36646',
            '#ec315c',
            '#e48ce4',
            '#5f65e3'*/
        ],
        tooltip: { trigger: "axis", axisPointer: { type: "shadow" } },
      },
      timeId: null,
      chartsWrapper: null,
    });

    onMounted(() => {
      console.log(ECharts, "ECharts");
      initialize();
    });
    onUnmounted(() => {
      console.log(23423);
      state.timeId && clearTimeout(state.timeId);
      state.instance.clear();
      state.timeId = null;
      state.instance = null;
      window.removeEventListener("resize", resize);
    });
    watch(props.option.series, (newVal, olVal) => {
      paint();
    });
    const initialize = async () => {
      let _this = this;
      const el = state["chartsWrapper"];
      // 先渲染页面,后渲染图表,防止初始化图表超出容器
      // await this.$nextTick();
      // state.instance = markRaw(ECharts.init(el));
      state.instance = ECharts.init(el);

      paint(state.finalOption);
      state.instance.on("click", function (params) {
        emit("click", params);
      });
      // 新增窗口监听
      window.addEventListener("resize", resize);
    };
    // 窗口大小改变时重绘
    const resize = () => {
      // 防抖延时
      if (state.timeId) {
        clearTimeout(state.timeId);
        state.timeId = null;
      }
      state.timeId = setTimeout(() => {
        state.instance.resize();
        state.timeId = null;
      }, 300);
    };
    // 绘制图表
    const paint = async () => {
      Object.assign(state.finalOption, props.option);
      console.log(state.finalOption,'state.finalOption');
      state.instance.setOption(state.finalOption);
    };
    /**
     * 提供给外部调用,用来调用echarts实例的方法
     * @param name echarts实例方法名称
     * @param args 不定长参数列表
     */
    const instanceMethod = (name, ...args) => {
      state.instance[name](...args);
    };
    return {
      ...toRefs(state),
      initialize,
      resize,
      paint,
      instanceMethod
    };
  },
});
</script>

<style scoped lang="scss">
.charts-wrapper {
  width: 100%;
  height: 100%;
  .charts-content {
    width: 100%;
    height: 100%;
  }

  .image-wrapper {
    color: rgba(0, 0, 0, 0.45);
    display: flex;
    flex-direction: column;
    align-items: center;
    justify-content: center;
  }

  .no-list-img {
    margin-bottom: 10px;
    width: 63px;
    height: 40px;
    background-repeat: no-repeat;
    background-image: url("");
  }
}
</style>

引用echarts界面

    <div style="width: 500px; height: 300px">
      <echarts :visible="true" :option="option"></echarts>
    </div>

<script >
export default {
  components: {
    Echarts: defineAsyncComponent(() => import("@/components/echarts")),
  },
    setup() {
        const state = reactive({
      option: {
        title: {
          text: "World Population",
        },
        tooltip: {
          trigger: "axis",
          axisPointer: {
            type: "shadow",
          },
        },
        legend: {
          type:'plain'
        },
        grid: {
          left: "3%",
          right: "4%",
          bottom: "3%",
          containLabel: true,
        },
        xAxis: {
          type: "value",
          boundaryGap: [0, 0.01],
        },
        yAxis: {
          type: "category",
          data: ["Brazil", "Indonesia", "USA", "India", "China", "World"],
        },
        series: [
          {
            name: "2011",
            type: "bar",
            data: [18203, 23489, 29034, 104970, 131744, 630230],
          },
          {
            name: "2012",
            type: "bar",
            data: [19325, 23438, 31000, 121594, 134141, 681807],
          },
        ],
      },
    });
      return {
      ...toRefs(state)
      }
  }
}

你可能感兴趣的:(echarts,javascript,vue.js)