获得naturalSize和列表分辨率视频m3u8

时间:2018-09-28 04:39:28

标签: ios swift hls

我播放视频m3u8。

我尝试使用let videoAssetSource = AVAsset(url: videoURL),但是videoAssetSource.tracks(withMediaType: .video).count总是返回0。 当我使用mp4链接时,此操作成功。

在播放视频时如何获得列表质量链接m3u8支持和更改质量。

2 个答案:

答案 0 :(得分:2)

您必须为分辨率创建自己的相应模型,但这样的代码应该可以工作。

    /// Downloads the stream file and converts it to the raw playlist.
    /// - Parameter completion: In successful case should return the `RawPlalist` which contains the url with which was the request performed
    /// and the string representation of the downloaded file as `content: String` parameter.
    func getPlaylist(from url: URL, completion: @escaping (Result<RawPlaylist, Error>) -> Void) {
        task = URLSession.shared.dataTask(with: url) { data, response, error in
            if let error = error {
                completion(.failure(error))
            } else if let data = data, let string = String(data: data, encoding: .utf8) {
                completion(.success(RawPlaylist(url: url, content: string)))
            } else {
                completion(.failure(PlayerException.MEDIA_ERR_DECODE)) // Probably an MP4 file.
            }
        }
        task?.resume()
    }

    /// Iterates over the provided playlist contetn and fetches all the stream info data under the `#EXT-X-STREAM-INF"` key.
    /// - Parameter playlist: Playlist object obtained from the stream url.
    /// - Returns: All available stream resolutions for respective bandwidth.
    func getStreamResolutions(from playlist: RawPlaylist) -> [StreamResolution] {
        var resolutions = [StreamResolution]()
        playlist.content.enumerateLines { line, shouldStop in
            let infoline = line.replacingOccurrences(of: "#EXT-X-STREAM-INF", with: "")
            let infoItems = infoline.components(separatedBy: ",")
            let bandwidthItem = infoItems.first(where: { $0.contains(":BANDWIDTH") })
            let resolutionItem = infoItems.first(where: { $0.contains("RESOLUTION")})
            if let bandwidth = bandwidthItem?.components(separatedBy: "=").last,
               let numericBandwidth = Double(bandwidth),
               let resolution = resolutionItem?.components(separatedBy: "=").last?.components(separatedBy: "x"),
               let strignWidth = resolution.first,
               let stringHeight = resolution.last,
               let width = Double(strignWidth),
               let height = Double(stringHeight) {
                resolutions.append(StreamResolution(maxBandwidth: numericBandwidth,
                                                    averageBandwidth: numericBandwidth,
                                                    resolution: CGSize(width: width, height: height)))
            }
        }
        return resolutions
    }
}

答案 1 :(得分:1)

您需要为观察者订阅播放器项目上的属性轨道:

//Define this variable globally
var observers:[NSKeyValueObservation]? = [NSKeyValueObservation]()

//Find tracks
let videoAssetSource = AVAsset(url: videoURL)
let playerItem = AVPlayerItem(asset: videoAssetSource)

let tracksObserver = self.playerItem.observe(\.tracks, options: [.old, .new]) { (item, change) in
     for track in item.tracks {
        let _assetTrack:AVAssetTrack? = track.assetTrack
        if let assetTrack = _assetTrack {
           if assetTrack.mediaType == .video {
              //we found a video track
           } 
        }
     }
}

//Keep observer reference
observers?.append(tracksObserver)

我正在使用基于Swift 4的基于块的键值观察器,但是如果需要,您可以使用observeValue(forKeyPath:...)。