Core Graphics Framework

一、简介

  对于 Core Graphics Framework,可以从两个方面初步感知,一是它在绘图系统中的地位,二是官方的介绍。

1、iOS绘图系统主要分为以下几层:

UIKit:最常用的视图框架,封装度最高,都是OC对象,处理与用户的交互

CoreAnimation:提供强大的2D和3D动画效果

CoreGraphics:主要绘图系统,常用于绘制自定义视图,纯C的API,使用Quartz2D做引擎

OpenGL-ES:主要用于游戏绘制,它是一套编程规范,具体由设备制造商实现

Graphics Hardware:图形硬件即GPU

image
  由图可见,最底层是图形硬件层Graphics Hardware;上层是OpenGL和CoreGraphics,提供一些接口来访问GPU;再上层是CoreAnimation,主要封装了一套视图动画的API;最上面的UIKit属于应用层,处理与用户的交互。

2、关于 Core Graphics 官方介绍及其翻译(非专业)如下:

  The Core Graphics framework is based on the Quartz advanced drawing engine. It provides low-level, lightweight 2D rendering with unmatched output fidelity. You use this framework to handle path-based drawing, transformations, color management, offscreen rendering, patterns, gradients and shadings, image data management, image creation, and image masking, as well as PDF document creation, display, and parsing.

  In macOS, Core Graphics also includes services for working with display hardware, low-level user input events, and the windowing system.

  Core Graphics framework 是一个基于Quartz的更先进的绘图引擎。它提供了低级别、轻量级、高保真的2D渲染。该框架可以用于操作基于路径的绘图、变换,颜色管理,脱屏渲染,模板,渐变,遮罩,图像数据管理,图像的创建、遮罩以及PDF文档的创建、显示和分析。

  在MacOS中,Core Graphics framework 还提供了显示硬件工作的服务、低级的用户输入事件和窗口系统。

二、图形上下文

  从官方简介中可以知道,Core Graphics 的主要功能是做视图渲染,简单地说,就是可以用它来绘制图形、变换图形、处理颜色、处理图片、操作PDF。

以下面的界面为例:

image
  对于这个界面的一般做法就是添加几个 label 和几个 button,再添加几个 view 充当表格的边线,这种做法没有什么技术难度,就是麻烦点,大家都会做。这里我们用 Core Graphics 实现,分为画线条、画图片、画文字三种,先看以下代码:
image
  这里只是简单贴了部分代码(文章末尾有Demo地址),作引题之用。其中需要注意以下几点:

(1) Core Graphics 的相关代码都以 CG 开头。

(2) 代码写在了 - (void)drawRect:(CGRect)rect { ... } 方法中。

(3) CGContextRef context = UIGraphicsGetCurrentContext(); 获取到的局部变量 context 被后面的代码多次用到。

可见使用 Core Graphics 首先要有一个CGContextRef,这个 CGContextRef 就是图形上下文,它相当于画板,封装了绘图信息(画了什么)和绘图状态(线条大小、颜色等),更决定了图形要绘制到什么地方。Core Graphics 提供了 5 种类型的图形上下文:

(1) Layer Graphics Context :图层上下文

(2) Bitmap Graphics Context :位图上下文

(3) PDF Graphics Context :PDF上下文

(4) Window Graphics Context :图形设备上下文

(5) Post Graphics Context :打印上下文

下面说说前3种,对于图形设备上下文和打印上下文,一般很少用到,就不做介绍了。

