微信小程序canvas手繪雷達圖

微信小程序已經(jīng)發(fā)布了兩個多月了,沒有想象中那么火。但是一些擁有多用戶量的APP應用也會抽出主業(yè)務流程做微信小程序版本。做過小程序開發(fā)的小伙伴們都知道微信小程序缺少類似echart.js的圖形庫,然而業(yè)內(nèi)echart.js等主流的h5圖形庫并不能在小程序上運行,所以遇到繪制圖形需求的時候我們只能用canvas繪畫。本文為大家分享的是以自定義組件的形式繪制雷達圖。

雷達圖

自定義組件

(1)建立組件文件

在工程根目錄建立component文件夾,里面建radar目錄,在radar里面建radar.js、radar.wxml、radar.wxss,3個文件


radar.js寫一個導出的方法:
<pre>
module.exports = {
radar
};
</pre>

radar.wxml寫一個模板:
<pre>
<template name="radar">
<view class="radar">
<canvas class="canvas" canvas-id="radarCanvas" />
</view>
</template>
</pre>

radar.wxss寫對應的樣式:
<pre>
.radar canvas {
width: 750rpx;
height: 750rpx;
background-color: #6fb9de;
}
</pre>

(2)在頁面中引入組件

比如我們要在index頁面引入組件,先在index.wxss引入樣式文件:
<pre>
@import "../../component/radar/radar.wxss"
</pre>

在index.wxml引入模板文件:
<pre>

<import src="../../component/radar/radar.wxml"/>

<template is="radar"/>
</pre>

