[73→100]iOS开发04:从绘制小黄人学习 用Quartz2D自定义UIView

iOS上怎么绘制如下的小黄人呢?

小黄人.png

在iOS系统中,UIView是最基础的显示控件,UILabel、UIButton都是它的子类。如果需要自定义新的UI控件样式,它也必须继承自UIView。

这里我们先自定义一个MinionsView试试,它的m文件如下:

#import "MinionsView.h"
@implementation MinionsView
/*
// Only override drawRect: if you perform custom drawing.
// An empty implementation adversely affects performance during animation.
- (void)drawRect:(CGRect)rect {
    // Drawing code
}
*/
@end

你会发现其中生成了一段被注释方法 drawRect ,而这正是自定义View最核心的方法。一般情况下它只在视图显示在屏幕上的时候被系统调用一次。

接下来的问题是怎么绘制出我们想要的View样式呢?

iOS给我们提供一个二维图形绘制引擎Quartz 2D

Quartz 2D是一个二维图形绘制引擎,支持iOS环境和Mac OS X环境。我们可以使用Quartz 2D API来实现许多功能,如基本路径的绘制、透明度、描影、绘制阴影、透明层、颜色管理、反锯齿、PDF文档生成和PDF元数据访问。在需要的时候,Quartz 2D还可以借助图形硬件的功能。
在Mac OS X中,Quartz 2D可以与其它图形图像技术混合使用,如Core Image、Core Video、OpenGL、QuickTime。例如,通过使用 QuickTime的GraphicsImportCreateCGImage函数,可以用 Quartz从一个 QuickTime图形导入器中创建一个图像。

Quartz 2D是一个纯C的库,接口完全面向过程,挺简单的。

一、绘制流程

基本流程3步:

  1. 获得图形上下文:
    CGContextRef context = UIGraphicsGetCurrentContext();
  2. 绘制图形。
  3. 将绘制的图形显示在UIView上。
CGContextFillPath(context); 
CGContextStrokePath(context);

这里面上下文有点像画笔,存放着画笔的宽度空心颜色实心颜色,为了避免频繁设置属性,可以保存到栈里面。

    // 保存CG上下文,放入栈中
    CGContextSaveGState(context);
    // 将CG上下文出栈,替换当前的上下文
    CGContextRestoreGState(context);

二、绘制基础元素

  1. 绘制线条
#pragma mark - 绘制线条
void drawline(){
    
    #pragma mark - 绘制斜线
    // 1:获得图形上下文
    CGContextRef context = UIGraphicsGetCurrentContext();
    // 保存CG上下文,放入栈中
    CGContextSaveGState(context);
    // 2:绘制图形
    // 设置一下线段的宽度
    CGContextSetLineWidth(context, 10);
    // 设置颜色
    CGContextSetRGBStrokeColor(context, 1, 0, 0, 1);
    // 设置起点
    CGContextMoveToPoint(context, 10, 10);
    // 设置线条终点
    CGContextAddLineToPoint(context, 100, 100);
    // 3:将绘制的图形显示在UIView上
    CGContextStrokePath(context);   // 以空心的方式画出
    
    #pragma mark - 绘制两条线
    // 将CG上下文出栈,替换当前的上下文
    CGContextRestoreGState(context);
    [[UIColor blueColor] set];
//    CGContextSetLineWidth(context, 10);
    // 设置线条头尾部的样式
    // kCGLineCapButt: 该属性值指定不绘制端点, 线条结尾处直接结束。这是默认值。
    // kCGLineCapRound: 该属性值指定绘制圆形端点, 线条结尾处绘制一个直径为线条宽度的半圆
    // kCGLineCapSquare: 该属性值指定绘制方形端点。
    CGContextSetLineCap(context, kCGLineCapRound);
    // 设置线段转折点的样式
    CGContextSetLineJoin(context, kCGLineJoinRound);
    // 画线
    CGContextMoveToPoint(context, 100, 120);
    CGContextAddLineToPoint(context, 150, 120);
    CGContextAddLineToPoint(context, 150, 180);
    CGContextAddLineToPoint(context, 200, 180);
    CGContextStrokePath(context);
}
  1. 绘制四边形
