预览视频是鼓励用户深层链接访问您的 TV 应用的好方法。 预览内容可以是简短的视频剪辑,也可以是完整的电影预告片。
在创建预览时,请注意以下准则:
- 不要在预览中显示广告。如果您在客户端拼接广告 请勿将其拼接到预览视频中。如果您在服务器端拼接广告 提供无广告视频的预览。
- 为获得最佳品质,预览视频的宽高比应为 16:9 或 4:3。请参阅 视频节目属性 了解推荐的预览视频尺寸。
- 当预览视频和海报图片具有不同的宽高比时,
在播放预览之前,主屏幕会将海报视图调整为视频的宽高比。
视频不是信箱模式的。例如,如果
海报图片宽高比是
ASPECT_RATIO_MOVIE_POSTER
(1:1.441) 但视频的宽高比是 16:9,那么海报视图就会转换为 16:9 的区域。 - 创建预览后,其内容可公开访问,也可以 受 DRM 保护。每种情形采用的步骤不同。当前页面 都说明了这一点
在主屏幕中播放预览
如果您使用任何视频类型来创建预览 ExoPlayer 支持 并且预览可公开访问,因此您可以直接在主屏幕上播放预览。
构建 PreviewProgram
将 setPreviewVideoUri()
与可公开访问的 HTTPS 搭配使用
如下例所示。预览可以是
视频或
audio。
val previewVideoUrl = Uri.parse("https://www.example.com/preview.mp4") val builder = PreviewProgram.Builder() builder.setChannelId(channelId) // ... .setPreviewVideoUri(previewVideoUrl)
Uri previewVideoUrl = Uri.parse("https://www.example.com/preview.mp4"); PreviewProgram.Builder builder = new PreviewProgram.Builder(); builder.setChannelId(channelId) // ... .setPreviewVideoUri(Uri.parse(previewVideoUrl));
在 Surface 上呈现预览
如果您的视频受 DRM 保护,或采用的是不受 DRM 支持的媒体类型
ExoPlayer,请使用 TvInputService
。
Android TV 主屏幕将 Surface
传递给您的服务
方法是调用 onSetSurface()
。您的应用会从 onTune()
直接在这个 Surface 上绘制视频。
直接 Surface 渲染可让应用控制渲染的内容及其呈现方式 。您可以叠加频道归因等元数据。
在清单中声明 TvInputService
应用必须提供 TvInputService
的实现
以便主屏幕能够呈现您的预览画面
在您的服务声明中,请包括一个 intent 过滤器,用于指定
TvInputService
作为要通过
intent。另外,请将服务元数据声明为单独的 XML 资源。通过
服务声明、intent 过滤器和服务元数据声明
如下例中所示:
<service android:name=".rich.PreviewInputService" android:permission="android.permission.BIND_TV_INPUT"> <!-- Required filter used by the system to launch our account service. --> <intent-filter> <action android:name="android.media.tv.TvInputService" /> </intent-filter> <!-- An XML file which describes this input. --> <meta-data android:name="android.media.tv.input" android:resource="@xml/previewinputservice" /> </service>
在单独的 XML 文件中定义服务元数据。
服务元数据文件位于 XML 资源目录中
并且必须与您在
清单。使用上一个示例中的清单条目,
在 res/xml/previewinputservice.xml
处创建一个 XML 文件,该文件包含一个空白
tv-input
标记:
<?xml version="1.0" encoding="utf-8"?>
<tv-input/>
TV 输入框架必须包含这个标记。不过, 仅用于配置直播频道由于您呈现的是视频 应为空。
创建视频 URI
指明您的预览视频应由您的应用呈现,而不是由您的应用呈现
Android TV 主屏幕时,您必须为 PreviewProgram
创建一个视频 URI。
URI 应以您的应用用于内容的标识符结尾,因此
可以稍后在 TvInputService
中检索该内容。
如果标识符是 Long
类型,请使用
TvContractCompat.buildPreviewProgramUri():
val id: Long = 1L // content identifier val componentName = new ComponentName(context, PreviewVideoInputService.class) val previewProgramVideoUri = TvContractCompat.buildPreviewProgramUri(id) .buildUpon() .appendQueryParameter("input", TvContractCompat.buildInputId(componentName)) .build()
Long id = 1L; // content identifier ComponentName componentName = new ComponentName(context, PreviewVideoInputService.class); previewProgramVideoUri = TvContractCompat.buildPreviewProgramUri(id) .buildUpon() .appendQueryParameter("input", TvContractCompat.buildInputId(componentName)) .build();
如果标识符不是 Long
类型,请使用以下代码构建 URI:
Uri.withAppendedPath()
:
val previewProgramVideoUri = Uri.withAppendedPath(PreviewPrograms.CONTENT_URI, "content-identifier") .buildUpon() .appendQueryParameter("input", TvContractCompat.buildInputId(componentName)) .build()
previewProgramVideoUri = Uri.withAppendedPath(PreviewPrograms.CONTENT_URI, "content-identifier") .buildUpon() .appendQueryParameter("input", TvContractCompat.buildInputId(componentName)) .build();
您的应用调用
onTune(Uri videoUri)
让 Android TV 启动预览视频。
创建服务
以下示例展示了如何扩展 TvInputService
以创建您自己的
PreviewInputService
。请注意,该服务使用 MediaPlayer
进行播放,
但您的代码可以使用任何可用的视频播放器。
import android.content.Context import android.media.MediaPlayer import android.media.tv.TvInputService import android.net.Uri import android.util.Log import android.view.Surface import java.io.IOException class PreviewVideoInputService : TvInputService() { override fun onCreateSession(inputId: String): TvInputService.Session? { return PreviewSession(this) } private inner class PreviewSession( internal var context: Context ) : TvInputService.Session(context) { internal var mediaPlayer: MediaPlayer? = MediaPlayer() override fun onRelease() { mediaPlayer?.release() mediaPlayer = null } override fun onTune(uri: Uri): Boolean { // Let the TvInputService know that the video is being loaded. notifyVideoUnavailable(VIDEO_UNAVAILABLE_REASON_TUNING) // Fetch the stream url from the TV Provider database // for content://android.media.tv/preview_program/val id = uri.lastPathSegment // Load your video in the background. retrieveYourVideoPreviewUrl(id) { videoUri -> if (videoUri == null) { Log.d(TAG, "Could not find video $id") notifyVideoUnavailable(TvInputManager.VIDEO_UNAVAILABLE_REASON_UNKNOWN) } try { mPlayer.setDataSource(getApplicationContext(), videoUri) mPlayer.prepare() mPlayer.start() notifyVideoAvailable() } catch (IOException e) { Log.e(TAG, "Could not prepare media player", e) notifyVideoUnavailable(TvInputManager.VIDEO_UNAVAILABLE_REASON_UNKNOWN) } } return true } override fun onSetSurface(surface: Surface?): Boolean { mediaPlayer?.setSurface(surface) return true } override fun onSetStreamVolume(volume: Float) { // The home screen may fade in and out the video's volume. // Your player should be updated accordingly. mediaPlayer?.setVolume(volume, volume) } override fun onSetCaptionEnabled(b: Boolean) { // enable/disable captions here } } companion object { private const val TAG = "PreviewInputService" } }
import android.content.Context; import android.media.MediaPlayer; import android.media.tv.TvInputService; import android.net.Uri; import android.support.annotation.Nullable; import android.util.Log; import android.view.Surface; import java.io.IOException; public class PreviewVideoInputService extends TvInputService { private static final String TAG = "PreviewVideoInputService"; @Nullable @Override public Session onCreateSession(String inputId) { return new PreviewSession(this); } private class PreviewSession extends TvInputService.Session { private MediaPlayer mPlayer; PreviewSession(Context context) { super(context); mPlayer = new MediaPlayer(); } @Override public boolean onTune(Uri channelUri) { // Let the TvInputService know that the video is being loaded. notifyVideoUnavailable(VIDEO_UNAVAILABLE_REASON_TUNING); // Fetch the stream url from the TV Provider database // for content://android.media.tv/preview_program/String id = uri.getLastPathSegment(); // Load your video in the background. retrieveYourVideoPreviewUrl(id, new MyCallback() { public void callback(Uri videoUri) { if (videoUri == null) { Log.d(TAG, "Could not find video" + id); notifyVideoUnavailable(TvInputManager.VIDEO_UNAVAILABLE_REASON_UNKNOWN); } try { mPlayer.setDataSource(getApplicationContext(), videoUri); mPlayer.prepare(); mPlayer.start(); notifyVideoAvailable(); } catch (IOException e) { Log.e(TAG, "Could not prepare media player", e); notifyVideoUnavailable(TvInputManager.VIDEO_UNAVAILABLE_REASON_UNKNOWN); } } }); return true; } @Override public boolean onSetSurface(@Nullable Surface surface) { if (mPlayer != null) { mPlayer.setSurface(surface); } return true; } @Override public void onRelease() { if (mPlayer != null) { mPlayer.release(); } mPlayer = null; } @Override public void onSetStreamVolume(float volume) { if (mPlayer != null) { // The home screen may fade in and out the video's volume. // Your player should be updated accordingly. mPlayer.setVolume(volume, volume); } } @Override public void onSetCaptionEnabled(boolean enabled) { // enable/disable captions here } } }