Quartz2D?

什么是Quartz2D?

Quartz 2D是一个二维绘图引擎,同时支持iOS和Mac系统

  • Quartz 2D能完成的工作
    • 绘制图形 : 线条\三角形\矩形\圆\弧等
    • 绘制文字
    • 绘制\生成图片(图像)
    • 读取\生成PDF
    • 截图\裁剪图片
    • 自定义UI控件

Quartz2D在iOS开发中的价值:

  • 绘制一些系统UIKit框架中不好展示的内容,例如饼图
  • 自定义一些控件
  • 不添加UI控件的情况下,使UI内容更丰富
  • ……

iOS中,大部分控件都是Quartz2D绘制出来的

图形上下文

[图片上传失败...(image-534e23-1521165872369)]

图形上下文就相当于画布,不同类型的画布就是决定着画得内容将展示在哪里。

  • Quartz2D提供了以下几种类型的Graphics Context:
    • Bitmap Graphics Context 位图上下文,在这个上下文上绘制或者渲染的内容,可以获取成图片(需要主动创建一个位图上下文来使用,使用完毕,一定要销毁)
    • PDF Graphics Context
    • Window Graphics Context
    • Layer Graphics Context 图层上下文,针对UI控件的上下文
    • Printer Graphics Context

drawRect:

为什么要实现drawRect:方法才能绘图到view上?
因为在drawRect:方法中才能取得跟view相关联的图形上下文

drawRect:中取得的上下文

在drawRect:方法中取得上下文后,就可以绘制东西到view上

View内部有个layer(图层)属性,drawRect:方法中取得的是一个Layer Graphics Context,因此,绘制的东西其实是绘制到view的layer上去了

View之所以能显示东西,完全是因为它内部的layer

drawRect:方法的调用?

  • 当view第一次显示到屏幕上时,系统会创建好一个跟当前view相关的Layer上下文
  • 系统会通过此上下文,在drawRect:方法中绘制好当前view的内容
  • 主动让view重绘内容的时候,调用setNeedsDisplay或者setNeedsDisplayInRect:。我们主动调用drawRect:方法是无效的。
  • 调用view的setNeedsDisplay或者setNeedsDisplayInRect:时。
  • 注意:setNeedsDisplay和setNeedsDisplayInRect:方法调用后,屏幕并不是立即刷新,而是会在下一次刷新屏幕的时候把绘制的内容显示出来。

也正是系统会在调用这个方法之前创建一个与该view相关的上下文,才让我们可以在drawRect:方法中绘制。注意:在其他地方拿不到view相关的上下文,所以不能实现绘制。

自定义view

如何利用Quartz2D绘制东西到view上?

  • 首先,得有图形上下文,因为它能保存绘图信息,并且决定着绘制到什么地方去
  • 其次,那个图形上下文必须跟view相关联,才能将内容绘制到view上面

自定义view的步骤:

  1. 新建一个类,继承自UIView
  2. 实现- (void)drawRect:(CGRect)rect方法,然后在这个方法中
  3. 取得跟当前view相关联的图形上下文
  4. 绘制相应的图形内容
  5. 利用图形上下文将绘制的所有内容渲染显示到view上面

常用拼接路径函数

  • 获取上下文
    CGContextRef triangle = UIGraphicsGetCurrentContext();

  • 新建一个起点
    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)

常用绘制路径函数

  • Mode参数决定绘制的模式
    void CGContextDrawPath(CGContextRef c, CGPathDrawingMode mode)

  • 绘制空心路径
    void CGContextStrokePath(CGContextRef c)

  • 绘制实心路径
    void CGContextFillPath(CGContextRef c)

提示:一般以CGContextDraw、CGContextStroke、CGContextFill开头的函数,都是用来绘制路径的

图形上下文栈的操作

  • 将当前的上下文copy一份,保存到栈顶(那个栈叫做”图形上下文栈”)
    void CGContextSaveGState(CGContextRef c)

  • 将栈顶的上下文出栈,替换掉当前的上下文
    void CGContextRestoreGState(CGContextRef c)

矩阵操作

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

  • 缩放
    void CGContextScaleCTM(CGContextRef c, CGFloat sx, CGFloat sy)

  • 旋转
    void CGContextRotateCTM(CGContextRef c, CGFloat angle)

  • 平移
    void CGContextTranslateCTM(CGContextRef c, CGFloat tx, CGFloat ty)

