echarts 立体柱状图 3D地球图 立体饼图

334 阅读8分钟

3D地球图示例

1718778234177.jpg

<template>

  <div ref="chartCom" id="chartCom" class="earthmap"></div>

</template>

<script setup>
import * as echarts from 'echarts';
import 'echarts-gl';
import { nextTick, onMounted, ref, watch, computed } from 'vue';
import { useStore } from 'vuex'
const store = useStore()
const emit = defineEmits(['loadingTrue', 'loadingFalse'])
import imgUrl from '../image/map2.jpg'
import imgUrl1 from '../image/mapBg.png'
import iconUrl from '../image/yellow.png'
import iconUrl1 from '../image/red.png'

import { ElMessageBox, ElMessage } from 'element-plus';
import { geocoordmap } from '../utils/nameMap.js'
import {
  selectAtlas,//地图数据
} from '@/web/apis/Analysisplatform/index.js'
const Iscollapse = computed(() => {
  return store.state.layout.collapse
})

// console.log(Iscollapse, "=========")
const chartCom = ref()
let myChart, chartDom, option;
const initChart = (cityData) => {

  let geocoordmapObj = geocoordmap()
  const cities = []
  for (let key in geocoordmapObj) {

    cityData.forEach(item => {
      if (key === item.countryName) {
        cities.push([item.countryName, geocoordmapObj[key]])
      }
    })
  }
  chartDom = document.getElementById('chartCom');
  myChart = echarts.init(chartDom);
  option = {
    backgroundColor: 'transparent',
    globe: {
      // top: '-20%',
      // globeRadius: 'auto',
      baseTexture: imgUrl,
      // heightTexture: 'https://cdn.jsdelivr.net/gh/apache/echarts-website@asf-site/examples/data-gl/asset/bathymetry_bw_composite_4k.jpg',
      displacementScale: 0.1,

      shading: 'color',
      atmosphere: {
        show: true
      },
      // // environment: imgUrl1,
      // realisticMaterial: {
      //   roughness: 1,
      //   metalness: 1
      // },
      // postEffect: {
      //   enable: true,
      //   SSAO: {
      //     enable: true,
      //     radius: 2
      //   }
      // },
      // light: {
      //   ambient: {
      //     intensity: 0.5
      //   },
      //   main: {
      //     // 主光源的颜色
      //     color: '#fff', // 光照颜色
      //     intensity: 0.5, // 光照强度
      //     shadow: true, // 是否显示阴影
      //     alpha: 0, // 主光源绕 x 轴,即上下旋转的角度
      //     beta: 0 //主光源绕 y 轴,即左右旋转的角度。
      //   },
      //   // 全局的环境光设置。
      //   ambient: {
      //     // /环境光的强度
      //     intensity: 0.5
      //   },

      // },
      viewControl: {
        autoRotate: false,
        targetCoord: [106.46, 20.92], // 定位到北京
        maxDistance: 166,
        minDistance: 166
        // maxDistance: 200,
        // minDistance: 200

      },

    },

    series: []
  };

  // cities = [
  //   ['缅甸', [96.077119, 19.763943]],
  //   ['中国', [116.407394, 39.904211]],
  //   ["泰国", [100.451117, 13.724061]],
  //   ["老挝", [102.632451, 17.975794]],
  //   ["柬埔寨", [104.891769, 11.545102]],
  //   ["", [67.97622, 50.757498],]
  // ];
  cities.push(['中国', [116.407394, 39.904211]])
  cities.forEach((item) => {
    option.series.push({
      name: item[0],
      type: 'scatter3D',
      coordinateSystem: 'globe',
      // symbol: 'image://',
      symbolSize: 0,
      label: {
        show: true,
        position: "top",
        distance: -20,
        // formatter: " ",
        formatter(params) {
          return " ";
        },

        textStyle: {
          color: "#fff",
          fongSize: 20,
          padding: [20, 15],
          backgroundColor: {
            image: item[0] === '中国' ? iconUrl1 : iconUrl,
          },
        },
      },

      encode: {
        value: 5,
      },
      emphasis: {
        label: {
          show: true,
          textStyle: {
            backgroundColor: {
              image: item[0] === '中国' ? iconUrl1 : iconUrl,
            },
          },
        },
      },

      data: [{
        name: item[0],
        value: [item[1][0], item[1][1], 0]
      }]
    });


    option.series.push({
      name: item[0],
      type: 'scatter3D',
      coordinateSystem: 'globe',
      // symbol: 'image://',
      symbolSize: 0,
      label: {
        show: true,
        position: "right",
        distance: -20,
        formatter: "{b}",
        textStyle: {
          color: '#fff',
          borderWidth: 0,
          // borderColor: '#fff',
          fontFamily: 'sans-serif',
          fontSize: 18,
          fontWeight: 700,

        },


        // textStyle: {
        //   color: "#fff",
        //   fontSize: 16,
        //   fontWeight: "700",
        //   // padding: [20, 0],
        //   backgroundColor: {
        //     // image: iconUrl,
        //     color: 'rgba(255,255,255,1)',
        //   },
        // },
      },
      data: [{
        name: item[0],
        value: [item[1][0], item[1][1], 0]
      }]
    });



    option.series.push({
      name: item[0],
      type: 'lines3D',
      coordinateSystem: 'globe',
      // symbol: 'image://',
      effect: {
        show: true,
        trailWidth: 10
      },
      blendMode: "lighter",
      lineStyle: {
        width: 3,
        // color: 'red',
        opacity: 1
      },



      data: [
        [[item[1][0], item[1][1], 0], [116.407394, 39.904211]]
      ]

    });



  });
  option && myChart.setOption(option);
  window.addEventListener("resize", () => {
    myChart.resize()
  });
}