1、图层上下文

  上面代码中,CGContextRef context = UIGraphicsGetCurrentContext(); 这句代码意思是获取当前上下文,即位于上下文栈栈顶的上下文,此时获取到的就是图层上下文,不同于其它上下文,它是由系统提供的,只能获取,不能手动创建,并且只能在 drawRect: 方法中获取。大概流程是,调用 drawRect: 方法时,系统生成图层上下文,并压入上下文栈,然后手动获取该图层上下文,又被各种使用,待绘制完成后,这个上下文就把绘制好的图形输出到对应的图层(即CALayer)上。

  需要解释一下 UIView 和 CALayer ,它们的继承关系是 UIView —> UIResponder —> NSObject  /  CALayer —> NSObject,而且 CALayer 是 UIView 的一个成员变量,所以 UIView 不仅可以响应交互事件,还能显示 CALayer 的内容。当要显示时,CALayer 会准备好一个 CGContextRef(图层上下文),然后调用它的 delegate (这里就是 UIView)的  drawLayer:inContext: 方法,并且传入已经准备好的 CGContextRe f对象,在drawLayer:inContext:方法中 UIView 又会调用自己的 drawRect: 方法。

2、位图上下文

  位图上下文一般用于图片处理,它比较灵活,可以手动创建,可以在任何地方使用。用法如下:
image

3、PDF上下文

PDF上下文也可以手动创建,可以在任何地方使用。生成PDF文档的一般步骤:

image

三、Core Graphics 应用

Core Graphics 一般用于以下操作(文章末尾有Demo地址):

(1) 绘制图形: 线条\矩形\圆\弧等

(2) 绘制文字

(3) 绘制\生成图片(图像)

(4) 截图\裁剪图片

(5) 读取\生成PDF

(6) 自定义UI控件

四、Core Graphics 部分API介绍

1、UIGraphics.h

// 获取当前栈顶的图层上下文(返回 not retained 对象)
CGContextRef __nullable UIGraphicsGetCurrentContext(void)

// 图层上下文入栈
void UIGraphicsPushContext(CGContextRef context)

// 图层上下文出栈
void UIGraphicsPopContext(void)

// 在当前图层上下文中填充指定的矩形区域,可以指定填充模式
void UIRectFillUsingBlendMode(CGRect rect, CGBlendMode blendMode)

// 在当前图形上下文中,用填充颜色和 kCGBlendModeCopy 混合模式来填充指定区域
void UIRectFill(CGRect rect)

// 绘制一个矩形框,可以指定绘制模式
void UIRectFrameUsingBlendMode(CGRect rect, CGBlendMode blendMode)

// 用填充颜色和 kCGBlendModeCopy 混合模式绘制一个矩形框
void UIRectFrame(CGRect rect)

// 剪切矩形区域内的路径
void UIRectClip(CGRect rect)

// 位图上下文
// 开启位图上下文
void UIGraphicsBeginImageContext(CGSize size)

// 开启位图上下文
void UIGraphicsBeginImageContextWithOptions(CGSize size, BOOL opaque, CGFloat scale)
// 获取位图上下文中绘制的图片
UIImage * __nullable UIGraphicsGetImageFromCurrentImageContext(void)

// 关闭位图上下文
void UIGraphicsEndImageContext(void)

// PDF上下文
// 开启 PDF上下文,path:PDF存储的路径,bounds:绘制的区域,documentInfo:PDF配置
BOOL UIGraphicsBeginPDFContextToFile(NSString *path, CGRect bounds, NSDictionary * __nullable documentInfo)

// 开启 PDF上下文,data:存储的路径,bounds:绘制的区域,documentInfo:PDF配置
void UIGraphicsBeginPDFContextToData(NSMutableData *data, CGRect bounds, NSDictionary * __nullable documentInfo)

// 关闭 PDF上下文
void UIGraphicsEndPDFContext(void)

// 开启一个 PDF page,page 有默认大小
void UIGraphicsBeginPDFPage(void)

// 开启一个 PDF page,可以对 page 的大小进行编辑,也可以设置其它属性
void UIGraphicsBeginPDFPageWithInfo(CGRect bounds, NSDictionary * __nullable pageInfo)

// 获取上下文的可绘制区域
CGRect UIGraphicsGetPDFContextBounds(void)

// 指定区域设置 url 链接,即点击此区域可以跳转到对应的链接
void UIGraphicsSetPDFContextURLForRect(NSURL *url, CGRect rect)

