DeviceStatsUV.vue 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. <template>
  2. <!-- 图表视图 -->
  3. <div ref="chartRef" style="height: 550px"></div>
  4. </template>
  5. <script setup lang="ts">
  6. import { ref, onMounted, watch, computed, nextTick, onUnmounted } from 'vue';
  7. import * as echarts from 'echarts';
  8. const props = defineProps({
  9. deviceStats: {
  10. type: Object,
  11. required: true
  12. }
  13. });
  14. const chartRef = ref(null);
  15. const loading = ref(false);
  16. const viewMode = ref('chart');
  17. let chart = null;
  18. const initChart = () => {
  19. if (!chartRef.value) return;
  20. chart = echarts.init(chartRef.value);
  21. updateChart();
  22. };
  23. // 格式化整数
  24. const formatNumber = (num: string | number) => {
  25. return Number(num).toLocaleString();
  26. };
  27. const updateChart = () => {
  28. if (!chart || !props.deviceStats) return;
  29. // 获取前数据并计算总数
  30. const data = props.deviceStats
  31. .sort((a, b) => Number(b.totalUsers) - Number(a.totalUsers));
  32. const total = data.reduce((sum, item) => sum + Number(item.totalUsers), 0);
  33. console.log(total, data)
  34. const option = {
  35. tooltip: {
  36. trigger: 'item',
  37. formatter: (params) => {
  38. const percent = ((params.value / total) * 100).toFixed(2);
  39. return `${params.name}<br/>访客数: ${formatNumber(params.value)}<br/>占比: ${percent}%`;
  40. }
  41. },
  42. legend: {
  43. orient: 'vertical',
  44. right: 10,
  45. top: 'center',
  46. type: 'scroll'
  47. },
  48. series: [
  49. {
  50. name: '用户数',
  51. type: 'pie',
  52. radius: ['40%', '70%'],
  53. avoidLabelOverlap: true,
  54. itemStyle: {
  55. borderRadius: 10,
  56. borderColor: '#fff',
  57. borderWidth: 2
  58. },
  59. label: {
  60. show: true,
  61. formatter: (params) => {
  62. const percent = ((params.value / total) * 100).toFixed(2);
  63. return `${params.name}\n${percent}%`;
  64. }
  65. },
  66. emphasis: {
  67. label: {
  68. show: true,
  69. fontSize: 14,
  70. fontWeight: 'bold'
  71. }
  72. },
  73. data: data.map(item => ({
  74. name: item.device,
  75. value: item.totalUsers
  76. }))
  77. }
  78. ]
  79. };
  80. chart.setOption(option);
  81. };
  82. // 监听数据变化
  83. watch(
  84. () => props.deviceStats,
  85. () => {
  86. updateChart();
  87. },
  88. { deep: true }
  89. );
  90. // 监听视图模式变化
  91. watch(viewMode, (newValue) => {
  92. if (newValue === 'chart') {
  93. // 在下一个 tick 后初始化图表,确保 DOM 已更新
  94. nextTick(() => {
  95. initChart();
  96. });
  97. }
  98. });
  99. onMounted(() => {
  100. if (viewMode.value === 'chart') {
  101. initChart();
  102. }
  103. });
  104. // 监听窗口大小变化
  105. window.addEventListener('resize', () => {
  106. if (viewMode.value === 'chart') {
  107. chart?.resize();
  108. }
  109. });
  110. // 组件卸载时清理
  111. onUnmounted(() => {
  112. chart?.dispose();
  113. window.removeEventListener('resize', () => {
  114. chart?.resize();
  115. });
  116. });
  117. </script>
  118. <style scoped>
  119. .ant-card {
  120. margin-bottom: 24px;
  121. }
  122. :deep(.ant-table-pagination) {
  123. margin: 16px 0;
  124. }
  125. </style>