自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Android進(jìn)階之MediaPlayer和TextureView封裝視頻播放器詳解(完美實現(xiàn)全屏、小窗)

移動開發(fā) Android
SurfaceView 以及 TextureView 均繼承于 android.view.View,屬于 Android 提供的控件體系的一部分。與普通 View 不同,它們都在獨立的線程中繪制和渲染。

[[419543]]

前言

上一篇文章我們介紹了SurfaceView和TextureView的基礎(chǔ)知識點;

SurfaceView 以及 TextureView 均繼承于 android.view.View,屬于 Android 提供的控件體系的一部分。與普通 View 不同,它們都在獨立的線程中繪制和渲染。所以,相比于普通的 ImageView 它們的性能更高,因此常被用在對繪制的速率要求比較高的應(yīng)用場景中,用來解決普通 View 因為繪制的時間延遲而帶來的掉幀的問題,比如用作相機(jī)預(yù)覽、視頻播放的媒介等;

今天我們就來簡單的用TextureView封裝下視頻播放器;

一、視頻播放器方案介紹

1、videoView+mediaPlayer

videoView繼承自SurfaceView。surfaceView是在現(xiàn)有View上創(chuàng)建一個新的Window,

內(nèi)容顯示和渲染是在新的Window中,這使得SurfaceView的繪制和刷新可以在單獨的線程中進(jìn)行。

由于SurfaceView的內(nèi)容是在新建的Window中,這使得SurfaceView不能放在RecyclerView或ScrollView中,一些View中的特性也無法使用。

2、textureView+mediaPlayer

textureView不會創(chuàng)建新的窗口,它的使用跟其他普通View一樣。

考慮到以后的可擴(kuò)展性,最終采用這個方案

3、為什么使用TextureView

TextureView是在4.0(API level 14)引入的,與SurfaceView相比,它不會創(chuàng)建新的窗口來顯示內(nèi)容。它是將內(nèi)容流直接投放到View中,并且可以和其它普通View一樣進(jìn)行移動,旋轉(zhuǎn),縮放,動畫等變化。TextureView必須在硬件加速的窗口中使用。

二、TextureView使用介紹

1、TextureView被創(chuàng)建后不能直接使用,必須將其添加到ViewGroup中。

2、TextureView必須要等SurfaceTexture準(zhǔn)備就緒才能起作用,這里通常需要給TextureView設(shè)置監(jiān)聽器SurfaceTextureListener。等待onSurfaceTextureAvailable回調(diào)后,才能使用

3、TextureView創(chuàng)建和初始化

  1. //初始化一個TextureView并添加至ViewGroup或找到你的TextureView 組件 
  2.   mTextureView=new TextureView(getContext()); 
  3.   //設(shè)置畫布監(jiān)聽 
  4.   textureView.setSurfaceTextureListener(this); 
  5.   //添加至布局 
  6.   fragment.addView(textureView,new LayoutParams(LayoutParams.MATCH_PARENT,LayoutParams.MATCH_PARENT, Gravity.CENTER)); 
  7.    /** 
  8.     * TextureView準(zhǔn)備好了回調(diào) 
  9.     * @param surface 內(nèi)部畫布渲染surface 
  10.     * @param width TextureView布局寬 
  11.     * @param height TextureView布局高 
  12.     */ 
  13.    @Override 
  14.    public void onSurfaceTextureAvailable(SurfaceTexture surface, int width, int height) { 
  15.        Logger.d(TAG,"onSurfaceTextureAvailable-->width:"+width+",height:"+height); 
  16.        //這里對畫面改變、轉(zhuǎn)場播放做了處理,聲明一個mSurfaceTexture ,在TextureView發(fā)生變化時更新 
  17.        if (mSurfaceTexture == null) { 
  18.            mSurfaceTexture = surface; 
  19.            //prepare(); 
  20.        } else { 
  21.            mTextureView.setSurfaceTexture(mSurfaceTexture); 
  22.        } 
  23.    } 
  24.    /** 
  25.     * TextureView寬高發(fā)生變化時回調(diào) 
  26.     * @param surface 內(nèi)部surface 
  27.     * @param width 新的TextureView布局寬 
  28.     * @param height 新的TextureView布局高 
  29.     */ 
  30.    @Override 
  31.    public void onSurfaceTextureSizeChanged(SurfaceTexture surface, int width, int height) { 
  32.        Logger.d(TAG,"onSurfaceTextureSizeChanged-->width:"+width+",height:"+height); 
  33.    } 
  34.    /** 
  35.     * TextureView銷毀時回調(diào) 
  36.     * @param surface 內(nèi)部surface 
  37.     * @return Most applications should return true
  38.     */ 
  39.    @Override 
  40.    public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) { 
  41.        Logger.d(TAG,"onSurfaceTextureDestroyed"); 
  42.        return null==mSurfaceTexture; 
  43.    } 
  44.    /** 
  45.     * TextureView刷新時回調(diào) 
  46.     * @param surface 內(nèi)部surface 
  47.     */ 
  48.    @Override 
  49.    public void onSurfaceTextureUpdated(SurfaceTexture surface) { 
  50.    } 

