CGContext-上下文(画布)的应用

在uiview的- (void)drawRect:(CGRect)rect方法之前会自动生成一个画布并放在栈顶。通过UIGraphicsGetCurrentContext()方法获取当前的画布。
苹果原本文档如下:

The current graphics context is nil
by default. Prior to calling its drawRect:
method, view objects push a valid context onto the stack, making it current. If you are not using a UIView
object to do your drawing, however, you must push a valid context onto the stack manually using the UIGraphicsPushContextfunction.

上下文的用处一般有三种,本文介绍第一种

  1. UIView调用drawRect:之前系统生成的,用于界面的绘制;
  2. UIGraphicsBeginImageContext(), 用于生成图片;
  3. UIGraphicsBeginPDFContextToFile和UIGraphicsBeginPDFContextToData,用于生成PDF;

1、画布的坐标转换(Coordinate space transformations)

当前变换矩阵CTM(current transformation matrix )用来改变坐标环境,实现画布的平移、缩放、旋转等操作。

  • 画布平移

/* Translate the current graphics state's transformation matrix (the CTM) by
`(tx, ty)'. */
CG_EXTERN void CGContextTranslateCTM(CGContextRef cg_nullable c,
CGFloat tx, CGFloat ty)
CG_AVAILABLE_STARTING(__MAC_10_0, __IPHONE_2_0);

  • 画布缩放

/* Scale the current graphics state's transformation matrix (the CTM) by
`(sx, sy)'. */
CG_EXTERN void CGContextScaleCTM(CGContextRef cg_nullable c,
CGFloat sx, CGFloat sy)
CG_AVAILABLE_STARTING(__MAC_10_0, __IPHONE_2_0);

  • 画布旋转

/* Rotate the current graphics state's transformation matrix (the CTM) by
`angle' radians. */
CG_EXTERN void CGContextRotateCTM(CGContextRef cg_nullable c, CGFloat angle)
CG_AVAILABLE_STARTING(__MAC_10_0, __IPHONE_2_0);

  • 将变换矩阵合并到画布中
    之前有写过关于仿射矩阵变换的简书文章,此处就是放射变换方法的一处应用场景

/* Concatenate the current graphics state's transformation matrix (the CTM)
with the affine transform `transform'. */
CG_EXTERN void CGContextConcatCTM(CGContextRef cg_nullable c,
CGAffineTransform transform)
CG_AVAILABLE_STARTING(__MAC_10_0, __IPHONE_2_0);

  • 获取当前变换矩阵

/* Return the current graphics state's transformation matrix. Returns
CGAffineTransformIdentity in case of inavlid context. */
CG_EXTERN CGAffineTransform CGContextGetCTM(CGContextRef cg_nullable c)
CG_AVAILABLE_STARTING(__MAC_10_0, __IPHONE_2_0);

实例

将一个字符串画到UIView 上,偏移tx=10,ty=10,放大2倍,旋转45度。

- (void)drawRect:(CGRect)rect {
    // Drawing code
    CGContextRef ct = UIGraphicsGetCurrentContext();
    // 放大 2 倍
    CGContextScaleCTM(ct, 2, 2);
    // 移动 10
    CGContextTranslateCTM(ct, 10, 10);
    // 旋转 绕45度
    CGContextRotateCTM(ct, M_PI_4);
    [@"将军走路" drawInRect:rect withAttributes:@{NSForegroundColorAttributeName:[UIColor blueColor],NSFontAttributeName:[UIFont systemFontOfSize:14],NSBackgroundColorAttributeName:[UIColor cyanColor]}];
    
}

最终效果如下:


E8864BF2-347B-4A83-854E-EDC5064B11C7.png

注意所有的操作都是针对画布坐标的变换,所以并不是中心旋转,而是坐标的变换。所以转换的顺序不一样,得到的结果也不一样。把代码调一下顺序,得到的结果如下:

    - (void)drawRect:(CGRect)rect {
    // Drawing code
    CGContextRef ct = UIGraphicsGetCurrentContext();
    // 旋转 绕45度
    CGContextRotateCTM(ct, M_PI_4);
    // 放大 2 倍
    CGContextScaleCTM(ct, 2, 2);
    // 移动 10
    CGContextTranslateCTM(ct, 10, 10);
    [@"将军走路" drawInRect:rect withAttributes:@{NSForegroundColorAttributeName:[UIColor blueColor],NSFontAttributeName:[UIFont systemFontOfSize:14],NSBackgroundColorAttributeName:[UIColor cyanColor]}];

}
017BEAC4-6418-4E90-9AC2-FD83CFECA0F4.png

2、上下文的绘图方式(Path drawing functions)

基本绘制方法

void CGContextDrawPath(CGContextRef cg_nullable c,
CGPathDrawingMode mode)

typedef CF_ENUM (int32_t, CGPathDrawingMode) {
kCGPathFill, //填充
kCGPathEOFill,//奇偶填充
kCGPathStroke,//划线
kCGPathFillStroke,//填充并划线
kCGPathEOFillStroke //奇偶填充并划线
};

苹果提供的一些方便的绘制方法
  • CGContextFillPath
  • CGContextEOFillPath
  • CGContextStrokePath
  • CGContextFillRect
  • CGContextFillRects
  • CGContextStrokeRect
  • CGContextStrokeRectWithWidth
  • CGContextClearRect
  • CGContextFillEllipseInRect
  • CGContextStrokeEllipseInRect
  • CGContextStrokeLineSegments