// 设置当跳转到本页时要显示的点
void UIGraphicsAddPDFContextDestinationAtPoint(NSString *name, CGPoint point)

// 设置当跳转到本页时要显示的区域
void UIGraphicsSetPDFContextDestinationForRect(NSString *name, CGRect rect)

2、CGContext.h

管理图形上下文

// 保存当前上下文的绘制状态
void CGContextSaveGState(CGContextRef cg_nullable c)

// 还原上次保存的上下文的绘制状态
void CGContextRestoreGState(CGContextRef cg_nullable c)

// 获取上下文 ID
CFTypeID CGContextGetTypeID(void)

// 图形上下文的引用计数+1
CGContextRef cg_nullable CGContextRetain(CGContextRef cg_nullable c)

// 图形上下文的引用计数-1
void CGContextRelease(CGContextRef cg_nullable c)

// 强制所有挂起的绘图操作在一个窗口上下文中立即被渲染到目标设备
void CGContextFlush(CGContextRef cg_nullable c)

// 将一个窗口的图像上下文内容更新,即所有的绘图操作都会在下次同步到窗口上
void CGContextSynchronize(CGContextRef cg_nullable c)

// 开始一个透明层,auxiliaryInfo:一般传NULL
void CGContextBeginTransparencyLayer(CGContextRef cg_nullable c, CFDictionaryRef __nullable auxiliaryInfo)

// 开始一个指定区域的透明层,auxiliaryInfo:一般传NULL
void CGContextBeginTransparencyLayerWithRect(CGContextRef cg_nullable c, CGRect rect, CFDictionaryRef __nullable auxInfo)

// 结束透明层
void CGContextEndTransparencyLayer(CGContextRef cg_nullable c)

CTM变换

// 上下文作缩放变换,sx:X轴的缩放系数, sy:Y轴的缩放系数
void CGContextScaleCTM(CGContextRef cg_nullable c, CGFloat sx, CGFloat sy)

// 上下文作偏移变换,tx:X轴的偏移量, ty:Y轴的偏移量
void CGContextTranslateCTM(CGContextRef cg_nullable c, CGFloat tx, CGFloat ty)

// 上下文作旋转变换,angle:旋转的角度
void CGContextRotateCTM(CGContextRef cg_nullable c, CGFloat angle)

// 上下文合并一个变换,transform:2D变换
void CGContextConcatCTM(CGContextRef cg_nullable c, CGAffineTransform transform)

// 获取上下文的变换
CGAffineTransform CGContextGetCTM(CGContextRef cg_nullable c)

设置路径的绘制状态

// 设置线宽
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)

// 设置虚线的样式,phase:从第几个点开始画起,lengths:虚线的样式 [10,5]表示每隔5个点就画一条长度为10的线,count:lengths数组的长度
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)

// 设置填充颜色
void CGContextSetFillColorWithColor(CGContextRef cg_nullable c, CGColorRef cg_nullable color)

// 设置描边颜色
void CGContextSetStrokeColorWithColor(CGContextRef cg_nullable c, CGColorRef cg_nullable color)

// 设置填充的颜色通道,space:颜色通道
void CGContextSetFillColorSpace(CGContextRef cg_nullable c, CGColorSpaceRef cg_nullable space)

// 设置描边的颜色通道,space:颜色通道
void CGContextSetStrokeColorSpace(CGContextRef cg_nullable c, CGColorSpaceRef cg_nullable space)

// 设置填充颜色,components:填充颜色的 RGBA 值
void CGContextSetFillColor(CGContextRef cg_nullable c, const CGFloat * cg_nullable components)

// 设置描边颜色,components:描边颜色的 RGBA 值
void CGContextSetStrokeColor(CGContextRef cg_nullable c, const CGFloat * cg_nullable components)