三、MediaPlayer介紹

1、重要的狀態(tài)

  • idle:空閑狀態(tài)。當(dāng)mediaPlayer沒有prepareAsync之前,就是處于idle狀態(tài)。
  • prepared:準(zhǔn)備好狀態(tài)。想要讓mediaPlayer開始播放,不能直接start,必須要先prepareSync。這期間mediaPlayer會一直在準(zhǔn)備preparing,直到進(jìn)入prepared狀態(tài)。
  • started:當(dāng)mediaPlayer準(zhǔn)備好,就可以調(diào)用mediaPlayer的start方法進(jìn)入started狀態(tài)。
  • paused:當(dāng)調(diào)用pause方法,進(jìn)入paused狀態(tài)。
  • completed:播放完成,進(jìn)入completed狀態(tài)。
  • error:播放錯誤。

2、重要的方法

  • prepareAsync:要想使用mediaPlayer,必須先調(diào)用prepareAsync。這是第一步。
  • start:開始
  • pause:暫停
  • reset:播放完成后,如想重新開始,調(diào)用該方法。

3、重要的回調(diào)

  • onSurfaceTextureAvailable:開始關(guān)聯(lián)mediaPlayer
  • onPrepared:此處開始調(diào)用mediaPlayer.start()
  • onInfo:播放開始后,視頻到底狀態(tài)如何,就是在onInfo中處理
  1. @Override 
  2.    public boolean onInfo(MediaPlayer mp, int what, int extra) { 
  3.        if (what == MediaPlayer.MEDIA_INFO_VIDEO_RENDERING_START) { 
  4.            // 播放器渲染第一幀 
  5.            mCurrentState = STATE_PLAYING; 
  6.            mController.onPlayStateChanged(mCurrentState); 
  7.        } else if (what == MediaPlayer.MEDIA_INFO_BUFFERING_START) { 
  8.            // MediaPlayer暫時不播放,以緩沖更多的數(shù)據(jù) 
  9.            if (mCurrentState == STATE_PAUSED || mCurrentState == STATE_BUFFERING_PAUSED) { 
  10.                mCurrentState = STATE_BUFFERING_PAUSED; 
  11.            } else { 
  12.                mCurrentState = STATE_BUFFERING_PLAYING; 
  13.            } 
  14.            mController.onPlayStateChanged(mCurrentState); 
  15.        } else if (what == MediaPlayer.MEDIA_INFO_BUFFERING_END) { 
  16.            // 填充緩沖區(qū)后,MediaPlayer恢復(fù)播放/暫停 
  17.            if (mCurrentState == STATE_BUFFERING_PLAYING) { 
  18.                mCurrentState = STATE_PLAYING; 
  19.                mController.onPlayStateChanged(mCurrentState); 
  20.            } 
  21.            if (mCurrentState == STATE_BUFFERING_PAUSED) { 
  22.                mCurrentState = STATE_PAUSED; 
  23.                mController.onPlayStateChanged(mCurrentState); 
  24.            } 
  25.        } else { 
  26.            LogUtil.d("onInfo ——> what:" + what); 
  27.        } 
  28.        return true
  29.    } 

