CoreGraphics画图

在iOS上画图主要有3种方法:

  • UIKit, 这是我们最常用的绘图方法,平时的UIButton、UIImageView就是使用UIkit实现的。
  • Core Graphics, 这是是苹果提供一套基于C的API框架,使用了Quartz2D作为绘图引擎。Quartz2D是既可以在iOS上使用,也可以在Mac OS上使用的垮平台2D绘图引擎。
  • OpenGL ES, 是种编程规范,也就是说它只定义了一套规范,具体的实现由设备制造商根据规范去做。

在上节UIBezierPath画一个拱形的tabBar图形章节中,我们使用它画了一个带拱形的View。可以看到UIBezierPath其实就是对Core Graphics的封装,直接调用API就可以简单的画直线、圆、曲线,省去了对Path的操作细节。

一. Core Graphics使用简介

Core Graphics是基于C语言的API,所以它不是面向对象的,在使用时需要一个图形上下(CGContext)。Context其实就是暂时缓存图形,等绘制完毕后将图形返还给所需的对象。

1.1 如何在UIView上使用Graphics绘图?

使用Core Graphics需要一个画布(CGContext),而UIView只能在-(void)drawRect:(CGRext)rect方法中获取当前UIView的相关联的图形上下文,所以Core Graphics只能在-(void)drawRect:(CGRext)rect方法中使用。

1.2 Core Graphics绘图七大步骤
  1. 获取上下文
  1. 设置画图起点
  2. 设置绘图类别:直线、圆、圆弧、矩形
  3. 设置绘图线条颜色和线条宽度
  4. 开始绘图
  5. 结束绘图
1.3 绘制实例
  • 绘制直线:
-(void)drawRect:(CGRect)rect
{
    // 1. 获取图像上下文
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    // 2. 设置起点
    CGContextMoveToPoint(ctx, 10, 100);
    // 3. 画直线
    CGContextAddLineToPoint(ctx, 100, 100);
    // 4. 设置线条颜色 红色
    CGContextSetRGBStrokeColor(ctx, 1.0, 0, 0, 1.0);
    // 5. 设置线条宽度
    CGContextSetLineWidth(ctx, 10);
    // 6. 设置线条的起点和终点的样式
    CGContextSetLineCap(ctx, kCGLineCapRound);
    // 7. 设置线条的转角的样式
    CGContextSetLineJoin(ctx, kCGLineJoinRound);
    // 8. 开始绘制
    CGContextStrokePath(ctx);
}
直线.png
  • 绘制矩形
-(void)drawRect:(CGRect)rect
{
    // 绘制四边形
    // 1.获取上下文
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    // 2.绘制四边形
    CGContextAddRect(ctx, CGRectMake(10, 10, 150, 100));
    [[UIColor colorWithRed:1.0 green:0 blue:0 alpha:1.0] set];
    // 3.渲染图形到layer上
    // 填充颜色
    CGContextFillPath(ctx);
    // 路径颜色
    //  CGContextStrokePath(ctx);
}
矩形.png
  • 绘制圆形
-(void)drawRect:(CGRect)rect
{
    // 画圆弧
    // 1.获取上下文
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    // 2.画圆弧
    // x/y 圆心
    // radius 半径
    // startAngle 开始的弧度
    // endAngle 结束的弧度
    // clockwise 画圆弧的方向 (0 顺时针, 1 逆时针)
    CGContextAddArc(ctx, 100, 100, 50, 0, M_PI * 2, 1);
//    CGContextClosePath(ctx);
    
    // 3.设置线条颜色
    [[UIColor redColor] set];
    // 4. 绘图
         CGContextStrokePath(ctx);
//    CGContextFillPath(ctx);
}
圆形.png

思考:

  1. CGContextStrokePath(ctx); 和 CGContextFillPath(ctx);有啥区别?
  1. 如何设置圆形的线条颜色以及实心圆的背景色?

1.4 如何在drawRect方法外绘图

使用UIGraphicsGetCurrentContext()绘图有个限制条件就是必须在-(void)drawRect:(CGRect)rect方法中使用,实际应用中显然不够灵活。那么如何在此方法以外的地方绘制图形了?常用的有两种方法:
a. UIGraphicsBeginImageContext(size)绘制位图
b. 贝塞尔曲线+CAShapeLayer绘图

  • 使用UIGraphicsBeginImageContext(CGSize size)绘图