#pragma mark - 绘制实心四边形
void drawR(){
    CGContextRef context = UIGraphicsGetCurrentContext();
    CGContextAddRect(context, CGRectMake(10, 10, 120, 180));
    [[UIColor purpleColor] setFill];
    CGContextFillPath(context);
}
  1. 绘制三角形
#pragma mark - 绘制空心三角形
void drawTriangle(){
    CGContextRef context = UIGraphicsGetCurrentContext();
    CGContextMoveToPoint(context, 0, 0);
    CGContextAddLineToPoint(context, 100, 100);
    CGContextAddLineToPoint(context, 150, 100);
    //关闭路径(连接起点和最后一个点)
    CGContextClosePath(context);
    [[UIColor redColor] set];
    CGContextStrokePath(context);
}
  1. 绘制圆
#pragma mark - 绘制圆
void drawCircle(){
    CGContextRef context = UIGraphicsGetCurrentContext();
    CGContextAddEllipseInRect(context, CGRectMake(50, 50, 100, 100));
    CGContextSetLineWidth(context, 10);
    CGContextStrokePath(context);
}
  1. 绘制圆弧
#pragma mark - 绘制圆弧
// 角度转弧度
CGFloat arc(CGFloat angle){
    return angle * (M_PI / 180);
}
void drawArc(){
    CGContextRef context = UIGraphicsGetCurrentContext();
    CGContextAddArc(context, 100, 100, 50, arc(90), arc(200), 1);
//    CGContextStrokePath(context);
    CGContextFillPath(context);
}
  1. 绘制文字
#pragma mark - 绘制文字
void drawText(){
    NSString *str = @"我是文字";
    NSMutableDictionary *attributes = [NSMutableDictionary dictionary];
    attributes[NSFontAttributeName] = [UIFont systemFontOfSize:20]; // 设置文字大小
    attributes[NSForegroundColorAttributeName] = [UIColor purpleColor]; // 设置文字颜色
    [str drawInRect:CGRectMake(100, 100, 100, 30) withAttributes:attributes];
}
  1. 绘制图片
#pragma mark - 绘制图片
void drawPicture(){
    UIImage *image = [UIImage imageNamed:@"1.png"];
//    [image drawAtPoint:CGPointMake(50, 50)];
//    [image drawInRect:CGRectMake(0, 0, 100, 100)];
    [image drawAsPatternInRect:CGRectMake(0, 0, 300, 300)]; //平铺
    NSString *str = @"我是图片";
    [str drawInRect:CGRectMake(30, 30, 100, 30) withAttributes:nil];
}
  1. 绘制贝塞尔曲线
#pragma mark - 贝塞尔曲线
void drawBezier(){
    CGContextRef context = UIGraphicsGetCurrentContext();
    // 设置起点
    CGContextMoveToPoint(context, 10, 10);
    // 设置2个控制点
    CGContextAddCurveToPoint(context, 120, 100, 180, 50, 190, 190);
    // 设置1个控制点
// CGContextAddQuadCurveToPoint(context, 150, 200, 200, 100);
    CGContextStrokePath(context);
}

三、挑战大BOSS:绘制小黄人

//
//  MinionsView.m
//  PandaiOSDemo
//
//  Created by shitianci on 16/7/7.
//  Copyright © 2016年 Panda. All rights reserved.
//

#import "MinionsView.h"
#define JYFRadius 70
#define JYFTopY 100
#define JYFColor(r,g,b) [UIColor colorWithRed:(r)/255.0 green:(g)/255.0 blue:(b)/255.0 alpha:1.0]


@implementation MinionsView

// Only override drawRect: if you perform custom drawing.
// An empty implementation adversely affects performance during animation.
- (void)drawRect:(CGRect)rect {
    // Drawing code
    CGContextRef context = UIGraphicsGetCurrentContext();
    drawBody(context, rect);
    drawMouse(context, rect);
    drawEyes(context, rect);
}