// 设置填充图案模式,pattern:填充图案模式,components:描边颜色的 RGBA 值
void CGContextSetFillPattern(CGContextRef cg_nullable c, CGPatternRef cg_nullable pattern, const CGFloat * cg_nullable components)

// 设置描边图案模式,pattern:描边图案模式,components:描边颜色的 RGBA 值
void CGContextSetStrokePattern(CGContextRef cg_nullable c, CGPatternRef cg_nullable pattern, const CGFloat * cg_nullable components)

// 设置图案模式的平铺范围,防止铺设一半的情况
void CGContextSetPatternPhase(CGContextRef cg_nullable c, CGSize phase)

// 设置填充颜色,依赖灰度颜色空间,gray:灰度,alpha:透明度
void CGContextSetGrayFillColor(CGContextRef cg_nullable c, CGFloat gray, CGFloat alpha)

// 设置描边颜色,依赖灰度颜色空间,gray:灰度,alpha:透明度
void CGContextSetGrayStrokeColor(CGContextRef cg_nullable c, CGFloat gray, CGFloat alpha)

// 设置填充颜色,依赖RGBA的颜色空间
void CGContextSetRGBFillColor(CGContextRef cg_nullable c, CGFloat red, CGFloat green, CGFloat blue, CGFloat alpha)

// 设置描边颜色,依赖RGBA的颜色空间
void CGContextSetRGBStrokeColor(CGContextRef cg_nullable c, CGFloat red, CGFloat green, CGFloat blue, CGFloat alpha)

// 设置填充颜色,依赖CMYK颜色空间
void CGContextSetCMYKFillColor(CGContextRef cg_nullable c, CGFloat cyan, CGFloat magenta, CGFloat yellow, CGFloat black, CGFloat alpha)

// 设置描边颜色,依赖CMYK颜色空间
void CGContextSetCMYKStrokeColor(CGContextRef cg_nullable c, CGFloat cyan, CGFloat magenta, CGFloat yellow, CGFloat black, CGFloat alpha)

// 设置渲染模式,intent:渲染模式(如:饱和度匹配),枚举值
void CGContextSetRenderingIntent(CGContextRef cg_nullable c, CGColorRenderingIntent intent)

构建路径

// 从新构建路径
void CGContextBeginPath(CGContextRef cg_nullable c)

// 移到某一点,准备画线
void CGContextMoveToPoint(CGContextRef cg_nullable c, CGFloat x, CGFloat y)

// 画线,从当前点画到新传入的点,x:点的X值, y:点的Y值
void CGContextAddLineToPoint(CGContextRef cg_nullable c, CGFloat x, CGFloat y)

// 画贝塞尔三次曲线,cp1x/cp1y:控制点1的X/Y坐标, cp2x/cp2y:控制点2的X/Y坐标, x/y:终点的X/Y坐标
void CGContextAddCurveToPoint(CGContextRef cg_nullable c, CGFloat cp1x, CGFloat cp1y, CGFloat cp2x, CGFloat cp2y, CGFloat x, CGFloat y)

// 画贝塞尔二次曲线,cpx/cpy:控制点的X/Y坐标,x/y:终点的X/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)

// 画多个矩形,rects:数组,形如:CGRect rects[] = {CGRect1,CGRect2},count:rects数组内容的数量
void CGContextAddRects(CGContextRef cg_nullable c, const CGRect * __nullable rects, size_t count)

// 画多条线,points:数组,形如:CGPoint points[] = {CGPoint1,CGPoint2},count:rects数组内容的数量
void CGContextAddLines(CGContextRef cg_nullable c, const CGPoint * __nullable points, size_t count)

// 画椭圆,rect:绘制的椭圆的大小
void CGContextAddEllipseInRect(CGContextRef cg_nullable c, CGRect rect)

// 画圆,x/y:圆心坐标点,radius:半径,startAngle/endAngle:起始角度/终止角度,clockwise:顺时针/逆时针
void CGContextAddArc(CGContextRef cg_nullable c, CGFloat x, CGFloat y, CGFloat radius, CGFloat startAngle, CGFloat endAngle, int clockwise)