绘图的核心步骤:

  1. 获得上下文
  2. 绘制/拼接绘图路径
  3. 将路径添加到上下文
  4. 渲染上下文

记住:所有的绘图,都是这个步骤,即使使用贝塞尔路径,也只是对这个步骤进行了封装。对于绘图而言,拿到上下文很关键。

贝塞尔路径

就是UIKit框架中,对绘图的封装。实际操作起来,使用贝塞尔路径,更为方便。

  • 用法与CGContextRef类似,但是oc对其进行了封装,更加面向对象。
  • 常用的方法:
    • 返回一个描述椭圆的路径:
      + (UIBezierPath *)bezierPathWithOvalInRect:(CGRect)rect;

    • 设置起始点:
      - (void)moveToPoint:(CGPoint)point;

    • 添加直线到一点:
      - (void)addLineToPoint:(CGPoint)point;

    • 三次贝塞尔曲线:
      - (void)addCurveToPoint:(CGPoint)endPoint controlPoint1:(CGPoint)controlPoint1 controlPoint2:(CGPoint)controlPoint2;

[图片上传失败...(image-f06941-1521165872367)]

  • 贝塞尔曲线:
    - (void)addQuadCurveToPoint:(CGPoint)endPoint controlPoint:(CGPoint)controlPoint;

    [图片上传失败...(image-c02bb8-1521165872365)]

  • 绘制圆弧:
    - (void)addArcWithCenter:(CGPoint)center radius:(CGFloat)radius startAngle:(CGFloat)startAngle endAngle:(CGFloat)endAngle clockwise:(BOOL)clockwise;

    [图片上传失败...(image-26cc2-1521165872365)]

  • 封闭闭路径:- (void)closePath;

裁剪核心代码

// 开启一个位图(图片)上下文 
//size:上下文尺寸
//opaque:不透明。一般是透明的,所以设置为NO
//scale:缩放,如果不缩放,设置为0就好
UIGraphicsBeginImageContextWithOptions(image.size, NO, 0);

// 描述圆形的路径
UIBezierPath *path = [UIBezierPath bezierPathWithOvalInRect:CGRectMake(0, 0, image.size.width, image.size.height)];

// 把圆形路径设置裁剪区域(将区域外的内容裁剪掉,是现实区域内的内容)
[path addClip];

// 绘制图片(先设置裁剪区域,再裁剪,才会有效果)
[image drawAtPoint:CGPointZero];

// 从上下文中内容生成一张图片
image = UIGraphicsGetImageFromCurrentImageContext();

// 关闭上下文(一定不要忘了关闭自己开启的上下文)
UIGraphicsEndImageContext();

截屏核心代码

// 开启一个跟屏幕一样大的尺寸的上下文
          UIGraphicsBeginImageContextWithOptions(caputeView.bounds.size, NO, 0);

// 获取自己创建的位图上下文
CGContextRef ctx = UIGraphicsGetCurrentContext();

// view之所以你能显示内容,是因为有图层,因此只要把图层画到上下文
// 图层只能渲染,不能绘制
[caputeView.layer renderInContext:ctx];

// 从上下文中生成一张新的图片
UIImage *image = UIGraphicsGetImageFromCurrentImageContext();

// 关闭上下文
UIGraphicsEndImageContext();

作者:Ljson
链接:https://www.jianshu.com/p/0e785269dccc
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

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

推荐阅读更多精彩内容

  • Quartz2D以及drawRect的重绘机制字数1487 阅读21 评论1 喜欢1一、什么是Quartz2D Q...
    PurpleWind阅读 767评论 0 3
  • 一、简介 对于 Core Graphics Framework,可以从两个方面初步感知,一是它在绘图系统中的...
    kReader阅读 1,711评论 0 1
  • 什么是Quartz2D? Quartz 2D是一个二维绘图引擎,同时支持iOS和Mac系统 Quartz 2D能完...
    Ljson阅读 11,533评论 2 83
  • 什么是Quartz2D 是一个二维的绘图引擎,同时支持iOS和Mac系统 Quartz2D的API是纯C语言的,它...
    Mario_ZJ阅读 580评论 0 1
  • 12.4. 不是滋味 有时候觉得不爽,不快乐,不幸福,却又说不出啥滋味……师父说,人生很多烦恼,就来自于“不是滋味...
    陌上花开wen阅读 276评论 0 3