Я хочу применить CIFilter к видео и сохранить видео с примененным фильтром. [экспорт занимает слишком много времени]

Я хочу применить CIFilter к видео и сохранить видео с примененным фильтром.

Я использую AVMutableVideoComposition (также пробовал с AVVideoComposition) для применения фильтра к видео, и он отлично работает с AVPlayer, но когда я экспортирую отфильтрованное видео, локальное сохранение занимает слишком много времени (20-секундное видео занимает около 6-8 минут). Пожалуйста, помогите мне, что я делаю неправильно... Спасибо!

let item = AVPlayerItem(asset: currentAsset!)
    let videoComposition = AVMutableVideoComposition(asset: currentAsset!) { request in
        let filter = CIFilter(name: "CIColorInvert")!
        let source = request.sourceImage.clampedToExtent()
        filter.setValue(source, forKey: kCIInputImageKey)

        let output = filter.outputImage!.cropped(to: request.sourceImage.extent)

        // Provide the filter output to the composition
        request.finish(with: output, context: nil)
    }
    item.videoComposition = videoComposition
    self.player.replaceCurrentItem(with: item)

    let exporter = AVAssetExportSession(asset: item.asset, presetName: AVAssetExportPresetHighestQuality)
    exporter?.videoComposition = videoComposition

    exporter?.outputFileType = .mp4
    let filename = "filename.mp4"

    let documentsDirectory = FileManager.default.urls(for: FileManager.SearchPathDirectory.documentDirectory, in: FileManager.SearchPathDomainMask.userDomainMask).last!
    let outputURL = documentsDirectory.appendingPathComponent(filename)
    exporter?.outputURL = outputURL
    exporter?.exportAsynchronously(completionHandler: {
        guard exporter?.status == .completed else {
            print("export failed: \(exporter?.error)")
            return
        }
        print("done: ",outputURL)
    }
    )

person Deep    schedule 12.02.2019    source источник
comment
Пробовали ли вы создать другой экземпляр композиции для сеанса экспорта?   -  person Frank Schlegel    schedule 12.02.2019
comment
Да, я уже пробовал отдельно для них обоих, но не получилось.   -  person Deep    schedule 12.02.2019
comment
Вы пробовали другой пресет качества? Этот вопрос кажется связанным: stackoverflow.com/questions/51036125   -  person Frank Schlegel    schedule 13.02.2019
comment
Пробовал все пресеты, но только сквозной экспорт видео (без фильтра), другие пресеты ничего не экспортировали. Я не понял, почему это происходит?   -  person Deep    schedule 15.02.2019


Ответы (1)


Здравствуйте! Попробуйте применить фильтр к видеофайлу

let documentsDirectory = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask)[0]
let savedoutputURL = documentsDirectory.appendingPathComponent("Filterd.mov")
try? FileManager.default.removeItem(at: savedoutputURL)

let urlVideofile = Bundle.main.url(forResource: "(your video file name)", withExtension: "mov")

let videoAsset = AVAsset(url: urlVideofile!)

let filter = CIFilter(name: "CIColorInvert")!
let composition = AVVideoComposition(asset: asset, applyingCIFiltersWithHandler: { request in

    let source = request.sourceImage.clampingToExtent()
    filter.setValue(source, forKey: kCIInputImageKey)

    // Crop the invert output to the bounds of the original image
    let output = filter.outputImage!.cropping(to: request.sourceImage.extent)

    // Provide the filter output to the composition
    request.finish(with: output, context: nil)
})
// Simultaneous Video playing although you can use as your's

let playerItem = AVPlayerItem(asset: videoAsset)
playerItem.videoComposition = composition

self.player = AVPlayer(playerItem: playerItem)

self.player.volume = 0.0

self.playerController.player = self.player

 present(self.playerController, animated: true, completion: {
      self.player.play()
})

Сохранение видео

let export = AVAssetExportSession(asset: videoAsset, presetName: AVAssetExportPresetHighestQuality)
    export!.outputFileType = AVFileType.mov
    export!.outputURL = savedoutputURL
    export!.videoComposition = composition

    export?.exportAsynchronously(completionHandler: {

        if export!.status.rawValue == 4 {
            print("Export failed -> Reason: \(export?.error!.localizedDescription))")
            print(export?.error!)
            return
        }
        print("done: ",savedoutputURL)

    })

Это 100% проверенный код, применяющий фильтр к 26-секундному видео примерно через 10-15 секунд

:)наслаждаться

person Mithilesh Kuamr    schedule 20.04.2020