-(void)viewDidLoad 
{
    [super viewDidLoad];
     UIImageView *imgView = [[UIImageView alloc] initWithFrame:CGRectMake(100, 100, 100, 100)];
    imgView.image = [UIImage imageNamed:@"liuYan.jpeg"];
    imgView.backgroundColor = [UIColor blueColor];
   UIImage *image =  [UIImage imageNamed:@"liuYan.jpeg"];
    NSLog(@"图片大小%f %f", image.size.width,image.size.height );
    // 我们重新绘图,将图片在100x100大小的画布上绘制
    CGSize size = CGSizeMake(100, 100);
    // 设置画图大小
    UIGraphicsBeginImageContext(CGSizeMake(size.width, size.height));
    [image drawInRect:CGRectMake(0,0,100, 100)];
    UIImage *scaledImage = UIGraphicsGetImageFromCurrentImageContext();
    NSLog(@"修改后大小%f %f", scaledImage.size.width,scaledImage.size.height );
    imgView.image = scaledImage;
    UIGraphicsEndImageContext();
}
liuyan.png

从代码可以看到我们使用UIGraphicsBeginImageContext(CGSize size)这个方法也可以绘制位图,而不需要在drawRect方法中绘制。
打印图片绘制前后大小:


绘制图片前后.png

从截图中可以看到我们将500x600大小的图片重新绘制后,产生的新图片scaledImage大小为100x100了。
UIGraphicsBeginImageContext(CGSize size)方法中的CGSize size参数即图形上下文中的画布大小,通过改变这个参数值,绘制出随意大小的图片。
-(void)drawInRect:(CGRect)rect; 方法中的rect参数告诉画布我需要在哪个点上画多大的图像。例如将 [image drawInRect:CGRectMake(0,0,100, 100)];改为 [image drawInRect:CGRectMake(25,25,50, 50)];


scaleImage.png

图像跑到了画布的中间位置,并且图像大小变成了50x50,但scaledImage仍然是100x100,说明画布的其他位置并没有东西填充。

P.S 使用此方法可以将一张图片进行缩放。

  • 贝塞尔曲线+CAShapeLayer绘图
-(void)viewDidLoad {
    [super viewDidLoad];
    UIView *rectView = [[UIView alloc] initWithFrame:CGRectMake(100, 100, 100, 100)];
    rectView.backgroundColor = [UIColor redColor];
    UIBezierPath *path = [UIBezierPath bezierPathWithRoundedRect:rectView.bounds byRoundingCorners:UIRectCornerTopLeft | UIRectCornerTopRight cornerRadii:CGSizeMake(20,5)];
    CAShapeLayer *layer = [CAShapeLayer layer];
    layer.frame = rectView.bounds;
    layer.path = path.CGPath;
    rectView.layer.mask = layer;
    [self.view addSubview:rectView];
}

部分带圆角的UIView.png

Core Graphic部分代码上传gitHub

1.5 Core Graphics画文字和图片

主要是 darwAtPoint和drawInRect 这两个方法的使用

// 画文字
-(void)drawRect:(CGRect)rect {
    // Drawing code
    
    NSString *str = @"Quartz 2D是一个二维绘图引擎,同时支持iOS和Mac系统";
    
    NSMutableDictionary *dict = [NSMutableDictionary dictionary];
    //设置文字大小
    dict[NSFontAttributeName] = [UIFont systemFontOfSize:12];
    //设置文字颜色
    dict[NSForegroundColorAttributeName] = [UIColor greenColor];
    //设置描边宽度
    dict[NSStrokeWidthAttributeName] = @2;
    //设置描边颜色
    dict[NSStrokeColorAttributeName] = [UIColor blueColor];
    //设置阴影
    NSShadow *shadow = [[NSShadow alloc] init];
    //设置阴影的便宜量
    shadow.shadowOffset = CGSizeMake(1, 1);
    //设置阴影颜色
    shadow.shadowColor = [UIColor greenColor];
    //设置阴影模糊程序
    shadow.shadowBlurRadius = 1;
    dict[NSShadowAttributeName] = shadow;
    /**
     AtPoint:文字所画的位置
     withAttributes:描述文字的属性.
     */
    //不会自动换行
    [str drawAtPoint:CGPointMake(0,0) withAttributes:dict];
    //会自动换行.
    [str drawInRect:self.bounds withAttributes:dict];
}
// 画图片
-(void)drawRect:(CGRect)rect {
    // Drawing code
    //1.加载图片
    UIImage *image = [UIImage imageNamed:@"liuYan.jpeg"];
    
    //绘制出来的图片,是保持原来图片大小
    [image drawAtPoint:CGPointZero];
    //把图片填充到这个rect当中.
    [image drawInRect:rect];
    //添加裁剪区域 .把超区裁剪区域以外都裁剪掉
//    UIRectClip(CGRectMake(0, 0, 50, 50));
    //平铺
//    [image drawAsPatternInRect:self.bounds];
    
//    //快速的画出一个矩形
//    [[UIColor blueColor] set];
//    UIRectFill(CGRectMake(10, 10, 100, 100));
}

