簡(jiǎn)單iOS折線圖

先看效果圖

line.gif

所用框架

CAShapeLayer與UIBezierPath配合畫線鹤树。畫線的兩種方法:

  • DrawRect:DrawRect屬于CoreGraphic框架,占用CPU椰拒,消耗性能大
  • CAShapeLayer:CAShapeLayer屬于CoreAnimation框架晶渠,通過GPU來渲染圖形,節(jié)省性能燃观。動(dòng)畫渲染直接提交給手機(jī)GPU褒脯,不消耗內(nèi)存

實(shí)現(xiàn)方法

  1. 初始化存放x軸的view及y軸的view
  2. 隨機(jī)產(chǎn)生x軸數(shù)據(jù)及y軸數(shù)據(jù)并賦值
NSMutableArray *xArray = [NSMutableArray new];  
NSMutableArray *yArray = [NSMutableArray new];
int x = 10+arc4random()%5;
    NSLog(@"x:%d",x);
    for (int i=1; i<x; i++) {
        [xArray addObject:[NSString stringWithFormat:@"%d",i]];
        int y1 = (arc4random()%10)-5;
        float y2 = (arc4random()%100) / 100.0;
        float y = (float)(y1+y2);
        NSLog(@"y1:%d y2:%f",y1,y2);
        [yArray addObject:[NSString stringWithFormat:@"%.2f",y]];
    }
 [self.lineChart setXvalue:xArray];
 [self.lineChart setDataArray:yArray];

3.在對(duì)應(yīng)的軸線上添加組件賦值,折線的實(shí)現(xiàn)首先對(duì)獲取到的y軸數(shù)據(jù)做處理
?3.1 取出最大值最小值

NSMutableArray *dataArray = [NSMutableArray new];
CGFloat max = [[yArray valueForKeyPath:@"@max.floatValue"] floatValue];
    _maxValue = max;
CGFloat min = [[yArray valueForKeyPath:@"@min.floatValue"] floatValue];
    _minValue = min;
if (max>=0) {
        _maxValue = max * 1.4;
}else {
        _maxValue = max * 0.6;
}
if (min>=0) {
        _minValue = min * 0.6;
}else {
        _minValue = min * 1.4;
}
CGFloat ava = (_maxValue-_minValue)/3;
[dataArray addObject:[NSString stringWithFormat:@"%.2f%%",_maxValue]];
[dataArray addObject:[NSString stringWithFormat:@"%.2f%%",(_maxValue-ava)]];
[dataArray addObject:[NSString stringWithFormat:@"%.2f%%",(_minValue+ava)]];
[dataArray addObject:[NSString stringWithFormat:@"%.2f%%",_minValue]];
return dataArray;

?3.2 畫背景線

CAShapeLayer *bgLayer = [CAShapeLayer layer];
bgLayer.strokeColor = self.bgLineColor.CGColor;
UIBezierPath *linePath = [UIBezierPath bezierPath];
linePath.lineWidth = 2.0;
//背景線
[yValue enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger i, BOOL * _Nonnull stop) {
    [linePath moveToPoint:CGPointMake(0 , self.lineHeight / (yValue.count - 1) * i)];
    [linePath addLineToPoint:CGPointMake(self.frame.size.width-self.yWidth ,self.lineHeight / (yValue.count - 1) * i)];
}];
bgLayer.path = linePath.CGPath;
[self.layer addSublayer:bgLayer];

?3.3 畫折線

///折線圖
_lineLayer.lineWidth = 1.0;
_lineLayer.strokeColor = COLORHEX(0x24d8fd).CGColor;
_lineLayer.fillColor = [UIColor clearColor].CGColor;
_fillLayer.fillColor = [COLORHEX(0x24d8fd) colorWithAlphaComponent:0.2].CGColor;
UIBezierPath *linePath = [UIBezierPath bezierPath];
UIBezierPath *fillpath = [UIBezierPath bezierPath];
//值差與高度的比例
CGFloat unitValue = (_maxValue - _minValue)/self.lineHeight;
//平均寬度
CGFloat unitWidth = (self.frame.size.width-self.yWidth)/(dataArray.count-1);
NSInteger startY = ABS(self.lineHeight - ([dataArray[0] floatValue]-self.minValue) / unitValue);
[linePath moveToPoint:CGPointMake(0, startY)];
[fillpath moveToPoint:CGPointMake(0, startY)];
[dataArray enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
    CGFloat xPosition = unitWidth * idx;
    CGFloat yPosition;
    if (unitValue == 0.00) {
        yPosition = 0.00;
    }else {
        yPosition = ABS(self.lineHeight - ([obj floatValue]-self.minValue) / unitValue);
    }    
    [linePath addLineToPoint:CGPointMake(xPosition, yPosition)];
    [fillpath addLineToPoint:CGPointMake(xPosition, yPosition)];
    //在最后一組數(shù)據(jù)時(shí)給填充路徑形成一個(gè)閉包
    if (idx == dataArray.count-1) {
        [fillpath addLineToPoint:CGPointMake(self.frame.size.width-self.yWidth, self.lineHeight)];
        [fillpath addLineToPoint:CGPointMake(0, self.lineHeight)];
        [fillpath moveToPoint:CGPointMake(0, startY)];
    }
}];

