iOS开发-贝塞尔曲线(UIBezierPath)和CGContextRef代码画图相关

此文章单方面对 贝塞尔曲线画图 核心动画 图层相关 方面做下整理,方便查看!

概念简介

(贝塞尔曲线)是应用于二维图形应用程序的数学曲线,本来有一阶二阶三阶......,但是iOS系统为我们简化了这些,通过简单的接口就可以画出平时常见的线段
下面我们来讲一讲UIBezierPath, UIBezierPath是对Core Graphics框架的一个封装,通过简单的调用借口可以画出常见的直线、矩形、椭圆、圆、圆弧、常见曲线......

UIBezierPath.h
///实例化一个新的Path
+ (instancetype)bezierPath;
///根据rect创建一个矩形路径Path(画笔轨迹:以rect的origin为起点,顺时针方向一周)
+ (instancetype)bezierPathWithRect:(CGRect)rect;
///根据rect创建一个椭圆或者圆(画笔轨迹:0->2π,顺时针一周,作为动画路径需要注意下)
+ (instancetype)bezierPathWithOvalInRect:(CGRect)rect;
画圆的位置和角度(顺时针).png
///创建一个圆角矩形,半径最大为矩形最小边长的一半(画笔轨迹:以rect的origin为起点,顺时针方向一周)
+ (instancetype)bezierPathWithRoundedRect:(CGRect)rect cornerRadius:(CGFloat)cornerRadius
圆角矩形.png
/**
  * @brief 创建一个矩形路径,某一个角为圆角(画笔轨迹:顺时针方向)
  * @param rect         矩形路径的 Frame
  * @param corners      UIRectCorner 枚举类型, 指定矩形的哪个角变为圆角
  * @param cornerRadii  矩形的圆角半径
  */
+ (instancetype)bezierPathWithRoundedRect:(CGRect)rect 
                        byRoundingCorners:(UIRectCorner)corners 
                              cornerRadii:(CGSize)cornerRadii;
某一个角为圆角png
/**
  *  @brief  创建圆或者圆弧路径(优势可以控制其实位置和画笔方向)
  *  @param  center      中心点
  *  @param  radius      半径
  *  @param  startAngle  开始位置(参照上面图:画圆的位置和角度(顺时针).png)
  *  @param  endAngle    结束位置(参照上面图:画圆的位置和角度(顺时针).png)
  *  @param  clockwise   是否顺时针
  */
+ (instancetype)bezierPathWithArcCenter:(CGPoint)center 
                                 radius:(CGFloat)radius 
                             startAngle:(CGFloat)startAngle 
                               endAngle:(CGFloat)endAngle  
                              clockwise:(BOOL)clockwise
///通过一个CGPathRef创建一个UIBezierPath对象
+ (instancetype)bezierPathWithCGPath:(CGPathRef)CGPath

///根据当前的UIBezierPath对象返回一个CGPathRef对象
@property(nonatomic) CGPathRef CGPath

用处:在CAShapeLayer上画图,需要的CGPathRef对象
/**
  *  @brief  将当前的path的currentPoint移到point这个点做为开始点
  *  如果当前有正在绘制的子路径, 该方法则会隐式的结束当前路径(closePath)
  *  对于大多数构造路径相关的方法而言, 在你绘制直线或曲线之前, 需要先调用这个方法
  */
- (void)moveToPoint:(CGPoint)point
/**
  *  @brief 该方法将会从currentPoint 到point链接一条直线
  *  注:在追加完这条直线后,该方法将会更新currentPoint为该点
  */
- (void)addLineToPoint:(CGPoint)point
/**
  * @brief  从 currentPoint到指定的endPoint追加一条二次贝塞尔曲线

  * 注意:调用该方法前,你要先设置currentPoint
        当添加完贝塞尔曲线后,该方法将会自动更新currentPoint为指定的结束点
  * @param  endPoint      终点
  * @param  controlPoint  控制点
  */
- (void)addQuadCurveToPoint:(CGPoint)endPoint 
               controlPoint:(CGPoint)controlPoint
