iOS - 從零開始繪制一個帶動畫的多邊形能力分布圖

能力分布圖

前言

能力分布圖是一個統(tǒng)計類經(jīng)常需要的一個需求专缠,需要一定的幾何計算,實(shí)現(xiàn)難度適中淑仆,在平時也很適合拿來用作iOS繪圖練手涝婉。
這里寫了一個具有一定可復(fù)用行比較高的多邊形能力分布圖,有需要的同學(xué)可以拿去用糯景,也可以當(dāng)做自己的練手Demo嘁圈。相關(guān)代碼我都放在我的Github-AWPolygonView上蟀淮,后續(xù)會有優(yōu)化更新最住,有什么問題歡迎大家提出想法,互相學(xué)習(xí)進(jìn)步怠惶。

正文

一涨缚、計算各個點(diǎn)的坐標(biāo)

畫圖一個核心步驟是計算需要的點(diǎn)的坐標(biāo)。需要的點(diǎn)的類型為:中心點(diǎn)、邊上的角脓魏、能力值的各個點(diǎn)兰吟。接下來開始計算各個點(diǎn)的坐標(biāo)。

1茂翔、中心的點(diǎn)

x坐標(biāo)就是View的寬度的一半
y坐標(biāo)就是View的高度的一半

    _centerX = self.bounds.size.width/2;
    _centerY = self.bounds.size.height/2;
2混蔼、邊上的角度

根據(jù)需要是幾邊形,通過在一個以View的中心點(diǎn)為圓心珊燎,view的寬度為直徑的圓上取點(diǎn)惭嚣。
然后通過角度和半徑 進(jìn)行計算獲得各個點(diǎn)的坐標(biāo)
cossin方法參數(shù)需要為弧度,將其轉(zhuǎn)換成傳入?yún)?shù)為角度的ANGLE_COS方法
公式為:

1弧度=180/π度
1度=π/180弧度

代碼就是:

#define ANGLE_COS(Angle) cos(M_PI / 180 * (Angle))
#define ANGLE_SIN(Angle) sin(M_PI / 180 * (Angle))

接著通過一個循環(huán)計算每個點(diǎn)的坐標(biāo)悔政。
點(diǎn)的坐標(biāo)計算代碼如下:

//self.radius 為半徑
//self.sideNum為總共多少個邊
//i為逆時針第i個點(diǎn)

CGPoint cornerPoint = CGPointMake(
_centerX - ANGLE_COS(90.0 - 360.0 /self.sideNum * i) * self.radius,
_centerY - ANGLE_SIN(90.0 - 360.0 /self.sideNum * i) * self.radius);

3晚吞、能力值點(diǎn)的坐標(biāo)

能力值的點(diǎn)不想邊上的點(diǎn)那么有規(guī)律,半徑長短不一谋国,但是角度跟邊上的點(diǎn)的角度是一樣的槽地,所以只要在邊上的點(diǎn)的基礎(chǔ)上乘上半徑長短比例就可以了。
代碼如下:

    //Values
    for (int i = 0; i < self.sideNum; i++) {
        if (self.values.count > i) {
            CGFloat valueRadius = [self.values[i] floatValue] * self.radius;
            CGPoint valuePoint =  CGPointMake(_centerX - ANGLE_COS(90.0 - 360.0 /self.sideNum * i) * valueRadius,
                                              _centerY - ANGLE_SIN(90.0 - 360.0 /self.sideNum * i) * valueRadius);
            [tempValuePoints addObject:[NSValue valueWithCGPoint:valuePoint]];
        }
    }

二芦瘾、進(jìn)行連線

1捌蚊、采用CoreGraphics方法

各個點(diǎn)的坐標(biāo)確定好了,接下來就是連線了旅急,可以采用CoreGraphics中的context繪制直線方法:

    CGContextMoveToPoint(CGContextRef  _Nullable c, CGFloat x, CGFloat y)
    CGContextAddLineToPoint(CGContextRef  _Nullable c, CGFloat x, CGFloat y)
    

這樣可以直接劃線比較簡單逢勾,但是只滿足不需要動畫的需求。如果需要進(jìn)行類似Stroke動畫變需要使用貝塞爾曲線

2藐吮、使用貝塞爾曲線與CAShaperLayer結(jié)合

初始化一個UIBezierPath

- (UIBezierPath *)bezierPath {
    if (!_bezierPath) {
        _bezierPath = [UIBezierPath bezierPath];
    }
    return _bezierPath;
}

通過這個bezierPath進(jìn)行畫路徑

[_bezierPath moveToPoint:firstPoint];
[_bezierPath addLineToPoint:point];

將貝塞爾曲線路徑賦值給CAShapeLayer的路徑

    self.shapeLayer.path = self.bezierPath.CGPath;

此時已經(jīng)劃出需要的圖案,接著是執(zhí)行動畫

