Swift-AVPlayer网络音频播放

import UIKit
import Foundation

class PublicFunction: NSObject {
    //MARK:播放时长
    static func convertTime(totalSeconds:Int) -> (String) {
        let seconds:Int = totalSeconds % 60
        let minutes:Int = (totalSeconds / 60) % 60
        let time:String = "\(String(format: "%02d", minutes)):\(String(format: "%02d", seconds))"
        return time
    }
import UIKit
import AVFoundation

class NetAudioPlayerTool: NSObject {

    //正在播放回调
    //currentTime       播放时间
    //currentProgress   播放进度
    typealias OnPlayingBlock =  (_ currentTime:Float64,_ currentProgress:Float) -> ()
    //准备播放回调
    //totalDuration     音频总时长
    typealias PrepareToPlayBlock =  (_ totalDuration:Float) -> ()
    //正在缓冲回调
    //bufferDuration    已缓冲的时长
    typealias OnBufferingBlock =  (_ bufferDuration:Float) -> ()
    //播放完成回调
    //flag              YES播放完成, NO播放失败
    typealias CompletePlayingBlock =  (_ flag:Bool) -> ()
    //缓存自动暂停回调, 用于更改播放按钮的样式
    //bufferDuration    isPlaying 是否正在播放, 如果没有播放表示正在缓冲
    typealias IsPlayingBlock =  (_ isPlaying:Float) -> ()
    
    var playingBlock:OnPlayingBlock?
    var prepareToPlayBlock:PrepareToPlayBlock?
    var bufferingBlock:OnBufferingBlock?
    var completePlayBlock:CompletePlayingBlock?
    var isPlayingBlock:IsPlayingBlock?
    
    var palyerItem:AVPlayerItem?
    //播放器
    var player:AVPlayer = AVPlayer()
    //播放路径
    var audioPath:String = ""
    //音量大小
    var volume:Float = 1.0
    // 只有当播放器状态为`ReadyToPlay`,才可以执行拖拽操作,否则crash.
    var canDraggingFlag:Bool = false
    
    //之前播放的进度
    var timeOffset:Float64 = 0
    
    //播放时间
    var currentTime:Float64 = 0
    //音频总时长
    var totalDuration:Float64 = 0
    
    override init() {
        super.init()
        self.addObserverForPlayer()
    }
    
    //MARK:如果在播放音频前有录音操作,需要重新设置音频会话,否则声音极小.
    func setPlaybackSession() {
        let session = AVAudioSession.sharedInstance()
        do {
            try session.setActive(true)
            try session.setCategory(AVAudioSession.Category.playback)
        } catch {
            print(error)
        }
    }
    
    //MARK:创建播放器
    func createAudioPlayer() {
        //创建媒体资源管理对象
        let Url = URL(string: self.audioPath)
        self.palyerItem = AVPlayerItem.init(url: Url!)
        //创建ACplayer:负责视频播放
        self.player = AVPlayer.init(playerItem: self.palyerItem)
        self.player.rate = 1.0
        self.player.volume = self.volume
        //与播放缓存相关的观测属性
        self.addObserverForPlayItem()
    }
    
    //MARK:开始播放
    func playAudio() {
        self.player.play()
    }
    
    //MARK:暂停播放
    func pauseAudio() {
        self.player.pause()
    }
    
    //MARK:停止播放
    func stopAudio() {
        self.destroyPlayer()
    }
    
    //MARK:改变播放进度
    func changeAudio(currentTime:Float64) {
        if currentTime>0 {
            let time:CMTime = CMTimeMakeWithSeconds(currentTime, preferredTimescale: 1 * Int32(NSEC_PER_SEC))
            self.player.seek(to: time, toleranceBefore: CMTime.zero, toleranceAfter:CMTime.zero)
            self.player.play()
        }
    }
    
    //MARK:Notification
    //播放完毕
    func addObserverForPlayer() {
        NotificationCenter.default.addObserver(self, selector: #selector(audioPlayCompletion), name: NSNotification.Name.AVPlayerItemDidPlayToEndTime, object: nil)
    }
    
    //MARK:Observer
    //观察 加载状态 加载进度
    func addObserverForPlayItem() {
        self.player.currentItem?.addObserver(self, forKeyPath: "status", options: .new, context: nil)
        self.player.currentItem?.addObserver(self, forKeyPath: "loadedTimeRanges", options: .new, context: nil)
        self.player.currentItem?.addObserver(self, forKeyPath: "playbackBufferEmpty", options: .new, context: nil)
        self.player.currentItem?.addObserver(self, forKeyPath: "playbackLikelyToKeepUp", options: .new, context: nil)
    }
    
    override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
        let playerItem:AVPlayerItem = object as! AVPlayerItem
        
        if keyPath == "status" {
            switch self.palyerItem?.status {
                case .readyToPlay ://准备播放
                    self.canDraggingFlag = true

                    self.totalDuration = CMTimeGetSeconds(playerItem.duration)
                    self.prepareToPlayBlock?(Float(self.totalDuration))
                    
                    if self.timeOffset>0 {
                        let time:CMTime = CMTimeMakeWithSeconds(self.timeOffset, preferredTimescale: 1 * Int32(NSEC_PER_SEC))
                        self.player.seek(to: time, toleranceBefore: CMTime.zero, toleranceAfter:CMTime.zero)
                    }
                    self.updatePlayProgress()//播放进度
                    
                    //self.playAudio()
                    break
                case .failed:
                    self.completePlayBlock?(false)
                    break
                case .unknown:
                    break
                default :
                    break
            }
        } else if keyPath == "loadedTimeRanges"{//获取最新缓存的区间
            let bufferInterval:TimeInterval = self.bufferedDuration()
            self.bufferingBlock?(Float(bufferInterval))
        } else if keyPath == "playbackBufferEmpty"{//正在缓存视频请稍等
            
        } else if keyPath == "playbackLikelyToKeepUp"{//缓存好了继续播放
            
        }
    }
    
