iOS开发波浪线图

波浪线图等同于折线图, 只是添加了填充, 另外实现波浪的效果.
实现同样自定义封装了view, 在UIBezierPath的分类中实现波浪的效果:

// .h
#import <UIKit/UIKit.h>

@interface LineChartView : UIView

@property (nonatomic, strong) NSArray *titleForYArr;

@property (nonatomic, strong) NSArray *titleForXArr;

@property (nonatomic, strong) NSArray *valueArr;

@property (nonatomic, strong) UIColor *lineColor;

- (instancetype)initWithFrame:(CGRect)frame;

- (void)startDraw;

@end
// .m
#import "LineChartView.h"
#import "UIBezierPath+category.h"

@interface LineChartView()<CAAnimationDelegate>

@property (nonatomic, strong) CAShapeLayer *lineChartLayer;

@end

@implementation LineChartView {
    CGFloat width;
    CGFloat height;
}

static CGFloat edgeLeft = 30;
static CGFloat edgeRight = 20;

static CGFloat edgeUp = 50;
static CGFloat edgeDown = 30;

- (instancetype)initWithFrame:(CGRect)frame {
    if (self = [super initWithFrame:frame]) {
        
        self.backgroundColor = [UIColor lightGrayColor];
        
        width = frame.size.width - edgeLeft - edgeRight;
        height = frame.size.height - edgeUp - edgeDown;
    }
    return self;
}
// 画出坐标轴
- (void)drawRect:(CGRect)rect {
    
    CGContextRef context = UIGraphicsGetCurrentContext();
    CGContextSetLineWidth(context, 1.0);
    CGContextSetRGBStrokeColor(context, 0.6, 0.6, 0.6, 1);
    CGContextMoveToPoint(context, edgeLeft, edgeUp);
    CGContextAddLineToPoint(context, edgeLeft, edgeUp + height);
    CGContextAddLineToPoint(context,edgeLeft + width, edgeUp + height);
    CGContextStrokePath(context);
}


#pragma mark 画折线图
- (void)dravLine{

    NSInteger yearNum = self.titleForXArr.count;
    if (yearNum <= 0) {
        return;
    }
    CGFloat widthForX = width / yearNum;
    CGFloat maxValue = [[self.titleForYArr lastObject] floatValue];
    
    if (maxValue <= 0) {
        return;
    }
    
    UIBezierPath *pathLine = [[UIBezierPath alloc] init];
    [pathLine moveToPoint:CGPointMake(edgeLeft + widthForX * 0.5, ((maxValue - [self.valueArr[0] floatValue]) / maxValue) * height + edgeUp)];
    
    // 创建折线点标记
    for (NSInteger i = 1; i < self.valueArr.count; i++) {
        
        CGPoint pointCenter = CGPointMake(edgeLeft + widthForX * (i + 0.5), (maxValue - [self.valueArr[i] floatValue]) / maxValue * height + edgeUp);
        
        [pathLine addLineToPoint:pointCenter];
    }
    // 实现波浪线
    pathLine = [pathLine smoothedPathWithGranularity:20];
    
    [pathLine addLineToPoint:CGPointMake(edgeLeft + width - widthForX * 0.5, height + edgeUp)];
    [pathLine addLineToPoint:CGPointMake(edgeLeft + widthForX * 0.5, height + edgeUp)];
    
    CAShapeLayer  *lineChartLayer = [CAShapeLayer layer];
    lineChartLayer.path = pathLine.CGPath;
    lineChartLayer.strokeColor = _lineColor.CGColor;
    lineChartLayer.fillColor = [[UIColor greenColor] CGColor];
    lineChartLayer.lineWidth = 0.0;
    lineChartLayer.lineCap = kCALineCapRound;
    lineChartLayer.lineJoin = kCALineJoinBevel;
    lineChartLayer.lineDashPhase = 5.0;
    lineChartLayer.miterLimit = 10.0;
    
    [self.layer addSublayer:lineChartLayer];
}

#pragma mark 创建x轴的数据
- (void)setTitleForXArr:(NSArray *)titleForXArr {
    _titleForXArr = titleForXArr;
    [self createLabelX];
}
- (void)createLabelX{

    NSInteger yearNum = self.titleForXArr.count;
    if (yearNum <= 0) {
        return;
    }
    CGFloat widthForX = width / yearNum;
    
    for (NSInteger i = 0; i < yearNum; i++) {
        
        UILabel *labelYear = [[UILabel alloc] initWithFrame:CGRectMake(edgeLeft + widthForX * i, height + edgeUp, widthForX, edgeDown)];
        labelYear.tag = 1000 + i;
        labelYear.text = self.titleForXArr[i];
        labelYear.font = [UIFont systemFontOfSize:14];
        labelYear.textAlignment = NSTextAlignmentCenter;
        [self addSubview:labelYear];
    }
}
#pragma mark 创建y轴数据及虚线
- (void)setTitleForYArr:(NSArray *)titleForYArr {
    _titleForYArr = titleForYArr;
    [self createLabelY];
    [self setLineDash];
}
- (void)createLabelY{

    NSInteger numForY = _titleForYArr.count;
    if (numForY <= 1) {
        return;
    }
    CGFloat widthForY = edgeLeft;
    CGFloat heightForY = height / (numForY - 1);
    
    for (NSInteger i = 0; i < numForY; i++) {
        
        UILabel *labelForY = [[UILabel alloc] initWithFrame:CGRectMake(0, edgeUp + (i - 0.5) * heightForY, widthForY, heightForY)];
        
        labelForY.tag = 2000 + i;
        labelForY.text = _titleForYArr[numForY - i - 1];
        labelForY.font = [UIFont systemFontOfSize:12];
        labelForY.textAlignment = NSTextAlignmentCenter;
        [self addSubview:labelForY];
    }
}
#pragma mark - 添加虚线
- (void)setLineDash {
    
    NSInteger numForY = _titleForYArr.count - 1;
    if (numForY <= 0) {
        return;
    }
    CGFloat heightForY = height / numForY;
    
    for (NSInteger i = 0; i < numForY; i++) {
        
        CAShapeLayer *dashLayer = [CAShapeLayer layer];
        dashLayer.strokeColor = [UIColor colorWithRed:0.6 green:0.6 blue:0.6 alpha:1].CGColor;
        dashLayer.lineWidth = 1.0;
        
        UIBezierPath *path = [[UIBezierPath alloc] init];
        path.lineWidth = 1.0;
        
        [path moveToPoint:CGPointMake(edgeLeft, edgeUp + i * heightForY)];
        [path addLineToPoint:CGPointMake(edgeLeft + width, edgeUp + i * heightForY)];
        CGFloat dash[] = {10,10};
        [path setLineDash:dash count:2 phase:10];
        [path stroke];
        dashLayer.path = path.CGPath;
        [self.layer addSublayer:dashLayer];
    }
}
#pragma mark - 开始画折线
- (void)startDraw {
    [self dravLine];
}

