ios AVFoundation框架给视频添加背景音乐的方法

/// 视频添加背景音乐
    /// - Parameters:
    ///   - videoURL: 视频文件URL
    ///   - audioURL: 音频文件URL
    ///   - startTime: 音频开始时间
    ///   - endTime: 音频结束时间
    ///   - isOriginal: 是否保留视频原音
    ///   - oriVolume: 视频原音音量
    ///   - newVolume: 新增背景音乐音量
    ///   - outputURL: 视频输出文件地址
    ///   - completion: 完成回掉
    static func addBackgroundMusic(videoURL : URL, audioURL : URL, startTime : Double, endTime : Double, isOriginal : Bool = true, oriVolume : Float, newVolume : Float, outputURL : String, completion : @escaping (_ success : Bool) -> Void) {
        
        if FileManager.default.fileExists(atPath: outputURL) {
            do {
                try FileManager.default.removeItem(atPath: outputURL)
            } catch {
                logger(item: "addBackgroundMusic remove exit file error")
            }
        }
        
        do {
            //导出路径
            let outputFileURL = URL.init(fileURLWithPath: outputURL)
            
            let nextClipStartTime = CMTime.zero
            
            //创建可变的音频视频组合
            let mixComposition = AVMutableComposition()
            
            //视频采集
            let videoAsset = AVURLAsset.init(url: videoURL)
            let videoTimeRange = CMTimeRange.init(start: .zero, duration: videoAsset.duration)
            let compositionVideoTrack = mixComposition.addMutableTrack(withMediaType: .video, preferredTrackID: kCMPersistentTrackID_Invalid)
            if compositionVideoTrack == nil {
                completion(false)
                return
            }
            try compositionVideoTrack!.insertTimeRange(videoTimeRange, of: videoAsset.tracks(withMediaType: .video).first!, at: nextClipStartTime)
            
            let start = CMTime.init(seconds: startTime, preferredTimescale: videoAsset.duration.timescale)
            let duration = CMTime.init(seconds: endTime - startTime, preferredTimescale: videoAsset.duration.timescale)
            let audioTimeRange = CMTimeRange.init(start: start, duration: duration)
            
            //创建最终混合的音频实例
            let audioMix = AVMutableAudioMix()
            
            //添加新的音频
            let audioAsset = AVURLAsset.init(url: audioURL)
            let newAudioTrack = mixComposition.addMutableTrack(withMediaType: .audio, preferredTrackID: kCMPersistentTrackID_Invalid)
            if compositionVideoTrack == nil {
                completion(false)
                return
            }
            try newAudioTrack!.insertTimeRange(audioTimeRange, of: audioAsset.tracks(withMediaType: .audio).first!, at: start)
            let newAudioInputParams = AVMutableAudioMixInputParameters.init(track: newAudioTrack!)
            newAudioInputParams.setVolumeRamp(fromStartVolume: newVolume, toEndVolume: 0.0, timeRange: CMTimeRange.init(start: .zero, duration: videoAsset.duration))
            newAudioInputParams.trackID = newAudioTrack!.trackID
            
            //视频文件原始音频通道
            if isOriginal {
                //视频声音采集(也可不执行这段代码不采集视频音轨,合并后的视频文件将没有视频)
                if videoAsset.tracks(withMediaType: .audio).first != nil {
                    if let originVoiceTrack = mixComposition.addMutableTrack(withMediaType: .audio, preferredTrackID: kCMPersistentTrackID_Invalid) {
                        try originVoiceTrack.insertTimeRange(videoTimeRange, of:videoAsset.tracks(withMediaType: .audio).first!, at: .zero)
                        let originAudioInputParams = AVMutableAudioMixInputParameters.init(track: originVoiceTrack)
                        originAudioInputParams.setVolumeRamp(fromStartVolume: newVolume, toEndVolume: 0.0, timeRange: CMTimeRange.init(start: .zero, duration: videoAsset.duration))
                        originAudioInputParams.trackID = originVoiceTrack.trackID
                        audioMix.inputParameters = [newAudioInputParams,originAudioInputParams]
                    } else {
                        audioMix.inputParameters = [newAudioInputParams]
                    }
                } else {
                    audioMix.inputParameters = [newAudioInputParams]
                }
            } else {
                audioMix.inputParameters = [newAudioInputParams]
            }
            
            //创建一个输出
            let assetExport = AVAssetExportSession.init(asset: mixComposition, presetName: AVAssetExportPresetHighestQuality)
            if assetExport == nil {
                completion(false)
                return
            }
            assetExport!.outputFileType = .mp4
            assetExport!.outputURL = outputFileURL
            assetExport!.shouldOptimizeForNetworkUse = true
            assetExport!.audioMix = audioMix
            
            assetExport!.exportAsynchronously {
                switch assetExport!.status {
                case .unknown:
                    break
                case .waiting:
                    break
                case .exporting:
                    break
                case .completed:
                    completion(true)
                    break
                case .failed:
                    //合成失败
                    completion(false)
                    break
                case .cancelled:
                    completion(false)
                    break
                @unknown default:
                    break
                }
            }
        } catch {
            completion(false)
        }
    }
    
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
敬告1:本课程基于最新版FFmpeg4.3.1, 是从一个商用产品中抽离而出,全程手把手教学。敬告2:本课程基于Java+Tomcat,C++方向的学员,不用购买此课程。 我将带领大家亲手开发一套WEB在线的基于Java语言的音视频转码系统 您将收获:用Java语言开发WEB在线音视频转码学习商用音视频产品的标准流程学习Java应用ffmpeg音视频的功能学习音视频的基础理论 学习SSH框架的灵活应用-------------------------------------------------------------------主要模块包括:1、标准文档:    需求说明书、系统分析、概要设计、数据结构、编码、测试、总结、面试技巧等。2、Java编码:   SSH框架,标准流程,可以自由扩展。3、音视频转码:最新版FFmpeg4.3.1,灵活的转码模板,可以自由扩展。4、音视频理论:讲解基础的音视频理论,帮助小白快速入门。5、面试技巧:    音视频方向的工作,薪资偏高,但要求苛刻。-------------------------------------------------------------------具体的课程目录参考如下: B/S音视频转码器简介与概览5 需求规格说明书5 系统分析与设计15 数据结构设计27数据库设计30Java+Tomcat+Eclipse开发环境配置35Java+SSH框架搭建与整合43界面设计49编码51左侧导航树58B/S音视频转码器总结与分析66 简历包装与面试技巧67-------------------------------------------------------------------音视频是一门很复杂的技术,涉及的概念、原理、理论非常多,很多初学者不学基础理论,而是直接做项目,往往会看到c/c++的代码时一头雾水,不知道代码到底是什么意思,这是为什么呢? 因为没有学习音视频的基础理论,就比如学习英语,不学习基本单词,而是天天听英语新闻,总也听不懂。所以呢,一定要认真学习基础理论,然后再学习播放器、转码器、非编、流媒体直播、视频监控、等等。  梅老师从事音视频与流媒体行业18年;曾在永新视博、中科大洋、百度、美国Harris广播事业部等公司就职,经验丰富;曾亲手主导广电直播全套项目,精通h.264/h.265/aac,曾亲自参与百度app上的网页播放器等实战产品。  

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值