潘志宝
23 小时以前 221918bba28d2384d03c596a68256d7832e4a0e0
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
<script lang="ts" setup>
import type { EChartsOption } from 'echarts'
import echarts from '@/plugins/echarts'
import { debounce } from 'lodash-es'
import 'echarts-wordcloud'
import { propTypes } from '@/utils/propTypes'
import { PropType } from 'vue'
import { useAppStore } from '@/store/modules/app'
import { isString } from '@/utils/is'
import { useDesign } from '@/hooks/web/useDesign'
 
import 'echarts/lib/component/markPoint'
import 'echarts/lib/component/markLine'
import 'echarts/lib/component/markArea'
 
defineOptions({ name: 'EChart' })
 
const { getPrefixCls, variables } = useDesign()
 
const prefixCls = getPrefixCls('echart')
 
const appStore = useAppStore()
 
const props = defineProps({
  options: {
    type: Object as PropType<EChartsOption>,
    required: true
  },
  width: propTypes.oneOfType([Number, String]).def(''),
  height: propTypes.oneOfType([Number, String]).def('500px')
})
 
const isDark = computed(() => appStore.getIsDark)
 
const theme = computed(() => {
  const echartTheme: boolean | string = unref(isDark) ? true : 'auto'
 
  return echartTheme
})
 
const options = computed(() => {
  return Object.assign(props.options, {
    darkMode: unref(theme)
  })
})
 
const elRef = ref<ElRef>()
 
let echartRef: Nullable<echarts.ECharts> = null
 
const contentEl = ref<Element>()
 
const styles = computed(() => {
  const width = isString(props.width) ? props.width : `${props.width}px`
  const height = isString(props.height) ? props.height : `${props.height}px`
 
  return {
    width,
    height
  }
})
 
const initChart = () => {
  if (unref(elRef) && props.options) {
    echartRef = echarts.init(unref(elRef) as HTMLElement)
    echartRef?.setOption(unref(options))
  }
}
 
watch(
  () => options.value,
  (options) => {
    if (echartRef) {
      echartRef?.setOption(options)
    }
  },
  {
    deep: true
  }
)
 
const resizeHandler = debounce(() => {
  if (echartRef) {
    echartRef.resize()
  }
}, 100)
 
const contentResizeHandler = async (e: TransitionEvent) => {
  if (e.propertyName === 'width') {
    resizeHandler()
  }
}
 
onMounted(() => {
  initChart()
 
  window.addEventListener('resize', resizeHandler)
 
  contentEl.value = document.getElementsByClassName(`${variables.namespace}-layout-content`)[0]
  unref(contentEl) &&
  (unref(contentEl) as Element).addEventListener('transitionend', contentResizeHandler)
})
 
onBeforeUnmount(() => {
  window.removeEventListener('resize', resizeHandler)
  unref(contentEl) &&
  (unref(contentEl) as Element).removeEventListener('transitionend', contentResizeHandler)
})
 
onActivated(() => {
  if (echartRef) {
    echartRef.resize()
  }
})
</script>
 
<template>
  <div ref="elRef" :class="[$attrs.class, prefixCls]" :style="styles"></div>
</template>