安裝echarts和echarts-gl
npm install echarts
npm install echarts-gl
echarts版本5.x的話需要對應(yīng)echarts-gl版本2.x
echarts版本4.x的話需要對應(yīng)echarts-gl版本1.x
指定版本命令?npm install echarts-gl@1.1.2
1.關(guān)鍵函數(shù),生成扇形的曲面參數(shù)方程,用于 series-surface
Documentation - Apache ECharts官網(wǎng)series-surface介紹?Documentation - Apache ECharts
getParametricEquation(startRatio, endRatio, isSelected, isHovered, k, h) {
? ? // 計算
? ? const midRatio = (startRatio + endRatio) / 2;
? ? const startRadian = startRatio * Math.PI * 2;
? ? const endRadian = endRatio * Math.PI * 2;
? ? const midRadian = midRatio * Math.PI * 2;
? ? // 如果只有一個扇形,則不實現(xiàn)選中效果。
? ? if (startRatio === 0 && endRatio === 1) {
? ? ? ? isSelected = false;
? ? }
? ? // 通過扇形內(nèi)徑/外徑的值,換算出輔助參數(shù) k(默認(rèn)值 1/3)
? ? k = 1;
? ? // 計算選中效果分別在 x 軸、y 軸方向上的位移(未選中,則位移均為 0)
? ? const offsetX = isSelected ? Math.cos(midRadian) * 0.1 : 0;
? ? const offsetY = isSelected ? Math.sin(midRadian) * 0.1 : 0;
? ? // 計算高亮效果的放大比例(未高亮,則比例為 1)
? ? const hoverRate = isHovered ? 1.05 : 1;
? ? // 返回曲面參數(shù)方程
? ? 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;
? ? ? ? },
? ? };
? }
2.構(gòu)建餅圖函數(shù),繪制3d圖。internalDiameterRatio為透明的空心占比,0就是普通餅1就鏤空
getPie3D(pieData, internalDiameterRatio) {
? ? const series = [];
? ? let sumValue = 0;
? ? let startValue = 0;
? ? let endValue = 0;
? ? const legendData = [];
? ? const k =
? ? ? ? typeof internalDiameterRatio !== 'undefined'
? ? ? ? ? ? ? (1 - internalDiameterRatio) / (1 + internalDiameterRatio)
? ? ? ? ? ? : 1 / 3;
? ? // 為每一個餅圖數(shù)據(jù),生成一個 series-surface 配置
? ? for (let i = 0; i < pieData.length; i += 1) {
? ? ? ? sumValue += pieData[i].value;
? ? ? ? const 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,
? ? ? ? ? ? },
? ? ? ? ? ? itemStyle:{}
? ? ? ? };
? ? ? ? if (typeof pieData[i].itemStyle !== 'undefined') {
? ? ? ? ? ? const itemStyle :any= {};
? ? ? ? ? ? if (typeof pieData[i].itemStyle.color !== 'undefined') {
? ? ? ? ? ? ? ? itemStyle.color = pieData[i].itemStyle.color;
? ? ? ? ? ? }
? ? ? ? ? ? if (typeof pieData[i].itemStyle.opacity !== 'undefined') {
? ? ? ? ? ? ? ? itemStyle.opacity = pieData[i].itemStyle.opacity;
? ? ? ? ? ? }
? ? ? ? ? ? seriesItem.itemStyle = itemStyle;
? ? ? ? }
? ? ? ? series.push(seriesItem);
? ? }
? ? // 使用上一次遍歷時,計算出的數(shù)據(jù)和 sumValue,調(diào)用 getParametricEquation 函數(shù),
? ? // 向每個 series-surface 傳入不同的參數(shù)方程 series-surface.parametricEquation,也就是實現(xiàn)每一個扇形。
? ? for (let i = 0; i < series.length; i += 1) {
? ? ? ? endValue = startValue + series[i].pieData.value;
? ? ? ? series[i].pieData.startRatio = startValue / sumValue;
? ? ? ? series[i].pieData.endRatio = endValue / sumValue;
? ? ? ? console.log(series[i].pieData.startRatio,
? ? ? ? ? ? series[i].pieData.endRatio,
? ? ? ? ? ? false, ?
? ? ? ? ? ? false,
? ? ? ? ? ? k,
? ? ? ? ? ? series[i].pieData.value)
? ? ? ? ? ? series[i].parametricEquation = this.getParametricEquation(
? ? ? ? ? ? series[i].pieData.startRatio,
? ? ? ? ? ? series[i].pieData.endRatio,
? ? ? ? ? ? false,
? ? ? ? ? ? false,
? ? ? ? ? ? k,
? ? ? ? ? ? 25//高度
? ? ? ? );
? ? ? ? startValue = endValue;
? ? ? ? legendData.push(series[i].name);
? ? }
? ? return series;
? }
3.繪制2d餅圖,添加label指引線
series.push({
? ? ? name: 'pie2d',
? ? ? type: 'pie',
? ? ? label: {
? ? ? ? ? opacity: 1,
? ? ? ? ? fontSize: 14,
? ? ? ? ? lineHeight: 20,
? ? ? },
? ? ? labelLine: {
? ? ? ? ? length: 50,
? ? ? ? ? length2: 50,
? ? ? },
? ? ? startAngle: -50, //起始角度,支持范圍[0, 360]。
? ? ? clockwise: false, //餅圖的扇區(qū)是否是順時針排布。上述這兩項配置主要是為了對齊3d的樣式
? ? ? radius: ['20%', '50%'],
? ? ? center: ['50%', '50%'],
? ? ? data: optionsData,
? ? ? itemStyle: {
? ? ? ? ? opacity: 0,
? ? ? },
? ? });
js代碼:文章來源:http://www.zghlxwxcb.cn/news/detail-593030.html
setPie(){
const optionsData = [
{
name: 'aa',
value: 20,
itemStyle: {
color: '#38ACEC',
// opacity: 1,
},
},
{
name: 'bb',
value: 15,
itemStyle: {
color: '#6960EC',
// opacity: 1,
},
},
{
name: 'cc',
value: 25,
itemStyle: {
color: '#6CBB3C',
// opacity: 1,
},
},
];
const series = this.getPie3D(optionsData, 0.5,);
series.push({
name: 'pie2d',
type: 'pie',
label: {
opacity: 1,
fontSize: 14,
lineHeight: 20,
},
labelLine: {
length: 50,
length2: 50,
},
startAngle: -50, //起始角度,支持范圍[0, 360]。
clockwise: false, //餅圖的扇區(qū)是否是順時針排布。上述這兩項配置主要是為了對齊3d的樣式
radius: ['20%', '50%'],
center: ['50%', '50%'],
data: optionsData,
itemStyle: {
opacity: 0,
},
});
this.option1 =
{
legend: {
tooltip: {
show: true,
},
data: ['aa', 'bb', 'cc'],
right: '2%',
textStyle: {
color: '#fff',
fontSize: 12,
},
},
tooltip: {
formatter: (params) => {
if (params.seriesName !== 'mouseoutSeries' && params.seriesName !== 'pie2d') {
let bfb = (
(this.option.series[params.seriesIndex].pieData.endRatio -
this.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;"></span>` +
`${bfb}%`
);
}
},
},
title: {
text: '3D 餅圖',
x: 'center',
top: '5%',
textStyle: {
color: '#fff',
fontSize: 22,
},
},
labelLine: {
show: true,
lineStyle: {
color: '#7BC0CB',
},
},
label: {
show: true,
position: 'outside',
formatter: ' \n{c} n5n3t3z%',
},
xAxis3D: {
min: -1,
max: 1,
},
yAxis3D: {
min: -1,
max: 1,
},
zAxis3D: {
min: -1,
max: 1,
},
grid3D: {
show: false,
boxHeight: 25, // 三維笛卡爾坐標(biāo)系在三維場景中的高度
viewControl: {
alpha: 45,
// beta: 1000,
distance: 300, //調(diào)整視角到主體的距離,類似調(diào)整zoom
// rotateSensitivity: 0, // 設(shè)置為0無法旋轉(zhuǎn)
zoomSensitivity: 0, // 設(shè)置為0無法縮放
panSensitivity: 0, // 設(shè)置為0無法平移
autoRotate: false, // 自動旋轉(zhuǎn)
},
},
series: series,
};
}
// 生成扇形的曲面參數(shù)方程,用于 series-surface.parametricEquation
getParametricEquation(startRatio, endRatio, isSelected, isHovered, k, h) {
// 計算
const midRatio = (startRatio + endRatio) / 2;
const startRadian = startRatio * Math.PI * 2;
const endRadian = endRatio * Math.PI * 2;
const midRadian = midRatio * Math.PI * 2;
// 如果只有一個扇形,則不實現(xiàn)選中效果。
if (startRatio === 0 && endRatio === 1) {
isSelected = false;
}
// 通過扇形內(nèi)徑/外徑的值,換算出輔助參數(shù) k(默認(rèn)值 1/3)
k = 1;
// 計算選中效果分別在 x 軸、y 軸方向上的位移(未選中,則位移均為 0)
const offsetX = isSelected ? Math.cos(midRadian) * 0.1 : 0;
const offsetY = isSelected ? Math.sin(midRadian) * 0.1 : 0;
// 計算高亮效果的放大比例(未高亮,則比例為 1)
const hoverRate = isHovered ? 1.05 : 1;
// 返回曲面參數(shù)方程
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;
},
};
}
//構(gòu)建餅圖數(shù)據(jù),繪制3d圖 internalDiameterRatio:透明的空心占比
getPie3D(pieData, internalDiameterRatio) {
const series = [];
let sumValue = 0;
let startValue = 0;
let endValue = 0;
const legendData = [];
const k =
typeof internalDiameterRatio !== 'undefined'
? (1 - internalDiameterRatio) / (1 + internalDiameterRatio)
: 1 / 3;
// 為每一個餅圖數(shù)據(jù),生成一個 series-surface 配置
for (let i = 0; i < pieData.length; i += 1) {
sumValue += pieData[i].value;
const 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,
},
itemStyle:{}
};
if (typeof pieData[i].itemStyle !== 'undefined') {
const itemStyle :any= {};
if (typeof pieData[i].itemStyle.color !== 'undefined') {
itemStyle.color = pieData[i].itemStyle.color;
}
if (typeof pieData[i].itemStyle.opacity !== 'undefined') {
itemStyle.opacity = pieData[i].itemStyle.opacity;
}
seriesItem.itemStyle = itemStyle;
}
series.push(seriesItem);
}
// 使用上一次遍歷時,計算出的數(shù)據(jù)和 sumValue,調(diào)用 getParametricEquation 函數(shù),
// 向每個 series-surface 傳入不同的參數(shù)方程 series-surface.parametricEquation,也就是實現(xiàn)每一個扇形。
for (let i = 0; i < series.length; i += 1) {
endValue = startValue + series[i].pieData.value;
series[i].pieData.startRatio = startValue / sumValue;
series[i].pieData.endRatio = endValue / sumValue;
console.log(series[i].pieData.startRatio,
series[i].pieData.endRatio,
false,
false,
k,
series[i].pieData.value)
series[i].parametricEquation = this.getParametricEquation(
series[i].pieData.startRatio,
series[i].pieData.endRatio,
false,
false,
k,
25//高度
);
startValue = endValue;
legendData.push(series[i].name);
}
return series;
}
html代碼:文章來源地址http://www.zghlxwxcb.cn/news/detail-593030.html
<div id="option" echarts [options]="option1" style="width:100%;height:100%"></div>
到了這里,關(guān)于用echarts實現(xiàn)3d餅圖的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!