4、MediaPlayer初始化和準(zhǔn)備播放

  1. mMediaPlayer = new MediaPlayer(); 
  2.    mMediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC); 
  3.    //設(shè)置準(zhǔn)備播放監(jiān)聽器,在onPrepared回調(diào)中開始播放 
  4.    mMediaPlayer.setOnPreparedListener(this); 
  5.    //...此處省去一系列監(jiān)聽設(shè)置 
  6.    //異步準(zhǔn)備 
  7.    mMediaPlayer.prepareAsync(); 
  8.    /** 
  9.     * 播放器準(zhǔn)備好了 
  10.     * @param mp 解碼器 
  11.     */ 
  12.    @Override 
  13.    public void onPrepared(MediaPlayer mp) { 
  14.        Logger.d(TAG,"onPrepared"); 
  15.        if(null!=mSurfaceTexture){ 
  16.            if(null!=mSurface){ 
  17.                mSurface.release(); 
  18.                mSurface=null
  19.            } 
  20.            mSurface =new Surface(mSurfaceTexture); 
  21.            mp.setSurface(mSurface); 
  22.        } 
  23.        //開始播放 
  24.        mp.start(); 
  25.    } 

四、封裝視頻播放器

1、封裝播放器

視頻播放控件應(yīng)該包含兩層:頂層是播放器的控制器mController,底層是播放視頻內(nèi)容的TextureView。這里將這兩層封裝在一個容器FrameLayout中;

  1. public VideoPlayer(@NonNull Context context, @Nullable AttributeSet attrs) { 
  2.       super(context, attrs); 
  3.       mContext = context; 
  4.       if (mNetworkChangeReceiver == null) { 
  5.           mNetworkChangeReceiver = new NetworkChangeReceiver(this); 
  6.       } 
  7.       allow4GFlag = false
  8.       init(); 
  9.   } 
  10.   private void init() { 
  11.       mContainer = new FrameLayout(mContext); 
  12.       mContainer.setBackgroundColor(Color.BLACK); 
  13.       LayoutParams params = new LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT); 
  14.       this.addView(mContainer, params); 
  15.   } 

addTextureView

  1. private void addTextureView() { 
  2.         mContainer.removeView(mTextureView); 
  3.         LayoutParams params = new LayoutParams( 
  4.                 ViewGroup.LayoutParams.MATCH_PARENT, 
  5.                 ViewGroup.LayoutParams.MATCH_PARENT, 
  6.                 Gravity.CENTER); 
  7.         mContainer.addView(mTextureView, 0, params); 

setController

  1. public void setController(IVideoController controller) { 
  2.        mContainer.removeView(mController); 
  3.        mController = controller; 
  4.        mController.reset(); 
  5.        mController.setVideoPlayer(this); 
  6.        LayoutParams params = new LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT); 
  7.        mContainer.addView(mController, params); 
  8.    } 