const getList = () => {
  emit('loadingTrue')
  selectAtlas().then(res => {
    emit('loadingFalse')
    if (res.code === 10001) {
      nextTick(() => {
        initChart(res.data)
      })
    } else {
      ElMessage({
        type: 'error',
        message: res.msg
      });
    }
  })
}

watch(() => Iscollapse.value, (n, o) => {
  nextTick(() => {
    myChart.resize()
  })
})
onMounted(() => {
  getList()
  // nextTick(() => {
  //   initChart()
  // })

})
</script>

<style lang="less" scoped>
.earthmap {
  height: 100%;
  width: 100%;
  // border: 1px solid red;
  // background: url("../image/mapBg.png") no-repeat center center;
  // background-size: contain;
}
</style>

立体柱状图示例

1718778256693.jpg

<template>
  <div id="jjtjqk" class="pieChart"></div>
</template>

<script setup>
import * as echarts from 'echarts';
import { nextTick, onMounted, watch, computed } from 'vue';
import { useStore } from 'vuex'
import { ElMessageBox, ElMessage } from 'element-plus';

const store = useStore()
import {
  selectInCountry,//入境国家统计情况
} from '@/web/apis/Analysisplatform/index.js'

const emit = defineEmits(['loadingTrue', 'loadingFalse'])
const Iscollapse = computed(() => {
  return store.state.layout.collapse
})

