import * as echarts from "echarts"; // import { init, registerMap, use } from 'echarts' // import { Lines3DChart, Map3DChart } from 'echarts-gl/charts' // import { Grid3DComponent } from 'echarts-gl/components' import "echarts-gl"; function getDigitsCount(number) { if (number === 0) return 1; return Math.floor(Math.log10(Math.abs(number))) + 1; } function calculateAddition(number) { const digitsCount = getDigitsCount(number); return Math.pow(10, digitsCount); } function getMaxAndAdd(arr) { const maxNumber = Math.max(...arr); const addition = calculateAddition(maxNumber); return maxNumber + addition; } // 近7月碳排放趋势 export const echartLineOptions = (data) => { let seriesDataOne = []; let seriesDataTwo = []; const xAxisData = data.map(({ time, ztTotalCo2, wschytpfqd }) => { seriesDataOne.push(ztTotalCo2); seriesDataTwo.push(wschytpfqd); return time; }); const option = { tooltip: { trigger: "axis", align: "left", }, legend: { x: "center", y: "26px", icon: "rect", itemWidth: 14, itemHeight: 8, textStyle: { fontSize: "12px", color: "rgba(217, 247, 255, 0.7)", }, data: ["碳排放量", "碳排放强度"], }, grid: { top: "20%", bottom: "20%", left: "10%", right: "10%", // containLabel: true, }, xAxis: { show: data.length, type: "category", data: xAxisData, boundaryGap: false, splitLine: { show: false, interval: "auto", lineStyle: { color: ["#fff"], }, }, axisTick: { show: false, }, axisLine: { lineStyle: { color: "#8890a0", }, }, axisLabel: { margin: 10, rotate: 45, fontSize: 12, color: "#fff", }, }, yAxis: [ { splitNumber: 3, show: data.length, type: "value", nameGap: 22, name: "tCO₂-eq", scale: true, axisLabel: { // margin: -10, color: "#fff", }, axisLine: { show: false, }, axisTick: { show: false, }, splitLine: { lineStyle: { color: "#535763", type: "dashed", }, }, }, { splitNumber: 3, show: data.length, type: "value", nameGap: 22, name: "kg CO₂-eq/m³", scale: true, axisLabel: { // margin: -20, color: "#fff", }, axisTick: { show: false, }, splitLine: { show: false, }, // min: (value) => { // return (value.min - 1).toFixed(1); // }, // max: (value) => {}, }, ], series: [ { name: "碳排放量", type: "line", yAxisIndex: 0, smooth: true, showSymbol: false, symbol: "circle", symbolSize: 6, data: seriesDataOne, // data: [44,14,66,77,72,99,11], areaStyle: { color: new echarts.graphic.LinearGradient( 0, 0, 0, 1, [ { offset: 0, color: "rgba(0, 255, 0, 0.3)", }, { offset: 1, color: "rgba(0, 255, 0, 0)", }, ], false ), }, itemStyle: { color: "#76fb4d", }, lineStyle: { width: 1.5, }, zLevel: 1, }, { name: "碳排放强度", type: "line", smooth: true, showSymbol: false, symbol: "circle", symbolSize: 6, data: seriesDataTwo, yAxisIndex: 1, // data: [14,44,56,67,22,19,41], areaStyle: { color: new echarts.graphic.LinearGradient( 0, 0, 0, 1, [ { offset: 0, color: "rgba(247, 147, 30, 1)", }, { offset: 1, color: "rgba(251, 176, 59, 0.2)", }, ], false ), }, itemStyle: { color: "rgba(255, 141, 7, 1)", }, lineStyle: { width: 1.5, }, zLevel: 12, }, ], }; return option; }; // 污水处理生化反应 export const echartGraphOptions = (d = []) => { const data = d.reverse(); let salvProName = [ "污水生物处理脱氮", "污水收集和处理", "外加碳源矿化", "化石源碳矿化", ]; let maxData = getMaxAndAdd(data); var colorList = [ ["rgba(0, 198, 255, 1)", "rgba(0, 198, 255, 0)"], ["rgba(42, 236, 60, 1)", "rgba(42, 236, 60, 0)"], ["rgba(255, 255, 0, 1)", "rgba(255, 255, 0, 0)"], ["rgba(255, 85, 0, 1)", "rgba(255, 85, 0, 0)"], ]; const options = { grid: { top: "10%", bottom: "5%", left: "5%", right: "5%", }, tooltip: { trigger: "axis", formatter: ([params]) => { return `${ params.name }
${params.data.value || ""}`; }, }, xAxis: [ { type: "value", // max: maxData, show: false, }, { type: "value", show: false, }, ], yAxis: [ { show: true, type: "category", axisTick: "none", axisLine: "none", position: "left", axisLabel: { show: true, margin: 10, align: "top", verticalAlign: "bottom", rich: { a: { padding: [24, 0, 14, 8], fontSize: 16, }, }, color: "#fff", formatter: function (params) { return `{a|${params || ""}}`; }, }, data: salvProName, }, { type: "category", axisTick: "none", axisLine: "none", axisLabel: { show: true, fontSize: 14, color: "red", inside: true, padding: [-20, 0, 20, 20], color: "#ffffff", fontSize: "12", fontFamily: "DIN", fontWeight: 700, rich: { a: { fontSize: 18, color: "#fff", fontFamily: "D-DIN-PRO-700-Bold", }, b: { fontSize: 14, paddingLeft: 10, color: "rgba(255,255,255,.5)", }, }, formatter: function (value) { let val = value; if (value === "null" || (!value && value !== 0)) { val = ""; } return `{a|${val}} {b|tCO₂-eq}`; }, }, data: data.reverse(), }, ], series: [ { // name: "进度部分", type: "bar", zlevel: 1, barWidth: 6, barGap: "0%", data: data.map((item, index) => { return { value: item, itemStyle: { color: new echarts.graphic.LinearGradient(1, 0, 0, 0, [ { offset: 0, color: colorList[index][0], }, { offset: 1, color: colorList[index][1], }, ]), }, }; }), }, { tooltip: { show: false, }, name: "背景部分", type: "bar", xAxisIndex: 1, barWidth: 6, barGap: "-100%", data: [1, 1, 1, 1], itemStyle: { color: "#D0DEEE", opacity: 0.1, }, }, // 进度条的小圆圈 { tooltip: { show: false, // 设置 tooltip 不显示 }, name: "小圈圈", type: "scatter", emphasis: { scale: false, }, symbol: "image://", symbolSize: [20, 20], itemStyle: { color: "#FFF", shadowColor: "rgba(255, 255, 255, 1)", shadowBlur: 5, borderWidth: 1, opacity: 1, }, z: 2, zlevel: 10, data: data, animationDelay: 500, }, ], }; return options; }; // 能源药剂 export const echart3DOptions = (data = []) => { const isEmpty = data.every((item) => !item); // 传入数据生成 option const dataList = [ { name: "电力消耗", val: data[0] || 0, itemStyle: { color: "rgba(245, 184, 77, 1)", }, }, { name: "药剂消耗", val: data[1] || 0, itemStyle: { color: "rgba(32, 159, 237, 1)", }, }, { name: "燃料消耗", val: data[2] || 0, itemStyle: { color: "rgba(128, 142, 199, 1)", }, }, { name: "净购入热力", val: data[3] || 0, itemStyle: { color: "rgba(239, 127, 53, 1)", }, }, ]; const heightProportion = 0.2; // 柱状扇形的高度比例 // 生成扇形的曲面参数方程,用于 series-surface.parametricEquation function getParametricEquation( startRatio, endRatio, isSelected, isHovered, k, height ) { // 计算 let midRatio = (startRatio + endRatio) / 3; 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.1 : 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); } return Math.sin(v) > 0 ? heightProportion * height : -1; }, }; } function getHeight3D (series, height) { series.sort((a, b) => { return b.pieData.value - a.pieData.value; }); return (height * 20) / series[0].pieData.value; } // 生成模拟 3D 饼图的配置项 function getPie3D(pieData, internalDiameterRatio) { let series = []; let sumValue = 0; let startValue = 0; let endValue = 0; let legendData = []; let linesSeries = []; // line3D模拟label指示线 let k = typeof internalDiameterRatio !== "undefined" ? (1 - internalDiameterRatio) / (1 + internalDiameterRatio) : 1 / 3; // 为每一个饼图数据,生成一个 series-surface 配置 for (let i = 0; i < pieData.length; i++) { sumValue += pieData[i].value; let seriesItem = { name: typeof pieData[i].name === "undefined" ? `series${i}` : pieData[i].name, type: "surface", parametric: true, wireframe: { show: false, }, pieData: pieData[i], pieStatus: { selected: false, hovered: false, k: k, }, z: 10, zlevel: 11, }; if (typeof pieData[i].itemStyle != "undefined") { let itemStyle = {}; typeof pieData[i].itemStyle.color != "undefined" ? (itemStyle.color = pieData[i].itemStyle.color) : null; typeof pieData[i].itemStyle.opacity != "undefined" ? (itemStyle.opacity = pieData[i].itemStyle.opacity) : null; seriesItem.itemStyle = itemStyle; } series.push(seriesItem); } // 使用上一次遍历时,计算出的数据和 sumValue,调用 getParametricEquation 函数, // 向每个 series-surface 传入不同的参数方程 series-surface.parametricEquation,也就是实现每一个扇形。 for (let i = 0; i < series.length; i++) { endValue = startValue + series[i].pieData.value; series[i].pieData.startRatio = startValue / sumValue; series[i].pieData.endRatio = endValue / sumValue; series[i].parametricEquation = getParametricEquation( series[i].pieData.startRatio, series[i].pieData.endRatio, false, false, k, series[i].pieData.value ); startValue = endValue; legendData.push(series[i].name); } series = series.concat(linesSeries); return series; } let total = 0; dataList.forEach((item) => { total += item.val; }); const series = getPie3D( dataList.map((item) => { item.value = Number(((item.val / total) * 100).toFixed(2)); return item; }), 0.6 ); const boxHeight = getHeight3D(series, 15); // 准备待返回的配置项,把准备好的 legendData、series 传入。 const option = { graphic: { elements: [ { type: "image", style: { image: "https://static.fuxicarbon.com/bigModel/pc/bg-3d.png", width: 266, height: 184, zIndex: -1, background: "red", }, left: "0%", top: "32%", }, ], }, legend: { data: dataList.map((item) => item.name), orient: "vertical", icon: "circle", top: "center", itemGap: 30, itemHeight: 10, itemWidth: 10, right: "4%", textStyle: { rich: { name: { width: 60, padding: [0, 20, 0, 0], fontSize: 14, color: "#fff", }, value: { fontSize: 14, color: "rgba(157, 219, 255, 1)", fontFamily: "D-DIN-PRO-700-Bold", }, unit: { fontSize: 14, color: "rgba(157, 219, 255, 1)", }, }, color: "red", fontSize: 14, }, formatter: function (name) { const value = dataList.find((item) => item.name === name).val; return `{name|${name}} {value|${value}} {unit|tCO₂-eq}`; }, }, tooltip: { formatter: (params) => { if ( params.seriesName !== "mouseoutSeries" && params.seriesName !== "pie2d" ) { return `${ params.seriesName }
${option.series[params.seriesIndex].pieData.val}`; } }, }, animation: true, label: { show: false, }, xAxis3D: { min: -1.5, max: 1.5, }, yAxis3D: { min: -1.5, max: 1.5, }, zAxis3D: { min: -1, max: 1, }, grid3D: { show: false, boxHeight: 3.5, top: "-6%", left: "-22%", width: "100%", viewControl: { distance: 180, alpha: 25, beta: 60, autoRotate: true, // 自动旋转 zoomSensitivity: 0, }, }, series: isEmpty ? [] : series, }; return option; }; // 污泥处理 - 生化反应 export const echartColumnarOptions = (data) => { var imgUrl = [ "image://", "image://", "image://", "image://", "image://", "image://", ]; var maxList = new Array(data.length).fill(getMaxAndAdd(data)); const CubeBack = echarts.graphic.extendShape({ shape: { x: 0, y: 0, }, buildPath: function (ctx, shape) { const xAxisPoint = shape.xAxisPoint; const c0 = [shape.x + 10, shape.y - 5]; //右上 const c1 = [shape.x - 10, shape.y - 5]; //左上 const c4 = [shape.x, shape.y - 5]; //上中点 const c2 = [xAxisPoint[0] - 10, xAxisPoint[1] - 5]; //左下 const c5 = [xAxisPoint[0], xAxisPoint[1]]; //下中点 const c3 = [xAxisPoint[0] + 10, xAxisPoint[1] - 5]; //右下 ctx .moveTo(c0[0], c0[1]) .lineTo(c4[0], c4[1]) .lineTo(c1[0], c1[1]) .lineTo(c2[0], c2[1]) .lineTo(c5[0], c5[1]) .lineTo(c3[0], c3[1]) .closePath(); }, }); const CubeLeft = echarts.graphic.extendShape({ shape: { x: 0, y: 0, }, buildPath: function (ctx, shape) { const xAxisPoint = shape.xAxisPoint; const c0 = [shape.x, shape.y]; const c1 = [shape.x - 12, shape.y - 7]; //左上角 const c2 = [xAxisPoint[0] - 12, xAxisPoint[1] - 6]; //右上角 const c3 = [xAxisPoint[0], xAxisPoint[1]]; ctx .moveTo(c0[0], c0[1]) .lineTo(c1[0], c1[1]) .lineTo(c2[0], c2[1]) .lineTo(c3[0], c3[1]) .closePath(); }, }); const CubeRight = echarts.graphic.extendShape({ shape: { x: 0, y: 0, }, buildPath: function (ctx, shape) { const xAxisPoint = shape.xAxisPoint; const c1 = [shape.x, shape.y]; const c2 = [xAxisPoint[0], xAxisPoint[1]]; const c3 = [xAxisPoint[0] + 12, xAxisPoint[1] - 6]; //右下 const c4 = [shape.x + 12, shape.y - 6]; ctx .moveTo(c1[0], c1[1]) .lineTo(c2[0], c2[1]) .lineTo(c3[0], c3[1]) .lineTo(c4[0], c4[1]) .closePath(); }, }); const CubeTop = echarts.graphic.extendShape({ shape: { x: 0, y: 0, }, buildPath: function (ctx, shape) { const c1 = [shape.x, shape.y]; const c2 = [shape.x + 12, shape.y - 7]; //右下 右 const c3 = [shape.x, shape.y - 15]; //右上 上 const c4 = [shape.x - 12, shape.y - 7]; ctx .moveTo(c1[0], c1[1]) .lineTo(c2[0], c2[1]) .lineTo(c3[0], c3[1]) .lineTo(c4[0], c4[1]) .closePath(); }, }); echarts.graphic.registerShape("CubeBack", CubeBack); echarts.graphic.registerShape("CubeLeft", CubeLeft); echarts.graphic.registerShape("CubeRight", CubeRight); echarts.graphic.registerShape("CubeTop", CubeTop); const option = { tooltip: { trigger: "axis", formatter: ([params]) => { return `${ params.name }
${ (!params.data && params.data != 0 ? "" : params.data) + "" }`; }, }, grid: { top: "25%", bottom: "5%", left: "5%", right: "5%", containLabel: true, }, xAxis: { data: [ "沼气泄露", "沼液", "好氧发酵", "污泥干化燃烧", "污泥燃烧", "污泥热解", ], axisLine: { show: false, }, offset: 0, axisTick: { show: false, }, axisLabel: { margin: 20, rotate: 45, fontSize: 12, color: "#fff", }, }, yAxis: { nameGap: 40, name: "tCO₂-eq", scale: true, axisLabel: { color: "#fff", }, splitNumber: 3, // interval: 20, type: "value", axisLine: { show: false, }, splitLine: { show: false, }, axisTick: { show: false, }, axisLabel: { show: true, color: "#fff", }, }, series: [ { type: "custom", renderItem: (params, api) => { const location = api.coord([api.value(0), api.value(1)]); return { type: "group", children: [ { type: "CubeLeft", shape: { api, xValue: api.value(0), yValue: api.value(1), x: location[0], y: location[1], xAxisPoint: api.coord([api.value(0), 0]), }, style: { fill: new echarts.graphic.LinearGradient(0, 0, 0, 1, [ { offset: 0, color: "rgba(4, 110, 188, 0.62)", }, { offset: 1, color: "rgba(184, 233, 255, 0.28)", }, ]), }, }, { type: "CubeRight", shape: { api, xValue: api.value(0), yValue: api.value(1), x: location[0], y: location[1], xAxisPoint: api.coord([api.value(0), 0]), }, style: { fill: new echarts.graphic.LinearGradient(0, 0, 0, 1, [ { offset: 0, color: "rgba(4, 110, 188, .1)", }, { offset: 1, color: "rgba(4, 110, 188, .3)", }, ]), }, }, { type: "CubeTop", shape: { api, xValue: api.value(0), yValue: api.value(1), x: location[0], y: location[1], xAxisPoint: api.coord([api.value(0), 0]), }, style: { fill: "rgba(4, 110, 188, .4)", }, }, ], }; }, tooltip: { show: false, }, data: maxList, }, { zlevel: 1, z: -1, color: "#3d98df", type: "custom", name: "", barWidth: 10, renderItem: (params, api) => { const location = api.coord([api.value(0), api.value(1)]); return { type: "group", children: [ { type: "CubeLeft", shape: { api, xValue: api.value(0), yValue: api.value(1), x: location[0], y: location[1], xAxisPoint: api.coord([api.value(0), 0]), }, style: { fill: new echarts.graphic.LinearGradient(0, 0, 0, 1, [ { offset: 0, color: "rgba(0, 183, 238, 0.82)", }, { offset: 1, color: "rgba(0, 176, 231, 0.1)", }, ]), }, }, { type: "CubeRight", shape: { api, xValue: api.value(0), yValue: api.value(1), x: location[0], y: location[1], xAxisPoint: api.coord([api.value(0), 0]), }, style: { fill: new echarts.graphic.LinearGradient(0, 0, 0, 1, [ { offset: 0, color: "rgba(0, 183, 238, 0.2)", }, { offset: 1, color: "rgba(0, 176, 231, 0.1)", }, ]), }, }, { type: "CubeTop", shape: { api, xValue: api.value(0), yValue: api.value(1), x: location[0], y: location[1], xAxisPoint: api.coord([api.value(0), 0]), }, style: { fill: "rgba(80,176,230, .9)", }, }, ], }; }, data: data, }, { name: "", type: "pictorialBar", z: 13, tooltip: { show: false, }, label: { show: false, position: [70, -60], color: "#74D3FF", formatter: function (params) { return "{a|" + params.data.val + "} {b|%}"; }, rich: { a: { fontSize: 22, verticalAlign: "bottom", fontWeight: "bolder", }, b: { fontSize: 12, verticalAlign: "bottom", fontWeight: "bolder", }, }, }, data: data.map((item, index) => { return { name: "", value: item, val: item, symbol: imgUrl[index], symbolPosition: "start", symbolOffset: [0, 10], symbolSize: [60, 60], color: "red", }; }), }, ], }; return option; };