// 画圆角,从当前点到点(x1,y1),再到点(x2,y2)构成一个角,把这个角变成圆角,去掉切点到点(x2,y2)间的线段,radius:圆角半径
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)

// 使用绘制当前路径时覆盖的区域作为当前CGContextRef中的新路径
void CGContextReplacePathWithStrokedPath(CGContextRef cg_nullable c)

绘制路径

// 绘制路径,mode:绘图模式
void CGContextDrawPath(CGContextRef cg_nullable c, CGPathDrawingMode mode)

// 填充模式绘制路径
void CGContextFillPath(CGContextRef cg_nullable c)

// 使用奇偶填充规则绘制路径
void CGContextEOFillPath(CGContextRef cg_nullable c)

// 描边模式绘制路径
void CGContextStrokePath(CGContextRef cg_nullable c)

// 填充模式绘制矩形
void CGContextFillRect(CGContextRef cg_nullable c, CGRect rect)

// 填充模式绘多个矩形,rects:数组,形如:CGRect rects[] = {CGRect1,CGRect2},count:rects数组内容的数量
void CGContextFillRects(CGContextRef cg_nullable c, const CGRect * __nullable rects, size_t count)

// 描边模式绘制矩形
void CGContextStrokeRect(CGContextRef cg_nullable c, CGRect rect)

// 描边模式绘制矩形,并设置线宽
void CGContextStrokeRectWithWidth(CGContextRef cg_nullable c, CGRect rect, CGFloat width)

// 清除指定区域内已经绘制的内容
void CGContextClearRect(CGContextRef cg_nullable c, CGRect rect)

// 填充模式绘制椭圆,rect:椭圆所在的区域
void CGContextFillEllipseInRect(CGContextRef cg_nullable c, CGRect rect)

// 描边模式绘制椭圆,rect:椭圆所在的区域
void CGContextStrokeEllipseInRect(CGContextRef cg_nullable c, CGRect rect)

// 描边模式绘制多条线段,points:数组,内容为偶数个,形如:CGPoint points[] = {CGPoint1,CGPoint2},count:rects数组内容的数量
void CGContextStrokeLineSegments(CGContextRef cg_nullable c, const CGPoint * __nullable points, size_t count)

操作路径

// 当前路径是否为空
bool CGContextIsPathEmpty(CGContextRef cg_nullable c)

// 获取当前路径中最后绘制的点
CGPoint CGContextGetPathCurrentPoint(CGContextRef cg_nullable c)

// 获取包含当前路径的最小矩形
CGRect CGContextGetPathBoundingBox(CGContextRef cg_nullable c)

// 复制当前路径
CGPathRef __nullable CGContextCopyPath(CGContextRef cg_nullable c)

// 当前路径是否包含指定的点,point:指定的点,mode:绘图模式
bool CGContextPathContainsPoint(CGContextRef cg_nullable c, CGPoint point, CGPathDrawingMode mode)

// 设置剪切路径(使用非零绕数规则)
void CGContextClip(CGContextRef cg_nullable c)

// 设置剪切路径(使用奇偶规则)
void CGContextEOClip(CGContextRef cg_nullable c)

// 重新设置剪切路径(使用非零绕数规则)
void CGContextResetClip(CGContextRef c);

// 设置剪切的区域,且绘制图片到此区域,rect:区域,mask:图片
void CGContextClipToMask(CGContextRef cg_nullable c, CGRect rect, CGImageRef cg_nullable mask)

// 获取剪切路径所在的最小区域
CGRect CGContextGetClipBoundingBox(CGContextRef cg_nullable c)

// 设置剪切的矩形区域
void CGContextClipToRect(CGContextRef cg_nullable c, CGRect rect)

// 设置多个剪切的矩形区域,rects:数组,形如:CGRect rects[] = {CGRect1,CGRect2},count:rects数组内容的数量

