Swift操作Quartz 2D進(jìn)行簡(jiǎn)單的繪圖與坐標(biāo)變換的教程
Quartz 2D簡(jiǎn)介
Quartz 2D是蘋(píng)果公司開(kāi)發(fā)的一個(gè)二維圖形繪制引擎,同時(shí)支持iOS和Mac系統(tǒng)。
它是一套基于C的API框架,提供了低級(jí)別、輕量級(jí)、高保真度的2D渲染。它能完成的工作有:
- 繪制圖形 : 線條\三角形\矩形\圓\弧等
- 繪制文字
- 繪制\生成圖片(圖像)
- 讀取\生成PDF
- 截圖\裁剪圖片
- 自定義UI控件
- …
Quartz 2D進(jìn)行繪圖
iOS繪圖技術(shù)主要有UIKit,Quartz 2D,Core Animation和OpenGL ES。我們平常對(duì)UIKit應(yīng)該不陌生,而Quartz 2D與UIKit的一個(gè)區(qū)別是:
Quartz 2D的坐標(biāo)原點(diǎn)在左下角,而UIKit的坐標(biāo)原點(diǎn)在左上角。
在開(kāi)始前作下準(zhǔn)備工作:創(chuàng)建一個(gè)新的Cocoa Touch Class,繼承自UIView,然后去StoryBoard把view視圖關(guān)聯(lián)下新創(chuàng)建的類(lèi)。
1.填充和描邊
重寫(xiě)繪圖方法drawRect(),添加代碼:
override func drawRect(rect: CGRect) {
//填充背景
UIColor.brownColor().setFill()
//填充矩形
UIRectFill(rect)
UIColor.whiteColor().setStroke()
//矩形描邊
let frame = CGRectMake(10, 24, 100, 300)
UIRectFrame(frame)
}
運(yùn)行效果:

2.繪制三角形
確定三個(gè)點(diǎn)就能繪制出三角形,當(dāng)然其他的圖形(如矩形)也是類(lèi)似。
在drawRect()里添加代碼:
override func drawRect(rect: CGRect) {
let context = UIGraphicsGetCurrentContext()
//繪制起始點(diǎn)
CGContextMoveToPoint(context, 120, 104)
//從起始點(diǎn)到這一點(diǎn)
CGContextAddLineToPoint(context, 150, 204)
CGContextAddLineToPoint(context, 200, 104)
//閉合路徑
CGContextClosePath(context)
UIColor.blackColor().setStroke()
UIColor.greenColor().setFill()
//繪制路徑
CGContextDrawPath(context, CGPathDrawingMode.FillStroke)
}
運(yùn)行效果:

依此類(lèi)推,大家可以試試怎么去畫(huà)長(zhǎng)方形,正方形和不規(guī)則多邊形。
3.繪制圖片和文字
首先準(zhǔn)備一張圖片放入工程中,注意不要放在Assets.xcassets文件夾下,因?yàn)檫@里尋找的路徑是在工程文件夾。而如果把圖片放在Assets.xcassets文件夾下,就要使用另外的一種方法。
在drawRect()里添加代碼:
override func drawRect(rect: CGRect) {
//繪制圖片和文字
//這種方式添加圖片需要把圖片放到根目錄下,而不是Assets.xcassets下
let imagePath = NSBundle.mainBundle().pathForResource("頭像004", ofType: "jpg")
let image = UIImage(contentsOfFile: imagePath!)
//具體位置根據(jù)你的圖片來(lái)調(diào)整
image?.drawInRect(CGRectMake(100,100, 200, 200))
let title = "頭像"
let font = UIFont.systemFontOfSize(44)
let attr = [NSFontAttributeName:font]
title.drawAtPoint(CGPointMake(100, 20), withAttributes: attr)
}
運(yùn)行效果:

Quartz 2D中的坐標(biāo)變換
注意:坐標(biāo)變換操作必須要在添加圖形之前,如果設(shè)置在添加圖形之后的話會(huì)無(wú)效。
我們先畫(huà)一個(gè)正方形做完參考:
override func drawRect(rect: CGRect) {
let context = UIGraphicsGetCurrentContext()
CGContextSetLineWidth(context, 2.0)
CGContextSetStrokeColorWithColor(context, UIColor.redColor().CGColor)
let rectangle = CGRectMake(125, 50, 50, 50)
CGContextAddRect(context, rectangle)
CGContextStrokePath(context)
}

1、平移
func CGContextTranslateCTM(c: CGContext?, _ tx: CGFloat, _ ty: CGFloat)
該方法相當(dāng)于把原來(lái)位于 (0, 0) 位置的坐標(biāo)原點(diǎn)平移到 (tx, ty) 點(diǎn)。在平移后的坐標(biāo)系統(tǒng)上繪制圖形時(shí),所有坐標(biāo)點(diǎn)的 X 坐標(biāo)都相當(dāng)于增加了 tx,所有點(diǎn)的 Y 坐標(biāo)都相當(dāng)于增加了 ty。
override func drawRect(rect: CGRect) {
let context = UIGraphicsGetCurrentContext()
CGContextSetLineWidth(context, 2.0)
CGContextSetStrokeColorWithColor(context, UIColor.redColor().CGColor)
CGContextTranslateCTM(context, -50, 25) // 向左向下平移
let rectangle = CGRectMake(125, 50, 50, 50)
CGContextAddRect(context, rectangle)
CGContextStrokePath(context)
}