二. UIBezierPath使用简介

UIBezierPath其实是对Core Graphics Path的封装, 需要与CAShapeLayer配合使用。

2.1 UIBezierPath使用六部曲
  1. 创建贝塞尔曲线
+(instancetype)bezierPath;
  1. 添加子路径
// 从point开始绘制路径
-(void)moveToPoint:(CGPoint)point;
  1. 设置绘图类别:直线、圆弧、矩形
// 添加一条直线路径
-(void)addLineToPoint:(CGPoint)point;
// 添加一个圆形路径
-(void)addArcWithCenter:(CGPoint)center radius:(CGFloat)radius startAngle:(CGFloat)startAngle
// 添加一段圆弧路径
-(void)addQuadCurveToPoint:(CGPoint)endPoint controlPoint:(CGPoint)controlPoint;
  1. 设置路径颜色
 [[UIColor redColor] set];
  1. 开始画路径
    [path stroke];
  1. 结束路径绘制
-(void)closePath;

主要就以上6个步骤就可以快速的画出图形。
注意:如果需要画一个实心View,可以使用 [[UIColor blackColor] setFill];设置填充颜色。

2.2 绘制实例
  • 直线
  -(void)drawRect:(CGRect)rect
{
    //1.获取图形上下文
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    //上下文的状态(线的粗细、颜色、链接样式等)
    //设置线的粗细
    CGContextSetLineWidth(ctx, 10);
    CGContextSetLineJoin(ctx, kCGLineJoinRound);
    CGContextSetLineCap(ctx, kCGLineCapRound);
    //2. 绘制路径
    UIBezierPath *path = [UIBezierPath bezierPath];
    //2.1 设置起点
    [path moveToPoint:CGPointMake(50, 250)];
    //2.2 添加一根线到终点
    [path addLineToPoint:CGPointMake(250, 50)];
    
    //画第二条直线(新起点)
    [path moveToPoint:CGPointMake(100, 250)];
    [path addLineToPoint:CGPointMake(250, 100)];
    
    //把上一条线的终点当做起点来画第二条线
    [path addLineToPoint:CGPointMake(250, 250)];
    [[UIColor redColor] setStroke];
    //3.把绘制的内容添加到上下文中
    // UIBezierPath:UIKit框架   --->   CGPathRef:CoreGraphics框架
    CGContextAddPath(ctx, path.CGPath);
    //4.把上下文的内容显示到view上(渲染到view的layer上)渲染的方式有两种Stroke(描边)、Fill(填充)
    CGContextStrokePath(ctx);
}
直线绘制.png
  • 矩形
-(void)drawRect:(CGRect)rect 
{
    //1.获取图形上下文
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    //设置颜色
    [[UIColor redColor]set];
    //2.绘制路径
    //画矩形
        UIBezierPath *path = [UIBezierPath bezierPathWithRect:CGRectMake(50, 50, 100, 100)];
    //画圆角矩形
//    UIBezierPath *path = [UIBezierPath bezierPathWithRoundedRect:CGRectMake(50, 50, 100, 100) cornerRadius:50];
    //3.把路径添加到上下文
    CGContextAddPath(ctx, path.CGPath);
    //4.把上下文渲染到view上
    //描边
    //    CGContextStrokePath(ctx);
    //填充
    CGContextFillPath(ctx);
}
矩形绘制.png

贝塞尔曲线部分代码上传到gitHub

Quartz2D内存管理

Quartz2D是C语言的框架,部分地方需要自己管理内存:
•使用含有“Create”或“Copy”的函数创建的对象,使用完后必须释放,否则将导致内存泄露
•使用不含有“Create”或“Copy”的函数获取的对象,则不需要释放
•如果retain了一个对象,不再使用时,需要将其release掉。
可以使用Quartz 2D的函数来指定retain和release一个对象。例如,如果创建了一个CGColorSpace对象,则使用函数CGColorSpaceRetain和CGColorSpaceRelease来retain和release对象。也可以使用Core Foundation的CFRetain和CFRelease。注意不能传递NULL值给这些函数

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

推荐阅读更多精彩内容