void CGContextClipToRects(CGContextRef cg_nullable c, const CGRect *  rects, size_t count)
绘制文本
// 设置字符间距
void CGContextSetCharacterSpacing(CGContextRef cg_nullable c, CGFloat spacing)

// 设置文本的绘制位置
void CGContextSetTextPosition(CGContextRef cg_nullable c, CGFloat x, CGFloat y)

// 获取当前上下文的文本的绘制位置
CGPoint CGContextGetTextPosition(CGContextRef cg_nullable c)

// 设置文本变换
void CGContextSetTextMatrix(CGContextRef cg_nullable c, CGAffineTransform t)

// 获取当前上下文的文本变换
CGAffineTransform CGContextGetTextMatrix(CGContextRef cg_nullable c)

// 设置文本绘制模式
void CGContextSetTextDrawingMode(CGContextRef cg_nullable c, CGTextDrawingMode mode)

// 设置字体
void CGContextSetFont(CGContextRef cg_nullable c, CGFontRef cg_nullable font)

// 设置字体大小
void CGContextSetFontSize(CGContextRef cg_nullable c, CGFloat size)

// 指定位置绘制对应的字符,CGGlyph数组个数和CGPoint数组个数要对应
void CGContextShowGlyphsAtPositions(CGContextRef cg_nullable c, const CGGlyph * cg_nullable glyphs, const CGPoint * cg_nullable Lpositions, size_t count)

// 设置抗锯齿开启或关闭
void CGContextSetShouldAntialias(CGContextRef cg_nullable c, bool shouldAntialias)

// 设置抗锯齿开启或关闭
void CGContextSetAllowsAntialiasing(CGContextRef cg_nullable c, bool allowsAntialiasing)

// 设置字体平滑
void CGContextSetShouldSmoothFonts(CGContextRef cg_nullable c, bool shouldSmoothFonts)

// 设置字体平滑
void CGContextSetAllowsFontSmoothing(CGContextRef cg_nullable c, bool allowsFontSmoothing)

// 设置启动或禁用子像素定位
void CGContextSetShouldSubpixelPositionFonts(CGContextRef cg_nullable c, bool shouldSubpixelPositionFonts)

// 设置是否允许子像素定位
void CGContextSetAllowsFontSubpixelPositioning(CGContextRef cg_nullable c, bool allowsFontSubpixelPositioning)

// 设置启动或禁用子像素细化
void CGContextSetShouldSubpixelQuantizeFonts(CGContextRef cg_nullable c, bool shouldSubpixelQuantizeFonts)

// 设置是否允许子像素细化
void CGContextSetAllowsFontSubpixelQuantization(CGContextRef cg_nullable c, bool allowsFontSubpixelQuantization)

// 设置字体,相当于 CGContextSetFont+CGContextSetFontSize
void CGContextSelectFont(CGContextRef cg_nullable c, const char * cg_nullable name, CGFloat size, CGTextEncoding textEncoding)

// 在当前点绘制指定文本
void CGContextShowText(CGContextRef cg_nullable c, const char * cg_nullable string, size_t length)

// 在指定点绘制指定文本
void CGContextShowTextAtPoint(CGContextRef cg_nullable c, CGFloat x, CGFloat y, const char * cg_nullable string, size_t length)

// 在当前点绘制一个数组的字形
void CGContextShowGlyphs(CGContextRef cg_nullable c, const CGGlyph * __nullable g, size_t count)

// 在指定点绘制一个数组的字形
void CGContextShowGlyphsAtPoint(CGContextRef cg_nullable c, CGFloat x, CGFloat y, const CGGlyph * __nullable glyphs, size_t count)

// 绘制具有不同的偏移量的一个数组字形
void CGContextShowGlyphsWithAdvances(CGContextRef cg_nullable c, const CGGlyph * __nullable glyphs, const CGSize * __nullable advances, size_t count)

绘制图像