@end
// 分类.h
#import <UIKit/UIKit.h>

@interface UIBezierPath (category)

- (UIBezierPath *)smoothedPathWithGranularity:(NSInteger)granularity;

@end
// 分类.m
#import "UIBezierPath+category.h"

@implementation UIBezierPath (category)

void getPointsFromBezier(void * info, const CGPathElement * element){
    NSMutableArray *bezierPoints = (__bridge NSMutableArray *)info;
    CGPathElementType type = element->type;
    CGPoint * points = element->points;
    
    if (type != kCGPathElementCloseSubpath) {
        [bezierPoints addObject:[NSValue valueWithCGPoint:points[0]]];
        if ((type != kCGPathElementAddLineToPoint) && (type != kCGPathElementMoveToPoint)) {
            [bezierPoints addObject:[NSValue valueWithCGPoint:points[1]]];
        }
    }
    if (type == kCGPathElementAddCurveToPoint) {
        [bezierPoints addObject:[NSValue valueWithCGPoint:points[2]]];
    }
}

- (NSMutableArray *)pointsFromBezierPath:(UIBezierPath *)bezierPath{
    NSMutableArray * points = [NSMutableArray array];
    CGPathApply(bezierPath.CGPath, (__bridge void *)points, getPointsFromBezier);
    return points;
}

- (UIBezierPath *)smoothedPathWithGranularity:(NSInteger)granularity{
    NSMutableArray * points = [NSMutableArray arrayWithArray:[self pointsFromBezierPath:self]];

    if (points.count < 4) {
        return [self copy];
    }
    
    [points insertObject:[points objectAtIndex:0] atIndex:0];
    [points addObject:[points lastObject]];
    
    UIBezierPath * smoothedPath = [self copy];
    [smoothedPath removeAllPoints];
    [smoothedPath moveToPoint:[(NSValue *)[points objectAtIndex:0] CGPointValue]];
    
    for (NSUInteger i = 1; i < points.count - 2; i++) {
        CGPoint p0 = [(NSValue *)[points objectAtIndex:i-1] CGPointValue];
        CGPoint p1 = [(NSValue *)[points objectAtIndex:i] CGPointValue];
        CGPoint p2 = [(NSValue *)[points objectAtIndex:i+1] CGPointValue];
        CGPoint p3 = [(NSValue *)[points objectAtIndex:i+2] CGPointValue];
        
        for (int i = 1; i < granularity; i++)
        {
            float t = (float) i * (1.0f / (float) granularity);
            float tt = t * t;
            float ttt = tt * t;
            
            CGPoint pi;
            pi.x = 0.5 * (2*p1.x+(p2.x-p0.x)*t + (2*p0.x-5*p1.x+4*p2.x-p3.x)*tt + (3*p1.x-p0.x-3*p2.x+p3.x)*ttt);
            pi.y = 0.5 * (2*p1.y+(p2.y-p0.y)*t + (2*p0.y-5*p1.y+4*p2.y-p3.y)*tt + (3*p1.y-p0.y-3*p2.y+p3.y)*ttt);
            [smoothedPath addLineToPoint:pi];
        }
        
        [smoothedPath addLineToPoint:p2];
    }
    
    [smoothedPath addLineToPoint:[(NSValue *)[points objectAtIndex:points.count - 1] CGPointValue]];
    return smoothedPath;
}

@end

// 使用
- (void)viewDidLoad {
    [super viewDidLoad];
    
    LineChartView *lineView = [[LineChartView alloc] initWithFrame:CGRectMake(0, 100, self.view.bounds.size.width, 190)];
    lineView.titleForYArr = @[@"0",@"1",@"2",@"3",@"4",@"5"];
    lineView.titleForXArr = @[@"2013年",@"2014年",@"2015年",@"2016年",@"2017年"];
    lineView.valueArr = @[@"0.8",@"1.9",@"4.0",@"1.3",@"2.5"];
    
    lineView.lineColor = [UIColor redColor];
    [self.view addSubview:lineView];
    
    [lineView startDraw];
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,444评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,019评论 4 62
  • 1.不存在变量提升 let & const 2.出现块级作用域 之前只有全局作用域和函数作用域
    国之大殇阅读 92评论 0 0
  • 下班回家比较早,换上运动服饰,准备找个场地跑步。小区的篮球场地,十几个中学生正在打篮球,健身器材又被一群老太太霸占...
    冷月花魂烘焙阅读 1,178评论 1 12