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

点击图例报错
解决方案
参考官方:你可以选择退出默认的深度响应式/只读转换模式,并原始的,未被代理的对象嵌入状态图中。它们可以根据情况灵活运用:有些值不应该是响应式的,例如复杂的第三方类实例或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)
      }
  }
}
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
Vue中添加OpenLayer图例时,可能会遇到“Cannot read properties of undefined (reading 'getCanvas')”错误。这通常是由于在创建地图之前尝试访问OpenLayer的方法或属性而导致的。为了解决这个问题,可以尝试以下几个步骤: 1. 确保在创建地图之前加载OpenLayer库,并且已经正确引入。 2. 确保在创建地图之前,已经正确初始化Vue应用程序,并且已经正确绑定全局属性。 3. 确保在创建地图之前,已经正确创建地图对象,并且已经正确设置图层和控件。 4. 确保在添加图例之前,已经正确创建了图例对象,并且已经正确设置图例的位置和样式。 5. 确保在添加图例之前,已经正确将图例对象添加到地图对象中。 以下是一个示例代码,演示如何在Vue中添加OpenLayer图例: ```javascript <template> <div id="map"></div> </template> <script> import { createApp } from 'vue' import App from './App.vue' import Map from 'ol/Map' import View from 'ol/View' import TileLayer from 'ol/layer/Tile' import OSM from 'ol/source/OSM' import Legend from 'ol-ext/control/Legend' export default { mounted() { const app = createApp(App) const map = new Map({ target: 'map', layers: [ new TileLayer({ source: new OSM() }) ], view: new View({ center: [0, 0], zoom: 2 }) }) const legend = new Legend({ title: 'Legend', style: 'border: 1px solid black; background-color: white; padding: 5px;', margin: 10, maxSize: 20, collapseLabel: '-', expandLabel: '+' }) map.addControl(legend) app.config.globalProperties.$http = axios app.mount('#app') } } </script> <style> #map { width: 100%; height: 100%; } </style> ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值