播放,將TextureView、MediaPlayer、Controller進(jìn)行初始化。待TextureView的數(shù)據(jù)通道SurfaceTexture準(zhǔn)備就緒后,打開播放器

  1. private void openMediaPlayer() { 
  2.         // 屏幕常亮 
  3.         mContainer.setKeepScreenOn(true); 
  4.         // 設(shè)置監(jiān)聽 
  5.         mMediaPlayer.setOnPreparedListener(this); 
  6.         mMediaPlayer.setOnVideoSizeChangedListener(this); 
  7.         mMediaPlayer.setOnCompletionListener(this); 
  8.         mMediaPlayer.setOnErrorListener(this); 
  9.         mMediaPlayer.setOnInfoListener(this); 
  10.         mMediaPlayer.setOnBufferingUpdateListener(this); 
  11.         mCurrentNetworkState = NetworkChangeReceiver.getNetworkStatus(CtripBaseApplication.getInstance()); 
  12.         mNetworkChangeReceiver.registerNetworkChangeBroadcast(); 
  13.         // 設(shè)置dataSource 
  14.         try { 
  15.             mMediaPlayer.setDataSource(mUrl); 
  16.             if (mSurface == null) { 
  17.                 mSurface = new Surface(mSurfaceTexture); 
  18.             } 
  19.             mMediaPlayer.setSurface(mSurface); 
  20.             mMediaPlayer.prepareAsync(); 
  21.             mCurrentState = STATE_PREPARING; 
  22.             mController.onPlayStateChanged(mCurrentState); 
  23.         } catch (IOException e) { 
  24.             e.printStackTrace(); 
  25.             LogUtil.e("打開播放器發(fā)生錯誤", e); 
  26.         } 
  27.     } 
  28.     private void initMediaPlayer() { 
  29.         if (mMediaPlayer == null) { 
  30.             mMediaPlayer = new MediaPlayer(); 
  31.             mMediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC); 
  32.         } 
  33.     } 
  34.     private void initTextureView() { 
  35.         if (mTextureView == null) { 
  36.             mTextureView = new TourTextureView(mContext); 
  37.             mTextureView.setSurfaceTextureListener(this);//此時回調(diào)onSurfaceTextureAvailable 
  38.         } 
  39.     } 
  40.     @Override 
  41.     public void onSurfaceTextureAvailable(SurfaceTexture surfaceTexture, int i, int i1) { 
  42.         if (mSurfaceTexture == null) { 
  43.             mSurfaceTexture = surfaceTexture; 
  44.             openMediaPlayer(); 
  45.         } else { 
  46.             mTextureView.setSurfaceTexture(mSurfaceTexture); 
  47.         } 
  48.     } 

播放邏輯寫完之后,具體UI展示邏輯在VideoPlayerController中。根據(jù)不同的狀態(tài)VideoPlayerController展示不同UI

  1. public static final int STATE_ERROR = -1;               //播放錯誤 
  2.    public static final int STATE_IDLE = 0;                 //播放未開始 
  3.    public static final int STATE_PREPARING = 1;            //播放準(zhǔn)備中 
  4.    public static final int STATE_PREPARED = 2;             //播放準(zhǔn)備就緒 
  5.    public static final int STATE_PLAYING = 3;              //正在播放 
  6.    public static final int STATE_PAUSED = 4;               //暫停播放 
  7.    public static final int STATE_BUFFERING_PLAYING = 5;    //正在緩沖 
  8.    public static final int STATE_BUFFERING_PAUSED = 6;     //正在緩沖 播放器 
  9.    public static final int STATE_COMPLETED = 7;            //播放完成 
  10.    public static final int STATE_NOTE_4G = 8;              //提示4G 
  11.    public static final int STATE_NOTE_DISCONNECT = 9;      //提示斷網(wǎng) 
  12.    public static final int MODE_NORMAL = 10;               //普通模式 
  13.    public static final int MODE_FULL_SCREEN = 11;          //全屏模式 
  14.    public static final int MODE_TINY_WINDOW = 13;          //小窗口模式 

2、全屏、小窗口播放的實現(xiàn)