//获取进境国家统计情况数据
const getselectInCountry = () => {
  var yData = [], xData = []
  emit('loadingTrue')
  selectInCountry().then(res => {
    emit('loadingFalse')
    if (res.code === 10001) {
      res.data.forEach(item => {
        yData.push(item.countNo)
        xData.push(item.countryName)
      });

      nextTick(() => {
        chartInit(yData, xData)
      })
    } else {
      ElMessage({
        type: 'error',
        message: res.msg
      });
    }

  })
}
var chartDom, myChart;
const chartInit = (yData, xData) => {
  chartDom = document.getElementById('jjtjqk');
  myChart = echarts.init(chartDom);
  var option;
  //组织数据
  let setData = function (data, constData, showData) {
    data.filter(function (item) {
      if (item) {
        constData.push(1);
        showData.push(item);
      } else {
        constData.push(0);
        showData.push({
          value: 1,
          itemStyle: {
            normal: {
              borderColor: 'rgba(0,0,0,0)',
              borderWidth: 2,
              color: 'rgba(0,0,0,0)'
            }
          }
        });
      }
    });
  };
  //组织颜色
  let setColor = function (colorArr) {
    let color = {
      type: 'linear',
      x: 0,
      x2: 1,
      y: 0,
      y2: 0,
      /* 此处决定阴暗面 若为横向柱状图则x,y轴调换
            x: 0,
            x2: 0,
            y: 0,
            y2: 1, */
      colorStops: [
        {
          offset: 0,
          color: colorArr[0]
        },
        {
          offset: 0.5,
          color: colorArr[0]
        },
        {
          offset: 0.5,
          color: colorArr[1]
        },
        {
          offset: 1,
          color: colorArr[1]
        }
      ]
    };
    return color;
  };

  let vehicle = yData;
  var barWidth = 30;
  var constData1 = [];
  var showData1 = [];
  setData(vehicle, constData1, showData1);
  var colorArr1 = ['#345A8B', '#387ABD', '#51C0DB'];
  var colorArr2 = ['#51C0DB', '#42D9D6', '#45F5F1'];
  var color1 = setColor(colorArr1);
  var option = {
    tooltip: {
      trigger: 'axis',
      axisPointer: {
        type: 'shadow'
      }
    },
    legend: {
      show: false
    },
    grid: {
      top: '15%',
      bottom: '15%'
    },
    xAxis: {
      type: 'category',
      axisLabel: {
        color: '#FFFFFF'
      },
      axisLine: {
        show: true,
        lineStyle: {
          color: '#1B3F66'
        }
      },
      axisTick: {
        show: false
      },
      data: xData
      // data: ['老挝', '越南', '柬埔寨', '泰国', '缅甸']
    },
    yAxis: {
      type: 'value',
      axisLabel: {
        color: '#FFFFFF'
      },
      axisLine: {
        show: true,
        lineStyle: {
          color: '#1B3F66'
        }
      },
      splitLine: {
        lineStyle: {
          color: '#1B3F66'
        }
      }
    },
    series: [
      {
        z: 1,
        type: 'bar',
        name: '进境国家统计情况',
        barGap: '-100%',
        barWidth: barWidth,
        itemStyle: {
          color: (params) => {
            if ((params.dataIndex + 1) % 2 == 0) {
              return new echarts.graphic.LinearGradient(0, 1, 0, 0, [
                { offset: 0.6, color: '#51C0DB' },
                { offset: 0, color: 'rgba(0, 20, 61, 0)' },
              ])
            } else {
              return new echarts.graphic.LinearGradient(0, 1, 0, 0, [
                { offset: 0.6, color: '#0062B3' },
                { offset: 0, color: 'rgba(0, 20, 61, 0)' },
              ])
            }

          }
        },
        data: vehicle //Y轴上的高度


      },

      // ---------------------------------------------
      {
        z: 2,
        name: '柱子1',
        type: 'pictorialBar',
        symbolPosition: 'start',
        data: vehicle, //此数据对应底部组件
        symbol: 'diamond', //底部组件形状,不写默认为椭圆
        // symbolOffset: [-barWidth / 2, '50%'], //与柱子的偏移角度
        symbolSize: [barWidth, (10 * (barWidth - 4)) / barWidth],
        itemStyle: {
          normal: {
            color: color1 //底面左右颜色(深,浅)
          }
        },
        tooltip: {
          show: false
        }
      },
      {
        z: 2,
        name: '柱子top',
        type: 'pictorialBar',
        symbolPosition: 'end',
        data: vehicle, //此数据对应顶部组件
        symbol: 'diamond',
        symbolOffset: [0, '-50%'],
        symbolSize: [barWidth, (10 * (barWidth - 4)) / barWidth],
        itemStyle: {
          normal: {
            color: (params) => {
              return (params.dataIndex + 1) % 2 == 0 ? colorArr2[2] : colorArr1[1]

            }
          }
        },
        tooltip: {
          show: false
        }
      },
      {
        z: 4,
        type: 'pictorialBar',
        name: '柱子Line',
        symbolPosition: 'center',
        symbolSize: [1, '93%'], //底面[宽,高]
        itemStyle: {
          borderRadius: [0, 0, 0, 0], //柱子四周圆角
          color: (params) => new echarts.graphic.LinearGradient(1, 1, 0, 0, [
            { offset: 0, color: 'rgba(0, 20, 61, 0.5)' },
            { offset: 0.5, color: 'rgba(0, 131, 235, 0.8)' },
          ])
        },
        data: vehicle, //Y轴上的高度
        tooltip: {
          show: false
        }
      },

    ]
  };


  option && myChart.setOption(option);
  window.addEventListener("resize", () => {
    myChart.resize()
  });



}

watch(() => Iscollapse.value, (n, o) => {
  nextTick(() => {
    myChart.resize()
  })
})
onMounted(() => {
  getselectInCountry()
})

// defineExpose({
//   chartInit
// })
</script>

<style lang="less" scoped>
.pieChart {
  height: 100%;
  width: 100%;
  // border: 1px solid rgb(255, 0, 0);
}
</style>

立体饼状图示例

1718778304759.jpg