// 绘制图片到指定区域
void CGContextDrawImage(CGContextRef cg_nullable c, CGRect rect, CGImageRef cg_nullable image)

// 平铺模式绘制图片到指定区域
void CGContextDrawTiledImage(CGContextRef cg_nullable c, CGRect rect, CGImageRef cg_nullable image)

// 获取当前上下文插值的质量水平(插值(Interpolation)是在不天生像素的环境下增长图像像素大小的一种方法,在周围像素色彩的根蒂根基上用算术公式计算亡失像素的色彩。)
CGInterpolationQuality CGContextGetInterpolationQuality(CGContextRef cg_nullable c)

// 设置插入像素的质量等级(填补对丢失像素的一种方法)
void CGContextSetInterpolationQuality(CGContextRef cg_nullable c, CGInterpolationQuality quality)

绘制PDF

// 绘制一页PDF
void CGContextDrawPDFPage(CGContextRef cg_nullable c, CGPDFPageRef cg_nullable page)

// 基于页面的图形上下文中开始了新的一页
void CGContextBeginPage(CGContextRef cg_nullable c, const CGRect * __nullable mediaBox)

// 在基于页面的图形上下文结束当前的页面
void CGContextEndPage(CGContextRef cg_nullable c)

// 绘制PDF文档
void CGContextDrawPDFDocument(CGContextRef cg_nullable c, CGRect rect, CGPDFDocumentRef cg_nullable document, int page)

绘制阴影渐变

// 设置彩色阴影,offset:阴影的位置,blur:模糊度,color:阴影颜色
void CGContextSetShadowWithColor(CGContextRef cg_nullable c, CGSize offset, CGFloat blur, CGColorRef __nullable color)

// 设置默认阴影,offset:阴影的位置,blur:模糊度
void CGContextSetShadow(CGContextRef cg_nullable c, CGSize offset, CGFloat blur)

// 绘制线性渐变色,gradient:渐变的颜色,startPoint:起点,endPoint:终点,options:渐变是否超越起始点和终止点
void CGContextDrawLinearGradient(CGContextRef cg_nullable c, CGGradientRef cg_nullable gradient, CGPoint startPoint, CGPoint endPoint, CGGradientDrawingOptions options)

// 绘制扩散渐变色,gradient:渐变的颜色,startCenter/startRadius:起始的圆心/半径,endCenter/endRadius:结束的圆心/半径,options:渐变是否超越起始点和终止点
void CGContextDrawRadialGradient(CGContextRef cg_nullable c, CGGradientRef cg_nullable gradient, CGPoint startCenter, CGFloat startRadius, CGPoint endCenter, CGFloat endRadius, CGGradientDrawingOptions options)

// 绘制CGShadingRef对象,CGShadingRef是渐变色对象
void CGContextDrawShading(CGContextRef cg_nullable c, cg_nullable CGShadingRef shading)

设备空间和用户空间之间的转换

// 获取Quartz转换用户空间和设备空间的仿射变换
CGAffineTransform CGContextGetUserSpaceToDeviceSpaceTransform(CGContextRef cg_nullable c)

// 用户空间的点转换到设备空间,像素要变化(乘/除 分辨率)
CGPoint CGContextConvertPointToDeviceSpace(CGContextRef cg_nullable c, CGPoint point)

// 设备空间的点转换到用户空间
CGPoint CGContextConvertPointToUserSpace(CGContextRef cg_nullable c, CGPoint point)

// 用户空间的CGSize转换到设备空间
CGSize CGContextConvertSizeToDeviceSpace(CGContextRef cg_nullable c, CGSize size)

// 设备空间的CGSize转换到用户空间
CGSize CGContextConvertSizeToUserSpace(CGContextRef cg_nullable c, CGSize size)

// 用户空间的CGRect转换到设备空间
CGRect CGContextConvertRectToDeviceSpace(CGContextRef cg_nullable c, CGRect rect)

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

推荐阅读更多精彩内容