Quartz2D使用详解

首先介绍一下使用Quartz2D能做些什么:

  • 绘制图形 : 线条\三角形\矩形\圆\弧等;
  • 绘制文字;
  • 绘制\生成图片(图像);
  • 读取\生成PDF;
  • 截图\裁剪图片;
  • 自定义UI控件;

图形上下文

图形上下文(Graphics Context):是一个CGContextRef类型的数据。
图形上下文的作用:

  • (1)保存绘图信息、绘图状态
  • (2)决定绘制的输出目标(绘制到什么地方去?)
    (输出目标可以是PDF文件、Bitmap或者显示器的窗口上)相同的一套绘图序列,指定不同的Graphics Context,就可将相同的图像绘制到不同的目标上。

Quartz2D提供了以下几种类型的Graphics Context:

  • (1)Bitmap Graphics Context
  • (2)PDF Graphics Context
  • (3)Window Graphics Context
  • (4)Layer Graphics Context
  • (5)Printer Graphics Context

使用Quartz2D自定义View

步骤:

  • (1)新建一个类,继承自UIView
  • (2)实现-(void)drawRect:(CGRect)rect方法,然后在这个方法中取得跟当前view相关联的图形上下文
    CGContextRef ctx = UIGraphicsGetCurrentContext();
  • (3)绘制相应的图形内容

画扇形

- (void)drawRect:(CGRect)rect {
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    CGContextMoveToPoint(ctx, 100, 100);
    CGContextAddLineToPoint(ctx, 100, 150);
    CGContextAddArc(ctx, 100, 100, 50, -M_PI_2, M_PI_2, 1);
    CGContextClosePath(ctx);
    [[UIColor redColor] set];
    CGContextFillPath(ctx);
}

**  注:**
//M_PI的含义:π
//M_PI * 2的含义:2π
//M_PI_2的含义:π/2
//M_PI / 2的含义:π/2
// 画的图形路径
//bezierPathWithArcCenter:弧所在的圆心
//radius:圆的半径
//startAngle:开始角度,圆的最右侧为0度
//endAngle:截至角度,向下为正,向上为负.
//clockwise:时针的方向,yes:顺时针 no:逆时针
`UIBezierPath *path = [UIBezierPath bezierPathWithArcCenter:self.center radius:radius startAngle:startA endAngle:endA clockwise:NO];`

注意4点:

  • 手动调用drawRect:方法,不会自动创建跟View相关联的上下文。应该调用setNeedsDisplay方法,系统底层会自动调用drawRect,告诉系统重新绘制View.这样,系统底层会自动创建跟View相关联的上下文
  • setNeedsDisplay底层会调用drawRect,并不是立马调用的.只是设了一个调用的标志.调用时刻是等下一次屏幕刷新时才去调用drawRect。屏幕每一秒刷新30-60秒次,所以1秒调用drawRect方法大概30-60次,速度非常快哦
  • view内部有个layer(图层)属性,drawRect:方法中取得的是一个Layer Graphics Context,因此,绘制的东西其实是绘制到view的layer上去了
  • View之所以能显示东西,完全是因为它内部的layer

Quartz2D重要函数

常用拼接路径函数

  • void CGContextMoveToPoint(CGContextRef c, CGFloat x, CGFloat y)新建一个起点
  • void CGContextAddLineToPoint(CGContextRef c, CGFloat x, CGFloat y) 添加新的线段到某个点
  • void CGContextAddRect(CGContextRef c, CGRect rect)添加一个矩形
  • void CGContextAddEllipseInRect(CGContextRef context, CGRect rect)添加一个椭圆
  • void CGContextAddArc(CGContextRef c, CGFloat x, CGFloat y,CGFloat radius, CGFloat startAngle, CGFloat endAngle, int clockwise)添加一个圆弧

常用绘制路径函数

一般以CGContextDrawCGContextStrokeCGContextFill开头的函数,都是用来绘制路径的。

  • void CGContextDrawPath(CGContextRef c, CGPathDrawingMode mode) Mode参数决定绘制的模式
  • void CGContextStrokePath(CGContextRef c)绘制空心路径
  • void CGContextFillPath(CGContextRef c)绘制实心路径

矩阵操作函数

利用矩阵操作,能让绘制到上下文中的所有路径一起发生变化。

  • 缩放:void CGContextScaleCTM(CGContextRef c, CGFloat sx, CGFloat sy)
  • 旋转: void CGContextRotateCTM(CGContextRef c, CGFloat angle)
  • 平移: void CGContextTranslateCTM(CGContextRef c, CGFloat tx, CGFloat ty)

其他常用函数

  • 设置线段宽度: CGContextSetLineWidth(ctx, 10);
  • 设置线段头尾部的样式:CGContextSetLineCap(ctx, kCGLineCapRound);
  • 设置线段转折点的样式: CGContextSetLineJoin(ctx, kCGLineJoinRound);
  • 设置颜色 : CGContextSetRGBStrokeColor(ctx, 1, 0, 0, 1);

画矩形,正方形

- (void)drawRect:(CGRect)rect {
   //1.获取上下文
   CGContextRef ctx = UIGraphicsGetCurrentContext();
   //2.描述路径
   UIBezierPath *path = [UIBezierPath bezierPathWithRect:CGRectMake(50, 50, 200, 200)];
   //3.把路径添加到上下文
   CGContextAddPath(ctx, path.CGPath);
   [[UIColor redColor] set];// 路径的颜色
   CGContextSetLineWidth(ctx, 10);
   //4.把上下文的内容渲染到View的layer.
   //CGContextStrokePath(ctx);// 描边路径
   CGContextStrokePath(ctx);// 填充路径
}

