アクティビティ(実際にはフラグメント)を一時停止して別のアプリに移動するたびに、onResumeで戻ると、ビデオの再生を再開しようとしますが、再生されません。空白の画面が表示されます。調査したところ、Logcatに次のように表示されます
E/BufferQueueProducer: [unnamed-23827-0] queueBuffer: BufferQueue has been abandoned
E/MediaPlayer: error (1, -38)
E/MediaPlayer: error (1, -38)
E/MediaPlayer: error (1, -38)
E/MediaPlayer: error (1, -38)
E/BufferQueueProducer: [unnamed-23827-0] connect(P): BufferQueue has been abandoned
これが私が履歴書の中で呼び出すコードです
player.seekTo(mVideoSeekPosition);
player.start();
参考:私はこの答えを私のケースに適用しようとしていますが、できません: BufferQueueが放棄されたときに何ができますか?
[〜#〜]更新[〜#〜]
一人で行くのに苦労しましたが、まだクラッシュしています。だから私は助けのためにコード全体を投稿しています
private void setupVideoPlayingSystem(View root) {
textureView = (TextureView) root.findViewById(R.id.textureView);
textureView.setSurfaceTextureListener(this);
}
@Override
public void onSurfaceTextureAvailable(SurfaceTexture surfaceTexture, int width, int height) {
Log.d(TAG, "onSurfaceTextureAvailable");
if (null == surface) {
Log.d(TAG, "new surface");
surface = new Surface(surfaceTexture);
mediaPlayer = new MediaPlayer();
mediaPlayer.setSurface(surface);
mediaPlayer.setLooping(false);
}
/*
outstandingVideoRequest is IOU for orentation change (verifed: onResume before onSurfaceTextureAvailable)
but for cold startup, must check mVideoUrl
*/
if (outstandingVideoRequest && null != mVideoUrl) {
outstandingVideoRequest = false;
playNewVideo(mVideoUrl);
}
}
@Override
public void onSurfaceTextureSizeChanged(SurfaceTexture surface, int width, int height) {
Log.d(TAG, "onSurfaceTextureSizeChanged");
}
@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
Log.d(TAG, "onSurfaceTextureDestroyed");
return false;//leave destruction for onDestroy
}
@Override
public void onSurfaceTextureUpdated(SurfaceTexture surface) {
}
private void playNewVideo(String url) {
if (null == mediaPlayer || null == surface) {
Log.d(TAG, "playNewVideo not ready");
synchronized (outstandingVideoRequest) {
Log.d(TAG, "playNewVideo outstandingVideoRequest");
outstandingVideoRequest = true;
}
} else {
try {
mediaPlayer.reset();
mediaPlayer.setDataSource(getContext(), Uri.parse(url));
mediaPlayer.setLooping(false);
mediaPlayer.prepareAsync();
mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer player) {
Log.d(TAG, "onPrepared changeMediaPlayerDatasource");
onReadyToPlay(player);
}
});
} catch (Exception e) {//IOException && IllegalStateException
Log.d(TAG, "textureview playNewVideo ERORR");
e.printStackTrace();
}
}
}
private void resumeVideoUponReturningFromAnotherActivity() {
if (null == mediaPlayer || null == surface) {
Log.d(TAG, "resumeVideoUponReturningFromAnotherActivity outstandingVideoRequest");
outstandingVideoRequest = true;
} else {
// playNewVideo(mVideoUrl);
Log.d(TAG, "resumeVideoUponReturningFromAnotherActivity go NOW");
mediaPlayer.setSurface(surface);
onReadyToPlay(mediaPlayer);
}
}
private void onReadyToPlay(MediaPlayer player) {
//play video
mProgressCircle.setVisibility(View.GONE);
showVideoOverlayChildren();
if (0 == mVideoSeekPosition) {
Log.d(TAG, "onReadyToPlay start");
player.start();
} else {
Log.d(TAG, "onReadyToPlay seek");
player.seekTo(mVideoSeekPosition);
player.start();
}
mHandler.postDelayed(new Runnable() {
@Override
public void run() {
Log.d(TAG, "postDelayed resumeVideo");
hideVideoOverlayChildren();
}
}, Constant.BEFORE_VIDEO_OVERLAY_DISAPPEAR);
}
private void destroyMediaPlayer() {
if (null != mediaPlayer) {//move to video todo
mediaPlayer.stop();
mediaPlayer.release();
mediaPlayer = null;
}
if (null != surface) {
surface.release();
surface = null;
}
}
private void pauseVideo() {
if (null != mediaPlayer) {
Log.d(TAG, "pause");
mediaPlayer.pause();
mVideoSeekPosition = mediaPlayer.getCurrentPosition();
}
}
private void stopVideo(){
if (null != mediaPlayer) {
Log.d(TAG, "stop video");
mediaPlayer.pause();
mVideoSeekPosition = mediaPlayer.getCurrentPosition();
mediaPlayer.stop();
}
}
@Override
public void onResume() {
super.onResume();
Log.d(TAG, "onResume");
mLocalBroadcastManager.registerReceiver(mVideoSelectionReceiver, mVideoSelectedIntentFilter);
resumeVideoUponReturningFromAnotherActivity();
}
アクティビティを切り替えたときに同じ問題が発生し、MediaPlayer(1971):エラー(100,0)も発生しました。 onSurfaceTextureDestroyed内にこれらの行を追加して解決しました
@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
if (mediaPlayer != null) {
mediaPlayer.stop();
mediaPlayer.release();
mediaPlayer = null;
}
return true;
}
コードにバグがあるようです。SurfaceTextureDestroyed()で、SurfaceまたはmediaPlayerをリセットしていません。再開すると、mediaPlayerもsurfaceもnullにならないため、resumeVideoUponReturningFromAnotherActivity()でサーフェスを設定し、再生開始を呼び出しますが、以前のSurfaceTextureDestroyedのため、サーフェスはすでに無効になっています。そのため、エラーが発生します。
これを修正するには、コールバックSurfaceTextureDestroyedでサーフェスをリセットする必要があります。再開したら、コールバックSurfaceTextureAvailableでサーフェスを再構築し、mediaPlayerに設定して、startを呼び出して再生します。コードは次のようになります。
public void onResume() {
if (mSurface == null) {
mResumeRequested = true;
return;
}
mMediaPlayer.start();
}
@Override
public void onSurfaceTextureAvailable(SurfaceTexture surface, int width, int height) {
mSurface = new Surface(surface);
if (mMediaPlayer != null) {
mMediaPlayer.setSurface(mSurface);
if (mResumeRequested) {
mMediaPlayer.start();
mResumeRequested = false;
}
}
}
@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
mSurface = null;
return false;
}
また、メディアプレーヤーをリセットする必要はまったくありません。リセットした場合は、再インスタンス化して再バッファリングする必要があります。これにより遅延が発生します。これにより、一時停止/再開の遅延が望ましくないため、ユーザーエクスペリエンスが損なわれます。
setSurface(null)
が便利だと思います。
TextureViewを使用して何かを表示する場合、_TextureView.SurfaceTextureListener
_コールバックonSurfaceTextureDestroyed
が呼び出されたときに、_camera2
_、MediaCodec
によってバインドされたSurfaceTexture/new Surface(SurfaceTexture)
の使用を停止する必要があります。またはMediaPlayer
。
このような
_@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
mediaPlayer.setDisplayer(null);
return false;//do not return true if you reuse it.
}
_