3、上下文绘制属性设置方法(Drawing attribute functions)

  • void CGContextSetLineWidth(CGContextRef cg_nullable c, CGFloat width)

  • void CGContextSetLineCap(CGContextRef cg_nullable c, CGLineCap cap)

  • void CGContextSetLineJoin(CGContextRef cg_nullable c, CGLineJoin join)

  • void CGContextSetMiterLimit(CGContextRef cg_nullable c, CGFloat limit)

  • void CGContextSetLineDash(CGContextRef cg_nullable c, CGFloat phase,
    const CGFloat * __nullable lengths, size_t count)

  • void CGContextSetFlatness(CGContextRef cg_nullable c, CGFloat flatness)

  • void CGContextSetAlpha(CGContextRef cg_nullable c, CGFloat alpha)

  • void CGContextSetBlendMode(CGContextRef cg_nullable c, CGBlendMode mode)

4、路径的构建(Path construction functions)

路径的基本构建方法
  • void CGContextBeginPath(CGContextRef cg_nullable c)

  • void CGContextMoveToPoint(CGContextRef cg_nullable c,
    CGFloat x, CGFloat y)

  • void CGContextAddLineToPoint(CGContextRef cg_nullable c,
    CGFloat x, CGFloat y)

  • void CGContextAddCurveToPoint(CGContextRef cg_nullable c, CGFloat cp1x,
    CGFloat cp1y, CGFloat cp2x, CGFloat cp2y, CGFloat x, CGFloat y)

  • void CGContextAddQuadCurveToPoint(CGContextRef cg_nullable c,
    CGFloat cpx, CGFloat cpy, CGFloat x, CGFloat y)

  • void CGContextClosePath(CGContextRef cg_nullable c)

在基本构建方法上,苹果有封装了一些方便的构建方法
  • void CGContextAddRect(CGContextRef cg_nullable c, CGRect rect)
  • void CGContextAddRects(CGContextRef cg_nullable c,
    const CGRect * __nullable rects, size_t count)
  • void CGContextAddLines(CGContextRef cg_nullable c,
    const CGPoint * __nullable points, size_t count)
  • void CGContextAddEllipseInRect(CGContextRef cg_nullable c, CGRect rect)
  • void CGContextAddArc(CGContextRef cg_nullable c, CGFloat x, CGFloat y,
    CGFloat radius, CGFloat startAngle, CGFloat endAngle, int clockwise)
  • void CGContextAddArcToPoint(CGContextRef cg_nullable c,
    CGFloat x1, CGFloat y1, CGFloat x2, CGFloat y2, CGFloat radius)
  • void CGContextAddPath(CGContextRef cg_nullable c,
    CGPathRef cg_nullable path)

实例

画虚线
  • 效果


    AA377882-6056-47A6-83C7-9A9BA7C6707C.png
  • 代码及注释

    //设置为虚线
//    CGFloat lengths[] = {2,2};
//    CGContextSetLineDash(ct, 0, lengths, 2);
    //默认线条的cap类型就是kCGLineCapButt,而且,虚线必须是这个类型
//    CGContextSetLineCap(ct, kCGLineCapButt);
//    CGContextSetLineCap(ct, kCGLineCapRound);
//    CGContextSetLineCap(ct, kCGLineCapSquare);

    CGPoint points[] = {
        CGPointMake(10, 10),CGPointMake(50, 10),
        CGPointMake(50, 10),CGPointMake(50, 50),
        CGPointMake(50, 50),CGPointMake(10, 50),
        CGPointMake(10, 50),CGPointMake(10, 10)
    };
    // 多点连线方法
    CGContextStrokeLineSegments(ct, points, 8);
画弧线
  • 效果


    81AD6E35-A85A-42EF-967E-A79D52952DA7.png
  • 代码
    从当前点开始画,然后画一个圆角。
    两个点加上当前点计算圆角的两条切线,来确定圆角的角度。
    半径参数决定圆角的大小
    CGContextMoveToPoint(ct, 10, 10);
    CGContextAddArcToPoint(ct,80,10,80,30,20);
    CGContextAddLineToPoint(ct, 80, 110);
画圆弧
  • 效果


    BB569983-B810-4089-890C-116ED2279AA9.png
  • 代码
CGContextAddArc(ct, 30, 30, 20, 0, M_PI_2*3, 0);
批量添加直线
  • 效果
    直线是连接着的,kCGLineJoinRound 设置后有连接样式改变


    24625B11-C3E2-4107-A735-27C73B33A7E2.png
  • 代码
// 设置连接点为圆角
    CGContextSetLineJoin(ct, kCGLineJoinRound);
    CGPoint points[] = {
        CGPointMake(10, 10),
        CGPointMake(50, 10),
        CGPointMake(50, 50),
        CGPointMake(10, 50),
        
    };
  
    CGContextAddLines(ct, points, 4);
    CGContextClosePath(ct);
    CGContextStrokePath(ct);
批量画直线

只是单独的画很多直线,但是相互并没有连接,利用kCGLineJoinRound 设置后并没有任何反应。

  • 效果


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

推荐阅读更多精彩内容