iOS 绘图-UIBezierPath

UIBezierPath主要用来绘制矢量图形,它是基于Core Graphics对CGPathRef数据类型和path绘图属性的一个封装。使用UIBezierPath可以定义简单的形状。


1、UIBezierPath的属性

1. CGPath:将UIBezierPath类转换成CGPath,类似于UIColor的CGColor
2. currentPoint:当前path的位置,可以理解为path的终点
3. lineWidth:path宽度
4. lineCapStyle:path端点样式枚举,有3种样式
    kCGLineCapButt,     // 无端点      
    kCGLineCapRound,    // 圆形端点  
    kCGLineCapSquare    // 方形端点(样式上和kCGLineCapButt是一样的,但是比kCGLineCapButt长一点)
5.  lineJoinStyle:拐角样式枚举,有3种样式
    kCGLineJoinMiter,    // 尖角
    kCGLineJoinRound,    // 圆角
    kCGLineJoinBevel     // 缺角
6.miterLimit:最大斜接长度(只有在使用kCGLineJoinMiter是才有效),边角的角度越小,斜接长度就会越大
  为了避免斜接长度过长,使用lineLimit属性限制
  如果斜接长度超过miterLimit,边角就会以KCALineJoinBevel类型来显示


2、UIBezierPath与Quartz2D

即UIBezierPath在drawRect中的使用,在实际中,我们多使用UIBezierPath来自定义视图。

矩形
- (void)drawRect:(CGRect)rect {
   
    CGFloat lineWidth = 3;
    
    //矩形(第一种方式)
    UIBezierPath *rectPath = [UIBezierPath bezierPathWithRect:CGRectMake(10, 10, 50,50 )];
    rectPath.lineWidth = lineWidth*2;
    rectPath.lineCapStyle = kCGLineCapRound;
    rectPath.lineJoinStyle = kCGLineJoinRound;
    //闭合路径
    [rectPath closePath];
    [[UIColor greenColor] set];
    //绘制边缘
    [rectPath stroke];
    
    //矩形(第二种方式)
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    UIBezierPath *pathRect = [UIBezierPath bezierPathWithRect:CGRectMake(70, 10, 50,50 )];
    CGContextAddPath(ctx, pathRect.CGPath);
    [[UIColor purpleColor] set];
    CGContextFillPath(ctx);

}
圆/弧
- (void)drawRect:(CGRect)rect {
   
    CGFloat lineWidth = 3;
    
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    //矩形圆
    UIBezierPath *roundPath = [UIBezierPath bezierPathWithRoundedRect:CGRectMake(130, 10, 50, 50) cornerRadius:10];
    CGContextAddPath(ctx, roundPath.CGPath);
    CGContextSetLineWidth(ctx, lineWidth);
    [[UIColor orangeColor] set];
    CGContextStrokePath(ctx);
    
    //实心圆
    UIBezierPath *arcPath = [UIBezierPath bezierPathWithArcCenter:CGPointMake(35, 95) radius:25 startAngle:0 endAngle:M_PI*2 clockwise:NO];
    [arcPath addLineToPoint:CGPointMake(35, 95)];
    [arcPath closePath];
    [[UIColor brownColor] set];
    [arcPath fill];
    CGContextSetRGBStrokeColor(ctx, 1, 0, 0, 1);
    //空心圆
    /*
     *  clockwise 1为顺时针  0为逆时针
     *  顺时针的理解:指定1创建顺时针弧或0创建逆时针弧(startAngle为0时为逆时针弧,非0为顺时针弧)
     */
    UIBezierPath *arcKPath = [UIBezierPath bezierPathWithArcCenter:CGPointMake(155, 95) radius:25 startAngle:M_PI_2 endAngle:M_PI clockwise:NO];
    CGContextAddPath(ctx, arcKPath.CGPath);
    CGContextSetLineWidth(ctx, lineWidth);
    [[UIColor lightGrayColor] set];
    CGContextStrokePath(ctx);
    
    float angle = 0;
    CGPoint center = CGPointMake(150, 160);
    for (int index = 1; index < 5; index++) {
        
        CGFloat startAngle = angle;
        CGFloat endAngle = index*M_PI_2;
        //实心圆
        UIBezierPath *arcPath = [UIBezierPath bezierPathWithArcCenter:center radius:30 startAngle:startAngle endAngle:endAngle clockwise:YES];
        [arcPath addLineToPoint:center];
        [arcPath closePath];
        [[self randColor] set];
        [arcPath fill];
        
        angle = endAngle;
    }
    
}

