ExoPlayer は、複数のコンテナ形式の HLS をサポートしています。含まれる音声と動画のサンプル形式もサポートされている必要があります(詳しくは、サンプル形式のセクションをご覧ください)。こちらのブログ投稿で説明されているように、HLS コンテンツ プロデューサーは高品質の HLS ストリームを生成することを強くおすすめします。
機能 | サポート対象 | コメント |
---|---|---|
コンテナ | ||
MPEG-TS | はい | |
FMP4/CMAF | はい | |
ADTS(AAC) | はい | |
MP3 | はい | |
字幕 /字幕 | ||
CEA-608 | はい | |
CEA-708 | はい | |
WebVTT | はい | |
メタデータ | ||
ID3 | はい | |
SCTE-35 | いいえ | |
コンテンツの保護 | ||
AES-128 | はい | |
AES-128 のサンプル | いいえ | |
Widevine | はい | API 19 以降(cenc スキーム)および 25 以降(cbcs スキーム) |
PlayReady SL2000 | はい | Android TV のみ |
サーバー管理 | ||
差分更新 | はい | |
再生リストの再読み込みをブロックする | はい | |
プリロード ヒントの読み込みをブロックする | はい | 長さが未定義のバイト範囲を除く |
広告挿入 | ||
サーバー ガイドによる広告挿入(インタースティシャル) | 部分的 | X-ASSET-URI の VOD のみ。ライブ配信と X-ASSET-LIST は後で追加されます。 |
IMA サーバーサイド広告とクライアントサイド広告 | はい | 広告挿入ガイド |
ライブ再生 | ||
通常のライブ再生 | はい | |
低レイテンシ HLS(Apple) | はい | |
低レイテンシ HLS(コミュニティ) | いいえ | |
共通メディア クライアント データ CMCD | はい | CMCD 統合ガイド |
MediaItem の使用
HLS ストリームを再生するには、HLS モジュールに依存する必要があります。
Kotlin
implementation("androidx.media3:media3-exoplayer-hls:1.6.0")
Groovy
implementation "androidx.media3:media3-exoplayer-hls:1.6.0"
次に、HLS 再生リスト URI の MediaItem
を作成し、プレーヤーに渡します。
Kotlin
// Create a player instance.
val player = ExoPlayer.Builder(context).build()
// Set the media item to be played.
player.setMediaItem(MediaItem.fromUri(hlsUri))
// Prepare the player.
player.prepare()
Java
// Create a player instance.
ExoPlayer player = new ExoPlayer.Builder(context).build();
// Set the media item to be played.
player.setMediaItem(MediaItem.fromUri(hlsUri));
// Prepare the player.
player.prepare();
URI が .m3u8
で終わらない場合は、MediaItem.Builder
の setMimeType
に MimeTypes.APPLICATION_M3U8
を渡して、コンテンツのタイプを明示的に指定できます。
メディア アイテムの URI は、メディア プレイリストまたはマルチバリアント プレイリストを参照できます。URI が複数の #EXT-X-STREAM-INF
タグを宣言するマルチバリアント プレイリストを指している場合、ExoPlayer は利用可能な帯域幅とデバイスの機能の両方を考慮して、バリアント間で自動的に適応します。
HlsMediaSource を使用する
より多くのカスタマイズ オプションを使用するには、MediaItem
ではなく HlsMediaSource
を作成してプレーヤーに直接渡します。
Kotlin
// Create a data source factory.
val dataSourceFactory: DataSource.Factory = DefaultHttpDataSource.Factory()
// Create a HLS media source pointing to a playlist uri.
val hlsMediaSource =
HlsMediaSource.Factory(dataSourceFactory).createMediaSource(MediaItem.fromUri(hlsUri))
// Create a player instance.
val player = ExoPlayer.Builder(context).build()
// Set the HLS media source as the playlist with a single media item.
player.setMediaSource(hlsMediaSource)
// Prepare the player.
player.prepare()
Java
// Create a data source factory.
DataSource.Factory dataSourceFactory = new DefaultHttpDataSource.Factory();
// Create a HLS media source pointing to a playlist uri.
HlsMediaSource hlsMediaSource =
new HlsMediaSource.Factory(dataSourceFactory).createMediaSource(MediaItem.fromUri(hlsUri));
// Create a player instance.
ExoPlayer player = new ExoPlayer.Builder(context).build();
// Set the HLS media source as the playlist with a single media item.
player.setMediaSource(hlsMediaSource);
// Prepare the player.
player.prepare();
マニフェストへのアクセス
現在のマニフェストを取得するには、Player.getCurrentManifest
を呼び出します。HLS の場合は、返されたオブジェクトを HlsManifest
にキャストする必要があります。Player.Listener
の onTimelineChanged
コールバックは、マニフェストが読み込まれるたびに呼び出されます。これは、オンデマンド コンテンツの場合は 1 回、ライブ コンテンツの場合は複数回行われます。次のコード スニペットは、マニフェストが読み込まれるたびにアプリが何かを行う方法を示しています。
Kotlin
player.addListener(
object : Player.Listener {
override fun onTimelineChanged(timeline: Timeline, @TimelineChangeReason reason: Int) {
val manifest = player.currentManifest
if (manifest is HlsManifest) {
// Do something with the manifest.
}
}
}
)
Java
player.addListener(
new Player.Listener() {
@Override
public void onTimelineChanged(
Timeline timeline, @Player.TimelineChangeReason int reason) {
Object manifest = player.getCurrentManifest();
if (manifest != null) {
HlsManifest hlsManifest = (HlsManifest) manifest;
// Do something with the manifest.
}
}
});
インタースティシャル広告付きの HLS ストリームを再生する
HLS 仕様では、メディア プレイリストにインタースティシャル情報を含めるために使用できる HLS インタースティシャルが定義されています。ExoPlayer はデフォルトでこれらのインタースティシャルを無視します。サポートを追加するには、HlsInterstitialsAdsLoader
を使用します。最初から仕様のすべての機能がサポートされているわけではありません。ストリーミングがサポートされていない場合は、GitHub で問題を報告し、ストリーミング URI をお知らせください。ストリーミングのサポートを追加いたします。
playlist API で MediaItem
を使用する
インタースティシャルで HLS ストリームを再生する最も便利な方法は、HlsInterstitialsAdsLoader.AdsMediaSourceFactory
を使用して ExoPlayer インスタンスを構築することです。これにより、Player
インターフェースの MediaItem
ベースの playlist API を使用して HLS インタースティシャルを再生できます。
ExoPlayer
の MediaSource.Factory
は、プレーヤー インスタンスをビルドするときにビルダーに挿入できます。
Kotlin
hlsInterstitialsAdsLoader = HlsInterstitialsAdsLoader(context)
// Create a MediaSource.Factory for HLS streams with interstitials.
var hlsMediaSourceFactory =
HlsInterstitialsAdsLoader.AdsMediaSourceFactory(
hlsInterstitialsAdsLoader,
playerView,
DefaultMediaSourceFactory(context),
)
// Build player with interstitials media source factory
player =
ExoPlayer.Builder(context)
.setMediaSourceFactory(hlsMediaSourceFactory)
.build()
// Set the player on the ads loader.
hlsInterstitialsAdsLoader.setPlayer(player)
playerView.setPlayer(player)
Java
hlsInterstitialsAdsLoader = new HlsInterstitialsAdsLoader(context);
// Create a MediaSource.Factory for HLS streams with interstitials.
MediaSource.Factory hlsMediaSourceFactory =
new HlsInterstitialsAdsLoader.AdsMediaSourceFactory(
hlsInterstitialsAdsLoader, playerView, new DefaultMediaSourceFactory(context));
// Build player with interstitials media source factory
player =
new ExoPlayer.Builder(context)
.setMediaSourceFactory(hlsMediaSourceFactory)
.build();
// Set the player on the ads loader.
hlsInterstitialsAdsLoader.setPlayer(player);
playerView.setPlayer(player);
このようなプレーヤーの設定では、HLS インタースティシャルを再生するには、プレーヤーに AdsConfiguration
を使用してメディア アイテムを設定します。
Kotlin
player.setMediaItem(
MediaItem.Builder()
.setUri("https://www.example.com/media.m3u8")
.setAdsConfiguration(
AdsConfiguration.Builder(Uri.parse("hls://interstitials"))
.setAdsId("ad-tag-0") // must be unique within playlist
.build())
.build())
player.prepare();
player.play();
Java
player.setMediaItem(
new MediaItem.Builder()
.setUri("https://www.example.com/media.m3u8")
.setAdsConfiguration(
new AdsConfiguration.Builder(Uri.parse("hls://interstitials"))
.setAdsId("ad-tag-0") // must be unique within playlist
.build())
.build());
player.prepare();
player.play();
メディアソースベースの API を使用する
または、デフォルトのメディアソース ファクトリーをオーバーライドせずに ExoPlayer インスタンスをビルドすることもできます。アプリは、インタースティシャルをサポートするために、HlsInterstitialsAdsLoader.AdsMediaSourceFactory
を直接使用して MediaSource
を作成し、メディアソースベースのプレイリスト API を使用して ExoPlayer に提供できます。
Kotlin
hlsInterstitialsAdsLoader = HlsInterstitialsAdsLoader(context)
// Create a MediaSource.Factory for HLS streams with interstitials.
var hlsMediaSourceFactory =
HlsInterstitialsAdsLoader.AdsMediaSourceFactory(hlsInterstitialsAdsLoader, playerView, context)
// Build player with default media source factory.
player = new ExoPlayer.Builder(context).build();
// Create an media source from an HLS media item with ads configuration.
val mediaSource =
hlsMediaSourceFactory.createMediaSource(
MediaItem.Builder()
.setUri("https://www.example.com/media.m3u8")
.setAdsConfiguration(
MediaItem.AdsConfiguration.Builder(Uri.parse("hls://interstitials"))
.setAdsId("ad-tag-0")
.build()
)
.build()
)
// Set the media source on the player.
player.setMediaSource(mediaSource)
player.prepare()
player.play()
Java
HlsInterstitialsAdsLoader hlsInterstitialsAdsLoader = new HlsInterstitialsAdsLoader(context);
// Create a MediaSource.Factory for HLS streams with interstitials.
MediaSource.Factory hlsMediaSourceFactory =
new HlsInterstitialsAdsLoader.AdsMediaSourceFactory(
hlsInterstitialsAdsLoader, playerView, context);
// Build player with default media source factory.
player = new ExoPlayer.Builder(context).build();
// Create an media source from an HLS media item with ads configuration.
MediaSource mediaSource =
hlsMediaSourceFactory.createMediaSource(
new MediaItem.Builder()
.setUri("https://www.example.com/media.m3u8")
.setAdsConfiguration(
new MediaItem.AdsConfiguration.Builder(Uri.parse("hls://interstitials"))
.setAdsId("ad-tag-0")
.build())
.build());
// Set the media source on the player.
exoPlayer.setMediaSource(mediaSource);
exoPlayer.prepare();
exoPlayer.play();
広告イベントをリッスンする
Listener
を HlsInterstitialsAdsLoader
に追加すると、HLS インタースティシャル再生に関するステータス変更に関するイベントをモニタリングできます。これにより、アプリまたは SDK は、再生された広告、読み込まれたアセットリスト、準備中の広告メディアソースを追跡したり、アセットリストの読み込みエラーや広告の準備エラーを検出したりできます。さらに、広告メディア ソースから出力されたメタデータを受信して、広告の再生の詳細な検証や広告の再生の進行状況のトラッキングを行うことができます。
Kotlin
class AdsLoaderListener : HlsInterstitialsAdsLoader.Listener {
override fun onStart(mediaItem: MediaItem, adsId: Any, adViewProvider: AdViewProvider) {
// Do something when HLS media item with interstitials is started.
}
override fun onMetadata(
mediaItem: MediaItem,
adsId: Any,
adGroupIndex: Int,
adIndexInAdGroup: Int,
metadata: Metadata,
) {
// Do something with metadata that is emitted by the ad media source of the given ad.
}
override fun onAdCompleted(
mediaItem: MediaItem,
adsId: Any,
adGroupIndex: Int,
adIndexInAdGroup: Int,
) {
// Do something when ad completed playback.
}
// ... See JavaDoc for further callbacks of HlsInterstitialsAdsLoader.Listener.
override fun onStop(mediaItem: MediaItem, adsId: Any, adPlaybackState: AdPlaybackState) {
// Do something with the resulting ad playback state when stopped.
}
}
Java
private class AdsLoaderListener
implements HlsInterstitialsAdsLoader.Listener {
// implement HlsInterstitialsAdsLoader.Listener
@Override
public void onStart(MediaItem mediaItem, Object adsId, AdViewProvider adViewProvider) {
// Do something when HLS media item with interstitials is started.
}
@Override
public void onMetadata(
MediaItem mediaItem,
Object adsId,
int adGroupIndex,
int adIndexInAdGroup,
Metadata metadata) {
// Do something with metadata that is emitted by the ad media source of the given ad.
}
@Override
public void onAdCompleted(
MediaItem mediaItem, Object adsId, int adGroupIndex, int adIndexInAdGroup) {
// Do something when ad completed playback.
}
// ... See JavaDoc for further callbacks
@Override
public void onStop(MediaItem mediaItem, Object adsId, AdPlaybackState adPlaybackState) {
// Do something with the resulting ad playback state when stopped.
}
}
使用可能なすべてのコールバックの詳細なドキュメントについては、HlsInterstitialsAdsLoader.Listener
の JavaDoc をご覧ください。
リスナーを広告読み込みツールに追加できます。
Kotlin
var listener = AdsLoaderListener();
// Add the listener to the ads loader to receive ad loader events.
hlsInterstitialsAdsLoader.addListener(listener);
Java
AdsLoaderListener listener = new AdsLoaderListener();
// Add the listener to the ads loader to receive ad loader events.
hlsInterstitialsAdsLoader.addListener(listener);
HlsInterstitialsAdsLoader
ライフサイクル
HlsInterstitialsAdsLoader
または HlsInterstitialsAdsLoader.AdsMediaSourceFactory
のインスタンスは、広告を読み込む必要がある複数のメディアソースを作成する複数のプレーヤー インスタンスで再利用できます。
インスタンスは、たとえば Activity
の onCreate
メソッドで作成し、複数のプレーヤー インスタンスで再利用できます。これは、1 つのプレーヤー インスタンスで同時に使用されている限り機能します。これは、アプリがバックグラウンドに移行され、プレーヤー インスタンスが破棄され、アプリがフォアグラウンドに戻ったときに新しいインスタンスが作成されるという一般的なユースケースに便利です。
Kotlin
// Create the ads loader instance (for example onCreate).
hlsInterstitialsAdsLoader = HlsInterstitialsAdsLoader(context);
// Build a player and set it on the ads loader (for example onStart).
player = ExoPlayer.Builder(context).build();
hlsInterstitialsAdsLoader.setPlayer(player);
// Release the player and unset it on the ads loader (for example onStop).
player.release();
hlsInterstitialsAdsLoader.setPlayer(null);
// Build another player and set it on the ads loader (for example onStart).
player = ExoPlayer.Builder(context).build();
hlsInterstitialsAdsLoader.setPlayer(player);
// Release the player and unset it on the ads loader (for example onStop).
player.release();
hlsInterstitialsAdsLoader.setPlayer(null);
// Release the ads loader when not used anymore (for example onDestroy).
hlsInterstitialsAdsLoader.release();
Java
// Create the ads loader instance (for example onCreate).
hlsInterstitialsAdsLoader = new HlsInterstitialsAdsLoader(context);
// Build a player and set it on the ads loader (for example onStart).
player = new ExoPlayer.Builder(context).build();
hlsInterstitialsAdsLoader.setPlayer(player);
// Release the player and unset it on the ads loader (for example onStop).
player.release();
hlsInterstitialsAdsLoader.setPlayer(null);
// Build another player and set it on the ads loader (for example onStart).
player = new ExoPlayer.Builder(context).build();
hlsInterstitialsAdsLoader.setPlayer(player);
// Release the player and unset it on the ads loader (for example onStop).
player.release();
hlsInterstitialsAdsLoader.setPlayer(null);
// Release the ads loader when not used anymore (for example onDestroy).
hlsInterstitialsAdsLoader.release();
通常は、広告ローダに次のプレーヤー インスタンスを設定する前に、古いプレーヤー インスタンスを解放してください。広告読み込みツール自体が解放されると、広告読み込みツールは使用できなくなります。
再生のカスタマイズ
ExoPlayer には、アプリのニーズに合わせて再生エクスペリエンスを調整するための複数の方法が用意されています。例については、カスタマイズページをご覧ください。
チャンクレス準備の無効化
デフォルトでは、ExoPlayer はチャンクレス準備を使用します。つまり、ExoPlayer はマルチバリアント プレイリストの情報のみを使用してストリームを準備します。これは、#EXT-X-STREAM-INF
タグに CODECS
属性が含まれている場合に機能します。
メディア セグメントに、マルチバリアント プレイリストで #EXT-X-MEDIA:TYPE=CLOSED-CAPTIONS
タグを使用して宣言されていない、マルチエディションの字幕トラックが含まれている場合は、この機能を無効にする必要があります。そうでない場合、これらの字幕トラックは検出されず、再生されません。次のスニペットに示すように、HlsMediaSource.Factory
でチャンクレス準備を無効にできます。ただし、ExoPlayer はこれらの追加トラックを見つけるためにメディア セグメントをダウンロードする必要があるため、起動時間が長くなります。代わりに、マルチバリアント プレイリストで字幕トラックを宣言することをおすすめします。
Kotlin
val hlsMediaSource =
HlsMediaSource.Factory(dataSourceFactory)
.setAllowChunklessPreparation(false)
.createMediaSource(MediaItem.fromUri(hlsUri))
Java
HlsMediaSource hlsMediaSource =
new HlsMediaSource.Factory(dataSourceFactory)
.setAllowChunklessPreparation(false)
.createMediaSource(MediaItem.fromUri(hlsUri));
高品質の HLS コンテンツを作成する
ExoPlayer を最大限に活用するには、HLS コンテンツを改善するためのガイドラインがあります。詳細については、ExoPlayer での HLS 再生に関する Medium 投稿をご覧ください。主なポイントは次のとおりです。
- 正確なセグメントの長さを使用する。
- 連続メディア ストリームを使用して、セグメント間でメディア構造が変更されないようにします。
#EXT-X-INDEPENDENT-SEGMENTS
タグを使用します。- 動画と音声の両方が含まれるファイルではなく、デマルチプレックスされたストリームを優先します。
- マルチバリエーション再生リストには、可能な限りすべての情報を含めます。
ライブ配信には、次のガイドラインが適用されます。
#EXT-X-PROGRAM-DATE-TIME
タグを使用します。#EXT-X-DISCONTINUITY-SEQUENCE
タグを使用します。- 長い公開期間を設定します。1 分以上が理想的です。