<template>

  <div class="earthmap">
    <div ref="chartCom3d" id="chartCom3d" class="PieCom"></div>
    <div class="rightText">
      <div class="rightText-item">
        <span class="lable">总数</span>
        <span class="valueNum">{{state.totleValue.countNo}}</span>
        <span class="dw"></span>
      </div>
      <div class="rightText-item" v-for="item,i in state.seriesData" :key="i">
        <span class="lable" :title="item.name">{{item.name}}</span>
        <span class="valueNum" :title="item.value">{{item.value}}</span>
        <span class="dw"></span>
      </div>

    </div>
  </div>

</template>

<script setup>
import * as echarts from 'echarts';
import 'echarts-gl';
import { nextTick, onMounted, ref, watch, computed, reactive } from 'vue';
import { useStore } from 'vuex'
const store = useStore()
import {
  selectInAnimalCount,//入境动物统计情况
} from '@/web/apis/Analysisplatform/index.js'
import { ElMessageBox, ElMessage } from 'element-plus';

const emit = defineEmits(['loadingTrue', 'loadingFalse'])

const Iscollapse = computed(() => {
  return store.state.layout.collapse
})


const fontSize = (res) => {
  const clientWidth =
    window.innerWidth ||
    document.documentElement.clientWidth ||
    document.body.clientWidth;
  if (!clientWidth) return;
  const fontSize = 40 * (clientWidth / 1920);
  return res * fontSize;
}



