index.ts 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. import type { EChartsOption } from 'echarts';
  2. import type { Ref } from 'vue';
  3. import { useTimeoutFn } from '/@/hooks/core/useTimeout';
  4. import { tryOnUnmounted } from '@vueuse/core';
  5. import { unref, nextTick, watch, computed, ref } from 'vue';
  6. import { useDebounceFn } from '@vueuse/core';
  7. import { useEventListener } from '/@/hooks/event/useEventListener';
  8. import { useBreakpoint } from '/@/hooks/event/useBreakpoint';
  9. import echarts from './echarts';
  10. import { useRootSetting } from '/@/hooks/setting/useRootSetting';
  11. export function useECharts(
  12. elRef: Ref<HTMLDivElement>,
  13. theme: 'light' | 'dark' | 'default' = 'light'
  14. ) {
  15. const { getDarkMode } = useRootSetting();
  16. let chartInstance: echarts.ECharts | null = null;
  17. let resizeFn: Fn = resize;
  18. const cacheOptions = ref<EChartsOption>({});
  19. let removeResizeFn: Fn = () => {};
  20. resizeFn = useDebounceFn(resize, 200);
  21. const getOptions = computed(
  22. (): EChartsOption => {
  23. if (getDarkMode.value !== 'dark') {
  24. return cacheOptions.value;
  25. }
  26. return {
  27. backgroundColor: 'transparent',
  28. ...cacheOptions.value,
  29. };
  30. }
  31. );
  32. function initCharts(t = theme) {
  33. const el = unref(elRef);
  34. if (!el || !unref(el)) {
  35. return;
  36. }
  37. chartInstance = echarts.init(el, t);
  38. const { removeEvent } = useEventListener({
  39. el: window,
  40. name: 'resize',
  41. listener: resizeFn,
  42. });
  43. removeResizeFn = removeEvent;
  44. const { widthRef, screenEnum } = useBreakpoint();
  45. if (unref(widthRef) <= screenEnum.MD || el.offsetHeight === 0) {
  46. useTimeoutFn(() => {
  47. resizeFn();
  48. }, 30);
  49. }
  50. }
  51. function setOptions(options: EChartsOption, clear = true) {
  52. cacheOptions.value = options;
  53. if (unref(elRef)?.offsetHeight === 0) {
  54. useTimeoutFn(() => {
  55. setOptions(unref(getOptions));
  56. }, 30);
  57. return;
  58. }
  59. nextTick(() => {
  60. useTimeoutFn(() => {
  61. if (!chartInstance) {
  62. initCharts(getDarkMode.value);
  63. if (!chartInstance) return;
  64. }
  65. clear && chartInstance?.clear();
  66. chartInstance?.setOption(unref(getOptions));
  67. }, 30);
  68. });
  69. }
  70. function resize() {
  71. chartInstance?.resize();
  72. }
  73. watch(
  74. () => getDarkMode.value,
  75. (theme) => {
  76. if (chartInstance) {
  77. chartInstance.dispose();
  78. initCharts(theme);
  79. setOptions(cacheOptions.value);
  80. }
  81. }
  82. );
  83. tryOnUnmounted(() => {
  84. if (!chartInstance) return;
  85. removeResizeFn();
  86. chartInstance.dispose();
  87. chartInstance = null;
  88. });
  89. return {
  90. setOptions,
  91. resize,
  92. echarts,
  93. };
  94. }