#pragma mark - 绘制身体
void drawBody(CGContextRef context,CGRect rect){
    // 上半圆
    CGFloat topX = rect.size.width * 0.5;
    CGFloat topY = JYFTopY;
    CGFloat topRadius = JYFRadius;
    CGContextAddArc(context, topX, topY, topRadius, 0, M_PI, 1);
    
    // 向下延伸的直线
    CGFloat starPoint = topX - topRadius;
    CGFloat leftHeight = JYFTopY;
    CGFloat endPoint = topY + leftHeight;
    CGContextAddLineToPoint(context, starPoint, endPoint);
    
    // 下半圆
    CGFloat bottomX = topX;
    CGFloat bottomY = endPoint;
    CGFloat bottomRadius = topRadius;
    CGContextAddArc(context, bottomX, bottomY, bottomRadius, M_PI, 0, 1);
    
    // 合并路径
    CGContextClosePath(context);
    
    // 设置颜色
    [JYFColor(252, 218, 0) set];
    
    CGContextFillPath(context);
    
}
#pragma mark - 绘制嘴巴
void drawMouse(CGContextRef context,CGRect rect){
    // 设置一个控制点
    CGFloat controlX = rect.size.width * 0.5;
    CGFloat controlY = rect.size.height * 0.4;
    
    // 设置当前点
    CGFloat marginX = 20;
    CGFloat marginY = 10;
    CGFloat currentX = controlX - marginX;
    CGFloat currentY = controlY - marginY;
    CGContextMoveToPoint(context, currentX, currentY);
    
    // 设置结束点
    CGFloat endX = controlX + marginX;
    CGFloat endY = currentY;
    
    // 绘制贝塞尔曲线
    CGContextAddQuadCurveToPoint(context, controlX, controlY, endX, endY);
    
    // 设置颜色
    [[UIColor blackColor] set];
    
    // 显示
    CGContextStrokePath(context);
}
#pragma mark - 绘制眼睛
void drawEyes(CGContextRef context,CGRect rect){
    // 黑色绑带
    
    // 起点
    CGFloat startPointX = rect.size.width * 0.5 - JYFRadius;
    CGFloat startPointY = JYFTopY;
    CGContextMoveToPoint(context, startPointX, startPointY);
    
    // 结束点
    CGFloat endX = startPointX + JYFRadius * 2;
    CGFloat endY = startPointY;
    CGContextAddLineToPoint(context, endX, endY);
    CGContextSetLineWidth(context, 15);
    
    [[UIColor blackColor] set];
    CGContextStrokePath(context);
    
    // 灰色镜框
    [JYFColor(61, 62, 66) set];
    CGFloat kuangRadius = JYFRadius * 0.4;
    CGFloat kuangX = rect.size.width * 0.5 - kuangRadius;
    CGFloat kuangY = startPointY;
    CGContextAddArc(context, kuangX + 25, kuangY, kuangRadius, 0, M_PI * 2, 0);
    CGContextFillPath(context);
    
    // 白色镜框
    [[UIColor whiteColor] set];
    CGFloat whiteRadius = kuangRadius * 0.7;
    CGFloat whiteX = kuangX;
    CGFloat whiteY = kuangY;
    CGContextAddArc(context, whiteX + 25, whiteY, whiteRadius, 0, M_PI * 2, 0);
    CGContextFillPath(context);
    
    // 黑色眼珠
    [[UIColor blackColor] set];
    CGFloat eyeRadius = whiteRadius * 0.5;
    CGFloat eyeX = whiteX;
    CGFloat eyeY = whiteY;
    CGContextAddArc(context, eyeX + 25, eyeY, eyeRadius, 0, M_PI * 2, 1);
    CGContextFillPath(context);
}

@end

参考资料

  1. Quartz 2D编程指南(1) - 概览
  2. 极客学院视频·IOS项目开发实战——绘制小黄人
  3. Quartz2D

Panda
2016-07-07

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

推荐阅读更多精彩内容