Core Image

Core Image是一个强大的框架,它能够让你轻松地对图像进行过滤。你能够通过修改图像的饱和度、色调或曝光率来获取各种特效。你也可以使用CPU或GPU更快地来处理图像数据,快到能够实时处理视频帧数据(video frame)。

Core Image过滤器能够以链式的方式结合将多个特效应用到一个图像或视频帧数据。多个过滤器能够组合成单个过滤器应用在一个图像。通过每次与每个过滤器比较,这使它非常有效地处理图像。

在这个教程中,你将会亲手实践如何使用Core Image.通过使用几个不同的过滤器,你会看到实时使用各种炫酷的特效是多么容易。

基础

在你开始之前,让我们先讨论在Core Image框架中几个很重要的类:

CIContext. 所有处理core image的工作都在CIContext完成。这个与Core Graphics或OpenGL context有几分相似。

CIImage. 这个类保存图像数据,它能够从UIImage,图像文件或像素数据等方式来创建。

CIFilter. CIFilter类有一个字典(dictionary)来定义特定属性的过滤器,比如常用过滤器有饱和度,颜色翻转,裁剪等属性,还有很多没列出来。

CoreImageFun

打开Xcode并使用iOS \ Application \ Single View Application模板来创建一个新工程。输入CoreImageFun作为Product Name,选择iPhone为Devices option和确定使用swift作为编程语言。

下载教程的资源,并添加图片image.png到工程

下一步,打开Main.storyboard你文件,拖拽一个image view到已存在的view,image view作为view的子视图。在Attributes Inspector中,设置image view的content mode属性为Aspect Fit,这样它就不会拉伸图片。

下一步,确定Document Outline(在Interface Builder里canvas左边)是可见的 - 你可以从menu中Editor \ Show Document Outline来启用。

按着control键从image view拖动到它的superview三次来添加三个约束(constraint):

添加一个Top Space to Layout Guide的约束,如果有必要的话,使用Size Inspector来设置约束的constant为0

添加一个Center Horizontally in Container的约束(同样也需要设置constant为0)

添加一个Equal Width约束

最后,为了约束image view的高度,按着control键从image view拖动到本身,然后添加一个Aspect Ratio约束,使用Size Inspector来设置它的multiplier的值为8:5对应宽高比和constant factor为0。最后,选择Editor \ Resolve Auto Layout Issues \ All Views in View Controller \ Update Frames,那么Interface Builder会根据这些约束来更新布局

下一步,打开Assistant Editor,然后确保它显示ViewController.swift。按着control键从UIImageView拖动到刚刚打开的ViewController类的大括号。命名outlet为imageView,然后点击connect。

编译和运行这个项目来确保目前进展顺利 - 你应该看到一个空屏。初始化设置完成,现在进入Core Image的世界。

基本图片过滤器

你将会通过使用CIFilter应用到图像和显示到屏幕来入门。每次你想将CIFilter应用到一个图像,都需要做四件事:

创建一个CIImage对象。CIImage有几个初始化方法,其中包括:CIImage(contentsOfURL:), CIImage(data:), CIImage(CGImage:), CIImage(bitmapData:bytesPerRow:size:format:colorSpace:)等,大多数你都会使用CIImage(contentsOfURL:)方法。

创建一个CIContext对象。一个CIContext是基于CPU或CPU,在初始化它时比较耗资源,所以需要复用它而不是多次创建。当你输出CIImage对象时,你会经常需要CIContext对象。

创建CIFilter对象。当你创建一个filter时,你需要配置多个属性来决定你使用的filter。

获取filter的输出。filter会输出一个CIImage类型的image给你 - 你可以使用CIContext将它转换为UIImage类型image。

让我们看看它是如何工作。添加以下代码到ViewController.swift的viewDidLoad():方法

// 1

let fileURL = NSBundle.mainBundle().URLForResource("image", withExtension: "png")

// 2

let beginImage = CIImage(contentsOfURL: fileURL)

// 3

let filter = CIFilter(name: "CISepiaTone")

filter.setValue(beginImage, forKey: kCIInputImageKey)

