还有另一种选择,使用 AVCaptureSession 创建 CIImage 实例,您可以在其中应用 CIFilters(其中有负载,从模糊到颜色校正到 VFX)。
这是一个使用 ComicBook 效果的示例。简而言之,创建一个 AVCaptureSession:
let captureSession = AVCaptureSession()
captureSession.sessionPreset = AVCaptureSessionPresetPhoto
创建一个AVCaptureDevice来代表摄像头,这里我设置的是后置摄像头:
let backCamera = AVCaptureDevice.defaultDeviceWithMediaType(AVMediaTypeVideo)
然后创建设备的具体实现并将其附加到会话。在 Swift 2 中,实例化 AVCaptureDeviceInput 可能会抛出错误,因此我们需要捕获它:
do
{
let input = try AVCaptureDeviceInput(device: backCamera)
captureSession.addInput(input)
}
catch
{
print("can't access camera")
return
}
现在,这里有一个小问题:虽然我们实际上并没有使用 AVCaptureVideoPreviewLayer,但它需要让示例委托工作,所以我们创建了其中一个:
// although we don't use this, it's required to get captureOutput invoked
let previewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
view.layer.addSublayer(previewLayer)
接下来,我们创建一个视频输出 AVCaptureVideoDataOutput,我们将使用它来访问视频源:
let videoOutput = AVCaptureVideoDataOutput()
确保自己实现了AVCaptureVideoDataOutputSampleBufferDelegate,我们可以在视频输出上设置sample buffer delegate:
videoOutput.setSampleBufferDelegate(self,
queue: dispatch_queue_create("sample buffer delegate", DISPATCH_QUEUE_SERIAL))
然后将视频输出附加到捕获会话:
captureSession.addOutput(videoOutput)
...最后,我们开始捕获会话:
captureSession.startRunning()
因为我们已经设置了委托,所以每次捕获帧都会调用 captureOutput。 captureOutput 传递一个 CMSampleBuffer 类型的样本缓冲区,只需两行代码即可将该数据转换为 CIImage 以供 Core Image 处理:
let pixelBuffer = CMSampleBufferGetImageBuffer(sampleBuffer)
let cameraImage = CIImage(CVPixelBuffer: pixelBuffer!)
...并且该图像数据被传递给我们的漫画书效果,该效果又用于填充图像视图:
let comicEffect = CIFilter(name: "CIComicEffect")
comicEffect!.setValue(cameraImage, forKey: kCIInputImageKey)
let filteredImage = UIImage(CIImage: comicEffect!.valueForKey(kCIOutputImageKey) as! CIImage!)
dispatch_async(dispatch_get_main_queue())
{
self.imageView.image = filteredImage
}
我有source code for this project available in my GitHub repo here。