Quartz2D_Day1_简单认识及使用

小编会开展一系列的Quartz2D专题的学习,而所记录的内容原文都是出自大师博客文顶顶感兴趣的可以直接看原文。

Quartz 2D 是一个二维绘图引擎,同时支持iOS和Mac系统
Quartz2D 能完成的工作

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

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

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

Quartz 2D 提供了一下几种类型的Graphics Context:

  • Bitmap Graphics Context
  • PDF Graphics Context
  • Window Graphics Context
  • Layer Graphics Context
  • Printer Graphics Context
image.png

学习目标:

* 如何利用Quartz 2D自定义view?(自定义UI控件)

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

* 如何利用Quart 2D绘制东西到view上?

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

// 注意:drawRect如果是手动调用的话,它是不会创建跟view相关联的上下文
// 刷新调用drawRect方法,可以手动调用:[self setNeedDisplay]

/// 作用:专门用来绘图
/// 什么时候调用:当View显示的时候调用
/// @param rect 当前View的bounds
- (void)drawRect:(CGRect)rect {
}

实例(2020年4月12日更新)

推荐使用UIBezierPath描述路径,更加灵活
比如画一个矩形

- (void)drawRect:(CGRect)rect {
    // 1.0 获取上下文
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    // 2.0 描述路径
    UIBezierPath *path = [UIBezierPath bezierPathWithRect:CGRectMake(50, 50, 100, 100)];
    [[UIColor redColor] set];
    // 3.0 把路径添加到上下文
    CGContextAddPath(ctx, path.CGPath);
    // 4.0 把上下文的内容渲染到view上
    CGContextFillPath(ctx);
}

   /**
            备注:已下的方法也能达到同样的效果
        因为使用[path fill] 底层的实现原理如下
        1.获取上下文-->2.描述路径-->3.把路径添加到上下文-->4.把上下文的内容渲染到layer上
     */
- (void)drawRect:(CGRect)rect {
    UIBezierPath *path = [UIBezierPath bezierPathWithRect:CGRectMake(50, 50, 100, 100)];
    [[UIColor redColor] set];
    [path fill];
}

bitMap位图上下文

bitMap位图上下文须手动去开启,开启多大的上下文,生成的图片就是多大,并且需要手动关闭上下文

`实例`:在一张图片上添加一个文字“@Liven”的文字水印

@interface ViewController ()
@property (nonatomic, strong, readwrite) UIImageView      *imageView;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 1. 加载图片
    UIImage *image = [UIImage imageNamed:@"测试.png"];
    // 2. 创建bitMap位图上下文
    // opaque 不透明度
    // scale 0表示根据屏幕的缩放比例
    UIGraphicsBeginImageContextWithOptions(image.size, 0, 0);
    // 3.将图片添加到位图上下文中
    [image drawAtPoint:CGPointZero];
    // 4.添加水印文字比如:@Liven
    NSString *str = @"@@Liven";
    [str drawAtPoint:CGPointZero withAttributes:nil];
    // 5.把上下文当中绘制的所有内容,生成一张图片
    UIImage *newImage = UIGraphicsGetImageFromCurrentImageContext();
    // 6.关闭位图上下文
    UIGraphicsEndImageContext();
    
    [self.view addSubview:self.imageView];
    self.imageView.image = newImage;
}

- (UIImageView *)imageView {
    if (!_imageView) {
        _imageView = [[UIImageView alloc]initWithFrame:self.view.bounds];
    }
    return _imageView;
}

@end


画图的步骤分三步
1、获取载体图形上下文:CGContextRef
在UIView的drawRect方法中获取的是关联Layer层的图形上下文
2、开始绘图
设置图形的形状,和图形状态(颜色、线宽等)
***3、展现图形****
即渲染图形,渲染的方式有两种:
CGContextStrokePath(ctx) ----空心
CGContextFillPath(ctx) ----实心

以下是Quartz2D的基本使用

#import "LineView.h"

@implementation LineView

- (void)drawRect:(CGRect)rect{
    
    //1.画直线
//    [self drawLine];
    
    //2.三角形
//    [self drawTriangle];
    
    //3.四边形
//    [self drawSquare];
    
    //4.圆
//    [self drawCircle];
    
    //5.扇形
    [self drawRoundCircle];
    
}

#pragma mark - 直线
- (void)drawLine{
    //1.取得和当前视图相关联的图形上下文(因为图形上下文决定绘制的输出目标)
    //如果是在drawRect方法中调用UIGraphicsGetCurrentContext方法获取出来的就是Layer的上下文
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    
    //2.绘制直线,保存绘图信息
    //  设置起点
    CGContextMoveToPoint(ctx, 20, 100);
    //  设置终点
    CGContextAddLineToPoint(ctx, 300, 100);
    //  设置绘图的状态
    //  设置线条的颜色为蓝色
    CGContextSetRGBStrokeColor(ctx, 0, 1, 0, 1);
    //  设置线条宽度
    CGContextSetLineWidth(ctx, 15);
    //  设置线条起点和终点的样式为圆角 cap:帽
    CGContextSetLineCap(ctx, kCGLineCapRound);
    //  设置线条的转角的样式为圆角 join:连接点
    CGContextSetLineJoin(ctx, kCGLineJoinRound);
    
    //3、渲染 (绘制出一条空心的线)
    //  注意:线条是不能渲染为实心的 CGContextFillPath(ctx)
    CGContextStrokePath(ctx);
}