2、縮放
func CGContextScaleCTM(c: CGContext?, _ sx: CGFloat, _ sy: CGFloat)
該方法控制坐標(biāo)系統(tǒng)在水平方向和垂直方向上進(jìn)行縮放。在縮放后的坐標(biāo)系統(tǒng)上繪制圖形時(shí),所有點(diǎn)的 X 坐標(biāo)都相當(dāng)于乘以 sx 因子,所有點(diǎn)的 Y 坐標(biāo)都相當(dāng)于乘以 sy 因子。
override func drawRect(rect: CGRect) {
let context = UIGraphicsGetCurrentContext()
CGContextSetLineWidth(context, 2.0)
CGContextSetStrokeColorWithColor(context, UIColor.redColor().CGColor)
CGContextScaleCTM(context, 0.5, 1)
let rectangle = CGRectMake(125, 50, 50, 50)
CGContextAddRect(context, rectangle)
CGContextStrokePath(context)
}

3、旋轉(zhuǎn)
func CGContextRotateCTM(c: CGContext?, _ angle: CGFloat)
該方法控制坐標(biāo)系統(tǒng)旋轉(zhuǎn) angle 弧度。在縮放后的坐標(biāo)系統(tǒng)上繪制圖形時(shí),所有坐標(biāo)點(diǎn)的 X、Y 坐標(biāo)都相當(dāng)于旋轉(zhuǎn)了 angle弧度之后的坐標(biāo)。
override func drawRect(rect: CGRect) {
let context = UIGraphicsGetCurrentContext()
CGContextSetLineWidth(context, 2.0)
CGContextSetStrokeColorWithColor(context, UIColor.redColor().CGColor)
CGContextRotateCTM(context, CGFloat(M_PI_4))
let rectangle = CGRectMake(125, 50, 50, 50)
CGContextAddRect(context, rectangle)
CGContextStrokePath(context)
}

注意:旋轉(zhuǎn)的時(shí)候,是整個(gè) layer 都旋轉(zhuǎn)了,所以 layer 看起來(lái)應(yīng)該是這樣的:

這個(gè)時(shí)候若想移動(dòng) view ,就應(yīng)該按照這個(gè)旋轉(zhuǎn)過(guò)的坐標(biāo)系來(lái)移動(dòng):
override func drawRect(rect: CGRect) {
let context = UIGraphicsGetCurrentContext()
CGContextSetLineWidth(context, 2.0)
CGContextSetStrokeColorWithColor(context, UIColor.redColor().CGColor)
CGContextRotateCTM(context, CGFloat(M_PI_4))
CGContextTranslateCTM(context, 0, -100) // 在新坐標(biāo)系中向上移動(dòng)100點(diǎn),視圖上看起來(lái)像是向右向上都移動(dòng)了
let rectangle = CGRectMake(125, 50, 50, 50)
CGContextAddRect(context, rectangle)
CGContextStrokePath(context)
}

- 詳解Swift的switch...case語(yǔ)句中break關(guān)鍵字的用法
- Swift編程中的switch...case語(yǔ)句實(shí)例解析
- 使用Swift實(shí)現(xiàn)iOS App中解析XML格式數(shù)據(jù)的教程
- Swift自定義iOS中的TabBarController并為其添加動(dòng)畫(huà)
- Swift實(shí)現(xiàn)iOS應(yīng)用中短信驗(yàn)證碼倒計(jì)時(shí)功能的實(shí)例分享
- Swift使用WKWebView在iOS應(yīng)用中調(diào)用Web的方法詳解
- 使用 Swift 語(yǔ)言編寫(xiě) Android 應(yīng)用入門(mén)
相關(guān)文章
swift版webview加載網(wǎng)頁(yè)進(jìn)度條效果
這篇文章主要為大家詳細(xì)介紹了swift實(shí)現(xiàn)webview加載網(wǎng)頁(yè)進(jìn)度條效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11
Swift下使用UICollectionView 實(shí)現(xiàn)長(zhǎng)按拖拽功能
拖拽排序是新聞?lì)惖腁pp可以說(shuō)是必有的交互設(shè)計(jì),如今日頭條,網(wǎng)易新聞等。這篇文章主要介紹了Swift下使用UICollectionView 長(zhǎng)按拖拽功能,需要的朋友可以參考下2017-03-03
Swift中通知中心(NotificationCenter)的使用示例
這篇文章主要給大家介紹了關(guān)于Swift中通知中心(NotificationCenter)使用的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2017-10-10
Swift如何為網(wǎng)頁(yè)承載頁(yè)面添加更多功能詳解
這篇文章主要給大家介紹了關(guān)于Swift如何為網(wǎng)頁(yè)承載頁(yè)面添加更多功能的相關(guān)資料,包括添加菊花加載的效果、添加跳轉(zhuǎn)到Safari的功能、添加復(fù)制鏈接的功能以及添加分享網(wǎng)頁(yè)的功能,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2018-05-05
用SwiftUI實(shí)現(xiàn)3D Scroll滾動(dòng)效果的實(shí)現(xiàn)代碼
這篇文章主要介紹了用SwiftUI實(shí)現(xiàn)3D Scroll效果的實(shí)現(xiàn)代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)2020-04-04
Swift實(shí)現(xiàn)簡(jiǎn)單計(jì)算器項(xiàng)目
這篇文章主要為大家詳細(xì)介紹了Swift實(shí)現(xiàn)簡(jiǎn)單計(jì)算器項(xiàng)目,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01
Swift利用Decodable解析JSON的一個(gè)小問(wèn)題詳解
這篇文章主要給大家介紹了關(guān)于Swift利用Decodable解析JSON的一個(gè)小問(wèn)題的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2018-04-04