在index.js引入js邏輯文件:
<pre>
// 1 引入雷達圖
import { radar } from '../../component/radar/radar'
// 2 page對象里加入組件
Page({
radar,
// 3 在onLoad函數(shù)里面調(diào)用組件
onLoad: function () {
this.radar.draw('');
}
</pre>

雷達圖組件的具體實現(xiàn)

(1)確定坐標系與中心點
canvas坐標系

canvas坐標系是X軸向右增加,Y軸向下增加的。

確定中心點:
<pre>
centerPoint = [rpx(375), rpx(375)];
context.moveTo(centerPoint[0], centerPoint[1]);
</pre>

根據(jù)雷達圖要分的比例項(即雷達圖的角數(shù))、網(wǎng)狀每一環(huán)的寬度、直角三角形的正弦sin計算網(wǎng)狀點的Y軸坐標,直角三角形的正弦cos計算網(wǎng)狀點的X軸坐標

網(wǎng)狀點的Y軸坐標(直角三角形的對角邊) = sin角度 x 網(wǎng)狀每一環(huán)的寬度(直角三角形的斜邊)
網(wǎng)狀點的X軸坐標(直角三角形的鄰角邊) = cos角度 x 網(wǎng)狀每一環(huán)的寬度(直角三角形的斜邊)

<pre>
for(n = 0; n < layerNum; n++) {
layerPoints[n] = [];
for(k = 0; k < angleNum; k++) {
context.moveTo(centerPoint[0], centerPoint[1]);
let offsetX = layerWidth * (n + 1) * getXParam(angleAvg * (k + 1) + angleOffset);
let offsetY = layerWidth * (n + 1) * getYParam(angleAvg * (k + 1) + angleOffset);
let distX = centerPoint[0] + offsetX;
let distY = centerPoint[1] + offsetY;
if(n == layerNum - 1) {
context.lineTo(distX, distY);
if(wordArr[k]) {
let wordOffsetX = offsetX >= 0 ? 1 : -1;
wordOffsetX = distX + wordOffsetX * wordOffset[k][0];
let wordOffsetY = offsetY >= 0 ? 1 : -1;
wordOffsetY = distY + wordOffsetY * wordOffset[k][1];
context.fillText(wordArr[k], wordOffsetX, wordOffsetY);
}
}
layerPoints[n][k] = [distX, distY];
}
}
</pre>

根據(jù)角度數(shù)來返回計算結果坐標偏移量的正負,是x軸是在左還是右,y軸是上還是下:
<pre>
let getXParam = (angle) => {
let param = 1;
if(angle >= 0 && angle < 90) {
param = 1;
} else if(angle >= 90 && angle < 180) {
param = -1;
angle = 180 - angle;
} else if(angle >= 180 && angle < 270) {
param = -1;
angle = angle - 180;
} else if(angle >= 270 && angle <= 360) {
param = 1;
angle = 360 - angle;
}

let angleCos = Math.cos(Math.PI / 180 \* angle);
if(angleCos < 0) {
    angleCos = angleCos \* -1;
}
return angleCos \* param;

};

let getYParam = (angle) => {
let param = 1;
if(angle >= 0 && angle < 90) {
param = 1;
} else if(angle >= 90 && angle < 180) {
param = 1;
angle = 180 - angle;
} else if(angle >= 180 && angle < 270) {
param = -1;
angle = angle - 180;
} else if(angle >= 270 && angle <= 360) {
param = -1;
angle = 360 - angle;
}

let angleSin = Math.sin(Math.PI / 180 \* angle);
if(angleSin < 0) {
    angleSin = angleSin \* -1;
}
return angleSin \* param;

};
</pre>

計算完網(wǎng)狀交點后,用context.moveTo把對應的點連起來就可以了,最后把數(shù)據(jù)在上面顯示,即把比例點連接成一個封閉圖形填充一個辦透明的顏色就可以了:

<pre>
// 繪制的雷達比例數(shù)據(jù)
let dataArr = [5, 2, 6, 2, 6, 0, 3];
// 繪制比例:
context.beginPath();
context.setStrokeStyle("rgba(77,168,213,0.85)");
context.setFillStyle("rgba(77,168,213,0.85)");

    let isFirstPoint = true;
    let tmpPoints = [];
    for(m = 0; m < angleNum; m++) {
        tmpPoints = centerPoint;
        if(dataArr[m] > 0) {
            for(n = 0; n < layerNum; n++) {
                if(dataArr[m] == (n + 1)) {
                    tmpPoints = layerPoints[n][m];
                    break;
                }
            }
        }
        if(isFirstPoint) {
            context.moveTo(tmpPoints[0], tmpPoints[1]);
            isFirstPoint = false;
        } else {
            context.lineTo(tmpPoints[0], tmpPoints[1]);
        }
    }

    context.fill();
    context.stroke();
    context.closePath();

</pre>

最終效果圖

微信小程序雷達圖完整代碼:
https://github.com/zhangxiongwu/smallAppRadar

結語:熟悉繪畫canvas與計算坐標點可以繪畫很多定制化需求的圖形。

您的意見是我改善的東西,歡迎評論提建議,如果對您有幫助,請點個贊,謝謝~~
菲麥前端專題,匯聚前端好文,邀您關注!

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,247評論 6 543
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,520評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,362評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,805評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,541評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,896評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,887評論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,062評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,608評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,356評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,555評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,077評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,769評論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,175評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,489評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,289評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,516評論 2 379

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,734評論 25 708
  • 當你對各種軍用雷達頭大的時候,推薦你讀讀這一篇。 艦載雷達 |型號|應用||:-:|:-:|:-:|:-:|:-:...
    好心態(tài)閱讀 7,039評論 0 7
  • 活著真好,這句話最近總出現(xiàn)在我腦子里,第一次是在一個充滿陽光的下午,在陽臺上,和家人在一起無所事事的曬太陽,雖然曬...
    正在生長閱讀 136評論 0 0
  • 學習一切,胸懷天下,我可以改變這世界!今天看到杜楠看《李光耀觀天下》,突然覺得,我也應該努力了解這世界,因為它如此...
    文字超能力閱讀 170評論 0 0
  • Zeldasvita閱讀 164評論 0 0