欢迎关注我创建的Android TV 简书专题,会定期给大家分享一些AndroidTV相关的内容:
Android TV
前言:
ExoPlayer
是google推出的一个应用级的播放框架。接下来我来对它做一个简单的介绍。
###一,为什么是ExoPlayer?
1.支持更多的格式
我们知道Android 提供的mediaPlayer只支持有限的格式,比如视频,它支持MP4,3GP。想要播放其他格式的视频还要进行相应的转换。ExoPlayer提供了强大的格式支持。
ExoPlayer支持的媒体格是在多个级别定义的。 从最低层到最高层,分别是:
单个媒体样本的格式(例如,视频帧或音频帧)。 我们称这些样本格式 。 请注意,典型的视频文件将包含至少两种样本格式的媒体; 一个用于视频(例如,H.264)和一个用于音频(例如,AAC)。
容纳媒体样本和相关元数据的容器的格式。 我们称这些容器格式 。 媒体文件具有单个容器格式(例如MP4),通常由文件扩展名指示。 请注意,对于某些仅音频格式(例如MP3),样本和容器格式可能相同。
自适应流技术,如DASH,SmoothStreaming和HLS。 这些不是媒体格式,但是仍然需要定义什么级别的支持ExoPlayer提供。
2.ExoPlayer是java语言编写的,相比与MediaPlayer用Native Code来说,可以更清楚的获得一些异常源和进行部分代码调试。
3.ExoPlayer是开源的,github上地址
ExoPlayer
遇到问题后可以提交Issues,最近遇到关于SimpleExoPlayer setVideoListener去监听OnVideoSizeChanged()时黑屏有声音,然后提交了一个Issues:
The surface show nothing after mSimpleExoPlayer.setVideoListener(mVideoListener)
,他们很快给了回复,另外开源的代码经过大家的验证,也会更加的健壮。
4.ExoPlayer有强大的开发文档
developer guide
,读者也可以根据文档更好的了解它。
###二,如何使用?
这里演示一个播放器的简单使用。
####1.在xml中
<LinearLayout
android:id="@+id/activity_main"
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="com.songwenju.useexoplayer.MainActivity">
<com.google.android.exoplayer2.ui.SimpleExoPlayerView
android:id="@+id/exoView"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ProgressBar
android:id="@+id/progressBar"
android:visibility="gone"
style="?android:attr/progressBarStyleLarge"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"/>
</com.google.android.exoplayer2.ui.SimpleExoPlayerView>
</LinearLayout>
这里使用了SimpleExoPlayerView,在其内部添加了一个ProgressBar。
####2.播放需要的步骤
初始化播放器:
* 初始化player
private
void
initPlayer
()
{
BandwidthMeter bandwidthMeter =
new
DefaultBandwidthMeter();
TrackSelection.Factory videoTackSelectionFactory =
new
AdaptiveTrackSelection.Factory(bandwidthMeter);
TrackSelector trackSelector =
new
DefaultTrackSelector(videoTackSelectionFactory);
LoadControl loadControl =
new
DefaultLoadControl();
mSimpleExoPlayer = ExoPlayerFactory.newSimpleInstance(
this
,trackSelector,loadControl);
mExoPlayerView = (SimpleExoPlayerView) findViewById(R.id.exoView);
mExoPlayerView.setPlayer(mSimpleExoPlayer);
开始播放:
private void playVideo() {
DefaultBandwidthMeter bandwidthMeter = new DefaultBandwidthMeter();
DataSource.Factory dataSourceFactory
= new DefaultDataSourceFactory(MainActivity.this,
Util.getUserAgent(MainActivity.this,"useExoplayer"),bandwidthMeter);
ExtractorsFactory extractorsFactory = new DefaultExtractorsFactory();
MediaSource videoSource = new ExtractorMediaSource(playerUri,dataSourceFactory,extractorsFactory,
null,null);
mSimpleExoPlayer.prepare(videoSource);
mSimpleExoPlayer.addListener(eventListener);
mSimpleExoPlayer.setPlayWhenReady(true);
监听播放器的状态:
private ExoPlayer.EventListener eventListener = new ExoPlayer.EventListener() {
@Override
public void onTimelineChanged(Timeline timeline, Object manifest) {
LogUtil.i(this,"onTimelineChanged");
@Override
public void onTracksChanged(TrackGroupArray trackGroups, TrackSelectionArray trackSelections) {
LogUtil.i(this,"onTracksChanged");
@Override
public void onLoadingChanged(boolean isLoading) {
LogUtil.i(this,"onLoadingChanged");
@Override
public void onPlayerStateChanged(boolean playWhenReady, int playbackState) {
LogUtil.i(this,"onPlayerStateChanged: playWhenReady = "+String.valueOf(playWhenReady)
+" playbackState = "+playbackState);
switch (playbackState){
case ExoPlayer.STATE_ENDED:
LogUtil.i(this,"Playback ended!");
setPlayPause(false);
mSimpleExoPlayer.seekTo(0);
break;
case ExoPlayer.STATE_READY:
mProgressBar.setVisibility(View.GONE);
LogUtil.i(this,"ExoPlayer ready! pos: "+mSimpleExoPlayer.getCurrentPosition()
+" max: "+stringForTime((int)mSimpleExoPlayer.getDuration()));
setProgress(0);
break;
case ExoPlayer.STATE_BUFFERING:
LogUtil.i(this,"Playback buffering!");
mProgressBar.setVisibility(View.VISIBLE);
break;
case ExoPlayer.STATE_IDLE:
LogUtil.i(this,"ExoPlayer idle!");
break;
@Override
public void onPlayerError(ExoPlaybackException error) {
LogUtil.i(this,"onPlaybackError: "+error.getMessage());
@Override
public void onPositionDiscontinuity() {
LogUtil.i(this,"onPositionDiscontinuity");
@Override
public void onPlaybackParametersChanged(PlaybackParameters playbackParameters) {
LogUtil.i(this,"MainActivity.onPlaybackParametersChanged."+playbackParameters.toString());
mSimpleExoPlayer.addListener(eventListener);
通过监听状态去显示loading条,做错误处理等。
Activity退出时,释放播放器:
@Override
protected void onPause() {
LogUtil.i(this,"MainActivity.onPause.");
super.onPause();
mSimpleExoPlayer.stop();
@Override
protected void onStop() {
LogUtil.i(this,"MainActivity.onStop.");
super.onStop();
mSimpleExoPlayer.release();
MainActivity完整的代码
public class MainActivity extends AppCompatActivity {
private SimpleExoPlayerView mExoPlayerView;
private SimpleExoPlayer mSimpleExoPlayer;
private Context mContext;
Uri playerUri = Uri.parse("https://storage.googleapis.com/android-tv/Sample%20videos/Demo%20Slam/Google%20Demo%20Slam_%20Hangin'%20with%20the%20Google%20Search%20Bar.mp4");
private ProgressBar mProgressBar;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
WindowManager.LayoutParams.FLAG_FULLSCREEN);
setContentView(R.layout.activity_main);
mContext = this;
mProgressBar = (ProgressBar) findViewById(R.id.progressBar);
initPlayer();
playVideo();
* 初始化player
private void initPlayer() {
BandwidthMeter bandwidthMeter = new DefaultBandwidthMeter();
TrackSelection.Factory videoTackSelectionFactory =
new AdaptiveTrackSelection.Factory(bandwidthMeter);
TrackSelector trackSelector =
new DefaultTrackSelector(videoTackSelectionFactory);
LoadControl loadControl = new DefaultLoadControl();
mSimpleExoPlayer = ExoPlayerFactory.newSimpleInstance(this,trackSelector,loadControl);
mExoPlayerView = (SimpleExoPlayerView) findViewById(R.id.exoView);
mExoPlayerView.setPlayer(mSimpleExoPlayer);
private void playVideo() {
DefaultBandwidthMeter bandwidthMeter = new DefaultBandwidthMeter();
DataSource.Factory dataSourceFactory
= new DefaultDataSourceFactory(MainActivity.this,
Util.getUserAgent(MainActivity.this,"useExoplayer"),bandwidthMeter);
ExtractorsFactory extractorsFactory = new DefaultExtractorsFactory();
MediaSource videoSource = new ExtractorMediaSource(playerUri,dataSourceFactory,extractorsFactory,
null,null);
mSimpleExoPlayer.prepare(videoSource);
mSimpleExoPlayer.addListener(eventListener);
mSimpleExoPlayer.setPlayWhenReady(true);
TextRenderer.Output mOutput = new TextRenderer.Output() {
@Override
public void onCues(List<Cue> cues) {
LogUtil.i(this,"MainActivity.onCues.");
private SimpleExoPlayer.VideoListener mVideoListener = new SimpleExoPlayer.VideoListener() {
@Override
public void onVideoSizeChanged(int width, int height, int unappliedRotationDegrees, float pixelWidthHeightRatio) {
LogUtil.i(this,"MainActivity.onVideoSizeChanged.width:"+width+", height:"+height);
@Override
public void onRenderedFirstFrame() {
LogUtil.i(this,"MainActivity.onRenderedFirstFrame.");
private ExoPlayer.EventListener eventListener = new ExoPlayer.EventListener() {
@Override
public void onTimelineChanged(Timeline timeline, Object manifest) {
LogUtil.i(this,"onTimelineChanged");
@Override
public void onTracksChanged(TrackGroupArray trackGroups, TrackSelectionArray trackSelections) {
LogUtil.i(this,"onTracksChanged");
@Override
public void onLoadingChanged(boolean isLoading) {
LogUtil.i(this,"onLoadingChanged");
@Override
public void onPlayerStateChanged(boolean playWhenReady, int playbackState) {
LogUtil.i(this,"onPlayerStateChanged: playWhenReady = "+String.valueOf(playWhenReady)
+" playbackState = "+playbackState);
switch (playbackState){
case ExoPlayer.STATE_ENDED:
LogUtil.i(this,"Playback ended!");
setPlayPause(false);
mSimpleExoPlayer.seekTo(0);
break;
case ExoPlayer.STATE_READY:
mProgressBar.setVisibility(View.GONE);
LogUtil.i(this,"ExoPlayer ready! pos: "+mSimpleExoPlayer.getCurrentPosition()
+" max: "+stringForTime((int)mSimpleExoPlayer.getDuration()));
setProgress(0);
break;
case ExoPlayer.STATE_BUFFERING:
LogUtil.i(this,"Playback buffering!");
mProgressBar.setVisibility(View.VISIBLE);
break;
case ExoPlayer.STATE_IDLE:
LogUtil.i(this,"ExoPlayer idle!");
break;
@Override
public void onPlayerError(ExoPlaybackException error) {
LogUtil.i(this,"onPlaybackError: "+error.getMessage());
@Override
public void onPositionDiscontinuity() {
LogUtil.i(this,"onPositionDiscontinuity");
@Override
public void onPlaybackParametersChanged(PlaybackParameters playbackParameters) {
LogUtil.i(this,"MainActivity.onPlaybackParametersChanged."+playbackParameters.toString());
* Starts or stops playback. Also takes care of the Play/Pause button toggling
* @param play True if playback should be started
private void setPlayPause(boolean play){
mSimpleExoPlayer.setPlayWhenReady(play);
private String stringForTime(int timeMs) {
StringBuilder mFormatBuilder;
Formatter mFormatter;
mFormatBuilder = new StringBuilder();
mFormatter = new Formatter(mFormatBuilder, Locale.getDefault());
int totalSeconds = timeMs / 1000;
int seconds = totalSeconds % 60;
int minutes = (totalSeconds / 60) % 60;
int hours = totalSeconds / 3600;
mFormatBuilder.setLength(0);
if (hours > 0) {
return mFormatter.format("%d:%02d:%02d", hours, minutes, seconds).toString();
} else {
return mFormatter.format("%02d:%02d", minutes, seconds).toString();
@Override
protected void onPause() {
LogUtil.i(this,"MainActivity.onPause.");
super.onPause();
mSimpleExoPlayer.stop();
@Override
protected void onStop() {
LogUtil.i(this,"MainActivity.onStop.");
super.onStop();
mSimpleExoPlayer.release();
gitHub实例:
useExoPlayer
。
关注微信公众号,定期为你推荐移动开发相关文章。
Android工程师
粉丝