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;
};