We developed different downloaders for handling HLS/FPS and MPEG-DASH content. They can all be used behind the common interface DownloaderAPI
DownloaderAPI
The main data type in DownloaderAPI
is a Download
.
let url = URL(string: "..")!
let config = PlayerConfiguration(with: url, contentType: "...")
let downloader = PRESTOplaySDK.shared.downloader()
let download = downloader.createDownload(config, headers: [:])
downloader.prepareDownload(download.uuid) { download, error in
guard let download else { return }
var res = false;
if let videoRendition = request.videoTracks[0].renditions[0] {
res = request.selectVideoRendition(videoRendition)
}
if let audioRendition = request.audioTracks[0].renditions[0] {
res = request.selectAudioRendition(audioRendition)
}
let delegate = DownloadDelegate({})
let error = downloader.startDownload(download, delegate: delegate)
}
To receive events related to a Download
we use a delegate:
class MyDownloadDelegate: DownloadDelegate {
public let complete: () -> Void
public init (_ complete: @escaping () -> Void) {
self.complete = complete
}
func didStateChange(_ download: MediaDownload) {
if (download.state == .success) {
complete()
}
}
func didProgressChange(_ download: MediaDownload) {
}
}
License
If you want to prefetch the license without forcing the user to be online during the first playback, use the method PRESTOplaySDK.shared.prefetcher(for:)
let prefetcher = PRESTOplaySDK.shared.prefetcher(for: config)
prefetcher?.prefetchKeys() { error in
if error != nil {
// Handle error
}
}
HLS Streams
For downloading HLS and FPS we created a wrapper around the native HLS downloader AVAssetDownloadTask
- enables querying and selection of stream bitrate
- enables querying of audio media streams and subtitle media tracks
- adds support for pre-selecting audio and subtitle tracks before download begins
- greatly simplifies API usage and the monitoring of downloading progress.
- supports preselection of the preferred media bitrate. If no suitable media bitrate is found, the highest bitrate will be selected. Only one media bitrate can be selected, meaning that only one video rendition will be downloaded
- background downloads are fully supported
- if the content is encrypted, it’s possible to prefetch the license before starting the download. It’s also possible to get the license after the download has completed (and without even starting the playback)
Background download tasks are handled by NSURLSession
underneath and there are differences between the test environment and the production environment.
For background downloads an instance of HLSDownloader should be kept in AppDelegate. This is the only supported way from Apple to preserve completionHandlers of background sessions in-between Application Lifecycle changes (in our example we use static class field, but it is also possible to use object variables in AppDelegate).
When additional tracks are selected, it’s not possible to know in advance the entire download size due to an API limitation. The download progress will restart from zero for each additional track. You can however differentiate between the main download (which includes the minimum set of tracks that can be played) and the additional tracks using the flag.
When using a master playlist, the CODECS attribute of the EXT-X-STREAM-INF tag may signal CODECs not supported for playback on the downloading device, but download of such streams will be possible. Check Apple FAQ for an updated list of recommended codecs.
HLS Download progress
Natively Apple’s API measure download progress in duration of the stream that is cached. The PRESTOplay SDK uses average bandwith provided in the main playlist to estimate the download size and progress.
Since AVERAGE-BANDWIDTH
is optional part of HLS playlist, the SDK fallbacks to mandatory BANDWIDTH
field.
BANDWITH
is a peak segment bit rate, so the download sizes can be overestimated (please see HLS Streaming standard to details).
We recommend to use native time-based download progress measures for HLS content.
MPEG-DASH Streams
DASH streams are downloaded using NSURLSession
with custom manifest parser library.
Additional track selection
Additional track selection can be done on Download
class instance.
We encourage to store all possible selection in your ViewController
class.
var audioTracks = [AudioTrack]()
var audioRenditions = [AudioRendition]()
var textTracks = [TextTrack]()
var videoRenditions = [VideoRendition]()
First you need to create the download.
let download = downloader.createDownload(persistableConfiguration, headers: [:])
When prepareDownload
step is finished you can get the list of all available to download tracks.
downloader.prepareDownload(download.uuid) { download, error in
...
download.audioTracks.forEach { track in
track.renditions.forEach { rendition in
self.audioRenditions.append(rendition)
}
}
self.audioTracks = download.audioTracks
self.textTracks = download.textTracks
download.videoTracks.forEach { track in
track.renditions.forEach { rendition in
self.videoRenditions.append(rendition)
}
}
...
}
Below you can find an universal method to select and unselect tracks for the download (for native HLS only the default video track is selected).
func selectTrack(_ indexPath: IndexPath , isSelected: Bool) {
if indexPath.section == audioSectionIndex {
let rendition = audioRenditions[indexPath.row]
if isSelected {
_ = download?.selectAudioRendition(rendition)
} else {
_ = download?.unselectAudioRendition(rendition)
}
} else if indexPath.section == textSectionIndex {
let textTrack = textTracks[indexPath.row]
if isSelected {
_ = download?.selectTextTrack(textTrack)
} else {
_ = download?.unselectTextTrack(textTrack)
}
} else if indexPath.section == renditionSectionIndex {
let rendition = videoRenditions[indexPath.row]
if isSelected {
_ = download?.selectVideoRendition(rendition)
} else {
_ = download?.unselectVideoRendition(rendition)
}
}
}