如何在Angular2中使用SVG自定義圖表?針對(duì)這個(gè)問(wèn)題,這篇文章詳細(xì)介紹了相對(duì)應(yīng)的分析和解答,希望可以幫助更多想解決這個(gè)問(wèn)題的小伙伴找到更簡(jiǎn)單易行的方法。
成都一家集口碑和實(shí)力的網(wǎng)站建設(shè)服務(wù)商,擁有專(zhuān)業(yè)的企業(yè)建站團(tuán)隊(duì)和靠譜的建站技術(shù),10年企業(yè)及個(gè)人網(wǎng)站建設(shè)經(jīng)驗(yàn) ,為成都上1000+客戶(hù)提供網(wǎng)頁(yè)設(shè)計(jì)制作,網(wǎng)站開(kāi)發(fā),企業(yè)網(wǎng)站制作建設(shè)等服務(wù),包括成都營(yíng)銷(xiāo)型網(wǎng)站建設(shè),品牌網(wǎng)站建設(shè),同時(shí)也為不同行業(yè)的客戶(hù)提供做網(wǎng)站、網(wǎng)站建設(shè)的服務(wù),包括成都電商型網(wǎng)站制作建設(shè),裝修行業(yè)網(wǎng)站制作建設(shè),傳統(tǒng)機(jī)械行業(yè)網(wǎng)站建設(shè),傳統(tǒng)農(nóng)業(yè)行業(yè)網(wǎng)站制作建設(shè)。在成都做網(wǎng)站,選網(wǎng)站制作建設(shè)服務(wù)商就選成都創(chuàng)新互聯(lián)。
demo:
html:
ts:
options = { type: 'line', //圖表類(lèi)型 xAxis: { //X軸的數(shù)據(jù) data: ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'] }, yAxis: { //X軸的數(shù)據(jù) data: [120, 220, 150, 111, -150, 55, 60], }, width: 600, //寬 height: 500, //高 dataPadding: 8 //條形圖之間的距離 };
效果:
源代碼:
import { Input, OnInit, ViewChild, Component, ViewEncapsulation, ElementRef, AfterViewInit, ChangeDetectorRef, } from '@angular/core'; import { NgoChartSvgParams, Scale, Axis, Chart } from './chart-svg-params'; @Component({ selector: 'ngo-chart-svg', templateUrl: './chart-svg.html', styleUrls: ['./chart-svg.scss'], encapsulation: ViewEncapsulation.Native }) export class NgoChartSvg implements OnInit, AfterViewInit { @Input() inputParams: NgoChartSvgParams; @ViewChild('svg') svg: ElementRef; @ViewChild('polyline') polyline: ElementRef; params: NgoChartSvgParams; AxisY: Axis; // Y軸 AxisX: Axis; // X軸 valueToPxRatio: number; // 值轉(zhuǎn)px的比率 Y0: number; // 坐標(biāo)軸 (0,0)的Y軸 Yscale: Array= []; // Y軸刻度值 Xscale: Array = []; // X軸刻度值 XgapWidth: number; // X軸刻度之間的間隙寬度 data: Array = []; color: string; type: string; polyLinePoints: string; polyLineLength: number; constructor( private ele: ElementRef, private cd: ChangeDetectorRef ) { } ... ngOnInit() { this.initParams(); const svg = this.svg.nativeElement; const _width = this.params.width; const _height = this.params.height; svg.setAttribute('width', _width); svg.setAttribute('height', _height); // 繪制 y軸 this.drawAxisY(); this.drawScaleY(); // 繪制 x軸 this.drawAxisX(); this.drawScaleX(); this.drawRect(); if (this.params.type === 'line') { this.drawLine(); } } ngAfterViewInit() { if (this.polyline) { this.polyLineLength = this.polyline.nativeElement.getTotalLength(); this.cd.detectChanges(); } } }
html
css
svg { background: rgba(0, 0, 0, 0.2); border: 1px solid black; } svg * { position: static; font-size: 16px; } ._polyline { fill: none; animation: lineMove 1.5s ease-in-out forwards; } @keyframes lineMove { to { stroke-dashoffset: 0; } }
一、初始化參數(shù)
//首先獲取傳入的參數(shù) @Input() inputParams; //初始化 const _params: NgoChartSvgParams = { xAxis: this.inputParams.xAxis, yAxis: this.inputParams.yAxis, type: this.inputParams.type ? this.inputParams.type : 'bar', width: this.inputParams.width ? this.inputParams.width : 700, height: this.inputParams.height ? this.inputParams.height : 500, dataPadding: this.inputParams.dataPadding !== undefined ? this.inputParams.dataPadding : 8, YscaleNo: this.inputParams.YscaleNo >= 3 ? this.inputParams.YscaleNo : 6, }; this.color = 'black'; this.type = _params.type; this.params = _params;
二:繪制坐標(biāo)軸Y軸
const _height = this.params.height; const _pad = this.params.padding; const _arrow = _pad + ',' + (_pad - 5) + ' ' + (_pad - 6) + ',' + (_pad + 12) + ' ' + (_pad + 6) + ',' + (_pad + 12); this.AxisY = { x1: _pad, y1: _height - _pad, x2: _pad, y2: _pad, arrow: _arrow };
三、繪制Y軸的刻度
const _height = this.params.height; const _width = this.params.width; // 顯示label的邊距 const _padding = this.params.padding; const _Ydata = this.params.yAxis.data; // 顯示的刻度數(shù) const _YscaleNo = this.params.YscaleNo; const _dataMax = this.getMinAndMaxData(_Ydata).dataMax; const _dataMin = this.getMinAndMaxData(_Ydata).dataMin; let _YminValue; let _YgapValue; if (_dataMin < 0) { _YgapValue = Math.ceil((_dataMax - _dataMin) / (_YscaleNo) / 10) * 10; _YminValue = Math.floor(_dataMin / _YgapValue) * _YgapValue; } else { _YgapValue = Math.ceil((_dataMax) / (_YscaleNo) / 10) * 10; _YminValue = 0; } // Y軸坐標(biāo)點(diǎn) const _y2 = this.AxisY.y2; const _y1 = this.AxisY.y1; const _x1 = this.AxisY.x1; // Y軸刻度的間隙寬度 const _YgapWidth = (_y1 - _y2) / (this.params.YscaleNo); this.valueToPxRatio = _YgapValue / _YgapWidth; // 坐標(biāo)軸(0,0)的Y軸坐標(biāo) const _Y0 = _y1 - Math.abs(_YminValue / this.valueToPxRatio); this.Y0 = _Y0; for (let i = 0; i < this.params.YscaleNo; i++) { const _obj: Scale = { x1: 0, x2: 0, y1: 0, y2: 0, label: '', value: 0 }; _obj.x1 = _x1; _obj.y1 = _y1 - _YgapWidth * i; _obj.x2 = _x1 + _width - 2 * _padding; _obj.y2 = _y1 - _YgapWidth * i; _obj.label = _YminValue + _YgapValue * i; this.Yscale.push(_obj); }
四、繪制X坐標(biāo)軸
const _width = this.params.width; // 顯示label的邊距 const _pad = this.params.padding; const _x2 = _width - _pad; const _y2 = this.Y0; const _arrow = (_x2 + 5) + ',' + _y2 + ' ' + (_x2 - 10) + ',' + (_y2 - 6) + ' ' + (_x2 - 10) + ',' + (_y2 + 6); this.AxisX = { x1: _pad, y1: _y2, x2: _x2, y2: _y2, arrow: _arrow };
五、繪制X軸刻度
const _width = this.params.width; const _Xdata = this.params.xAxis.data; const _Ydata = this.params.yAxis.data; const Y0 = this.Y0; const _x1 = this.AxisX.x1; const _x2 = this.AxisX.x2; const XgapWidth = ((_x2 - _x1) / (this.params.xAxis.data.length + 1)); this.XgapWidth = XgapWidth; for (let i = 0; i < _Xdata.length; i++) { const _obj: Scale = { x1: 0, x2: 0, y1: 0, y2: 0, value: 0, label: '' }; _obj.y1 = Y0; _obj.y2 = Y0 + 5; _obj.label = _Xdata[i]; _obj.value = _Ydata[i]; _obj.x1 = _x1 + XgapWidth * (i + 1); _obj.x2 = _x1 + XgapWidth * (i + 1); this.Xscale.push(_obj);
六、繪制矩形
const _value = this.params.yAxis.data; const _dataPadding = this.params.dataPadding; const _XgapWidth = this.XgapWidth; for (let i = 0; i < _value.length; i++) { const element = _value[i]; const _obj: Chart = { x: 0, y: 0, w: 0, h: 0, value: 0 }; _obj.w = _XgapWidth - 2 * _dataPadding; _obj.x = this.Xscale[i].x1 - _obj.w - _dataPadding; _obj.h = Math.abs(this.Xscale[i].value / this.valueToPxRatio); _obj.value = this.Xscale[i].value; if (this.Xscale[i].value >= 0) { _obj.y = this.Y0 - (this.Xscale[i].value) / this.valueToPxRatio; } else { _obj.y = this.Y0; } this.data.push(_obj); } }
七、繪制折線
const _data = this.data; let _str = ''; _data.forEach(ele => { if (ele.value < 0) { ele.y = ele.y + ele.h; } _str += (ele.x + ele.w / 2) + ',' + ele.y + ' '; }); this.polyLinePoints = _str;
關(guān)于如何在Angular2中使用SVG自定義圖表問(wèn)題的解答就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,如果你還有很多疑惑沒(méi)有解開(kāi),可以關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道了解更多相關(guān)知識(shí)。