filter.setValue(0.5, forKey: kCIInputIntensityKey)

// 4

let newImage = UIImage(CIImage: filter.outputImage)

self.imageView.image = newImage

让我们逐段分析以上代码:

这行代码创建一个NSURL对象,它保存image文件的路径

通过CIImage(contentsOfURL:)构造器来创建CIImage对象

创建CIFilter对象。CIFilter构造器将name作为参数,然后用dictionary来指定filter的键值对。每个filter都拥有唯一的键和多个有效值。CISepiaTonefilter只接收两个值kCIInputImageKey(CIImage)和kCIInputIntensityKey(在0到1之间的一个浮点数)。这里你设置它的值为0.5。如果没有值提供的话,大多数filters都有默认值。但CIImage没有默认值,它必须要提供。

通过使用filter的outputImage属性能够轻松地获取CIImage。一旦你有个输出的CIImage,你需要将它转换为UIImage。使用UIImage(CIImage:)构造器来创建UIImage。如果你已经将CIImage转换为UIImage,你就能将image显示到image view

把它放进Context

在你向前看之前,有个优化建议你应该需要知道。

之前我曾提及过,你需要有个CIContext来使用CIFilter,但以上例子都没有提及过这个对象。结果变成了由UIImage(CIImage:)构造器来为你处理所有的工作。UIImage它创建一个CIContext,然后用它来执行所有过滤图像的操作,这让你更加容易地使用Core Image的API。

但这种方式有一个主要缺点就是每次使用时都创建一个新的CIContext对象。CIContext对象是为了复用来提高性能。如果你想用一个slider来更新filter的值,是以每次创建一个新的CIContext来实现更新filter的方式,那么运行速度将会很慢。

让我们以恰当的方式来实现。在viewDidLoad()里删除步骤4的代码,然后用一下代码来代替:

// 1

let context = CIContext(options:nil)

// 2

let cgimg = context.createCGImage(filter.outputImage, fromRect: filter.outputImage.extent())

// 3

let newImage = UIImage(CGImage: cgimg)

self.imageView.image = newImage

我们再一次逐段分析代码:

你设置CIContext对象,然后用它来画一个CGImage。CIContext(options:)构造器以NSDictionary为参数,它指定一些选项,比如:颜色格式,或者context是否运行在CPU或GPU。对于这个app来说,默认值是可以的,所以你可以传递nil这个值进去。

context调用createCGImage(outputImage:fromRect:)方法在给定CIImage参数并返回一个新的CGImage示例。

你根据刚刚获取的CIImage,使用UIImage(CGImage:)构造器来创建一个UIImage对象。注意,当我们使用CIImage对象之后,没有必要显式地释放它,虽然在Objective-C需要这样做。但在Swift中,ARC会自动释放Core Foundation的对象。

编译和运行,确保项目进展顺利

在这个例子中,自己创建CIContext与不创建没什么不同。但下个部分中,你会看到当你动态地修改filter时,为什么CIContext对性能影响很大。

修改Filter值

目前还不错,但这只是使用Core Image filters的入门方式。让我们添加一个slider,然后设置它以便你能够实时调整filter的设置

打开Main.storyboard,选取一个slider,将它拖放到image view的上面,并水平对齐。选中view,然后点击Editor \ Resolve Auto Layout Issues \ Selected Views \ Reset to Suggested Constraints,添加需要的宽约束。

确保Assistant Editor可见和显示ViewController.swift,然后按着control键从slider拖动到之前添加的@IBOutlet下面,设置name为amountSlider,然后点击Connect

当你还选中slider时,也让我们连接slider到一个action method。再一次按着control键从slider拖动到ViewController类的}上面。设置Connection为Action,name为amountSliderValueChanged,确保Event设置为Value Changed,然后点击Connect

每次slider改变时,你需要根据不同的值来重新创建image filter。然而,你不想重复整个耗时且没效率的过程。你需要在你的类改变几样东西,那么你就要在viewDidLoad方法中创建一些对象并保存。