二次贝塞尔曲线.png
/**
  *  @brief 从currenPoint到指定的endPoint追加一条三次贝塞尔曲线

  *  注意:调用该方法前,你要先设置currentPoint
         当添加完贝塞尔曲线后,该方法将会自动更新currentPoint为指定的结束点
  *  @param  endPoint        终点
  *  @param  controlPoint1   控制点1
  *  @param  controlPoint2   控制点2
  */
- (void)addCurveToPoint:(CGPoint)endPoint 
          controlPoint1:(CGPoint)controlPoint1 
          controlPoint2:(CGPoint)controlPoint2
三次贝塞尔曲线.png
///闭合路径
- (void)closePath
///删除当前路径中所有的点
- (void)removeAllPoints
///追加一段路劲
- (void)appendPath:(UIBezierPath *)bezierPath
///绘图路径中当前的起点
@property(nonatomic,readonly) CGPoint currentPoint
///绘图的线宽
@property(nonatomic) CGFloat lineWidth
///path线的首尾边幅样式
@property(nonatomic) CGLineCap lineCapStyle
///曲线连接点样式
@property(nonatomic) CGLineJoin lineJoinStyle
///内角和外角距离,只有当连接点样式为 kCGLineJoinMiter时才会生效,最大限制为10
@property(nonatomic) CGFloat miterLimit

/**
  *  @brief  渲染精度 默认值为 0.6
  *  该属性的值用来测量真实曲线的点和渲染曲线的点的最大允许距离
     值越小,渲染精度越高,会产生相对更平滑的曲线,但是需要花费更多的计算时间
     值越大导致则会降低渲染精度,这会使得渲染的更迅速

  *  注意:我们需要渲染速度的时候,可以适当增大该值
  */
@property(nonatomic) CGFloat flatness
/**
  * @param pattern  该属性是一个C语言的数组,其中每一个元素都是 CGFloat
  *  数组中的元素代表着线段每一部分的长度,第一个元素代表线段的第一条线,第二个元素代表线段中的第一个间隙.
     这个数组中的值是轮流的,举个例子:
     声明一个数组 CGFloat pattern[] = @{3.0, 1.0, 4.0, 1.5}; 
     实际画出来的虚线第一段长度为3.0,间隔1.0,第二段长度为4.0,间隔为1.5 
     第三段重头再来一遍,依次类推

  * @param count  这个参数是 pattern 数组的个数
  * @param phase  这个参数代表着,虚线从哪里开始绘制
     举个例子:如果pattern[] = @{3.0, 1.0},我们想让以虚线间隔开头,可以这是phase为3.0
  */
- (void)setLineDash:(nullable const CGFloat *)pattern 
              count:(NSInteger)count 
              phase:(CGFloat)phase
///填充路径包裹的部分颜色,没有闭合会隐式闭合填充
- (void)fill
///填充画笔路径颜色
- (void)stroke

先来画几个路径来看下,画在drawRect里面
代码画图.png

在屏幕中间添加一个正方形view,然后在view上画出这个图,本来想画个五角星的,但是坐标算起来实在麻烦,就画了个四角星凑合凑合,下面用两种方式来画,UIBezierPath和CGContextRef,两者差不多,下面上代码

  • UIBezierPath画图