- (UIColor *)randColor
{
    //rand(),就返回一个类型为int的整数,其范围是0到RAND_MAX
    float red = rand() / (float)RAND_MAX;
    float green = rand() / (float)RAND_MAX;
    float blue = rand() / (float)RAND_MAX;
    
    return [UIColor colorWithRed:red green:green blue:blue alpha:1];
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    [self setNeedsDisplay];
}

线条
- (void)drawRect:(CGRect)rect {
   
    CGFloat lineWidth = 3;
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    //线条
    UIBezierPath *linePath = [UIBezierPath bezierPath];
    [linePath moveToPoint:CGPointMake(10, 130)];
    [linePath addLineToPoint:CGPointMake(60, 190)];
    [linePath moveToPoint:CGPointMake(60, 130)];
    [linePath addLineToPoint:CGPointMake(10, 190)];
    CGContextAddPath(ctx, linePath.CGPath);
    CGContextSetLineWidth(ctx, lineWidth*3);
    CGContextSetLineCap(ctx, kCGLineCapSquare);
    CGContextSetLineJoin(ctx, kCGLineJoinBevel);
    [[UIColor blackColor] set];
    CGContextStrokePath(ctx);
}

效果

UIBezierPath.png
曲线
塞尔曲线原理.png
- (void)drawRect:(CGRect)rect {
    
    //1.获取跟View相关联的上下文.】
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    //2.描述路径
    UIBezierPath *path = [UIBezierPath  bezierPath];
    //画曲线,设置起点.还有一个控制点(用来控制曲线的方向跟弯曲程度)
    //设置起点
    //A: CGPointMake(10, 150)
    [path moveToPoint:CGPointMake(10, 150)];
    //添加一要曲线到某个点
    //C:CGPointMake(200, 150)    B:CGPointMake(150, 10)
    [path addQuadCurveToPoint:CGPointMake(200, 150) controlPoint:CGPointMake(150, 10)];
    //3.把路径添加到上下文当中
    CGContextAddPath(ctx, path.CGPath);
    //4.把上下文的内容渲染View上
    CGContextStrokePath(ctx);
    
}

2、UIBezierPath与CAShapeLayer

UIBezierPath与CAShapeLayer的结合,是我们开发中最常见的,功能很强大。

CAShapeLayer与UIBezierPath的关系

CAShapeLayer中shape代表形状的意思,所以需要形状才能生效
贝塞尔曲线可以创建基于矢量的路径,而UIBezierPath类是对CGPathRef的封装
贝塞尔曲线给CAShapeLayer提供路径,CAShapeLayer在提供的路径中进行渲染。路径会闭环,所以绘制出了Shape
用于CAShapeLayer的贝塞尔曲线作为path,其path是一个首尾相接的闭环的曲线,即使该贝塞尔曲线不是一个闭环的曲线

例如设置圆角:

- (void)setFillet
{
    UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(100, 100, 100, 100)];
    imageView.image = [UIImage imageNamed:@"test"];
    UIBezierPath *maskPath = [UIBezierPath bezierPathWithRoundedRect:imageView.bounds byRoundingCorners:UIRectCornerAllCorners cornerRadii:imageView.bounds.size];

    CAShapeLayer *maskLayer = [[CAShapeLayer alloc]init];
    //设置大小
    maskLayer.frame = imageView.bounds;
    //设置图形样子
    maskLayer.path = maskPath.CGPath;
    imageView.layer.mask = maskLayer;
}

自定义视图

- (void)addCustomizeView
{
    CGFloat H = self.view.frame.size.height;
    CGFloat W = self.view.frame.size.width;
    CGFloat layerHeight = self.view.frame.size.height * 0.1;
    
    UIBezierPath *bezier = [UIBezierPath bezierPath];
    [bezier moveToPoint:CGPointMake(0, H-layerHeight)];
    [bezier addLineToPoint:(CGPointMake(0, H - 1))];
    [bezier addLineToPoint:(CGPointMake(W, H - 1))];
    [bezier addLineToPoint:CGPointMake(W, H-layerHeight)];
    [bezier addQuadCurveToPoint:CGPointMake(0, H-layerHeight) controlPoint:CGPointMake(W/2, H-layerHeight-40)];
    
    CAShapeLayer *shapeLayer = [CAShapeLayer layer];
    shapeLayer.lineWidth = 3;
    shapeLayer.path = bezier.CGPath;
    shapeLayer.fillColor = [UIColor grayColor].CGColor;
    shapeLayer.strokeColor = [UIColor greenColor].CGColor;
    [self.view.layer addSublayer:shapeLayer];
 }

效果

customizeView.png

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,029评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,395评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,570评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,535评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,650评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,850评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,006评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,747评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,207评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,536评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,683评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,342评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,964评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,772评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,004评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,401评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,566评论 2 349

推荐阅读更多精彩内容