?3.4 動(dòng)畫效果

CABasicAnimation *ani = [ CABasicAnimation animationWithKeyPath : NSStringFromSelector ( @selector (strokeEnd))];
ani.fromValue = @0;
ani.toValue = @1;
ani.duration = 2.0;
[_lineLayer addAnimation:ani forKey:NSStringFromSelector(@selector(strokeEnd))];
    __block typeof(&*self) ws = self;
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
    [ws.layer addSublayer:ws.fillLayer];
});

demo地址

https://github.com/RedBluePeas/line_test

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末缆毁,一起剝皮案震驚了整個(gè)濱河市番川,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌脊框,老刑警劉巖颁督,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異浇雹,居然都是意外死亡沉御,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門昭灵,熙熙樓的掌柜王于貴愁眉苦臉地迎上來吠裆,“玉大人,你說我怎么就攤上這事烂完∈愿恚” “怎么了?”我有些...
    開封第一講書人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵抠蚣,是天一觀的道長(zhǎng)祝旷。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么怀跛? 我笑而不...
    開封第一講書人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任奇昙,我火速辦了婚禮,結(jié)果婚禮上敌完,老公的妹妹穿的比我還像新娘储耐。我一直安慰自己,他們只是感情好滨溉,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開白布什湘。 她就那樣靜靜地躺著,像睡著了一般晦攒。 火紅的嫁衣襯著肌膚如雪闽撤。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,292評(píng)論 1 301
  • 那天脯颜,我揣著相機(jī)與錄音哟旗,去河邊找鬼。 笑死栋操,一個(gè)胖子當(dāng)著我的面吹牛闸餐,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播矾芙,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼舍沙,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了剔宪?” 一聲冷哼從身側(cè)響起拂铡,我...
    開封第一講書人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎葱绒,沒想到半個(gè)月后感帅,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡地淀,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年失球,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片骚秦。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡她倘,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出作箍,到底是詐尸還是另有隱情,我是刑警寧澤前硫,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布胞得,位于F島的核電站,受9級(jí)特大地震影響屹电,放射性物質(zhì)發(fā)生泄漏阶剑。R本人自食惡果不足惜跃巡,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望牧愁。 院中可真熱鬧素邪,春花似錦、人聲如沸猪半。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)磨确。三九已至沽甥,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間乏奥,已是汗流浹背摆舟。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留邓了,地道東北人恨诱。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像骗炉,于是被迫代替她去往敵國(guó)和親胡野。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354

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

  • 1痕鳍、通過CocoaPods安裝項(xiàng)目名稱項(xiàng)目信息 AFNetworking網(wǎng)絡(luò)請(qǐng)求組件 FMDB本地?cái)?shù)據(jù)庫(kù)組件 SD...
    陽(yáng)明先生_X自主閱讀 15,980評(píng)論 3 119
  • 1 CALayer IOS SDK詳解之CALayer(一) http://doc.okbase.net/Hell...
    Kevin_Junbaozi閱讀 5,150評(píng)論 3 23
  • 人在硫豆,門外。 心想笼呆,門內(nèi)熊响。 出門,在外诗赌。 內(nèi)心汗茄,想人。
    將覺者閱讀 210評(píng)論 4 4
  • 不錯(cuò)
    不知道怎么取閱讀 253評(píng)論 0 0
  • 年輕時(shí)铭若,她什么都怕洪碳;到中年時(shí),她什么都不怕叼屠。一個(gè)女人瞳腌,看到房間里有蛇,有慌有怕有無奈镜雨,慌的是里房竟有蛇而且不知何時(shí)...
    Ronvalyn閱讀 174評(píng)論 0 0