最重要的一件事就是当你需要使用CIContext时,你只需复用它。如果你每次都重新创建它,你的程序将会变得很慢。另一件事就是保存CIFilter和CIImage,CIImage主要保存原始的图像。每次输出都会产生新的CIImage,但你刚开始使用的图像都会保持不变。

你需要添加几个实例变量来完成这个任务。添加以下三个属性到你的ViewController类:

var context: CIContext!

var filter: CIFilter!

var beginImage: CIImage!

请注意,你已经用!语法来声明那些值为implicitly-unwrapped optionals,因为直到viewDidLoad才初始化它们。你也可以用?,但采用那种方式是为了当你使用那几个实例变量时,防止optionals为nil。implicitly-unwrapped语法由于不用到处使用!标识来访问变量,它让代码更加易读。

viewDidLoad改变代码,那么它使用这些属性而不是使用新的局部变量,代码如下:

beginImage = CIImage(contentsOfURL: fileURL)

filter = CIFilter(name: "CISepiaTone")

filter.setValue(beginImage, forKey: kCIInputImageKey)

filter.setValue(0.5, forKey: kCIInputIntensityKey)

let outputImage = filter.outputImage

context = CIContext(options:nil)

let cgimg = context.createCGImage(outputImage, fromRect: outputImage.extent())

现在你将会实现changeValue方法。你在这个方法需要做的是修改CIFilter dictionary的inputIntentsity键对应的值。

一旦你修改这个值,你需要重复这几个步骤:

从CIFilter获取输出的CIImage

将CIImage转换为CGImage

将CGImage转换为UIImage,然后将它显示在image view

用以下代码代替amountSliderValueChanged(sender:)方法:

@IBAction func amountSliderValueChanged(sender: UISlider) {

let sliderValue = sender.value

filter.setValue(sliderValue, forKey: kCIInputIntensityKey)

let outputImage = filter.outputImage

let cgimg = context.createCGImage(outputImage, fromRect: outputImage.extent())

let newImage = UIImage(CGImage: cgimg)

self.imageView.image = newImage

}

你会注意到,你已经将方法定义中参数类型从AnyObject转换为UISlider。你只用这个方法来从UISlider获取值,以便你改变值。如果你不管它,默认是AnyObject,你需要将它转换为UISlider,否则下一行代码就会抛出错误。

你可以从slider获取浮点数的值。你的slider默认设置为0.5,最小值为0,最大值为1。通过slider设置CIFilter是多么方便。

CIFilter有多个方法允许你在dictionary根据不同键来设置多个值。而这里,你刚设置inputIntensity键对应的值,而这个值是从slider获取。Swift自动将CGFloat值转换为NSNumber对象,来符合setValue(value:forKey:)方法的使用。

剩下的代码看起来很熟悉,因为它与viewDidLoad方法的逻辑一样。你将会多次使用这段代码。从现在起,你会用amountSliderValueChanged(sender:)方法来将CIFilter输出的图像渲染到ImageView。

编译和运行,你可以修改slider值来实时地改变图像。

从相册获取图片

现在你可以改变filter的值,事情也开始变得有趣。但是,如果你不喜欢这张花朵的图片。你可以设置UIImagePickerController来从相册选取图片放进你的app来使用。

你需要创建一个button来跳转到相册视图,所以打开Main.storyboard,拖动一个button到scene的右底部,并改变按钮文字为"Photo Album"。像之前一样,使用Auto Layout来Reset to Suggested Constraints。button应该在slider的右下边。

确保Assistant Editor是可见和显示ViewController.swift,然后按着control键从button拖动到}的上面。设置Connection为Action,name为loadPhoto,确保Event设置为Touch Up Inside,最后点击Connect

loadPhoto方法实现如下:

@IBAction func loadPhoto(sender : AnyObject) {

let pickerC = UIImagePickerController()

pickerC.delegate = self

self.presentViewController(pickerC, animated: true, completion: nil)

}

第一行代码主要是创建一个UIImagePickerController对象。然后设置image picker的delegate为self(ViewController)

你会这里得到一个警告。你需要声明ViewController遵循UIImagePickerControllerDelegateUINavigationControllerDelegate协议。

仍在ViewController.swift文件,在文件顶部改变类的定义,代码如下:

class ViewController: UIViewController, UINavigationControllerDelegate, UIImagePickerControllerDelegate {

而方法实现如下:

func imagePickerController(picker: UIImagePickerController!, didFinishPickingMediaWithInfo info: NSDictionary!) {

self.dismissViewControllerAnimated(true, completion: nil);

println(info);

}

UIImagePickerControllerDelegate方法还没完成 - 它只是一个占位符来打印被选择图片的信息。注意,不管你怎样实现UIImagePickerControllerDelegate这个方法,你必须在实现中显式地dimiss UIImagePickerController。如果你不这样做的话,你就会永远地盯着image picker。

编译和运行这个app,点击button。它会跳转到相册任你选择图片。

如果你在模拟器运行app,你可以不会有图片。在模拟器(或在没有摄像头的设备),你可以使用Safari来保存图片到相册。打开Safari,查找一张图片,点着并长按,你就会有一个选项来保存图片。下一次你运行你app,它将会出现在你图片库。

在控制台中,你选择完一张图片之后,就看到类似以下的打印信息:

{

UIImagePickerControllerMediaType = "public.image";

UIImagePickerControllerOriginalImage = " size {1165, 770} orientation 0 scale 1.000000";

UIImagePickerControllerReferenceURL = "assets-library://asset/asset.PNG?id=DCFE1435-2C01-4820-9182-40A69B48EA67&ext=PNG";

}

注意,它有一个dictionary入口,对应就是用户选择的“原始图片”。这个就是你想拽取和过滤的东西。

现在你已经有方法选取图片了,你怎样使用它作为你的beginImage呢?

很简答,只需将delegate方法修改成以下代码:

func imagePickerController(picker: UIImagePickerController!, didFinishPickingMediaWithInfo info: NSDictionary!) {

self.dismissViewControllerAnimated(true, completion: nil);

let gotImage = info[UIImagePickerControllerOriginalImage] as UIImage

beginImage = CIImage(image:gotImage)

filter.setValue(beginImage, forKey: kCIInputImageKey)

self.amountSliderValueChanged(amountSlider)

}

你需要从你选择的图片来创建一个新的CIImage。你可以通过在dictionary的UIImagePickerControllerOriginalImage键获取值,从而获取UIImage的表示。注意,最好就是用UIImagePickerControllerOriginalImage这个常量,而不是硬编码的字符串,因为Apple可能在将来会改变这个键的名字。

你需要通过CIImage(image:)构造器将image转换为CIImage对象。然后在filter dictionary设置键,那么就可以创建新的CIImage。

最后一行代码看起来有点奇怪。还记得我说过,怎样运行changeValue方法,最新的值被设置在filter,然后更新image view。

好,你需要再做一次,所以你只是调用changeValue方法。即使slider的值还没改变,你仍然可以使用那个方法的代码来完成工作。你可以分解那段代码到自己的方法(如果你想做得更加复杂来避免混淆),但这种情况下,你的目的是为了复用amountSliderValueChanged方法。传递amountSlider作为sender以致它有正确的值使用。

编译和运行,你能从相册中更新任何图片。

如果你创建完美的深褐色图片,你怎样才能保存它。你可以截图,但最恰当的方式就是保存已过滤的图片到相册。

保存到相册

为了保存到相册,你需要使用AssetsLibraryframework。在ViewController.swift文件顶部添加以下导入语句:

import AssetsLibrary

有一件你需要知道的事就是当你保存图片到相册,它需要花费几秒时间,即使在你关闭app之后仍在继续处理图片。

由于当你切换到另一个app的时候,GPU就停止处理,这将变成一个问题。如果图片还没完成保存,当你迟点找到它的时候,它将不在那里。

有一个解决方案就是使用基于CPU的CIContext来渲染。默认选择就是使用GPU,因为它处理速度更快,但你不想为了添加保存功能而降低过滤性能。而现在,你会创建另一个CIContext来保存image。注意,软件渲染将不会在模拟器正常工作。

添加一个新button到你的app,这个button会让你保存修改后的图片。打开Main.storyboard,添加一个新的button,修改button标题为“Save to Album”。将button放在slider的左边,然后添加相应的约束。

然后像上一次一样,连接button到一个新方法savePhoto(sender:),方法实现如下:

@IBAction func savePhoto(sender: AnyObject) {

// 1

let imageToSave = filter.outputImage

// 2

let softwareContext = CIContext(options:[kCIContextUseSoftwareRenderer: true])

// 3

let cgimg = softwareContext.createCGImage(imageToSave, fromRect:imageToSave.extent())

// 4

let library = ALAssetsLibrary()

library.writeImageToSavedPhotosAlbum(cgimg,

metadata:imageToSave.properties(),

completionBlock:nil)

}

在这段代码块中:

从filter获取CIImage的输出。

创建一个新的,基于软件的,使用CPU渲染器的CIContext对象。

生成CGImage。

保存CIImage到photo library。

编译和运行app(记住要运行在实际设备,因为你已经使用软件渲染),现在你可以永久地保存完美的图像到你的photo library。

Image元数据是什么?

让我们讨论一下关于image元数据。用手机拍照出来的图片有很多关联它的数据,例如GPS坐标,图片格式,和方向。

方向是一样特殊的东西你需要保存。加载UIImage变成CIImage,渲染成CGImage,然后转换回一个UIImage这个过程是为了从image抓取元数据。为了保存方向,你需要保存它然后传递它作为UIImage构造器参数。

添加一个新属性到ViewController类定义:

var orientation: UIImageOrientation = .Up

下一步,添加下面代码到imagePickerController(picker:didFinishPickingMediaWithInfo:)方法里,设置beginImage代码行的上面:

orientation = gotImage.imageOrientation

这样就会保存原始image方向的属性。

最后,修改在amountSliderValueChanged方法里,那段创建UIImage来设置imageView对象的代码:

let newImage = UIImage(CGImage: cgimg, scale:1, orientation:orientation)

现在,如果你拍一张照片而不是默认方向,它将会默认保存。

还有哪些Filters是可用的?

CIFilter API在Mac OS有超过160个,在iOS 8有126个关于filters的API。

为了查找有哪些filters在给定的设备是可用的,你可以使用CIFilter的filterNamesInCategory(kCICategoryBuiltIn)方法。这个方法会返回一个关于filter名字的数组。

除此之外,每个filter有一个attributes()方法返回一个dictionary,dictionary包含关于filter的信息。这些信息包括filter的名字,分类,filter接受哪些输入,默认输入,和这些输入接受哪些值。

让我们将所有东西放在一个类的方法,这个方法记录当前所有可用的filters信息。添加这个方法到ViewController类的定义:

func logAllFilters() {

let properties = CIFilter.filterNamesInCategory(kCICategoryBuiltIn)

println(properties)

for filterName: AnyObject in properties {

let fltr = CIFilter(name:filterName as String)

println(fltr.attributes())

}

}

这个方法仅仅通过filterNamesInCategory()方法来获取filters的数组。首先,它打印名字的列表。然后,遍历列表获取每个名字,根据名字实例化filter和打印它的属性dictionary。

viewDidLoad()方法最后调用这个方法:

self.logAllFilters()

你会看到很多filters在控制台被列出:

[CIAttributeFilterDisplayName: Color Monochrome, inputColor: {

CIAttributeClass = CIColor;

CIAttributeDefault = "(0.6 0.45 0.3 1)";

CIAttributeType = CIAttributeTypeColor;

}, inputImage: {

CIAttributeClass = CIImage;

CIAttributeType = CIAttributeTypeImage;

}, CIAttributeFilterCategories: (

CICategoryColorEffect,

CICategoryVideo,

CICategoryInterlaced,

CICategoryNonSquarePixels,

CICategoryStillImage,

CICategoryBuiltIn

), inputIntensity: {

CIAttributeClass = NSNumber;

CIAttributeDefault = 1;

CIAttributeIdentity = 0;

CIAttributeSliderMax = 1;

CIAttributeSliderMin = 0;

CIAttributeType = CIAttributeTypeScalar;

}, CIAttributeFilterName: CIColorMonochrome]

哇,有很多的filters!它会给你一些灵感在你自己的app中尝试其他filters。

更加复杂的Filter链

现在我们已经知道在iOS平台有哪些可用的filter,是时候创建更加复杂的filter链了。为了做到这样,你会创建一个专门的方法来处理CIImage。它会接受一个CIImage作为参数,过滤它,让它看起来像一张老旧的图片,然后返回一个已经修改的CIImage。

添加以下方法到ViewController:

func oldPhoto(img: CIImage, withAmount intensity: Float) -> CIImage {

// 1

let sepia = CIFilter(name:"CISepiaTone")

sepia.setValue(img, forKey:kCIInputImageKey)

sepia.setValue(intensity, forKey:"inputIntensity")

// 2

let random = CIFilter(name:"CIRandomGenerator")

// 3

let lighten = CIFilter(name:"CIColorControls")

lighten.setValue(random.outputImage, forKey:kCIInputImageKey)

lighten.setValue(1 - intensity, forKey:"inputBrightness")

lighten.setValue(0, forKey:"inputSaturation")

// 4

let croppedImage = lighten.outputImage.imageByCroppingToRect(beginImage.extent())

// 5

let composite = CIFilter(name:"CIHardLightBlendMode")

composite.setValue(sepia.outputImage, forKey:kCIInputImageKey)

composite.setValue(croppedImage, forKey:kCIInputBackgroundImageKey)

// 6

let vignette = CIFilter(name:"CIVignette")

vignette.setValue(composite.outputImage, forKey:kCIInputImageKey)

vignette.setValue(intensity * 2, forKey:"inputIntensity")

vignette.setValue(intensity * 30, forKey:"inputRadius")

// 7

return vignette.outputImage

}

这里会逐段讲解:

以相同的方式设置深褐色的filter。你在这个方法传递一个浮点数来设置深褐色效果的强度。这个值是由slider提供。

设置一个创建随机噪音模式的filter,filter效果如下:

它好像没有接受任何参数。你会使用这个噪音模式来添加纹理到你的最终“老旧相片”的外观。

修改随机噪音生成器的输出。你想改变的它的灰度和减轻它一点以至于效果没有那么炫酷。你会注意到,输入image的键被设置到随机filter的outputImage属性。这是一种传递一个filter的输出作为下个输入的方便方式。

imageByCroppingToRect()接受被裁剪的rect作为参数,然后返回CIImage。在这种情况下,你需要裁剪CIRandomGenerator filter的输出,因为它无限地碎片化。在某种情况下,如果你不裁剪它,你会得到一个错误信息,说filters有‘an infinite extent’。CIImages实际上是不包含image数据,它们创建它时将它描述成一个'recipe'。直到你在CIContext调用一个方法时数据才实际被处理。

将深褐色filter的输出和CIRandomGenerator filter的输出结合起来。这个filter执行像在photoshop层‘Hard Light’设置完全一样的操作。大多数在photoshop(但不是全部)的filter选项都可以通过使用Core Image来实现。

在这个合成的输出中,运行一个vignette filter,它会使你的图片边缘变暗。你使用从slider获取的值来设置半径和效果的强度。

最后,返回最后filter的输出。

关于filter chain讲到这里。你现在应该有一个如何形成复杂filter chain的想法。通过组合多个Core Image filters成各种各样的chains,你就能实现无穷多种的特效。

下一件事就是在amountSliderValueChanged()方法实现。改变这两行代码:

filter.setValue(sliderValue, forKey: "inputIntensity")

let outputImage = filter.outputImage

成这行代码:

let outputImage = self.oldPhoto(beginImage, withAmount: sliderValue)

这只是用新的,更复杂的filter方法来代替之前深褐色效果。你可以传递slder的值给强度,和你使用在viewDidLoad方法中被设置的beginImage作为输入的CIImage。编译和运行,你应该得到一个更加完善的旧图片效果,这种效果会是深褐色,有一点噪音和一点渐晕。

这种噪音效果会是更加微妙,但我会留给你自行试验。现在你完全有能力随意操纵Core Image,尽管去尝试吧!

你可能感兴趣的:(Core Image)