Android使用MediaPlayer和TextureView實(shí)現(xiàn)視頻無(wú)縫切換
需求描述
比如廣告播放,每個(gè)視頻15秒,視頻之間切換的時(shí)候,性能不太好的機(jī)器可能會(huì)黑屏一段時(shí)間,體驗(yàn)不太好,接下來(lái)我們就是要解決這個(gè)黑屏問(wèn)題。
解決方案
使用兩個(gè)surfaceView方式,經(jīng)過(guò)測(cè)試不行
使用一個(gè)MediaPlayer,在MediaPlayer上面加一層ImageView,每次播放完成后,獲取視頻的最后一幀的圖像給ImageView,視頻切換完成,ImageView隱藏,如此往復(fù)循環(huán),可行
實(shí)踐
1.獲取視頻流圖片方式,通過(guò)MediaMetadataRetriever,測(cè)試發(fā)現(xiàn),部分機(jī)器獲取的Bitmap可能為空,無(wú)法解決,放棄
2.使用TextureView方式,可以獲取當(dāng)前幀的Bitmap,可行,下面貼代碼
package com.winson.blog.video;
import android.graphics.Bitmap;
import android.graphics.SurfaceTexture;
import android.media.MediaPlayer;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.util.Log;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.Surface;
import android.view.TextureView;
import android.view.View;
import android.view.ViewGroup;
import android.widget.FrameLayout;
import android.widget.ImageView;
import java.io.IOException;
public class VideoFragment extends Fragment {
public static final String TAG = VideoFragment.class.getSimpleName();
String TEST_PATH = Environment.getExternalStorageDirectory().getAbsolutePath() + "/testmp4.mp4";
int mIndex = 0;
String path1 = TEST_PATH;
String[] paths = new String[]{TEST_PATH, TEST_PATH, TEST_PATH, TEST_PATH, TEST_PATH, TEST_PATH};
boolean destory;
Handler mHandler;
Runnable mPlayRun;
FrameLayout content;
TextureView textureView;
ImageView frameImage;
MediaPlayer mediaPlayer;
Bitmap lastFrameBitmap;
public void updateResources(String[] paths) {
this.paths = paths;
if(mHandler != null && mPlayRun!= null){
mHandler.post(mPlayRun);
}
}
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mHandler = new Handler();
mediaPlayer = new MediaPlayer();
mPlayRun = new Runnable() {
@Override
public void run() {
if (mediaPlayer == null || destory) {
return;
}
mediaPlayer.pause();
mediaPlayer.reset();
try {
String path = paths[mIndex % paths.length];
mIndex++;
mediaPlayer.setDataSource(getActivity(), Uri.parse(path));
mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer arg0) {
mediaPlayer.start();
frameImage.setVisibility(View.GONE);
}
});
mediaPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
@Override
public void onCompletion(MediaPlayer mp) {
frameImage.setVisibility(View.VISIBLE);
Bitmap currentFrameBitmap = textureView.getBitmap();
frameImage.setImageBitmap(currentFrameBitmap);
if (lastFrameBitmap != null) {
lastFrameBitmap.recycle();
}
lastFrameBitmap = currentFrameBitmap;
mHandler.post(mPlayRun);
}
});
mediaPlayer.prepareAsync();
} catch (Exception e) {
e.printStackTrace();
}
}
};
}
public void release() {
mHandler.removeCallbacks(mPlayRun);
if (mediaPlayer != null) {
mediaPlayer.pause();
mediaPlayer.release();
}
}
public Bitmap getBitmap() {
return textureView == null ? null : textureView.getBitmap();
}
@Override
public void onDestroy() {
super.onDestroy();
release();
}
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
content = new FrameLayout(getActivity());
FrameLayout.LayoutParams lp = new FrameLayout.LayoutParams(FrameLayout.LayoutParams.MATCH_PARENT, FrameLayout.LayoutParams.MATCH_PARENT);
lp.gravity = Gravity.LEFT | Gravity.TOP;
textureView = new TextureView(getActivity());
textureView.setLayoutParams(lp);
content.addView(textureView);
frameImage = new ImageView(getActivity());
frameImage.setScaleType(ImageView.ScaleType.FIT_XY);
frameImage.setLayoutParams(lp);
content.addView(frameImage);
textureView.setSurfaceTextureListener(new TextureView.SurfaceTextureListener() {
@Override
public void onSurfaceTextureAvailable(SurfaceTexture surface, int width, int height) {
Surface s = new Surface(surface);
mediaPlayer.setSurface(s);
}
@Override
public void onSurfaceTextureSizeChanged(SurfaceTexture surface, int width, int height) {
}
@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
return false;
}
@Override
public void onSurfaceTextureUpdated(SurfaceTexture surface) {
}
});
return content;
}
@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
testPlay();
}
public void testPlay() {
// mediaPlayer.pause();
// mediaPlayer.reset();
//
// try {
// mediaPlayer.setDataSource(getActivity(), Uri.parse(TEST_PATH));
// mediaPlayer.prepare();
// mediaPlayer.start();
// } catch (IOException e) {
// e.printStackTrace();
// }
mHandler.post(mPlayRun);
}
}
相關(guān)鏈接,github地址
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
android使用FlipAnimation實(shí)現(xiàn)3D垂直翻轉(zhuǎn)動(dòng)畫(huà)
這篇文章主要為大家詳細(xì)介紹了android使用FlipAnimation實(shí)現(xiàn)3D垂直翻轉(zhuǎn)動(dòng)畫(huà),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12
Android提高之SQLite分頁(yè)表格實(shí)現(xiàn)方法
這篇文章主要介紹了Android提高之SQLite分頁(yè)表格實(shí)現(xiàn)方法,在項(xiàng)目開(kāi)發(fā)中有很高的實(shí)用價(jià)值,需要的朋友可以參考下2014-08-08
Android實(shí)現(xiàn)3秒鐘自動(dòng)關(guān)閉界面
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)3秒鐘自動(dòng)關(guān)閉界面,以支付成功為例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-02-02
Unity同步/異步調(diào)用Android的方法實(shí)例
unity在Android端開(kāi)發(fā)的時(shí)候,免不了要調(diào)用Java,下面這篇文章主要給大家介紹了關(guān)于Unity同步/異步調(diào)用Android的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2021-08-08
SwipeRefreshLayout+RecyclerView實(shí)現(xiàn)上拉刷新和下拉刷新功能
這篇文章主要介紹了SwipeRefreshLayout+RecyclerView實(shí)現(xiàn)上拉刷新和下拉刷新功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01
Android實(shí)現(xiàn)擴(kuò)展Menu的方法
這篇文章主要介紹了Android實(shí)現(xiàn)擴(kuò)展Menu的方法,涉及Android操作menu菜單的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10
Android編程之canvas繪制各種圖形(點(diǎn),直線(xiàn),弧,圓,橢圓,文字,矩形,多邊形,曲線(xiàn),圓角矩形)
這篇文章主要介紹了Android編程之canvas繪制各種圖形的方法,涉及Android使用Canvas類(lèi)中常用繪圖方法的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-12-12
Android socket如何實(shí)現(xiàn)文件列表動(dòng)態(tài)訪(fǎng)問(wèn)
本文介紹Android socket實(shí)現(xiàn)文件列表動(dòng)態(tài)訪(fǎng)問(wèn),訪(fǎng)問(wèn)文件夾之后通過(guò)listview展示,并在點(diǎn)擊文件夾后進(jìn)入文件夾,獲得其內(nèi)容,有此需求的朋友可以參考下2021-06-06
Android實(shí)現(xiàn)Toast提示框圖文并存的方法
這篇文章主要介紹了Android實(shí)現(xiàn)Toast提示框圖文并存的方法,實(shí)例分析了Toast提示框的參數(shù)設(shè)置及圖文調(diào)用的相關(guān)技巧,需要的朋友可以參考下2016-01-01