- (void)addStrokeEndAnimationToLayer:(CAShapeLayer *)layer {
    CABasicAnimation *animation = [CABasicAnimation animationWithKeyPath:@"strokeEnd"];
    animation.fromValue = @0;
    animation.toValue = @1;
    animation.duration = self.animationDuration;
    [layer addAnimation:animation forKey:@"stokeEndAnimation"];
}
//調(diào)用動畫
[self addStrokeEndAnimationToLayer:self.shapeLayer];

三逃贝、展示效果

接下來是展示動畫效果:

animation.gif

相關(guān)代碼我都放在我的Github-AWPolygonView上谣辞,后續(xù)會有優(yōu)化更新,有什么問題歡迎大家提出想法沐扳,互相學(xué)習(xí)進(jìn)步泥从。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市沪摄,隨后出現(xiàn)的幾起案子躯嫉,更是在濱河造成了極大的恐慌,老刑警劉巖杨拐,帶你破解...
    沈念sama閱讀 211,884評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件祈餐,死亡現(xiàn)場離奇詭異,居然都是意外死亡哄陶,警方通過查閱死者的電腦和手機(jī)帆阳,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,347評論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來屋吨,“玉大人蜒谤,你說我怎么就攤上這事山宾。” “怎么了鳍徽?”我有些...
    開封第一講書人閱讀 157,435評論 0 348
  • 文/不壞的土叔 我叫張陵资锰,是天一觀的道長。 經(jīng)常有香客問我阶祭,道長绷杜,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,509評論 1 284
  • 正文 為了忘掉前任胖翰,我火速辦了婚禮接剩,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘萨咳。我一直安慰自己懊缺,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,611評論 6 386
  • 文/花漫 我一把揭開白布培他。 她就那樣靜靜地躺著鹃两,像睡著了一般。 火紅的嫁衣襯著肌膚如雪舀凛。 梳的紋絲不亂的頭發(fā)上俊扳,一...
    開封第一講書人閱讀 49,837評論 1 290
  • 那天,我揣著相機(jī)與錄音猛遍,去河邊找鬼馋记。 笑死,一個胖子當(dāng)著我的面吹牛懊烤,可吹牛的內(nèi)容都是我干的梯醒。 我是一名探鬼主播,決...
    沈念sama閱讀 38,987評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼腌紧,長吁一口氣:“原來是場噩夢啊……” “哼茸习!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起壁肋,我...
    開封第一講書人閱讀 37,730評論 0 267
  • 序言:老撾萬榮一對情侶失蹤号胚,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后浸遗,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體猫胁,經(jīng)...
    沈念sama閱讀 44,194評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,525評論 2 327
  • 正文 我和宋清朗相戀三年乙帮,在試婚紗的時候發(fā)現(xiàn)自己被綠了杜漠。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,664評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖驾茴,靈堂內(nèi)的尸體忽然破棺而出盼樟,到底是詐尸還是另有隱情,我是刑警寧澤锈至,帶...
    沈念sama閱讀 34,334評論 4 330
  • 正文 年R本政府宣布晨缴,位于F島的核電站,受9級特大地震影響峡捡,放射性物質(zhì)發(fā)生泄漏击碗。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,944評論 3 313
  • 文/蒙蒙 一们拙、第九天 我趴在偏房一處隱蔽的房頂上張望稍途。 院中可真熱鬧,春花似錦砚婆、人聲如沸械拍。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,764評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽坷虑。三九已至,卻和暖如春埂奈,著一層夾襖步出監(jiān)牢的瞬間迄损,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,997評論 1 266
  • 我被黑心中介騙來泰國打工账磺, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留芹敌,地道東北人。 一個月前我還...
    沈念sama閱讀 46,389評論 2 360
  • 正文 我出身青樓垮抗,卻偏偏與公主長得像党窜,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子借宵,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,554評論 2 349

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

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,790評論 25 707
  • 在iOS中隨處都可以看到絢麗的動畫效果,實(shí)現(xiàn)這些動畫的過程并不復(fù)雜矾削,今天將帶大家一窺ios動畫全貌壤玫。在這里你可以看...
    每天刷兩次牙閱讀 8,471評論 6 30
  • 在iOS中隨處都可以看到絢麗的動畫效果,實(shí)現(xiàn)這些動畫的過程并不復(fù)雜哼凯,今天將帶大家一窺iOS動畫全貌欲间。在這里你可以看...
    F麥子閱讀 5,104評論 5 13
  • 早春 綠草絨絨細(xì)嫩柔, 無邊春色誘清眸断部。 桃花不待春雷響猎贴, 早送紅霞入蕙樓。 春到邊城 雨潤桃紅碧水悠, 邊城春到...
    繁花落盡深眸閱讀 851評論 15 9
  • 可能是因?yàn)榕嗽诜质种蠖紩o師自通的變成哲學(xué)家她渴。我無意義的嘮叨和‘感悟’有點(diǎn)多达址,頗有不吐不快的沖動。 讓我稍微提...
    陸小冉閱讀 1,420評論 0 0