const chartCom3d = ref()
let myChart, chartDom;
const initChart = (data) => {
  if (data.length == 0) return;
  chartDom = document.getElementById('chartCom3d');
  myChart = echarts.init(chartDom);
  let maxvalue = Math.max.apply(data, data.map(item => item.value))
  let colors = [
    "rgba(16, 155, 229, 0.5)",
    "RGBA(209, 228, 74, 0.5)",
    "RGBA(183, 49, 81,0.5)",
    "RGBA(44, 231,255,0.5)",
  ];

  //写一个方法来计算环形的高度
  const getPie3D = (pieData, internalDiameterRatio) => {
    //internalDiameterRatio:透明的空心占比
    let series = [];
    let sumValue = 0;
    let startValue = 0;
    let endValue = 0;
    let k = 1 - internalDiameterRatio;
    pieData.sort((a, b) => {
      return b.value - a.value
    })
    pieData.forEach((item, i) => {
      sumValue += item.value;
      let seriesItem = {
        name: typeof pieData[i].name === "undefined"
          ? `series${i}`
          : pieData[i].name,
        type: "surface",
        parametric: true,
        wireframe: {
          show: false,
        },
        pieData: item,
        pieStatus: {
          selected: false,
          hovered: false,
          k: k,
        },
        radius: "20%",
        center: ["10%", "20%"],
      };
      if (typeof colors[i] != "undefined") {
        let itemStyle = {};
        typeof colors[i] != "undefined"
          ? (itemStyle.color = colors[i])
          : null;
        seriesItem.itemStyle = itemStyle;

      }
      series.push(seriesItem);
    })

    series.forEach((item, i) => {
      endValue = startValue + item.pieData.value;
      item.pieData.startRatio = startValue / sumValue;
      item.pieData.endRatio = endValue / sumValue;
      item.parametricEquation = getParametricEquation(item.pieData.startRatio, item.pieData.endRatio, false,
        false,
        k,
        item.pieData.value);
      startValue = endValue;

    })
    let boxHeight = getHeight3D(series, 15); //通过传参设定3d饼/环的高度,26代表26px
    // 准备待返回的配置项,把准备好的 legendData、series 传入。
    let option = {
      labelLine: {
        show: true,
        lineStyle: {
          color: "#7BC0CB",
        },
      },
      legend: {
        show: false,
      },
      label: {
        show: false,
        formatter: "",
      },
      tooltip: {
        show: true,
        formatter: (params) => {
          if (
            params.seriesName !== "mouseoutSeries" &&
            params.seriesName !== "pie2d"
          ) {
            let bfb = (
              (option.series[params.seriesIndex].pieData.endRatio -
                option.series[params.seriesIndex].pieData.startRatio) *
              100
            ).toFixed(2);
            return (
              `${params.seriesName}<br/>` +
              `<span style="display:inline-block;margin-right:5px;border-radius:10px;width:10px;height:10px;background-color:${params.color};"></span>` +
              `${bfb}%` + `<span style="display:inline-block;margin-left:20px;">数量:${option.series[params.seriesIndex].pieData.value}</span> `

            );
          }
        },
        position: function (point) { // 自定义 tooltip 的显示位置
          return {
            left: point[0], // 水平方向的偏移
            top: point[1] // 垂直方向的偏移
          };
        }
      },
      xAxis3D: {
        min: -1,
        max: 1,
      },
      yAxis3D: {
        min: -1,
        max: 1,
      },
      zAxis3D: {
        min: -1,
        max: 1,
      },
      grid3D: {
        show: false,
        boxHeight: boxHeight,

        // boxHeight: maxvalue > 999 ? (maxvalue > 9999 ? maxvalue * 0.0000001 : maxvalue * 0.00001) : (maxvalue < 100 ? (maxvalue < 10 ? maxvalue * -2 : maxvalue * 0.05) : maxvalue * 0.001), //圆环的高度  根据数据来算
        // left: -fontSize(1.8),
        // top: -fontSize(1.12), //3d饼图的位置
        viewControl: {
          //3d效果可以放大、旋转等,请自己去查看官方配置
          alpha: 25, //角度
          distance: 220, //调整视角到主体的距离,类似调整zoom
          rotateSensitivity: 0, //设置为0无法旋转
          zoomSensitivity: 0, //设置为0无法缩放
          panSensitivity: 0, //设置为0无法平移
          autoRotate: false, //自动旋转
        },
      },
      series: series,



    }
    return option;


  }

  const getHeight3D = (series, height) => {
    series.sort((a, b) => {
      //给series的值从大到小排序
      return (b.pieData.value - a.pieData.value);
    })
    //动态计算 图形高度 series[0].pieData.value是最大的值
    return (height * 25 / series[0].pieData.value) > 10 ? 20 : (height * 25 / series[0].pieData.value)

  }

  const getParametricEquation = (
    startRatio,
    endRatio,
    isSelected,
    isHovered,
    k,
    h
  ) => {
    // 计算
    let midRatio = (startRatio + endRatio) / 2;
    let startRadian = startRatio * Math.PI * 2;
    let endRadian = endRatio * Math.PI * 2;
    let midRadian = midRatio * Math.PI * 2;
    // 如果只有一个扇形,则不实现选中效果。
    if (startRatio === 0 && endRatio === 1) {
      isSelected = false;
    }
    // 通过扇形内径/外径的值,换算出辅助参数 k(默认值 1/3)
    k = typeof k !== "undefined" ? k : 1 / 3;
    // 计算选中效果分别在 x 轴、y 轴方向上的位移(未选中,则位移均为 0)
    let offsetX = isSelected ? Math.cos(midRadian) * 0.1 : 0;
    let offsetY = isSelected ? Math.sin(midRadian) * 0.1 : 0;
    // 计算高亮效果的放大比例(未高亮,则比例为 1)
    let hoverRate = isHovered ? 1.05 : 1;
    // 返回曲面参数方程
    return {
      u: {
        min: -Math.PI,
        max: Math.PI * 3,
        step: Math.PI / 32,
      },
      v: {
        min: 0,
        max: Math.PI * 2,
        step: Math.PI / 20,
      },
      x: function (u, v) {
        if (u < startRadian) {
          return (
            offsetX +
            Math.cos(startRadian) * (1 + Math.cos(v) * k) * hoverRate
          );
        }
        if (u > endRadian) {
          return (
            offsetX +
            Math.cos(endRadian) * (1 + Math.cos(v) * k) * hoverRate
          );
        }
        return offsetX + Math.cos(u) * (1 + Math.cos(v) * k) * hoverRate;
      },
      y: function (u, v) {
        if (u < startRadian) {
          return (
            offsetY +
            Math.sin(startRadian) * (1 + Math.cos(v) * k) * hoverRate
          );
        }
        if (u > endRadian) {
          return (
            offsetY +
            Math.sin(endRadian) * (1 + Math.cos(v) * k) * hoverRate
          );
        }
        return offsetY + Math.sin(u) * (1 + Math.cos(v) * k) * hoverRate;
      },
      z: function (u, v) {
        if (u < -Math.PI * 0.5) {
          return Math.sin(u);
        }
        if (u > Math.PI * 2.5) {
          return Math.sin(u) * h * 0.1;
        }
        return Math.sin(v) > 0 ? 1 * h * 0.1 : -1;
      },
    };
  }

  let option = getPie3D(data, 0.8);
  option.series.push({
    name: "pie2d",
    type: "pie",
    labelLine: {
      show: false,
      length: 15,
      length2: 15,
    },
    startAngle: -fontSize(0.3), //起始角度,支持范围[0, 360]。
    clockwise: false, //饼图的扇区是否是顺时针排布。上述这两项配置主要是为了对齐3d的样式
    radius: [fontSize(1.3) + "%", fontSize(1.4) + "%"],
    center: [fontSize(1.5) + "%", fontSize(1.8) + "%"], //指示线的位置
    data: data,
    itemStyle: {
      opacity: 0,
    },
  });
  option && myChart.setOption(option);
  window.addEventListener("resize", () => {
    myChart.resize()
  });
}