實現(xiàn)全屏:將mContainer移除,并添加到android.R.content中,并設(shè)置成橫屏

  1. @Override 
  2.    public void enterFullScreen() { 
  3.        if (mCurrentMode == MODE_FULL_SCREEN) return
  4.        // 隱藏ActionBar、狀態(tài)欄,并橫屏 
  5.        TourVideoUtil.hideActionBar(mContext); 
  6.        TourVideoUtil.scanForActivity(mContext) 
  7.                .setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE); 
  8.        new Handler().post(new Runnable() { 
  9.            @Override 
  10.            public void run() { 
  11.                ViewGroup contentView = (ViewGroup) TourVideoUtil.scanForActivity(mContext) 
  12.                        .findViewById(android.R.id.content); 
  13.                if (mCurrentMode == MODE_TINY_WINDOW) { 
  14.                    contentView.removeView(mContainer); 
  15.                } else { 
  16.                    TourVideoPlayer.this.removeView(mContainer); 
  17.                } 
  18.                LayoutParams params = new LayoutParams( 
  19.                        ViewGroup.LayoutParams.MATCH_PARENT, 
  20.                        ViewGroup.LayoutParams.MATCH_PARENT); 
  21.                contentView.addView(mContainer, params); 
  22.            } 
  23.        }); 
  24.        mCurrentMode = MODE_FULL_SCREEN; 
  25.        mController.onPlayModeChanged(mCurrentMode); 
  26.    } 

實現(xiàn)小窗口:將mContainer移除,添加到android.R.content中,并設(shè)置寬高

  1. @Override 
  2.    public void enterTinyWindow() { 
  3.        if (mCurrentMode == MODE_TINY_WINDOW) return
  4.        this.removeView(mContainer); 
  5.        new Handler().post(new Runnable() { 
  6.            @Override 
  7.            public void run() { 
  8.                ViewGroup contentView = (ViewGroup) TourVideoUtil.scanForActivity(mContext) 
  9.                        .findViewById(android.R.id.content); 
  10.                // 小窗口的寬度為屏幕寬度的60%,長寬比默認(rèn)為16:9,右邊距、下邊距為8dp。 
  11.                FrameLayout.LayoutParams params = new FrameLayout.LayoutParams( 
  12.                        (int) (CommonUtil.getScreenWidth(mContext) * 0.6f), 
  13.                        (int) (CommonUtil.getScreenWidth(mContext) * 0.6f * 9f / 16f)); 
  14.                params.gravity = Gravity.TOP | Gravity.START; 
  15.                params.topMargin = CommonUtil.dp2px(mContext, 48f); 
  16.                contentView.addView(mContainer, params); 
  17.            } 
  18.        }); 
  19.        mCurrentMode = MODE_TINY_WINDOW; 
  20.        mController.onPlayModeChanged(mCurrentMode); 
  21.    } 

總結(jié)

關(guān)于視頻播放器封裝的知識點還有很多,今天知識簡單的介紹了下封裝的步驟和思路;

大家如果想自己封裝可以參考網(wǎng)上NiceVieoPlayer;

以后會繼續(xù)講解關(guān)于視頻播放器的知識點;

本文轉(zhuǎn)載自微信公眾號「Android開發(fā)編程」

責(zé)任編輯:姜華 來源: Android開發(fā)編程
相關(guān)推薦

2021-08-25 07:43:17

AndroidSurfaceViewTextureView

2017-03-01 14:01:31

android多媒體音樂代碼

2011-07-20 16:21:20

iPhone 視頻 播放器

2022-08-16 17:37:06

視頻播放器鴻蒙

2011-09-06 10:46:19

QT播放器

2015-05-21 15:25:42

VLC播放器

2024-04-23 08:24:05

音頻Android播放

2022-06-21 14:41:38

播放器適配西瓜視頻

2012-06-04 13:44:08

2011-09-05 18:08:01

MTK音頻播放器

2021-08-10 20:41:33

AndroidApp流程

2011-09-09 11:28:35

Android Mus

2021-10-19 14:27:07

鴻蒙HarmonyOS應(yīng)用

2021-10-21 16:00:07

鴻蒙HarmonyOS應(yīng)用

2022-11-12 08:26:04

VLC視頻播放器裁剪視頻

2023-04-06 13:47:47

2010-06-11 12:53:56

openSUSE播放器

2021-08-17 13:41:11

AndroidView事件

2018-05-25 14:37:58

2021-09-02 07:00:01

Glide流程Android
點贊
收藏

51CTO技術(shù)棧公眾號