画圆形

- (void)drawRect:(CGRect)rect {
 //1.获取上下文
 CGContextRef ctx = UIGraphicsGetCurrentContext();
 //2.描述路径
 // cornerRadius:圆角半径。矩形的宽高都为200,如果圆角为100,那么两个角之间弧线上任意一点到矩形中心的距离都为100,所以为圆形
 UIBezierPath *path = [UIBezierPath bezierPathWithRoundedRect:CGRectMake(50, 50, 200, 200) cornerRadius:100];
 //3.把路径添加到上下文
 CGContextAddPath(ctx, path.CGPath);
 [[UIColor redColor] set];// 路径的颜色
 //4.把上下文的内容渲染到View的layer.
 // CGContextStrokePath(ctx);// 描边路径
 CGContextFillPath(ctx);// 填充路径
}

画圆角矩形

- (void)drawRect:(CGRect)rect {
  
 //1.获取上下文
 CGContextRef ctx = UIGraphicsGetCurrentContext();
 //2.描述路径
 // cornerRadius:圆角半径。
 UIBezierPath *path = [UIBezierPath bezierPathWithRoundedRect:CGRectMake(50, 50, 200, 200) cornerRadius:50];
 //3.把路径添加到上下文
 CGContextAddPath(ctx, path.CGPath);
  
 [[UIColor redColor] set];// 路径的颜色
  
 //4.把上下文的内容渲染到View的layer.
 CGContextStrokePath(ctx);// 描边路径
 //CGContextFillPath(ctx);// 填充路径
  
}

画直线

- (void)drawRect:(CGRect)rect {
 //1.获取跟View相关联的上下文(uigraphics开头)
 CGContextRef ctx = UIGraphicsGetCurrentContext();
 //2.描述路径
 //一条路径可以绘制多条线 路径:path  路径绘制多条线:path使用了两次(2次的起点到终点),都是将线添加到某个点
 UIBezierPath *path = [UIBezierPath bezierPath];
 //设置起点
 [path moveToPoint:CGPointMake(50, 150)];
 //添加一根线Line到某个点
 [path addLineToPoint:CGPointMake(250, 50)];
 //画第二根线
 [path moveToPoint:CGPointMake(50, 250)];
 [path addLineToPoint:CGPointMake(250, 100)];
 //设置线宽
 CGContextSetLineWidth(ctx, 20);
 //设置线的连接样式
 CGContextSetLineJoin(ctx, kCGLineJoinBevel);
 //设置线的顶角样式
 CGContextSetLineCap(ctx, kCGLineCapRound);// 圆角线条
 //设置线条颜色
 [[UIColor redColor] set];
 //3.把路径添加到上下文
 CGContextAddPath(ctx, path.CGPath);
 //4.把上下文当中绘制的内容渲染到跟View关联的layer
 CGContextStrokePath(ctx);
}

绘制文字

- (void)drawRect:(CGRect)rect {
  
 NSString *str = @"HelloWorld";
  
 NSMutableDictionary *dict = [NSMutableDictionary dictionary];
 //设置字体
 dict[NSFontAttributeName] = [UIFont systemFontOfSize:30];
 //设置颜色
 dict[NSForegroundColorAttributeName] = [UIColor redColor];
 //设置描边
 dict[NSStrokeColorAttributeName] = [UIColor blueColor];
 dict[NSStrokeWidthAttributeName] = @3;
 //设置阴影
 NSShadow *shadow = [[NSShadow alloc] init];
 shadow.shadowColor = [UIColor greenColor];
 shadow.shadowOffset = CGSizeMake(-2, -2);
 shadow.shadowBlurRadius = 3;
 dict[NSShadowAttributeName] = shadow;
  
 //设置文字的属性
 //drawAtPoint不会自动换行
 //[str drawAtPoint:CGPointMake(0, 0) withAttributes:dict];
 //drawInRect会自动换行
 [str drawInRect:self.bounds withAttributes:dict];
  
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335

推荐阅读更多精彩内容

  • Quartz2D以及drawRect的重绘机制字数1487 阅读21 评论1 喜欢1一、什么是Quartz2D Q...
    PurpleWind阅读 752评论 0 3
  • --绘图与滤镜全面解析 概述 在iOS中可以很容易的开发出绚丽的界面效果,一方面得益于成功系统的设计,另一方面得益...
    韩七夏阅读 2,684评论 2 10
  • 在iOS中随处都可以看到绚丽的动画效果,实现这些动画的过程并不复杂,今天将带大家一窥ios动画全貌。在这里你可以看...
    每天刷两次牙阅读 8,418评论 6 30
  • 今天来讲一只熊,一只大名鼎鼎的熊。哈哈,对啦,就是可爱到不行男女生都会爱的熊本熊(。•ㅅ•。) 可能由于本人比...
    宋小帆阅读 392评论 0 0
  • 分镜 从视觉角度来讲,人眼得到的是一个二维的平面,因此有分镜说。 1.将发生事件的整个地区设为一个对象,镜头可以出...
    顾左右而呯呯阅读 435评论 0 1