ios - 将 CIFilters 切换到视频
问题描述
有人可以帮助我了解在不重新启动视频播放器的情况下切换 CIFilters 的正确方法吗?
我在视图中播放本地视频。如果我在集合视图中点击一个单元格,视频将更改 CIFilter。
我的代码
let filter = CIFilter(name: "CIPhotoEffectNoir")!
let asset = AVAsset(url: fooURL)
let item = AVPlayerItem(asset: asset)
item.videoComposition = AVVideoComposition(asset: asset, applyingCIFiltersWithHandler: { request in
let source = request.sourceImage.clampedToExtent()
filter.setValue(source, forKey: kCIInputImageKey)
let output = filter.outputImage
request.finish(with: output!, context: nil)
})
player = AVPlayer(playerItem: item)
let playerLayer = AVPlayerLayer(player: player)
playerLayer.frame = videoViewDetail.bounds
videoViewDetail.layer.addSublayer(playerLayer)
player?.play()
代码效果很好,效果正确应用
但我有一个包含许多 CIFilters 的集合视图。在点击一个单元格时,我找不到将过滤器切换到视频的方法。如果我使用新过滤器重新创建一个新播放器并将其添加为当前“addsublayer”的替代品,播放器将重新启动视频。
@IBAction func tap(_ sender: UITapGestureRecognizer) {
let location = sender.location(in: self.collectionView)
let indexPath = self.collectionView.indexPathForItem(at: location)
if let index = indexPath {
// code to switch to another CIFilter like for example "CISepiaTone"
}
}
在不重新启动视频的情况下将 CIFilters 更改为正在播放的视频的最佳方法是什么可以使用新过滤器保存视频吗?
谢谢!
解决方案
AVVideoComposition
为播放器显示的每个视频帧连续调用您在创建时提供的处理程序块。这意味着您只需切换该块内使用的过滤器。
实现这一点的最简单方法是从块外部引用过滤器而不是捕获它。然后您可以在运行时更改引用。
例如,假设您的代码在某个视图控制器方法中运行,您可以执行以下操作:
class MyViewController: UIViewController {
var filter: CIFilter = CIFilter(name: "CIPhotoEffectNoir")!
func createPlayer() {
let asset = AVAsset(url: fooURL)
let item = AVPlayerItem(asset: asset)
item.videoComposition = AVVideoComposition(asset: asset, applyingCIFiltersWithHandler: { [weak self] request in
guard let self = self else {
request.finish(with error: SomeError)
return
}
let source = request.sourceImage.clampedToExtent()
self.filter.setValue(source, forKey: kCIInputImageKey)
let output = self.filter.outputImage
request.finish(with: output!, context: nil)
})
player = AVPlayer(playerItem: item)
let playerLayer = AVPlayerLayer(player: player)
playerLayer.frame = videoViewDetail.bounds
videoViewDetail.layer.addSublayer(playerLayer)
player?.play()
}
// ...
}
然后,您只需更改即可轻松更改过滤器self.filter
。
(您可能希望同步对过滤器的访问,以避免并发问题。)
推荐阅读
- windows - 在 Windows 中激活 conda 环境时如何更改目录
- java - 自动记录在 catch 块中,仅当注释异常变量时,对于 Java 或 Kotlin
- mongodb - 带有条件,排序和限制的mongodb组
- jquery - jQuery + .each() + .post()... 同步运行?
- node.js - 我怎样才能只看到 Sequelize.js 生成的 SQL?
- amazon-web-services - Hyperledger Fabric 应用程序中的 Orderer 断开连接
- node.js - ExpressJS Handlebars 在文件更改时更新部分
- google-api - Ruby on rails:Google API dailyLimitExceededUnreg
- android - 如何在 Android N 中从 URI 获取文件路径
- keras - 自定义损失函数会影响 Keras 中的反向传播吗?