#pragma mark - 三角形
- (void)drawTriangle{
    //1.获取图形上下文
    CGContextRef ctxt = UIGraphicsGetCurrentContext();
    
    //2.绘制三角形
    CGContextMoveToPoint(ctxt, 20, 100);
    CGContextAddLineToPoint(ctxt, 20, 300);
    CGContextAddLineToPoint(ctxt, 200, 200);
    //  关闭起点和终点
    //  闭合起点和终点还有一种方式就是:在终点到起点处再画多一条线
    CGContextClosePath(ctxt);
    
    //3.渲染图形到layer上
    CGContextStrokePath(ctxt);
    
}
#pragma mark - 画四边形
- (void)drawSquare{
    //1.获取图形上下文
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    
    //2.绘制图形
    CGContextAddRect(ctx, CGRectMake(20, 20, 150, 100));
    //  设置绘图状态 (状态的设置必须跟图形的类型一致(空心还是实心))
    //  空心
//    CGContextSetRGBStrokeColor(ctx, 1, 0, 0, 1);
    //  实心
    CGContextSetRGBFillColor(ctx, 1, 0, 0, 1);
    //  第二种设置图形状态的方法
//    [[UIColor redColor]setStroke];    //空心
//    [[UIColor redColor]setFill]   //实心

    //3.渲染
    //  空心的
//    CGContextStrokePath(ctx);
    //  实心的
    CGContextFillPath(ctx);
}

#pragma mark - 画圆
- (void)drawCircle{
    //1、
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    
    //2、
    // 第一个和第二个参数是设置圆心,第三个是半径,第四个是设置起始角度,第五个是设置终止角度,第六个是顺时针还是逆时针
    // 0:顺时针   1:逆时针
    CGContextAddArc(ctx, 100, 100, 50, 0, 2*M_PI, 0);
    //用椭圆的方法画圆,矩形的内切圆
    //CGContextAddEllipseInRect(ctx, CGRectMake(250, 300, 100, 100));
    //3.渲染
    CGContextStrokePath(ctx);   //空心
    CGContextFillPath(ctx);
}

#pragma mark - 扇形
- (void)drawRoundCircle{
    //1.
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    
    //2.
    CGContextMoveToPoint(ctx, 100, 100);
    CGContextAddLineToPoint(ctx, 100, 150);

    CGContextAddArc(ctx, 100, 100, 50, M_PI_2, M_PI, 0);
    CGContextClosePath(ctx)

    //3.
    CGContextStrokePath(ctx);

}

文字和图片

#import "LabelAndImageView.h"

@implementation LabelAndImageView

- (void)drawRect:(CGRect)rect{
    
    //1.文字
//    [self drawString];

    //2.图片
    [self drawImage];
}

#pragma mark - 绘制文字
- (void)drawString{
    NSString *str = @"好像好像好像好像好像好像好像好像好像好像好像好像好像";
    
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    
    CGContextAddRect(ctx, CGRectMake(10, 100, 300, 200));
    [[UIColor redColor]setStroke];
    
    CGContextStrokePath(ctx);
    
    NSMutableDictionary *md = [NSMutableDictionary dictionary];
    //设置文字颜色
    md[NSForegroundColorAttributeName] = [UIColor blackColor];
    //设置文字背景颜色
    md[NSBackgroundColorAttributeName] = [UIColor lightGrayColor];
    //设置文字大小
    md[NSFontAttributeName] = [UIFont systemFontOfSize:18];
    
    //绘制文字,会自动换行,当文字超出rect的范围就不显示
    [str drawInRect:CGRectMake(10, 100, 300, 200) withAttributes:md];
    //在某一点插入文字,但是文字不会换行
//    [str drawAtPoint:CGPointMake(10, 100) withAttributes:md];

}

- (void)drawImage{
    
    //1.加载图片到内存
    UIImage *image = [UIImage imageNamed:@"image.jpeg"];
    
    //2.利用drawAsPatternInRect方法绘制图片到layer,是通过平铺原有照片
//    [image drawAsPatternInRect:self.frame];
    //drawInRect是拉伸压缩图片尺寸填充
//    [image drawInRect:self.frame];
    //以原图的尺寸,在某点插入图片
    [image drawAtPoint:CGPointMake(20, 20)];
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,657评论 6 505
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,889评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,057评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,509评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,562评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,443评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,251评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,129评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,561评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,779评论 3 335
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,902评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,621评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,220评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,838评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,971评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,025评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,843评论 2 354

推荐阅读更多精彩内容

  • Quartz2D以及drawRect的重绘机制字数1487 阅读21 评论1 喜欢1一、什么是Quartz2D Q...
    PurpleWind阅读 773评论 0 3
  • 简述: 1、Quartz2D是什么Quartz2D是二维绘图引擎,同时支持IOS和Mac 2、Quartz2D能做...
    LitterL阅读 641评论 0 6
  • 第一步:先科普一下基础知识: Core Graphics是基于C的API,可以用于一切绘图操作 Core Grap...
    真爱要有你才完美阅读 2,447评论 0 1
  • 大家跟着导游乘坐自由女神号游轮,在游轮上观赏代表美国国魂标志的【自由女神像】 看到大家兴奋的合影拍照,我想之前在另...
    Mary妹善阅读 230评论 0 0
  • 前段时间小易脸上长了很多痘痘。开始吃了中药,后来又弄了擦的药。 但是效果都不是特别的理想。然后又去找一位奶奶按摩,...
    俊妈李利阅读 179评论 0 0