vue中使用keep-alive緩存后處於disactivated狀態的echarts表格自適應功能(resize)失效的解決辦法
在vue開發中的時候,使用keep-alive監聽的頁面,echart或者地圖相關的模塊,測試提出了一個bug,頁面偶現空白。
經過排查發現重現步驟如下:
離開當前頁面以后,再其他tab頁面切換了窗口大小,再回到原頁面,發現echart不顯示了,再重新改變一下窗口大小,就又恢復正常
原因:
之前為了實現echarts的自適應,添加了一個監聽器監測window窗口是否發生變化,如果變化了就執行echarts自帶的resize方法來改變一下echarts的大小,並且將這個方法習慣性地寫在了生成echarts的方法中。但是,我們使用了keep-alive,進行緩存后情況就有點不一樣了。我們繪制Echarts的方法是寫在mounted生命周期中的,而切換頁面並會不會進入destroyed生命周期,而是進入disactivated生命周期,這使得mounted中的函數還是激活狀態,我們之前添加的監聽器仍然在工作,但是但是但是,之前緩存頁面的dom沒了,打印出來的高度為0,我們的eventListener監聽到了別人家的dom變了,高度為0,所以圖表也為0。
然后,當切換回已緩存的頁面時,bug就來了。echarts消失了。
然后你再改變一下寬口大小,echarts又回來了。。。
這是因為你之前你改變dom的時候,echarts重畫失敗了,可是你切回來的時候,dom雖然相對於你切出去之前可能變了,但是對於監聽器來說,他只關注於當前dom大小是否改變了。所以剛切回來的時候echarts的resize方法不會被激活。而只要稍稍改變一下dom的大小,就可以激活resize方法。
解決方案如下:
方法一:引入element-resize-detector插件進行處理
<template> <div ref="chart"></div> </template> <script> import elementResizeDetector from "element-resize-detector"; import * as echarts from "echarts/core"; import { LineChart, BarChart, PieChart, GraphChart, LinesChart} from "echarts/charts"; import { GridComponent, SingleAxisComponent, TooltipComponent, AxisPointerComponent, TitleComponent, LegendComponent } from "echarts/components"; import { CanvasRenderer } from "echarts/renderers"; echarts.use([ TitleComponent, TooltipComponent, LegendComponent, SingleAxisComponent, GridComponent, AxisPointerComponent, BarChart, LineChart, PieChart, GraphChart, LinesChart, CanvasRenderer ]); export default { name: "viChart", props: { options: { type: Object, required: true }, isClear: { type: Boolean, default: true } }, data() { return { myChart: null, erd: null, }; }, beforeDestroy() { this.dispose(); }, methods: { watchOptions() { this.$watch("options", this.setOption, { deep: true }); }, init() { this.myChart = echarts.init(this.$refs.chart); this.setOption(); }, resize() { if (!this.myChart) return; this.myChart.resize(); }, setOption() { if (!this.myChart || Object.keys(this.options).length === 0) return; if (this.isClear) { this.myChart.clear(); } this.myChart.setOption(this.options); // this.resize(); }, dispose() { if (!this.myChart) return; this.myChart.dispose(); } }, mounted() { this.init(); this.watchOptions(); this.erd = elementResizeDetector(); this.erd.listenTo(this.$refs.chart, this.resize); }, beforeDestroy() { this.erd.removeListener(this.$refs.chart); } }; </script>
方法二:使用指令方法,監聽DOM元素寬度變化。該方法是定時循環調用時間監聽,判斷寬高變化,比較消耗資源,不建議。
<template>
<div ref="chart" v-resize="resize"></div>
</template>
<script>
import * as echarts from "echarts/core";
import { LineChart, BarChart, PieChart, GraphChart, LinesChart} from "echarts/charts";
import { GridComponent, SingleAxisComponent, TooltipComponent, AxisPointerComponent, TitleComponent, LegendComponent } from "echarts/components";
import { CanvasRenderer } from "echarts/renderers";
echarts.use([
TitleComponent,
TooltipComponent,
LegendComponent,
SingleAxisComponent,
GridComponent,
AxisPointerComponent,
BarChart,
LineChart,
PieChart,
GraphChart,
LinesChart,
CanvasRenderer
]);
export default {
name: "viChart",
props: {
options: {
type: Object,
required: true
},
isClear: {
type: Boolean,
default: true
}
},
data() {
return {
myChart: null
};
},
mounted() {
this.init();
this.watchOptions();
},
beforeDestroy() {
this.unbind();
this.dispose();
},
methods: {
watchOptions() {
this.$watch("options", this.setOption, { deep: true });
},
init() {
this.myChart = echarts.init(this.$refs.chart);
this.setOption();
},
unbind() {
window.removeEventListener("resize", this.resize);
},
resize() {
if (!this.myChart) return;
this.myChart.resize();
},
setOption() {
if (!this.myChart || Object.keys(this.options).length === 0) return;
if (this.isClear) {
this.myChart.clear();
}
this.myChart.setOption(this.options);
},
dispose() {
if (!this.myChart) return;
this.myChart.dispose();
}
},
directives: { // 使用局部注冊指令的方式。主要處理keep-alive情況下,切換到其他tab情況下,再切回原頁面,echart不見了
resize: { // 指令的名稱
bind(el, binding) { // el為綁定的元素,binding為綁定給指令的對象
let width = ""; let height = "";
function isReize() {
const style = document.defaultView.getComputedStyle(el); // 獲取對象的css樣式,返回的是一個CSS樣式對象
if (width !== style.width || height !== style.height) {
binding.value(); // 關鍵
}
width = style.width;
height = style.height;
}
el.__vueSetInterval__ = setInterval(isReize, 300);
},
unbind(el) {
clearInterval(el.__vueSetInterval__);
}
}
},
};
</script>
參考文檔:
https://blog.csdn.net/kiwon1993/article/details/102638808
https://blog.csdn.net/csl125/article/details/115075643