- (void)drawRect:(CGRect)rect {
    
    //圆环所在矩形的宽度
    float width = MIN(rect.size.width, rect.size.height);
    //圆环宽度
    float circleLW = 10;
    //星角距离圆环内边的距离
    float starEdgeC = 10.0;
    //星星线宽
    float starLW = 4.0;
    //星角距离圆环所在矩形边框的距离
    float starER = circleLW + starEdgeC + starLW * 0.5;
    //星星所在正方形宽度
    float starWidth = rect.size.width - starER * 2;
    //星星所在矩形宽度的1/8
    float partW = starWidth / 8;
    
    //1. 先画圆
    UIBezierPath *circlePath = [UIBezierPath bezierPathWithArcCenter:CGPointMake(width*0.5, width*0.5)
                                                              radius:(width-circleLW) * 0.5
                                                          startAngle:0
                                                            endAngle:2 * M_PI
                                                           clockwise:YES];
    circlePath.lineWidth = circleLW;
    [[UIColor redColor] setStroke];
    [circlePath stroke];
    
    //2. 再画四角星
    UIBezierPath *starPath = [UIBezierPath bezierPath];
    [starPath moveToPoint:CGPointMake(starER, width * 0.5)];
    [starPath addLineToPoint:CGPointMake(width*0.5-partW, width*0.5-partW)];
    [starPath addLineToPoint:CGPointMake(width*0.5, starER)];
    [starPath addLineToPoint:CGPointMake(width*0.5+partW, width*0.5-partW)];
    [starPath addLineToPoint:CGPointMake(width-starER,width*0.5)];
    [starPath addLineToPoint:CGPointMake(width*0.5+partW, width*0.5+partW)];
    [starPath addLineToPoint:CGPointMake(width*0.5, width-starER)];
    [starPath addLineToPoint:CGPointMake(width*0.5-partW, width*0.5+partW)];
    [starPath closePath];
    /**设置线段交接处,内角和外角的距离
    starPath.lineJoinStyle = kCGLineJoinMiter;
    starPath.miterLimit = 1;
     */
    starPath.lineWidth = starLW;
    [[UIColor cyanColor] setStroke];
    [starPath stroke];
    [[UIColor yellowColor] setFill];
    [starPath fill];
}
  • CGContextRef画图
- (void)drawRect:(CGRect)rect {
    
    //圆环所在矩形的宽度
    float width = MIN(rect.size.width, rect.size.height);
    //圆环宽度
    float circleLW = 10;
    //星角距离圆环内边的距离
    float starEdgeC = 10.0;
    //星星线宽
    float starLW = 4.0;
    //星角距离圆环所在矩形边框的距离
    float starER = circleLW + starEdgeC + starLW * 0.5;
    //星星所在正方形宽度
    float starWidth = rect.size.width - starER * 2;
    //星星所在矩形宽度的1/8
    float partW = starWidth / 8;
    
    CGContextRef circleCtx = UIGraphicsGetCurrentContext();
    CGContextAddArc(circleCtx, width * 0.5, width * 0.5, (width - circleLW) * 0.5, 0, 2 * M_PI, 1);
    CGContextSetLineWidth(circleCtx, circleLW);
    CGContextSetStrokeColorWithColor(circleCtx, [UIColor redColor].CGColor);
    CGContextStrokePath(circleCtx);
    
    CGContextRef starCtx = UIGraphicsGetCurrentContext();
    CGContextMoveToPoint(starCtx, starER, width * 0.5);
    CGContextAddLineToPoint(starCtx, width * 0.5 - partW, width * 0.5 - partW);
    CGContextAddLineToPoint(starCtx, width * 0.5, starER);
    CGContextAddLineToPoint(starCtx, width * 0.5 + partW, width * 0.5 - partW);
    CGContextAddLineToPoint(starCtx, width - starER, width * 0.5);
    CGContextAddLineToPoint(starCtx, width * 0.5 + partW, width * 0.5 + partW);
    CGContextAddLineToPoint(starCtx, width * 0.5, width - starER);
    CGContextAddLineToPoint(starCtx, width * 0.5 - partW, width * 0.5 + partW);
    CGContextClosePath(starCtx);
    CGContextSetLineWidth(starCtx, starLW);
    
    CGContextSetStrokeColorWithColor(starCtx, [UIColor cyanColor].CGColor);
    CGContextSetFillColorWithColor(starCtx, [UIColor yellowColor].CGColor);
    
    /** 即填充页画线,下面的两个方法只能共存一个,所以要用下方没注释的方法来画
    CGContextStrokePath(starCtx);
    CGContextFillPath(starCtx);
    */
    CGContextDrawPath(starCtx, kCGPathFillStroke);
}

通过上面两种画图方式的比较,我更喜欢用UIBezierPath对象来画图

画成UIImage对象

平时开发中可能有些简单的图,我们可以用代码画出来存到缓存中,也省了拖图片进工程

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

推荐阅读更多精彩内容