const state = reactive({
  totleValue: {},
  seriesData: [
    // {
    //   name: "印度尼西亚",
    //   value: 100,
    // },
    // {
    //   name: "越南",
    //   value: 10,
    // },
    // {
    //   name: "柬埔寨",
    //   value: 10,
    // },
    // {
    //   name: "泰国",
    //   value: 5,
    // },

  ]
})


const getList = () => {
  emit('loadingTrue')
  selectInAnimalCount().then(res => {
    emit('loadingFalse')
    if (res.code === 10001) {
      let seriesData = [],
        allValue = {};//总数的数据

      res.data.forEach(item => {
        if (item.countryName === '总数') {
          allValue = item
        } else {
          seriesData.push({
            name: item.countryName,
            value: item.countNo
          })
        }
      })
      state.seriesData = seriesData
      state.totleValue = allValue

      nextTick(() => {
        initChart(state.seriesData)
      })

    } else {
      ElMessage({
        type: 'error',
        message: res.msg
      });
    }
  })

}




watch(() => Iscollapse.value, (n, o) => {
  nextTick(() => {
    myChart.resize()
  })
})
onMounted(() => {
  getList()

})
</script>

<style lang="less" scoped>
.earthmap {
  height: 100%;
  width: 100%;
  overflow: hidden;
  //   border: 1px solid red;

  display: flex;
  justify-content: center;
  .PieCom {
    height: 100%;
    width: 50%;
    // border: 1px solid red;
    background: url("../image/9.png") no-repeat bottom center;
    background-size: contain;
  }
  .rightText {
    height: 100%;
    width: 50%;
    padding-top: 8%;
    padding-left: 5%;
    .rightText-item {
      color: #fff;
      height: 18%;
      display: flex;
      align-items: center;
      // border: 1px solid red;
      &:nth-child(2) {
        // border: 1px solid red;
        .valueNum {
          background: linear-gradient(
            0deg,
            #f6f594 0%,
            #ffffff 80%,
            #ffffff 100%
          );
          -webkit-background-clip: text;
          -webkit-text-fill-color: transparent;
        }
      }
      &:nth-child(3) {
        // border: 1px solid red;
        .valueNum {
          background: linear-gradient(
            0deg,
            #97fdff 0%,
            #ffffff 80%,
            #ffffff 100%
          );
          -webkit-background-clip: text;
          -webkit-text-fill-color: transparent;
        }
      }
      &:nth-child(4) {
        // border: 1px solid red;
        .valueNum {
          background: linear-gradient(
            0deg,
            #ffdede 0%,
            #ffffff 80%,
            #ffffff 100%
          );
          -webkit-background-clip: text;
          -webkit-text-fill-color: transparent;
        }
      }
      &:nth-child(5) {
        // border: 1px solid red;
        .valueNum {
          background: linear-gradient(
            0deg,
            #8dc0ef 0%,
            #ffffff 80%,
            #ffffff 100%
          );
          -webkit-background-clip: text;
          -webkit-text-fill-color: transparent;
        }
      }

      .lable {
        font-family: PingFang SC;
        font-weight: 500;
        font-size: 14px;
        color: #ffffff;
        width: 60px;
        // border: 1px solid red;
        white-space: nowrap;
        overflow: hidden;
        text-overflow: ellipsis;
      }
      .valueNum {
        margin-left: 10px;
        font-family: pmzd;
        font-weight: 600;
        font-size: 24px;
        letter-spacing: 2px;
        // margin-left: 10%;
        max-width: calc(100% - 70px);
        // border: 1px solid red;
        white-space: nowrap;
        overflow: hidden;
        text-overflow: ellipsis;
        text-align: left;
      }
      .dw {
        font-family: PingFang SC;
        font-weight: 500;
        font-size: 12px;
        color: #ffffff;
        line-height: 40px;
        opacity: 0.5;
        margin-left: 10%;
      }
    }
  }
}
</style>