    //播放进度
    func updatePlayProgress() {
        self.player.addPeriodicTimeObserver(forInterval: CMTimeMake(value: 1, timescale: 1), queue: DispatchQueue.main) { (time) in
            //当前正在播放的时间
            self.currentTime = CMTimeGetSeconds(time)
            self.playingBlock?(self.currentTime,Float(self.currentTime))
        }
    }
    
    //MARK:播放完毕
    @objc func audioPlayCompletion() {
        if self.player.currentItem?.status == AVPlayerItem.Status.readyToPlay {
            self.player.seek(to: CMTime.zero) { finished in
                self.completePlayBlock?(true)
            }
        }
    }
    
    //MARK:计算缓冲进度
    func bufferedDuration() -> (TimeInterval) {
        let loadTimeArray = self.palyerItem!.loadedTimeRanges
        //获取最新缓存的区间
        let newTimeRange : CMTimeRange = loadTimeArray.first as! CMTimeRange
        let startSeconds = CMTimeGetSeconds(newTimeRange.start);
        let durationSeconds = CMTimeGetSeconds(newTimeRange.duration);
        let totalBuffer = startSeconds + durationSeconds;//缓冲总长度
        //print("当前缓冲时间:\(totalBuffer)")
        return totalBuffer
    }
    
    //MARK:释放 播放器
    func destroyPlayer() {
        self.player.pause()
        self.currentTime = 0.0
        self.player.currentItem?.cancelPendingSeeks()
        self.player.currentItem?.asset.cancelLoading()
        self.player.replaceCurrentItem(with: nil)
    }
    
    //MARK:移除通知
    func removeObserverFromPlayer() {
        self.player.currentItem?.removeObserver(self, forKeyPath: "status")
        self.player.currentItem?.removeObserver(self, forKeyPath: "loadedTimeRanges")
        self.player.currentItem?.removeObserver(self, forKeyPath: "playbackBufferEmpty")
        self.player.currentItem?.removeObserver(self, forKeyPath: "playbackLikelyToKeepUp")
        
        NotificationCenter.default.removeObserver(self, name: NSNotification.Name.AVPlayerItemDidPlayToEndTime, object: nil)
    }
}
let netPlayer = NetAudioPlayerTool()
//线上音频 播放时间
var currentTime:Float = 0
//线上音频 音频总时长
var totalDuration:Float = 0
//MARK:创建线上播放器
func createOnlineAudioPlayer() {
    self.netPlayer.audioPath = "https://webfs.ali.kugou.com/202108021358/f8e446476833e4c950f58d07762c4827/KGTX/CLTX001/84ea1667187279849794a9cc96fa0d27.mp3"
    self.netPlayer.createAudioPlayer()
}
//MARK:Player Block
func addNetPlayerBlock() {
    //网络音频 准备播放时更新UI
    netPlayer.prepareToPlayBlock = { (totalDuration) in
        print("音频总时长:\(PublicFunction.convertTime(totalSeconds: Int(totalDuration)))")
        self.totalDuration = totalDuration
    }
    
    //网络音频 缓冲时更新UI
    netPlayer.bufferingBlock = { [weak self] (bufferInterval) in
        print("当前缓冲时间:\(PublicFunction.convertTime(totalSeconds: Int(bufferInterval)))")
        print("当前缓冲进度:\(Float(bufferInterval / self!.totalDuration))")
    }
    
    //网络音频 播放时更新UI
    netPlayer.playingBlock = { (currentTime,currentProgress) in
        print("当前播放时长---\(PublicFunction.convertTime(totalSeconds: Int(currentTime)))")
        print("当前播放进度:\(currentProgress)")
    }
    
    //网络音频 播放完成更新UI
    netPlayer.completePlayBlock = { (isSuccess) in
        print("播放完毕")
    }
}

//开始播放
netPlayer.playAudio()
//暂停播放
netPlayer.pauseAudio()
//拖动修改进度
netPlayer.changeAudio(currentTime: Float64(sender.value))
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,013评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,205评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,370评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,168评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,153评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,954评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,271评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,916评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,382评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,877评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,989评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,624评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,209评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,199评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,418评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,401评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,700评论 2 345

推荐阅读更多精彩内容